【发布时间】:2014-05-28 16:50:47
【问题描述】:
我从SE QUE得到以下信息 将 SIGCHLD 的配置显式设置为 SIG_IGN 会导致任何随后终止的子进程立即从系统中删除,而不是被转换为僵尸。
据我所知,要读取子状态,进程表中需要有子 pid。所以有必要在进程表中有僵尸子进程来读取子状态。
所以我想编写信号处理程序,它将删除“将 SIGCHLD 的处置设置为 SIG_IGN”
我使用下面的代码(在 fork 之前)避免在终止后立即删除子进程:但我仍然无法获得子进程并且 waitpid 返回 -1 和 ECHILD。
#include <errno.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
static siginfo_t sig_info;
static volatile sig_atomic_t sig_num;
static void *sig_ctxt;
static void catcher(int signum, siginfo_t *info, void *vp)
{
sig_num = signum;
sig_info = *info;
sig_ctxt = vp;
}
static void set_handler(int signum)
{
struct sigaction sa;
sa.sa_flags = SA_SIGINFO;
sa.sa_sigaction = catcher;
sigemptyset(&sa.sa_mask);
if (sigaction(signum, &sa, 0) != 0)
{
int errnum = errno;
fprintf(stderr, "Failed to set signal handler (%d: %s)\n", errnum, strerror(errnum));
exit(1);
}
}
static void prt_interrupt(FILE *fp)
{
if (sig_num != 0)
{
fprintf(fp, "Signal %d from PID %d\n", sig_info.si_signo, (int)sig_info.si_pid);
sig_num = 0;
}
}
请提出一些想法,我被这个阻止了。
【问题讨论】:
-
如果您阅读关于 wait(2) 的注释,您会发现显式设置 SIG_IGN 会导致立即收获孩子。但是,默认操作(也是“忽略”)不会执行此操作。因此,要获得僵尸,您所要做的就是不为 CIGCHLD 设置信号操作,或者,AFAIK,将其设置为 SIG_DFL。
-
如何不为 SIGCHLD 或 AFAIK 设置信号操作?将其设置为 SIG_DFL?您能否提供示例实现链接?
-
只需不要为 SIGCHLD 设置处理程序!在您显示的代码中,切勿调用 set_handler(SIGCHLD)。我可以从个人经验告诉你,如果你不使用 SIGCHLD,你会得到僵尸。
标签: c linux signals waitpid sigchld