【问题标题】:Sharing an array of structs using mmap使用 mmap 共享结构数组
【发布时间】:2014-02-08 17:35:40
【问题描述】:

我正在尝试创建一个在父进程和子进程之间共享的结构数组。尝试访问数组数据时出现分段错误。

我确信问题与我使用指针的方式有关,因为这是我不太习惯的领域。

请注意,我删除了大部分看起来不相关的代码。

/* 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++;
}

【问题讨论】:

  • 你的调试器说了什么?
  • @H2CO3 我没有运行任何调试器,因为这是一个网络应用程序,我不知道如何使用调试工具。错误返回是segmentation fault

标签: c arrays struct mmap


【解决方案1】:

table 被 mmap 分配后,您还没有初始化它的内容。所以它包含垃圾。所以strcmp(table[i].name, packet_reg.data) 有很大的机会遍历分配的缓冲区并访问例如未分配的内存。

  1. 正确初始化表;
  2. 使用 strncmp 进行比较。

【讨论】:

  • 你能详细说明我应该如何初始化table吗?我确实初始化了base_tabletable 应该指向同一段内存。澄清一下:我在调用 mmap 后尝试向共享内存写入一个值,但我得到了同样的错误(段错误)。谢谢。
  • base_tabletable 无关,它们是不同的对象。根据 mmap man 的说法:“如果 addr 不为 NULL,那么内核会将其作为关于在何处放置映射的提示;在 Linux 上,映射将在附近的页面边界处创建。新映射的地址返回为通话的结果。”所以你应该调用 mmap 并调用你为 base_table 而不是 table 的初始化代码。
  • 是的,很好的一点,我完全错过了返回的内存地址不同。但是,手册页确实说当使用MAP_ANONYMOUS 时,值被初始化为0。我的主要问题是我在需要PROT_READ | PROT_WRITE 时使用了PROT_NONE。如果您可以修改答案以提及这一点,我愿意接受。谢谢。
猜你喜欢
  • 1970-01-01
  • 2016-04-14
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2011-02-17
  • 2013-01-11
  • 2016-11-07
  • 1970-01-01
相关资源
最近更新 更多