5

私の目標は、メイン プロセスとその「フォーク」の子プロセスを相互通信することです。通信はシグナル配信によって行われます。

私の問題は、最初の子が 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);
    }
}
4

1 に答える 1

3

この問題は、親がフォークした直後に子プロセスにシグナルを送信し、子プロセスがシグナルをブロックしていないという事実に関係している可能性が最も高いです。したがってsigsuspend()、子プロセスを呼び出すときまでに、シグナルはすでに子に配信されており、シグナルが決して来ないのを待っているだけです。sleep()信号の送信を開始する前に、メイン プロセスで を 1 ~ 2 秒間呼び出すことで、この理論をすばやくテストできます。コードは現在構造化されているsigsuspend()ため、待機しているシグナルのシグナルハンドラーがないと正しく機能しないことに注意してください...したがって、このようなシグナルを操作する場合は、次のことをお勧めします。

  1. 親プロセスでは、親プロセスと子プロセス間の通信に使用する予定のすべてのシグナルをブロックします。これを呼び出す必要がありますsigprocmask()
  2. 親に子プロセスをフォークさせる
  3. sigwait()子プロセスでは、通信に使用されているブロックされたシグナルを含むシグナルセットを使用して呼び出すだけです...sigsuspend()ここで行っていることは必要ありません。
  4. 親プロセスが子プロセスにシグナルを送信した後、sigwait()子プロセスの応答を待機するために呼び出すこともできます。

動作するコードの例を次に示します: http://ideone.com/TRcqga

于 2012-12-05T22:40:56.720 に答える