【问题标题】:Child process receives parent's SIGINT子进程收到父进程的 SIGINT
【发布时间】:2013-08-17 04:31:58
【问题描述】:

我有一个使用 Qt 框架的简单程序。 它使用 QProcess 来执行 RAR 并压缩一些文件。在我的程序中,我正在捕捉 SIGINT 并在它发生时在我的代码中做一些事情:

signal(SIGINT, &unix_handler);

SIGINT 发生时,我检查 RAR 进程是否完成,如果没有,我将等待它......问题是(我认为)RAR 进程也会得到 SIGINT,这意味着我的程序在压缩所有文件之前就退出了。

有没有办法运行 RAR 进程,这样当我的程序收到它时它不会收到 SIGINT

谢谢

【问题讨论】:

  • 目前我在 Debian 5.0.8 上测试它

标签: c++ c linux unix


【解决方案1】:

如果您在 Unix 系统上使用 Ctrl+C 生成 SIGINT,则信号将发送到整个 process group

您需要使用setpgidsetsid将子进程放到不同的进程组中,这样子进程就不会收到控制终端产生的信号了。


[编辑:]

请务必仔细阅读setpgid 页面的基本原理部分。在这里插入所有潜在的竞争条件有点棘手。

要保证 100% 不会将SIGINT 传递给您的子进程,您需要执行以下操作:

#define CHECK(x) if(!(x)) { perror(#x " failed"); abort(); /* or whatever */ }
/* Block SIGINT. */
sigset_t mask, omask;
sigemptyset(&mask);
sigaddset(&mask, SIGINT);
CHECK(sigprocmask(SIG_BLOCK, &mask, &omask) == 0);

/* Spawn child. */
pid_t child_pid = fork();
CHECK(child_pid >= 0);
if (child_pid == 0) {
    /* Child */
    CHECK(setpgid(0, 0) == 0);
    execl(...);
    abort();
}
/* Parent */
if (setpgid(child_pid, child_pid) < 0 && errno != EACCES)
    abort(); /* or whatever */
/* Unblock SIGINT */
CHECK(sigprocmask(SIG_SETMASK, &omask, NULL) == 0);

严格来说,这些步骤中的每一步都是必要的。如果用户在调用fork 后立即点击Ctrl+C,您必须阻止该信号。您必须在孩子中调用setpgid,以防execl 在父母有时间做任何事情之前发生。您必须在父级中调用setpgid,以防 父级 运行并且有人在 子级之前点击 Ctrl+C em> 有时间做任何事。

上面的序列很笨拙,但它确实处理了 100% 的竞争条件。

【讨论】:

  • 非常感谢。我认为这就是我的问题的答案。我会尽快尝试的;)
  • 它正在工作,我刚刚添加了正确的包含和这一行:setpgid(rar.pid(), 0);
  • 没问题。请注意,setpgid 充满了竞争条件,正确使用它有点微妙。我已更新我的答案以使其更完整。
  • 感谢您提供更多信息 ;)
  • @Jan:嗯,我不同意assert 适用于“仅当您的程序中存在错误时才为真”的表达式。在这种情况下,setpgid(0,0) 只有在 C 库或内核中存在错误时才会失败,并且使用assert 检查这种“不可能”的情况是完全合理的。但是您对副作用有很好的看法。我以为assert(x) 保证会评估x,但我错了。我已经恢复到你的版本。
【解决方案2】:

你在你的处理程序中做什么?只有某些 Qt 函数可以从 unix 信号处理程序中安全调用。文档中的This page 标识了它们是什么。

主要问题是处理程序将在主 Qt 事件线程之外执行。该页面还提出了一种处理此问题的方法。我更喜欢让处理程序将自定义事件“发布”到应用程序并以这种方式处理它。我发布了一个答案,描述了如何实现自定义事件here

【讨论】:

  • 感谢您提供此页面,我不知道。我只是调用退出并等待我的线程,然后退出应用程序。它适用于除 RAR 之外的所有内容,所以我认为这不是问题。但我会读这个页面。再次感谢