1

我正在处理一个家庭作业问题,使用 fork() 和共享内存对象在 C 中编写 collat​​z 猜想的实现,在子进程中执行计算并在父进程中打印结果。我对C不是很熟悉,所以我学习了很多工具。使用 gdb,当我尝试访问父进程中的对象时,我发现了一个段错误。我正在使用这是我的代码:

#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <sys/mman.h>
#include <math.h>

int main(int argc, const char* argv[])
{

const int SIZE  = 4096;
const char *name = "SM";
int shm_fd;
void  *ptr;

pid_t child;


if ((argc != 2) || (strtol(argv[1],NULL, 10) <= 0))
{
        printf("Invalid usage: requires 1 positive integer parameter\n");
        return -1;
}

child = fork();
if(child >=0 )
{
        if (child == 0)
        {
                shm_fd = shm_open(name, O_CREAT || O_RDWR, 0666);
                ftruncate(shm_fd, SIZE);
                ptr = mmap(0, SIZE, PROT_WRITE, MAP_SHARED, shm_fd, 0);
                int currentval = strtol(argv[1],NULL,10);
                sprintf(ptr,"%d",currentval);
                ptr++;/*  floor(log10(abs(currentval))) + 1;*/
                while (currentval > 1)
                {
                        sprintf(ptr, ", ");
                        ptr += 2;
                        if (currentval % 2 == 1)
                        {
                                currentval = currentval * 3 + 1;
                                sprintf(ptr, "%d", currentval);
                                ptr++;
                        }
                        else
                        {
                                currentval = currentval/2;
                                sprintf(ptr, "%d", currentval);
                                ptr++;
                        }
                }
                sprintf(ptr, "\n");
                ptr+= 1;
                return 0;
        }
        else
        {
                wait();

                shm_fd = shm_open(name, O_RDONLY, 0666);
                ptr = mmap(0, SIZE, PROT_READ, MAP_SHARED, shm_fd, 0);
                printf("%s\n",(char *)ptr);
                shm_unlink(name);
                return 0;
        }
}
else
{
        printf("error creating child process\n");
}
return 0;
}

我以前从未调试过段错误,所以任何建议都会受到欢迎。提前致谢。

4

3 回答 3

1

I found the issue. It was doing a logical OR instead of a bitwise OR, so was creating a bad file descriptor.

shm_fd = shm_open(name, O_CREAT || O_RDWR, 0666);

should have been

shm_fd = shm_open(name, O_CREAT | O_RDWR, 0666);

于 2013-10-02T17:35:44.557 回答
1

我知道这与您提出的问题无关,但是自从我遇到这个问题以来,对于其他可能遇到的其他人可能会发现以下有用:

在阅读和打印 Collat​​z 猜想的系列时,如果举的例子是 8,上面的程序运行良好,因为它的答案是:8, 4, 2, 1 但是如果你尝试运行它 10,它的答案是:10、5、16、8、4、2、1,而上面的程序只会显示:1、5、16、8、4、2、1。因为它一次读取一个字符。所以这里需要做一点修改。

int shift = 0;
char store[sizeof(int)];
while (currentval > 1)
{
        sprintf(ptr, ", ");
        ptr += 2;
        if (currentval % 2 == 1)
        {
                currentval = currentval * 3 + 1;
        }
        else
        {
                currentval = currentval/2;
        }
        sprintf(store, "%s", currentval);  /* Convert the number to a string */
        sprintf(ptr, "%s", store); /* Store the string */ 
        shift = strlen(store);
        ptr += shift;
}
sprintf(ptr, "\n");
ptr+= 1;            
于 2015-07-15T22:06:29.687 回答
0

您能否粘贴使用 gdb 创建的核心文件的回溯:

gdb -c <core file> <executable>

它将在 gdb 中打开核心,然后键入 "bt" 。这将为您提供问题所在的确切位置。您也可以在此处粘贴该输出。

于 2013-09-30T04:07:29.297 回答