siglongjmp、sigsetjmp

相关函数:longjmp, siglongjmp, setjmp 
表头文件:#include <setjmp.h> 
函数定义:int sigsetjmp(sigjmp_buf env, int savesigs) 
函数说明:sigsetjmp()会保存目前堆栈环境,然后将目前的地址作一个记号,
而在程序其他地方调用siglongjmp()时便会直接跳到这个记号位置,然后还原堆栈,继续程序的执行。 
参数env为用来保存目前堆栈环境,一般声明为全局变量 
参数savesigs若为非0则代表搁置的信号集合也会一块保存 
当sigsetjmp()返回0时代表已经做好记号上,若返回非0则代表由siglongjmp()跳转回来。 
返回:若直接调用则为0,若从siglongjmp调用返回则为非0

实例:

[cpp]  view plain copy
  1. #include <stdio.h>  
  2. #include <signal.h>  
  3. #include <setjmp.h>  
  4. #include <unistd.h>  
  5. #include <sys/time.h>  
  6.   
  7. sigjmp_buf jmp_env;  
  8.   
  9. static void connect_alarm(int)  
  10. {  
  11.     siglongjmp(jmp_env, 1);  
  12. }  
  13.   
  14. int main()  
  15. {  
  16.     // 当超时时间sec_timeout大于等于运行时间run_time时会跳过printf("running...\n");  
  17.     int sec_timeout = 3;  
  18.     int run_time = 2;  
  19.   
  20.     printf("timeout = %d, run time = %d\n", sec_timeout, run_time);  
  21.     if (sec_timeout)  
  22.     {  
  23.         // 超过用alarm函数设置的时间时产生此信号,调用connect_alarm函数  
  24.         signal(SIGALRM, connect_alarm);  
  25.         alarm(sec_timeout);  
  26.         printf("set timeout\n");  
  27.         if (sigsetjmp(jmp_env, 1))  
  28.         {  
  29.             printf("timeout\n");  
  30.             goto out;  
  31.         }  
  32.     }  
  33.   
  34.     sleep(run_time);  
  35.     printf("running...\n");  
  36.   
  37. out:  
  38.     if (sec_timeout)  
  39.     {  
  40.         // 取消先前设置的闹钟  
  41.         alarm(0);  
  42.         printf("cancel timeout\n");  
  43.     }  
  44.   
  45.     return 0;  
  46. }  









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

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

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

字会发生什么呢?

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

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

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <setjmp.h>  
  2. int sigsetjmp(sigjmp_buf env, int savemask); //若直接调用则返回0,若从siglongjmp调用返回则返回非0值。  
  3. void siglongjmp(sigjmp_buf env, int val);  
在sigsetjmp中增加了一个参数,如果savemask非0,则sigsetjmp在env中保存进程的当前信号屏蔽字。调用siglongjmp时,

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


实践:

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <stdio.h>  
  2. #include <setjmp.h>  
  3. #include <signal.h>  
  4.   
  5. static sigjmp_buf jmpbuf;  
  6.   
  7. static void sighandle(int signo){  
  8.         printf("starting sighandle\n");  
  9.         sleep(10);  
  10.         siglongjmp(jmpbuf,1);  
  11. }  
  12.   
  13. int main(void){  
  14.         if(signal(SIGUSR1, sighandle) == SIG_ERR){  
  15.                 perror("signal");  
  16.                 return -1;  
  17.         }  
  18.         printf("start main\n");  
  19.         if(sigsetjmp(jmpbuf,1)){  
  20.                 printf("return from sighandle\n");  
  21.         }  
  22.         while(1){  
  23.                 pause();  
  24.         }  
  25.         return 0;  
  26. }  

运行结果:

[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函数:

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <stdio.h>  
  2. #include <setjmp.h>  
  3. #include <signal.h>  
  4.   
  5. static sigjmp_buf jmpbuf;  
  6.   
  7. static void sighandle(int signo){  
  8.         printf("starting sighandle\n");  
  9.         sleep(10);  
  10.         longjmp(jmpbuf,1);  
  11. }  
  12.   
  13. int main(void){  
  14.         if(signal(SIGUSR1, sighandle) == SIG_ERR){  
  15.                 perror("signal");  
  16.                 return -1;  
  17.         }  
  18.         printf("start main\n");  
  19.         if(setjmp(jmpbuf,0)){  
  20.                 printf("return from sighandle\n");  
  21.         }  
  22.         while(1){  
  23.                 pause();  
  24.         }  
  25.         return 0;  
  26. }  
运行结果:

[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]#

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


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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值