在 cpp 多线程程序中,sigwait
在创建所有线程并初始化程序之后,我们在 main 函数的末尾使用 Linux 实现的副本:
sigset_t mask;
struct sigaction sa;
signal(SIGPIPE, SIG_IGN);
sigemptyset(&mask);
sigaddset(&mask, SIGINT);
sa.sa_flags = SA_SIGINFO | SA_NODEFER;
sa.sa_sigaction = handler;
sigaction(SIGINT, &sa, NULL);
// Copy/paste of sigwait source code
// but keeping full siginfo (instead of just returning signo)
// and adding ETIMEDOUT as a do/while continue condition
siginfo_t si = {};
int ret = 0;
do
{
ret = sigtimedwait (&mask, &si, 0);
if(ret == -1)
{
if(errno == ETIMEDOUT)
{
printf("sigtimedwait timeout\n");
}
}
}
while (ret < 0 && (errno == EINTR || errno == ETIMEDOUT));
sigwait
实施参考:
int
__sigwait (const sigset_t *set, int *sig)
{
siginfo_t si;
int ret;
do
ret = __sigtimedwait (set, &si, 0);
/* Applications do not expect sigwait to return with EINTR, and the
error code is not specified by POSIX. */
while (ret < 0 && errno == EINTR);
if (ret < 0)
return errno;
*sig = si.si_signo;
return 0;
}
sigtimedwait
声明供参考:
int sigtimedwait(const sigset_t *restrict set,
siginfo_t *restrict info,
const struct timespec *restrict timeout);
sigtimedwait
设置errno
为ETIMEDOUT
当我们的程序在高负载下运行时,即进程使用> 50%的CPU,几乎所有可用内存都被缓存用于SD卡写入操作。
这种情况并不总是可重复的,而是在持续长达 3 分钟的高负载序列中随机发生一次或多次。在常规操作期间不会发生这种情况。
我们在armv7l 上运行Linux 4.14。
即使将所有可能的信号添加到 sigaddset 和 sigaction 之后,我们的 sigactionhandler
也不会被调用。查看siginfo_t si
,所有值均未设置 - 读数为零。
我在 Linux 手册中找不到任何关于sigtimedwait
返回的内容。ETIMEDOUT
有谁知道可能是什么原因造成的?