0

在这个小代码示例中,我在 MPI(使用 C++)中发现了一些意外行为:

int rank, size;
MPI_Init(&argc, &argv);
MPI_Comm_rank(MPI_COMM_WORLD, &rank);
MPI_Comm_size(MPI_COMM_WORLD, &size);

if(rank == 1) {
    int *sendDone = (int*)malloc(sizeof(int));    
    *sendDone = 1;
    MPI_Ssend(sendDone,1,MPI_INT, 0, 1,MPI_COMM_WORLD);  
    foo();    
} else {
    int *rcvDone = (int*)malloc(sizeof(int));
    bar();
    while(*rcvDone != 1) {
        MPI_Recv(rcvDone,1,MPI_INT, MPI_ANY_SOURCE, 1, MPI_COMM_WORLD, MPI_STATUS_IGNORE);
    }
    cout << *rcvDone << endl;
}
MPI_Finalize();

它使用以下命令编译和运行:

mpic++ sim.cc -o sim
mpirun -n 2 ./sim

代码应按以下顺序执行:

bar();
foo();

因为在执行bar()之后 Process #0 开始接收。但实际上,foo()有时会在bar()完成之前开始。有人可以向我解释一下并解决问题吗?

4

1 回答 1

2
  1. 您还没有说如何检查首先调用哪个函数。在屏幕上计算某些内容并不能保证显示消息的正确顺序(至少在使用 MPI 时)。

  2. 您不需要将 MPI_Recv 放入循环中,因为它是一个阻塞函数。当您没有为 *rcvDone 分配起始值时,甚至不推荐它。

  3. 将 malloc 与一些 MPI 函数一起使用是不安全的。阅读http://www.mcs.anl.gov/research/projects/mpi/www/www3/MPI_Ssend.html上的“线程和中断安全”部分

于 2013-07-11T14:22:26.857 回答