《UNIX环境高级编程》笔记--sigsetjmp和siglongjmp函数

setjmp和longjmp函数用于非局部跳转,在信号处理程序中经常调用longjmp函数以返回到程序的主循环中,而不是从该处理

程序返回。但是调用longjmp有一个问题,当捕捉到一个信号时,进入进行处理函数,此时当前信号被自动加到进程的信号

屏蔽字中。这阻止了后来产生的这种信号中断该信号处理程序。如果用longjmp跳出信号处理程序,那么对此进程的信号屏蔽

字会发生什么呢?

POSIX.1并没有说明setjmp和longjmp对信号屏蔽字的作用,而是定义了两个新函数sigsetjmp和siglongjmp。在信号处理程序

进行非局部转移时应该使用这两个函数。

#include <setjmp.h>
int sigsetjmp(sigjmp_buf env, int savemask); //若直接调用则返回0,若从siglongjmp调用返回则返回非0值。
void siglongjmp(sigjmp_buf env, int val);
在sigsetjmp中增加了一个参数,如果savemask非0,则sigsetjmp在env中保存进程的当前信号屏蔽字。调用siglongjmp时,

如果带非0 savemask的sigsetjmp调用已经保存了env,则siglongjmp从其中恢复保存的信号屏蔽字。


实践:

#include <stdio.h>
#include <setjmp.h>
#include <signal.h>

static sigjmp_buf jmpbuf;

static void sighandle(int signo){
        printf("starting sighandle\n");
        sleep(10);
        siglongjmp(jmpbuf,1);
}

int main(void){
        if(signal(SIGUSR1, sighandle) == SIG_ERR){
                perror("signal");
                return -1;
        }
        printf("start main\n");
        if(sigsetjmp(jmpbuf,1)){
                printf("return from sighandle\n");
        }
        while(1){
                pause();
        }
        return 0;
}

运行结果:

[root@yanPC apue]# ./a.out &
[1] 16536
[root@yanPC apue]# start main
kill -SIGUSR1 16536
starting sighandle
[root@yanPC apue]# return from sighandle
kill -SIGUSR1 16536
[root@yanPC apue]# starting sighandle
return from sighandle

从信号处理函数跳出后,再发送信号,信号处理函数能够被执行,说明跳出后,信号屏蔽被解除。


修改使用setjmp函数:

#include <stdio.h>
#include <setjmp.h>
#include <signal.h>

static sigjmp_buf jmpbuf;

static void sighandle(int signo){
        printf("starting sighandle\n");
        sleep(10);
        longjmp(jmpbuf,1);
}

int main(void){
        if(signal(SIGUSR1, sighandle) == SIG_ERR){
                perror("signal");
                return -1;
        }
        printf("start main\n");
        if(setjmp(jmpbuf,0)){
                printf("return from sighandle\n");
        }
        while(1){
                pause();
        }
        return 0;
}
运行结果:

[root@yanPC apue]# ./a.out &
[1] 16545
[root@yanPC apue]# start main
kill -SIGUSR1 16545
starting sighandle
[root@yanPC apue]# return from sighandle
kill -SIGUSR1 16545
[root@yanPC apue]#

从信号处理函数跳出后,再发送信号,信号处理函数不能执行,说明跳出后,信号依然被阻塞着。

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值