您好我是MPI编程的新手。我试图将两个矩阵相乘(NxN矩阵(A)和Nx1(B)矩阵)以得到结果C矩阵(Nx1)。每个进程应该计算矩阵C中的行(元素),但是只有进程0(我的主进程)正确计算,因为它似乎没有等待其他进程完成计算。我也不确定非主过程是否正确地发回结果(或者如果他们甚至需要?)。这是我的代码:
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include "mpi.h"
#define PRINT_VECS 1
#define MAX_RAND 100
#define MASTER 0
#define COLUMNS_B 1
#define N 4
void init_vec(int *vec, int len);
void print_vec(const char *label, int *vec, int len);
void init_vec(int *vec, int len)
{
int i;
for (i = 0; i < len; i++)
{
vec[i] = rand() % MAX_RAND;
}
}
void print_vec(const char *label, int *vec, int len)
{
#if PRINT_VECS
printf("%s", label);
int i;
for (i = 0; i < len; i++)
{
printf("%d ", vec[i]);
}
printf("\n\n");
#endif
}
void init_matrix(int** matrix, int rows, int cols)
{
int i,j;
for (i = 0; i < rows; i++)
{
for (j = 0; j < cols; j++)
{
matrix[i][j] = rand() % MAX_RAND;
}
}
}
void print_matrix(int** matrix, int rows, int cols)
{
int i;
for (i = 0; i < rows; i++)
{
printf("|");
int j;
for (j = 0; j < cols; j++)
{
printf("%d ", matrix[i][j]);
}
printf("|\n");
}
}
int main(int argc, char *argv[])
{
int my_rank;
int num_procs;
MPI_Init(&argc, &argv);
MPI_Comm_rank(MPI_COMM_WORLD, &my_rank); //grab this process's rank
MPI_Comm_size(MPI_COMM_WORLD, &num_procs); //grab the total num of processes
int results[num_procs]; // used to store the partial sum computed
int rows, cols, colsB;
rows = N;
cols = N;
colsB = COLUMNS_B;
int **A; // N x N Matrix
int B[N]; // N x 1 Matrix
int **C; // N x 1 Matrix
double start_time; // use these for timing
double stop_time;
if (my_rank == MASTER)
{
printf("Number of processes: %d\n", num_procs);
printf("N: %d\n", N);
srand(time(NULL));
// init A
int i;
A = malloc(rows * sizeof *A);
for (i = 0; i < rows; i++)
{
A[i] = malloc(cols * sizeof *A[i]);
}
init_matrix(A, rows, cols);
printf("Matrix A:\n");
print_matrix(A, rows, cols);
// init B
init_vec(B, N);
print_vec("Matrix B:\n", B, N);
// init C
C = malloc(rows * sizeof *C);
for (i = 0; i < rows; i++)
{
C[i] = malloc(colsB * sizeof *C[i]);
}
start_time = MPI_Wtime();
}
MPI_Bcast(B, N, MPI_INT, 0, MPI_COMM_WORLD);
//MPI_Bcast(A, N, MPI_INT, 0, MPI_COMM_WORLD);
int row = my_rank;
int my_sum = 0;
int i;
if (my_rank < N)
{
for (i = 0; i < N; i++)
{
int num = A[row][i] * B[i];
my_sum = my_sum + num;
}
C[row] = &my_sum;
printf("HAI FROM PROCESS %d! I will calculate row %d. My calculation: %d\n", my_rank, row, my_sum);
}
//MPI_Gather(&C, 1, MPI_INT, results, 1, MPI_INT, 0, MPI_COMM_WORLD);
if (my_rank == MASTER)
{
stop_time = MPI_Wtime();
printf("\nMatrix C:\n");
print_matrix(C, rows, colsB);
printf("Total time (sec): %f\n", stop_time - start_time);
}
MPI_Finalize();
return EXIT_SUCCESS;;
}
我很确定我很接近,但我只是遗漏了一些东西。我已经尝试添加一些注释掉的语句,同时广播A矩阵,和/或调用MPI_GATHER,但似乎没有任何东西给出除了主过程之外的任何过程的结果,所以显然我仍然做错了。以下是一些示例输出:
Number of processes: 28
N: 4
Matrix A:
|11 30 69 24 |
|83 38 66 71 |
|68 71 27 33 |
|58 5 50 10 |
Matrix B:
1 58 81 44
HAI FROM PROCESS 0! I will calculate row 0. My calculation: 8396
Matrix C:
|8396 |
|-2107258888 |
|-2107258920 |
|-2107258888 |
Total time (sec): 0.000078
所以proc 0正在计算正确,但是我的错误信息是proc 1正在获得seg错误,我无法弄清楚原因。我得到的错误是:
mpirun noticed that process rank 1 with PID 0 exited on signal 11 (Segmentation fault).
非常感谢任何帮助!
答案 0 :(得分:1)
在C
中,2D矩阵是一个指针数组和这是行主要,这意味着一个2D矩阵是一个指针数组,一个给定的指针指向一行(不一列)。
话虽如此,MPI
期望行在连续的内存中,因此,指针数组不适合MPI。
静态数组非常适合,但如果需要动态数组,则需要一次性分配矩阵&#34;然后手动构建指针数组。
所以你可以替换
A = malloc(rows * sizeof *A);
for (i = 0; i < rows; i++)
{
A[i] = malloc(cols * sizeof *A[i]);
}
与
A = (int **)malloc(cols * sizeof(int *);
A[0] = (int *)malloc(cols * rows * sizeof(int));
for (i = 1; i < cols; i++)
{
A[i] = A[i-1] + rows;
}
然后您将能够使用
广播您的矩阵MPI_Bcast(A[0], cols*rows, MPI_INT, 0, MPI_COMM_WORLD);
请参阅MPI_Bcast a dynamic 2d array了解冗长的解释
fwiw,我不明白你对C
数据的逻辑。
据我所知,这是一个简单的向量,所以声明它是这样的,不要使用指针(因为MPI将无法正确处理它们)
答案 1 :(得分:1)
这是您的程序已修复问题:
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include "mpi.h"
#define PRINT_VECS 1
#define MAX_RAND 100
#define MASTER 0
#define COLUMNS_B 1
#define N 4
void init_vec(int *vec, int len);
void print_vec(const char *label, int *vec, int len);
void init_vec(int *vec, int len)
{
int i;
for (i = 0; i < len; i++)
{
vec[i] = rand() % MAX_RAND;
}
}
void print_vec(const char *label, int *vec, int len)
{
#if PRINT_VECS
printf("%s", label);
int i;
for (i = 0; i < len; i++)
{
printf("%d ", vec[i]);
}
printf("\n\n");
#endif
}
void init_matrix(int** matrix, int rows, int cols)
{
int i,j;
for (i = 0; i < rows; i++)
{
for (j = 0; j < cols; j++)
{
matrix[i][j] = rand() % MAX_RAND;
}
}
}
void print_matrix(int** matrix, int rows, int cols)
{
int i;
for (i = 0; i < rows; i++)
{
printf("|");
int j;
for (j = 0; j < cols; j++)
{
printf("%d ", matrix[i][j]);
}
printf("|\n");
}
}
int main(int argc, char *argv[])
{
int my_rank;
int num_procs;
MPI_Init(&argc, &argv);
MPI_Comm_rank(MPI_COMM_WORLD, &my_rank); //grab this process's rank
MPI_Comm_size(MPI_COMM_WORLD, &num_procs); //grab the total num of processes
int results[num_procs]; // used to store the partial sum computed
int rows, cols, colsB, k;
rows = N;
cols = N;
colsB = COLUMNS_B;
int **A; // N x N Matrix
int B[N]; // N x 1 Matrix
int C[N]; // N x 1 Matrix
// Allocate memory for the NxN matrix on all processes
A = (int**) malloc(N * sizeof(int*));
for(k=0;k<N;k++)
A[k]= (int*) malloc(N * sizeof(int));
double start_time; // use these for timing
double stop_time;
if (my_rank == MASTER)
{
printf("Number of processes: %d\n", num_procs);
printf("N: %d\n", N);
srand(time(NULL));
// Initilize arrays on root only
init_matrix(A, rows, cols);
printf("Matrix A:\n");
print_matrix(A, rows, cols);
init_vec(B, N);
print_vec("Matrix B:\n", B, N);
start_time = MPI_Wtime();
}
// Be consistent with names vs. values to avoid bugs
MPI_Bcast(B, N, MPI_INT, MASTER, MPI_COMM_WORLD);
for (k=0; k<N; k++)
MPI_Bcast(&(A[k][0]), N, MPI_INT, MASTER, MPI_COMM_WORLD);
int row = my_rank;
int my_sum = 0;
int i,num;
if (my_rank < N)
{
for (i = 0; i < N; i++)
{
num = A[row][i] * B[i];
my_sum = my_sum + num;
}
C[row] = my_sum;
printf("HAI FROM PROCESS %d! I will calculate row %d. My calculation: %d\n", my_rank, row, my_sum);
}
MPI_Gather(&C[row], 1, MPI_INT, &C[row], 1, MPI_INT, MASTER, MPI_COMM_WORLD);
if (my_rank == MASTER)
{
stop_time = MPI_Wtime();
print_vec("Matrix C:\n", C, N);
printf("Total time (sec): %f\n", stop_time - start_time);
}
// Free matrix A
for(k=0;k<N;k++)
free(A[k]);
free(A);
MPI_Finalize();
return EXIT_SUCCESS;
}
如评论中所述,在这种情况下,您需要为所有流程的所有矩阵分配内存。该过程与程序仅在根进程上执行的A和B的初始化不同。这里A使用malloc
分配,而C是静态分配的,并且进一步用作矢量,方式与B相同。这不是必需的,但似乎是一个更好的选择,因为C是一维数组并且在它的本质等同于B.
B像以前一样广播到所有进程,但是程序使用MASTER
代替0,所以当你有机会改变MASTER
的值时,它的所有出现也会改变。这通常是一种很好的编程习惯,并且它在新代码中的任何地方都适用。
A以一种简单但肯定不如@Gilles Gouailardet建议的方式广播 - 该节目只是单独播放A的每一行,
for (k=0; k<N; k++)
MPI_Bcast(&(A[k][0]), N, MPI_INT, MASTER, MPI_COMM_WORLD);
这与行主要排序和连续访问A中第k行的这N个元素的事实有关。如果按列发送A,则会失败。
剩余的更改是将my_sum
的值分配给C[row]
,C[row] = my_sum;
以及收集操作,而不是指向它的指针:
MPI_Gather(&C[row], 1, MPI_INT, &C[row],
1, MPI_INT, MASTER, MPI_COMM_WORLD);
此处每个进程将其值C[row]
发送到根进程上的C[row]
。使用print_vec
函数在根目录上打印C.