pclose()间歇性地返回SIGPIPE

时间:2014-04-26 07:22:49

标签: linux multithreading signals posix popen

当执行以下C程序并且SIGUSR1被重复发送到正在运行的进程时,pclose()调用有时会返回13. 13对应于我系统上的SIGPIPE。< / p>

为什么会这样?

我正在使用while true; do kill -SIGUSR1 <process-id>; done向程序发送SIGUSR1。该程序在Ubuntu 14.04上执行。

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

void handler(int i) {}

void* task(void*)
{
    FILE *s;
    char b [BUFSIZ];
    while (1) {
        if ((s = popen("echo hello", "r")) == NULL) {
            printf("popen() failed\n");
        }
        while (fgets(b, BUFSIZ, s) != NULL) ;
        if (int r = pclose(s)) {
            printf("pclose() failed (%d)\n", r);
        }
    }

    return 0;
}

int main(int argc, char **argv)
{
    struct sigaction action;
    action.sa_handler = handler;
    sigemptyset(&action.sa_mask);
    action.sa_flags = 0;
    sigaction(SIGUSR1, &action, NULL);

    pthread_t tid;
    pthread_create(&tid, 0, task, NULL);
    pthread_join(tid, NULL);
}

1 个答案:

答案 0 :(得分:2)

fgets被信号中断时会发生这种情况。该程序不会读取管道到最后并关闭它。另一个程序然后是SIGPIPE。

正确的管道读取操作是:

 do {
    while (fgets(b, BUFSIZ, s) != NULL) ;
 } while (errno == EINTR);