gpt4 book ai didi

在 BSD 上使用 sigtimedwait() 捕获 SIGCHLD

转载 作者:行者123 更新时间:2023-11-30 17:50:50 25 4
gpt4 key购买 nike

我在 FreeBSD 上使用 sigtimedwait() 捕获 SIGCHLD 信号时遇到问题。以下源代码在 Debian GNU/Linux 7 上运行良好,但为我提供了在 FreeBSD 9.1 上暂时不可用的资源:

#include <stdio.h>
#include <signal.h>
#include <errno.h>
#include <stdlib.h>
#include <time.h>

int main() {
sigset_t set;
pid_t pid;

printf("SIGCHLD is %i\n", SIGCHLD);

sigemptyset(&set);
sigaddset(&set, SIGCHLD);
sigprocmask(SIG_BLOCK, &set, NULL);

pid = fork();

if(pid == -1) {
printf("fork failed: %s\n", strerror(errno));
exit(1);
} else if(pid) {
sigset_t set2;
siginfo_t siginfo;
struct timespec timeout = {3, 0};
int signal;

sigemptyset(&set2);
sigaddset(&set2, SIGCHLD);

signal = sigtimedwait(&set2, &siginfo, &timeout);

if(signal == -1) {
printf("sigtimedwait failed: %s\n", strerror(errno));
exit(2);
} else {
printf("received signal %i from %i with status %i\n", signal, siginfo.si_pid, siginfo.si_status);
}
} else {
sleep(1);
exit(123);
}

return 0;
}

Linux 上的输出:

SIGCHLD is 17
received signal 17 from 27600 with status 123

FreeBSD 上的输出:

SIGCHLD is 20
sigtimedwait failed: Resource temporarily unavailable

使用 signal() 在 BSD 上工作得很好,但这并不是我想要的。我错过了什么?

最佳答案

我认为这是 FreeBSD 中的一个内核/库错误。看起来 sigtimedwait 没有报告信号,因为默认情况下它被忽略。所以你可以做两件事

  1. 为 SIGCHLD 事件安装信号处理程序。即使它永远不会被调用,因为您阻止了信号,它也会解决该错误。

  2. 将 kqueue 与 EVFILT_SIGNAL 一起使用,在这种情况下肯定可以工作,但不可移植(因此您需要 ifdef)

对于 2,这是等效的代码

     int kq = kqueue();
struct kevent ke;
EV_SET(&ke, SIGCHLD, EVFILT_SIGNAL, EV_ADD, 0, 0, NULL);
kevent(kq, &ke, 1, NULL, 0, NULL);
if (kevent(kq, NULL, 0, &ke, 1, &timeout) == 1) {
signal = ke.ident;
}
else {
// Catches errors in the add, timeout, and kevent wait
signal = -1;
}
close(kq);
// note that siginfo is not populated, there is no way to populate it using kqueue.

关于在 BSD 上使用 sigtimedwait() 捕获 SIGCHLD,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17012206/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com