sigsuspend() 不对信号作出反应

xsu*_*ira 5 c posix fork signals suspend

我的目标是相互通信主要进程及其“叉”子进程。通信是通过信号传递完成的。

当第一个孩子在等待 SIGUSR1 信号时卡住等待时,我的问题就出现了。

我不知道为什么它会卡在这一点上。甚至,如果我通过控制台发送信号,该子进程似乎没有注意到。

有人可以帮我吗?


代码来了

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <signal.h>

int N = 5;
int _pipe[2];
pid_t children[5];

void main(){
    pid_t parent_pid;
    pid_t pid;
    int i = 0;

    sigset_t set;
    sigfillset(&set);

    parent_pid = getpid();
    fprintf(stderr,"I am main process, here comes my pid %u\n",getpid());

    if (0>pipe(_pipe)) fprintf(stderr,"Error when creating pipe");

    //Start creating child processes
    while (i < N){
            pid = fork();
            if (pid == 0){
                close(_pipe[1]);
            break;
        }
        else{
            fprintf(stderr,"Created child with pid %u\n",pid);
            children[i] = pid;
            write(_pipe[1],&pid,sizeof(pid_t));
        }
        i = i+1;
    }

    i = 0;

    // What main process does
    if (pid>0){
        close(_pipe[0]);
        close(_pipe[1]);

        sigdelset(&set,SIGUSR2);

        sigdelset(&set,SIGTERM);
        sigdelset(&set,SIGKILL);

        // Main process sends signal to each child
        while(i < N){           
            kill(children[i],SIGUSR1);
            fprintf(stderr,"Sent SIGUSR1 to child %u\n",children[i]);
            // .. Now just wait for SIGUSR2 arrival
            sigsuspend(&set);

            i = i+1;
        }
    }
    // What children do
    else{
        // Wait for main process SIGUSR1 delivery
        sigdelset(&set,SIGUSR1);
        sigsuspend(&set);

        fprintf(stderr, "SIGUSR1 arrived child %u from its father",getpid());

        // Once SIGUSR1 has arrived, pipe is read N times
        while((i < N) && (read(_pipe[0],&pid,sizeof(pid_t))>0)){
            children[i] = pid;
            i = i+1;
        }
        close(_pipe[0]);

        // After reading pipe, a reply is sent to parent process
        kill(parent_pid,SIGUSR2);
    }
}
Run Code Online (Sandbox Code Playgroud)

Jas*_*son 3

该问题很可能与以下事实有关:父进程在派生子进程后立即向子进程发送信号,并且子进程没有阻止该信号。因此,当您调用sigsuspend()子进程时,信号已经传递给子进程,现在它只是坐在那里等待永远不会到来的信号。sleep()您可以通过在主进程开始发送信号之前调用一两秒钟来快速测试这个理论。请记住,由于您的代码现在是结构化的,sigsuspend()如果没有您正在等待的信号的信号处理程序,将无法正常工作......所以我建议在处理这样的信号时执行以下操作:

  1. 在父进程中,阻止您计划用于父进程和子进程之间通信的所有信号。sigprocmask()这个你需要打电话询问。
  2. 让父进程 fork 子进程
  3. 在子进程中,只需sigwait()使用包含用于通信的阻塞信号的信号集进行调用...您不需要sigsuspend()在这里执行的操作。
  4. 父进程向子进程发送信号后,它也可以调用sigwait()等待子进程的回复。

这是一个有效的代码示例:http://ideone.com/TRcqga