图说系统调用-sigsuspend

引入 sigsuspend 是为了解决 sigprocmaskpause 之间存在时间窗口,从而可能引发信号丢失的问题。这是一个原子操作。

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***
!*****

如果觉得对你有帮助的话,一个免费的赞是对我最大的鼓励。咱们下期再见!

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值