发送到孤立的 linux 进程组的 SIGCONT 和 SIGHUP 的顺序

Tra*_*acy 3 c unix linux signals process

APUE 说

由于进程组在父进程终止时是孤立的,POSIX.1 要求新孤立进程组中的每个停止的进程(就像我们的孩子一样)被发送挂断信号(SIGHUP),然后是继续信号(SIGCONT)

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <signal.h>
#include <errno.h>
#define errexit(msg) do{ perror(msg); exit(EXIT_FAILURE); } while(0)
static void sig_hup(int signo)
{
    printf("SIGHUP received, pid = %d\n", getpid());
}
static void sig_cont(int signo)
{
    printf("SIGCONT received, pid = %d\n", getpid());
}
static void sig_ttin(int signo)
{
    printf("SIGTTIN received, pid = %d\n", getpid());
}
static void pr_ids(char *name)
{
    printf("%s: pid = %d, ppid = %d, pgrp = %d, tpgrp = %d\n",
           name, getpid(), getppid(), getpgrp(), tcgetpgrp(STDIN_FILENO));
}
int main(int argc, char *argv[])
{
    char c;
    pid_t pid;
    setbuf(stdout, NULL);
    pr_ids("parent");
    if ((pid = fork()) < 0) {
            errexit("fork error");
    } else if (pid > 0) { /* parent */
            sleep(5);
            printf("parent exit\n");
            exit(0);
    } else { /* child */
            pr_ids("child...1");
            signal(SIGCONT, sig_cont);
            signal(SIGHUP, sig_hup);
            signal(SIGTTIN, sig_ttin);
            kill(getpid(), SIGTSTP);
            //sleep(10);
            pr_ids("child...2");
            if (read(STDIN_FILENO, &c, 1) != 1) {
                    printf("read error from controlling TTY, errno = %d\n", 
                                                             errno);
            }
            printf("child exit\n");
    }
    exit(0);
  }
Run Code Online (Sandbox Code Playgroud)

程序输出:

parent: pid = 2036, ppid = 1959, pgrp = 2036, tpgrp = 2036
child...1: pid = 2037, ppid = 2036, pgrp = 2036, tpgrp = 2036
parent exit
xiejingfeng@xiekupuejing/ $ SIGCONT received, pid = 2037
SIGHUP received, pid = 2037
child...2: pid = 2037, ppid = 1, pgrp = 2036, tpgrp = 1959
read error from control TTY, errno = 5
child exit

输出不像书中所说的那样预期,因为程序首先接收 SIGCONT 然后 SIGHUP,这对我来说很困惑,你们能帮我吗?

提前致谢。

pil*_*row 6

SIGHUP 在子进程恢复之前无法传递。当一个进程停止时,除了 SIGCONT 和 SIGKILL 之外的所有信号传递都被挂起。

因此,SIGHUP 确实首先到达,但在 SIGCONT 唤醒进程执行之前无法处理它。

  • @Tracy,不客气。如果答案有帮助,请毫不犹豫地投票和/或接受它,谢谢。 (2认同)