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]#
从信号处理函数跳出后,再发送信号,信号处理函数不能执行,说明跳出后,信号依然被阻塞着。