曲径通幽论坛

 找回密码
 立即注册
搜索
查看: 7818|回复: 0
打印 上一主题 下一主题

[信号] sigsetjmp() 和 siglongjmp() -- 带有信号屏蔽字恢复的返回

[复制链接]

4918

主题

5880

帖子

3万

积分

GROAD

曲径通幽,安觅芳踪。

Rank: 6Rank: 6

积分
34395
跳转到指定楼层
楼主
发表于 2009-7-7 22:31:41 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
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() 之前过来了,由于信号处理函数将因为程序没有保存返回点的栈状态信息而返回到未知地点或程序崩溃!
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|曲径通幽 ( 琼ICP备11001422号-1|公安备案:46900502000207 )

GMT+8, 2025-5-4 01:07 , Processed in 0.076696 second(s), 23 queries .

Powered by Discuz! X3.2

© 2001-2013 Comsenz Inc.

快速回复 返回顶部 返回列表