c - MPI 调试,段错误?

标签 c debugging mpi

编辑:我的问题类似于 C, Open MPI: segmentation fault from call to MPI_Finalize(). Segfault does not always happen, especially with low numbers of processes ,所以如果你回答那个,那将是很好的,无论哪种方式。 . .

我希望得到一些调试以下代码的帮助:

int main(){
        long* my_local;
        long n, s, f;
        MPI_Init(NULL, NULL);
        MPI_Comm_size(MPI_COMM_WORLD, &comm_sz);
        MPI_Comm_rank(MPI_COMM_WORLD, &my_rank);

    if(my_rank == 0){
        /*  Get size n from user                            */
        printf("Total processes: %d\n", comm_sz);
        printf("Number of keys to be sorted?  ");
        fflush(stdout);
        scanf("%ld", &n);

        /*  Broadcast size n to other processes             */
        MPI_Bcast(&n, 1, MPI_LONG, 0, MPI_COMM_WORLD);

        /*  Create n/comm_sz keys                           
             NOTE! some processes will have 1 extra key if
              n%comm_sz != 0                                */
        create_Keys(&my_local, my_rank, comm_sz, n, &s, &f);
    }

    if(my_rank != 0){
        /* Receive n from process 0                         */
        MPI_Bcast(&n, 1, MPI_LONG, 0, MPI_COMM_WORLD);

        /*  Create n/comm_sz keys                           */
        create_Keys(&my_local, my_rank, comm_sz, n, &s, &f);
    }

        /* The offending function, f is a long set to num elements of my_local*/
        Odd_Even_Tsort(&my_local, my_rank, f, comm_sz);

        printf("Process %d completed the function", my_rank);
        MPI_Finalize();
        return 0;
}

void Odd_Even_Tsort(long** my_local, int my_rank, long my_size, int comm_sz)
{
    long nochange = 1;
    long phase = 0;
    long complete = 1;
    MPI_Status Stat;
    long your_size = 1;

    long* recv_buf = malloc(sizeof(long)*(my_size+1));
    printf("rank %d has size %ld\n", my_rank, my_size);

    while (complete!=0){
        if((phase%2)==0){
            if( ((my_rank%2)==0) && my_rank < comm_sz-1){
            /*  Send right                          */
                MPI_Send(&my_size, 1, MPI_LONG, my_rank+1, 0, MPI_COMM_WORLD);
                MPI_Send(*my_local, my_size, MPI_LONG, my_rank+1, 0, MPI_COMM_WORLD);
                MPI_Recv(&your_size, 1, MPI_LONG, my_rank+1, 0,  MPI_COMM_WORLD, &Stat);
                MPI_Recv(&recv_buf, your_size, MPI_LONG, my_rank+1, 0,  MPI_COMM_WORLD, &Stat);
            }
            if( ((my_rank%2)==1) && my_rank < comm_sz){
            /*  Send left                          */
                MPI_Recv(&your_size, 1, MPI_LONG, my_rank-1, 0,  MPI_COMM_WORLD, &Stat);
                MPI_Recv(&recv_buf, your_size, MPI_LONG, my_rank-1, 0,  MPI_COMM_WORLD, &Stat);
                MPI_Send(&my_size, 1, MPI_LONG, my_rank-1, 0, MPI_COMM_WORLD);
                MPI_Send(*my_local, my_size, MPI_LONG, my_rank-1, 0, MPI_COMM_WORLD);
            }
        }
        phase ++;
        complete = 0;
    }

    printf("Done!\n");
    fflush(stdout);
}

我得到的错误是:

[ubuntu:04968] *** Process received signal ***
[ubuntu:04968] Signal: Segmentation fault (11)
[ubuntu:04968] Signal code: Address not mapped (1)
[ubuntu:04968] Failing at address: 0xb
--------------------------------------------------------------------------
mpiexec noticed that process rank 1 with PID 4968 on node ubuntu exited on signal 11 (Segmentation fault).

我莫名其妙的是,函数后面的打印语句仍然显示,但是如果我注释掉函数,就没有错误。那么,我在哪里遇到段错误?我在使用 mpiexec -n 2 ./a.out 和大于 9 的“n”大小时遇到​​错误。

如果您真的想要完整的可运行代码,请告诉我。真的,我希望的不是精确答案,而是更多如何使用 gdb/valgrind 工具来调试这个问题和其他类似问题(以及如何读取它们的输出)。

(是的,我意识到“排序”功能尚未排序)。

最佳答案

这里的问题很简单,但很难发现,除非你使用调试器或打印出详尽的调试信息:

查看调用MPI_Recv 的代码。 recv_buf 变量应该作为参数而不是 &recv_buf 提供。

  MPI_Recv(   recv_buf  , your_size, MPI_LONG, my_rank-1, 0,  MPI_COMM_WORLD, &Stat);

其余的似乎没问题。

关于c - MPI 调试,段错误?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10407771/

相关文章:

python - 派生进程能否与 "main"MPI 通信器通信

visual-studio-2010 - 防止 Visual Studio 2010 MPI 群集调试器在每次执行之前将所有项目二进制文件复制到临时用户文件夹

objective-c - 取消之前调用的 dispatch_async

c - 这里发生了什么? sizeof(short_int_variable + char_variable)

调用 sprintf 函数

php - 在 php 中有没有办法用相应的值转储 "all"变量名?

android - 适用于Android IDE的Android错误/崩溃检查插件

c++ - 运行调试器时,带有空格字符的程序参数被 Eclipse 破坏

fortran - MPI_CART_CREATE 没有特定的子例程

c - 逐字读取文件