backtrace_symbols无法打印导致信号的函数

时间:2014-01-23 08:28:18

标签: c++ macos stack-trace callstack signal-handling

我正在为我的C ++应用程序实现一个简单的崩溃记录器:

static void handler(int, siginfo_t * info, void *) {
    void *array[1000];

    switch (info->si_signo) {
    case SIGILL:
        Logger() << "Received SIGILL";
        break;
    case SIGSEGV:
        Logger() << "Received SIGSEGV";
        break;
    case SIGBUS:
        Logger() << "Received SIGBUS";
        break;
    case SIGSYS:
        Logger() << "Received SIGSYS";
        break;
    default:
        break;
    }

    // get void*'s for all entries on the stack
    const size_t size = backtrace(array, 1000);

    // print out all the frames
    char ** symbols = backtrace_symbols(array, size);
    for(size_t i = 0; i < size; ++i)
    {
        Logger() << symbols[i];
    }

    free(symbols);
    exit(EXIT_FAILURE);
}

struct sigaction SignalAction;
SignalAction.sa_flags = SA_SIGINFO;
SignalAction.sa_sigaction = handler;
sigemptyset(&SignalAction.sa_mask);
sigaction(SIGSEGV, &SignalAction, NULL);
sigaction(SIGBUS, &SignalAction, NULL);
sigaction(SIGILL, &SignalAction, NULL);

我还没有在Linux上测试它,但是在OS X上我感兴趣的特定跟踪项目,即thatr引起信号的那个项目是被打印(条目号2):

: " Received SIGSEGV" 
: " 0   App                       0x0000000100253d15 _ZL7handleriP9__siginfoPv + 229" 
: " 1   libsystem_platform.dylib  0x00007fff8ff0f5aa _sigtramp + 26" 
: " 2   ???                       0x000000000000000c 0x0 + 12" 
: " 3   App                       0x000000010000dfa7 _ZN11CMainWindow13initShortcutsEv + 231" 
: " 4   App                       0x000000010000d059 _ZN11CMainWindowC2EP7QWidget + 1001" 
: " 5   App                       0x00000001000091d9 main + 6217" 
: " 6   App                       0x00000001000070a5 _start + 227" 
: " 7   App                       0x0000000100006fc1 start + 33" 

为什么会发生这种情况,我可以解决这个问题吗?

P上。 S.这是一个调试版本。没有实际的段错误,它是用raise(SIGSEGV)模拟的。 raise是从某个方法调用的,而该方法又来自MainWindow::initShortcuts

2 个答案:

答案 0 :(得分:4)

使用信号处理程序构建崩溃记录器的想法存在根本缺陷。来自sigaction man page的“注意”部分:

  

以下功能可以是可重入的,也可以是不可中断的   信号和异步信号安全。因此应用程序可以调用   它们没有限制,不受信号捕捉功能的影响:

     

[...异步信号安全功能列表省略...]

     

以上列表中没有的所有功能都被认为是不安全的   尊重信号。也就是说,这种功能的行为   从信号处理程序调用时未定义。一般来说,   信号处理程序应该只做一个标志;大多数其他   行动不安全。

在信号处理程序中调用backtrace()backtrace_symbols()free()exit()都不安全。几乎可以肯定,Loggeroperator<<内的内容也不安全。

至于为什么信号激发函数不在回溯中,显然是由内核设置信号处理程序上下文的原因。通过检查由调用语句和函数序言构造的堆栈帧来获得回溯。信号处理程序不作为普通函数调用调用,因此没有特别的理由期望堆栈帧链可以正确设置。

事实上,信号处理程序可以在与普通代码完全不同的堆栈上运行。请参阅sigaltstack()

答案 1 :(得分:3)

条目号2有非常特殊的地址。它是0x000000000000000c,你必须知道有中断描述表。根据{{​​3}},它可能是Stack Fault。

然而正如Ken指出的那样,做你正在做的事情是不好的。这就像试图记住当你的主动脉被撕裂时撞到你的汽车牌照。你能做到但是为了什么?

也许不要尝试学习如何使用strace http://en.wikipedia.org/wiki/Interrupt_Descriptor_Table