/* -*- Mode: C; c-basic-offset:4 ; indent-tabs-mode:nil ; -*- */ /* * (C) 2009 by Argonne National Laboratory. * See COPYRIGHT in top-level directory. */ /* Based on a test case contributed by Michael Hofmann. * * This test makes sure that zero counts with non-zero-sized types on the * send (recv) side match and don't cause a problem with non-zero counts and * zero-sized types on the recv (send) side when using MPI_Alltoallw and * MPI_Alltoallv. */ /* TODO test intercommunicators as well */ #include #include #include #include "mpitest.h" int main(int argc, char *argv[]) { int sendbuf, recvbuf; int *sendcounts; int *recvcounts; int *sdispls; int *rdispls; MPI_Datatype sendtype; MPI_Datatype *sendtypes; MPI_Datatype *recvtypes; int rank = -1; int size = -1; int i; MPI_Init(&argc, &argv); MPI_Comm_size(MPI_COMM_WORLD, &size); MPI_Comm_rank(MPI_COMM_WORLD, &rank); sendtypes = malloc(size * sizeof(MPI_Datatype)); recvtypes = malloc(size * sizeof(MPI_Datatype)); sendcounts = malloc(size * sizeof(int)); recvcounts = malloc(size * sizeof(int)); sdispls = malloc(size * sizeof(int)); rdispls = malloc(size * sizeof(int)); if (!sendtypes || !recvtypes || !sendcounts || !recvcounts || !sdispls || !rdispls) { printf("error, unable to allocate memory\n"); goto fn_exit; } MPI_Type_contiguous(0, MPI_INT, &sendtype); MPI_Type_commit(&sendtype); for (i = 0; i < size; ++i) { sendtypes[i] = sendtype; sendcounts[i] = 1; sdispls[i] = 0; recvtypes[i] = MPI_INT; recvcounts[i] = 0; rdispls[i] = 0; } /* try zero-counts on both the send and recv side in case only one direction is broken for some reason */ MPI_Alltoallw(&sendbuf, sendcounts, sdispls, sendtypes, &recvbuf, recvcounts, rdispls, recvtypes, MPI_COMM_WORLD); MPI_Alltoallw(&sendbuf, recvcounts, rdispls, recvtypes, &recvbuf, sendcounts, sdispls, sendtypes, MPI_COMM_WORLD); #if MTEST_HAVE_MIN_MPI_VERSION(2,2) /* pass MPI_IN_PLACE and different but compatible types rank is even/odd */ if (rank % 2) MPI_Alltoallw(MPI_IN_PLACE, NULL, NULL, NULL, &recvbuf, recvcounts, rdispls, recvtypes, MPI_COMM_WORLD); else MPI_Alltoallw(MPI_IN_PLACE, NULL, NULL, NULL, &recvbuf, sendcounts, sdispls, sendtypes, MPI_COMM_WORLD); #endif /* now the same for Alltoallv instead of Alltoallw */ MPI_Alltoallv(&sendbuf, sendcounts, sdispls, sendtypes[0], &recvbuf, recvcounts, rdispls, recvtypes[0], MPI_COMM_WORLD); MPI_Alltoallv(&sendbuf, recvcounts, rdispls, recvtypes[0], &recvbuf, sendcounts, sdispls, sendtypes[0], MPI_COMM_WORLD); #if MTEST_HAVE_MIN_MPI_VERSION(2,2) if (rank % 2) MPI_Alltoallv(MPI_IN_PLACE, NULL, NULL, MPI_DATATYPE_NULL, &recvbuf, recvcounts, rdispls, recvtypes[0], MPI_COMM_WORLD); else MPI_Alltoallv(MPI_IN_PLACE, NULL, NULL, MPI_DATATYPE_NULL, &recvbuf, sendcounts, sdispls, sendtypes[0], MPI_COMM_WORLD); #endif MPI_Type_free(&sendtype); if (rank == 0) printf(" No Errors\n"); fn_exit: if (rdispls) free(rdispls); if (sdispls) free(sdispls); if (recvcounts) free(recvcounts); if (sendcounts) free(sendcounts); if (recvtypes) free(recvtypes); if (sendtypes) free(sendtypes); MPI_Finalize(); return 0; }