首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >在linux中处理SIGBUS

在linux中处理SIGBUS
EN

Stack Overflow用户
提问于 2013-10-17 00:07:11
回答 1查看 3.6K关注 0票数 4

在我的一个程序中,当我试图访问一个mmap编辑的内存位置时,我会点击" SIGBUS“,该位置无法获得内存页(因为底层物理内存耗尽),并且程序由于SIGBUS而崩溃。

我计划注册一个SIGBUG信号处理程序以避免崩溃。但是,我不希望从SIGBUS处理程序中退出()程序。我正在试着看看是否有合适的报告ENOMEM,并继续该计划与其他工作。

我能做以下几件事吗?代码如下所示:

代码语言:javascript
复制
mem_p->head = MY_HEAD_MAGIC;   /* this line could trigger SIGBUS */
if (sigbus_happened) {
    sigbus_happened = FALSE;
    do_something_else();   
    return ENOMEM;
}

信号处理程序:

代码语言:javascript
复制
void signal_handler (int sig)
{
   if (sig == SIGBUS)
      sigbus_happened = TRUE;
}   

上面的内容会有效吗?不会崩溃?

谢谢。

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2013-10-17 00:49:14

您显示的代码可能会违背您的期望,这是有危险的。这是因为编译器可以自由地排列代码,以便在赋值sigbus_happened之前“记住”mem_p->head的值。因此,即使在执行信号处理程序时,您的代码也可能没有检测到已经设置了标志。至少,您需要使变量volatile

最好只是检查一下mmap()调用是否失败。您可以通过检查调用是否返回值MAP_FAILED来完成此操作。如果调用失败,请不要尝试访问指针值。

捕获SIGBUS的尝试会提醒您异常处理。C不具有C++风格的异常处理(尽管存在模仿它们的宏包,如节目单 )。但是,以一种更像异常工作方式的方式遵循您的模型的一种方法是使用setjmp()longjmp()setjmp()保存现有堆栈上下文并返回0longjmp()将代码返回到保存的上下文,并使setjmp()返回一个非0值。

从信号处理程序中,最好使用POSIX sigsetjmp()siglongjmp(),以便在调用信号处理程序之前被C运行时或操作系统阻塞的任何信号在返回到保存的上下文时被重置为它们所拥有的值。

代码语言:javascript
复制
jmp_buf *sigbus_jmp; // global

void signal_handler (int sig)
{
   if (sig == SIGBUS) {
      if (sigbus_jmp) siglongjmp(*sigbus_jmp, 1);
      // no one to catch the error, so abort
      abort();
    }
}

    //...
    jmp_buf sigbus_jmpbuf;
    sigbus_jmp = &sigbus_jmpbuf;
    if (sigsetjmp(sigbus_jmpbuf, 1) == 0) {
        // try
        mem_p->head = MY_HEAD_MAGIC;   /* this line could trigger SIGBUS */
    } else {
        // catch
        do_something_else();   
        return ENOMEM;
    }
    sigbus_jmp = 0;
票数 4
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/19416054

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档