Ответ 1
Просто, чтобы усилить точки Джоэля:
Это намного проще, если вы распределите свои массивы так, чтобы они были смежными (что-то C "многомерные массивы" не дают вам автоматически:)
int **alloc_2d_int(int rows, int cols) {
int *data = (int *)malloc(rows*cols*sizeof(int));
int **array= (int **)malloc(rows*sizeof(int*));
for (int i=0; i<rows; i++)
array[i] = &(data[cols*i]);
return array;
}
/*...*/
int **A;
/*...*/
A = alloc_2d_init(N,M);
Затем вы можете отправлять и получать весь массив NxM с помощью
MPI_Send(&(A[0][0]), N*M, MPI_INT, destination, tag, MPI_COMM_WORLD);
и когда вы закончите, освободите память с помощью
free(A[0]);
free(A);
Кроме того, MPI_Recv
- это получение блокировки, а MPI_Send
может быть блокировкой отправки. Одна вещь, которая означает, по мнению Джоэла, состоит в том, что вам определенно не нужны барьеры. Кроме того, это означает, что если у вас есть шаблон отправки/получения, как указано выше, вы можете попасть в тупик - все отправляются, никто не получает. Безопаснее:
if (myrank == 0) {
MPI_Send(&(A[0][0]), N*M, MPI_INT, 1, tagA, MPI_COMM_WORLD);
MPI_Recv(&(B[0][0]), N*M, MPI_INT, 1, tagB, MPI_COMM_WORLD, &status);
} else if (myrank == 1) {
MPI_Recv(&(A[0][0]), N*M, MPI_INT, 0, tagA, MPI_COMM_WORLD, &status);
MPI_Send(&(B[0][0]), N*M, MPI_INT, 0, tagB, MPI_COMM_WORLD);
}
Другой, более общий подход - использовать MPI_Sendrecv
:
int *sendptr, *recvptr;
int neigh = MPI_PROC_NULL;
if (myrank == 0) {
sendptr = &(A[0][0]);
recvptr = &(B[0][0]);
neigh = 1;
} else {
sendptr = &(B[0][0]);
recvptr = &(A[0][0]);
neigh = 0;
}
MPI_Sendrecv(sendptr, N*M, MPI_INT, neigh, tagA, recvptr, N*M, MPI_INT, neigh, tagB, MPI_COMM_WORLD, &status);
или неблокировать отправку и/или получение.