引入 sigsuspend 是为了解决 sigprocmask 到 pause 之间存在时间窗口,从而可能引发信号丢失的问题。这是一个原子操作。
int sigsuspend(const sigset_t *sigmask);
/* - 将进程的信号屏蔽字设置为 sigmask 指向的值
* - 在接收到一个信号之前,调用进程被挂起
* - 在捕捉到信号并且从信号处理函数返回后,sigsuspend 返回
* - 进程的信号屏蔽字恢复为调用 sigsuspend 之前的值 */
⚠️:此调用没有成功返回值。总是返回 -1,并将 errno 设置为 EINTR 。
sigsuspend 系统调用搭配 sigprocmask 保证了在执行关键代码区时不受指定的信号打断,在关键代码区执行完以后,不丢失执行关键区时可能发生的信号,进程可以转而执行信号处理程序和之后的任务。
如何更好地理解这个函数?下面是一张简明的图示:
在这里插入图片描述
下面是一个简单的 信号驱动 小程序,演示 sigsuspend 的使用:
#include <stdio.h>
#include <stdlib.h>
#include <signal.h>
#include <unistd.h>
static void
sig_int(int signal)
{
printf("!");
fflush(NULL);
}
int
main(void)
{
if (signal(SIGINT, sig_int) == SIG_ERR)
printf("signal err!\n"), exit(1);
int i, j;
sigset_t newset, oldset;
sigemptyset(&newset);
sigaddset(&newset, SIGINT);
for (i = 0; i < 100; ++i) {
/* 阻止 SIGINT 打断关键区代码执行 */
sigprocmask(SIG_BLOCK, &newset, &oldset);
/* 关键区代码 */
{
for (j = 0; j < 5; j++) {
printf("*");
fflush(NULL);
sleep(1);
}
printf("\n");
}
/* 可以收到 SIGINT 信号了!
* 并且恢复信号屏蔽字为 newset 中的信号 */
sigsuspend(&oldset);
/* 恢复原来进程的信号屏蔽字 */
sigprocmask(SIG_SETMASK, &oldset, NULL);
}
return 0;
}
执行结果:
*****
^C!*****
^C!*^C*^C^C***
!*****
如果觉得对你有帮助的话,一个免费的赞是对我最大的鼓励。咱们下期再见!