在 http://www.groad.net/bbs/read.php?tid=919
中使用了 setjmp()/longjmp() 函数,但这里有一个问题,就是在使用 longjmp() 跳出信号处理程序时不会自动将信号屏蔽码修改回原来的屏蔽码,从而引起该信号的永久屏蔽。对此,可以用 sigsetjmp() 和 siglongjmp() 来解决,两个函数的原型如下:
[C++] 纯文本查看 复制代码 #include <setjmp.h>
int setjmp(jmp_buf env);
int sigsetjmp(sigjmp_buf env, int savesigs);
测试代码1:
[C++] 纯文本查看 复制代码 #include <stdio.h>
#include <setjmp.h>
#include <signal.h>
#include <unistd.h>
#define ENV_UNSAVE 0
#define ENV_SAVED 1
int flag_saveenv = ENV_UNSAVE;
jmp_buf env;
int flag = 0;
void handler_sigrtmin15(int signo)
{
if (flag_saveenv == ENV_UNSAVE) {
printf("进入到信号处理函数\n");
flag = 1;
return;
}
printf("recv SIGRTMIN+15\n");
sleep(10);
printf("in handler_sigrtmin15, after sleep\n");
siglongjmp(env, 1); //返回到env处,注意第二个参数的值
}
int main()
{
/*设置返回点*/
switch (sigsetjmp(env, 1)) { /*sigsetjmp的第二个参数只要非 0 即可*/
case 0:
printf("return 0\n");
if (flag == 1) {
printf("没通过siglongjmp() 而回到 case 0\n");
flag = 0;
}
// flag_saveenv = ENV_SAVED;
break;
case 1:
if (flag == 1) {
printf("没通过siglongjmp() 而回到case 1\n");
flag = 0;
}
printf("return from SIGRTMIN+15\n");
break;
default:
if (flag == 1) {
printf("没通过siglongjmp() 而回到case default\n");
flag = 0;
}
printf("return else\n");
break;
}
if (flag == 1) {
printf("没通过siglongjmp() 而回到 signal() 之前\n");
flag = 0;
}
signal(SIGRTMIN+15, handler_sigrtmin15);
if (flag == 1) {
printf("没通过siglongjmp() 而回到 signal() 之后\n");
flag = 0;
}
while(1)
;
return 0;
}
说明:
在上面的代码中,如果发送 SIGRTMIN+15 信号,则信号处理函数对信号进行处理。如果是别的信号,一般默认下会结束进程。如发送一个 SIGRTMIN+15 信号:beyes@linux-beyes:~/C> kill -s SIGRTMIN+15 23382 程序做出处理:beyes@linux-beyes:~/C> ./right_return.exe
return 0
进入到信号处理函数 发送别的信号看看:beyes@linux-beyes:~/C> kill -s SIGRTMIN 23382 接着上面的输出,程序提示得到了发送过来的信号,就自行退出了:实时信号 0
beyes@linux-beyes:~/C>
上面,把 flag_saveenv = ENV_SAVED; 这句的屏蔽取消,那么程序就会根据 sigsetjmp() 的设置,然后 siglongjmp() 进行相应的跳转。
注意,如果在 sigsetjmp() 之前就将 flag_saveenv 设置为 ENV_SAVED,那么信号一旦在 sigsetjmp() 之前过来了,由于信号处理函数将因为程序没有保存返回点的栈状态信息而返回到未知地点或程序崩溃! |