使用mmap共享结构体数组

3
我正在尝试创建一个结构体数组,该数组在父进程和子进程之间共享。但是当我尝试访问数组数据时,出现了分段错误。
我确信问题与指针的使用方式有关,因为这是我不太熟悉的领域。
请注意,我已编辑掉了大部分似乎不相关的代码。
/* structure of Registration Table */
struct registrationTable{
    int port;
    char name[MAXNAME];
    int req_no;
};

main() {

    /* the registrationTable is to be a shared memory space with each child
    process able to access and update. No concurrency controls are 
     implemented. The parent process is responsible for cleaning up after
     the kiddies.
 */
struct registrationTable base_table[REG_TABLE_SIZE];
for (int i = 0; i < REG_TABLE_SIZE; i++) {
    base_table[i].req_no = 0;
    memset(base_table[i].name, '\0', MAXNAME);
    base_table[i].port = 0;
}

printf("\nMapping Shared Memory\n");

//set up shared memory space
//void *mmap(void *addr, size_t length, int prot, int flags,
    //              int fd, off_t offset);
//      addr = NONE, prot = PROT_NONE, flags = MAP_SHARED 
struct registrationTable *table = mmap(base_table, sizeof(base_table),
                        PROT_READ | PROT_WRITE, 
                        MAP_SHARED | MAP_ANONYMOUS,
                        -1, 0);

while(1){
    pid_t child = fork();

    if (child == 0) {//is child

        for(int i = 0; i < REG_TABLE_SIZE; i++) {

            printf("\nExamining table looking for client at %s port: %d\n", 
                    packet_reg.data, clientAddr.sin_port);

            printf("\ntable[1].req_no: %d", ta[i].req_no);

                            //segmentation fault on next line
            if (strcmp(table[i].name, packet_reg.data) == 0 
                    && table[i].port == clientAddr.sin_port) {
                table[i].req_no++;
}

你的调试器说了什么? - user529758
@H2CO3 我还没有运行任何调试器,因为这是一个网络应用程序,我不确定如何使用调试工具。错误返回是“段错误”。 - Destruktor
1个回答

3

在使用mmap分配内存后,您没有初始化table的内容。因此它包含垃圾数据。因此,strcmp(table[i].name, packet_reg.data)有很大的机会超过分配的缓冲区并访问未分配的内存。

  1. 正确地初始化表格;
  2. 在比较时使用strncmp。

你能详细说明一下我应该如何初始化 table 吗?我已经初始化了 base_table,而且 table 应该指向同一段内存。为了澄清:在调用 mmap 后,我尝试向共享内存写入值,但是出现了相同的错误(段错误)。谢谢。 - Destruktor
base_tabletable没有任何关系,它们是不同的对象。根据mmap手册:“如果addr不为NULL,则内核将其视为有关放置映射的提示;在Linux上,映射将在附近的页面边界处创建。新映射的地址作为调用的结果返回。”因此,您应该调用mmap并调用您针对base_table的初始化代码,但是针对table进行操作。 - c-smile
是的,非常好的观点,我完全忽视了返回的内存地址是不同的这一点。然而,手册确实说当使用MAP_ANONYMOUS时,这些值会被初始化为0. 我的主要问题是我在需要使用PROT_READ | PROT_WRITE时使用了PROT_NONE. 如果您能修改您的答案提到这一点,我愿意接受它。谢谢。 - Destruktor

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接