为什么在 c linux 中尝试使用共享内存的 IPC 时出现分段错误

Why am i getting segmentation fault when trying IPC using shared memory in c linux

问题:在 3 个进程之间实现 IPC,第一个进程创建共享内存,然后向第二个和第三个进程发送信号,以便它们可以附加到该共享内存,现在让第二个进程写入一些东西到共享内存,然后让它向第三个进程发送一个信号,以便第三个进程读取第二个进程写入的内容。

我的代码:

#include <signal.h> 
#include <stdio.h> 
#include <stdlib.h> 
#include <string.h> 
#include <sys/ipc.h> 
#include <sys/shm.h> 
#include <sys/types.h> 
#include <unistd.h> 

static int pid0, pid1, pid2, shmid, status;

struct memory { 
    char msg[100];
}; 

struct memory* shmptr; 
struct sigaction act0, act1, act2;

// Handler For P0
void handler0(int signum) 
{ 
    if (signum == SIGUSR1) { 
        printf("Allowing P1 And P2 To Get Attached\n\n");
        kill(pid1, SIGUSR1); // Allow P1 To Get Attached
        kill(pid2, SIGUSR1); // Allow P2 To Get Attached
    } 

    if (signum == SIGUSR2) { 
        kill(pid1, SIGINT); // Kill P1 
        kill(pid2, SIGINT); // Kill P2 
    } 
} 

// Handler For P1
void handler1(int signum) 
{ 
    if (signum == SIGUSR1) { 
        
        // Attaching To The Shared Memory 
        shmptr = (struct memory*) shmat(shmid, 0, 0); 

        printf("P1 - Attached to: %d\n\n", shmid); 
    } 
} 

// Handler For P2
void handler2(int signum) 
{ 
    if (signum == SIGUSR1) { 
        
        // Attaching To The Shared Memory 
        shmptr = (struct memory*) shmat(shmid, 0, 0); 

        printf("P2 - Attached to: %d\n\n", shmid); 
    } 

    if (signum == SIGUSR2) { 

        printf("Message Read From Shared Memory Is: "); 
        puts(shmptr->msg); 

        status = 1;

        // Detaching P2 From The Shared Memory
        shmdt((void*)shmptr);
        
        kill(pid0, SIGUSR2);
    }
} 

int main() 
{
    status = 0;

    printf("P0\n\n");

    pid0 = getpid();

    // Key Of Shared Memory 
    key_t key = 123456;

    // Shared Memory Creation
    printf("Creating Shared Memory\n\n"); 
    shmid = shmget(key, sizeof(struct memory), IPC_CREAT | 0666); 

    // Attaching P0 To The Shared Memory 
    shmptr = (struct memory*)shmat(shmid, 0, 0); 
    printf("P0 - Attached to: %d\n\n", shmid);

    // Attaching Handler Function To Signal SIGUSR1 For Process 0
    act0.sa_handler = handler0;
    sigaction(SIGUSR1, &act0, NULL);
    printf("Forking...\n\n");
    
    pid1 = vfork();

    if(pid1 == 0){ // Inside P1

        printf("P1\n\n");
        
        // Attaching Handler Function To Signal SIGUSR1 For Process 1
        act1.sa_handler = handler1;
        sigaction(SIGUSR1, &act1, NULL);

        pid2 = vfork();

        if(pid2 == 0){ // Inside P2

            printf("P2\n\n");

            // Attaching Handler Functions To Signals SIGUSR1 And SIGUSR2 For Process 2
            act2.sa_handler = handler2;
            sigaction(SIGUSR1, &act2, NULL);
            sigaction(SIGUSR2, &act2, NULL);
            kill(pid0, SIGUSR1); // Tell P0 That P1 And P2 Are Ready To Get Attached To The Shared Memory
            
        } else { // Inside P1

            printf("P1 Again\n\n");

            // Writing A Message To The Shared Memory
            printf("Enter A Message: \n");
            fgets(shmptr->msg, 100, stdin); 

            // Sending The Message To Process 2
            kill(pid2, SIGUSR2); 
            printf("Sent Message To P2\n\n"); // this gets printed and then a segmentation fault error occurs
        }

    } else { // Inside P0
    
        printf("P0 - Waiting..\n\n");
        while(1){
            if(status == 1)
                break;
        }
        // Detaching P0 From The Shared Memory
        shmdt((void*)shmptr);

        shmctl(shmid, IPC_RMID, (struct shmid_ds *) NULL);
    }
}

执行 printf("Sent Message To P2\n\n"); 后出现分段错误

请告诉我为什么,我想不通

谢谢

您不能使用 vfork() 执行此操作。

阅读手册页,尤其是靠近顶部的部分

the behavior is undefined if the process created by vfork() either modifies any data other than a variable of type pid_t used to store the return value from vfork(), or returns from the function in which vfork() was called, or calls any other function before successfully calling _exit(2) or one of the exec(3) family of functions.