表头文件:#include <setjmp.h>
函数定义:int sigsetjmp(sigjmp_buf env, int savesigs)
函数说明:sigsetjmp()会保存目前堆栈环境,然后将目前的地址作一个记号,
而在程序其他地方调用siglongjmp()时便会直接跳到这个记号位置,然后还原堆栈,继续程序的执行。
参数env为用来保存目前堆栈环境,一般声明为全局变量
参数savesigs若为非0则代表搁置的信号集合也会一块保存
当sigsetjmp()返回0时代表已经做好记号上,若返回非0则代表由siglongjmp()跳转回来。
返回:若直接调用则为0,若从siglongjmp调用返回则为非0
实例:
- #include <stdio.h>
- #include <signal.h>
- #include <setjmp.h>
- #include <unistd.h>
- #include <sys/time.h>
- sigjmp_buf jmp_env;
- static void connect_alarm(int)
- {
- siglongjmp(jmp_env, 1);
- }
- int main()
- {
- // 当超时时间sec_timeout大于等于运行时间run_time时会跳过printf("running...\n");
- int sec_timeout = 3;
- int run_time = 2;
- printf("timeout = %d, run time = %d\n", sec_timeout, run_time);
- if (sec_timeout)
- {
- // 超过用alarm函数设置的时间时产生此信号,调用connect_alarm函数
- signal(SIGALRM, connect_alarm);
- alarm(sec_timeout);
- printf("set timeout\n");
- if (sigsetjmp(jmp_env, 1))
- {
- printf("timeout\n");
- goto out;
- }
- }
- sleep(run_time);
- printf("running...\n");
- out:
- if (sec_timeout)
- {
- // 取消先前设置的闹钟
- alarm(0);
- printf("cancel timeout\n");
- }
- return 0;
- }
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);
如果带非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]#
从信号处理函数跳出后,再发送信号,信号处理函数不能执行,说明跳出后,信号依然被阻塞着。