我正在为Android(仅限ARM)编写此内容,但我相信原则对于通用Linux也是相同的。
我试图从信号处理程序中捕获堆栈跟踪,以便在我的应用程序崩溃时记录它。这是我使用<unwind.h>
想出来的方法。
初始化:
struct sigaction signalhandlerDescriptor;
memset(&signalhandlerDescriptor, 0, sizeof(signalhandlerDescriptor));
signalhandlerDescriptor.sa_flags = SA_SIGINFO;
signalhandlerDescriptor._u._sa_sigaction = signalHandler;
sigaction(SIGSEGV, &signalhandlerDescriptor, 0);
代码本身:
struct BacktraceState
{
void** current;
void** end;
void* pc;
};
inline _Unwind_Reason_Code unwindCallback(struct _Unwind_Context* context, void* arg)
{
BacktraceState* state = static_cast<BacktraceState*>(arg);
state->pc = (void*)_Unwind_GetIP(context);
if (state->pc)
{
if (state->current == state->end)
return _URC_END_OF_STACK;
else
*state->current++ = reinterpret_cast<void*>(state->pc);
}
return _URC_NO_REASON;
}
inline size_t captureBacktrace(void** addrs, size_t max, unsigned long pc)
{
BacktraceState state = {addrs, addrs + max, (void*)pc};
_Unwind_Backtrace(unwindCallback, &state);
personality_routine();
return state.current - addrs;
}
inline void dumpBacktrace(std::ostream& os, void** addrs, size_t count)
{
for (size_t idx = 0; idx < count; ++idx) {
const void* addr = addrs[idx];
const char* symbol = "";
Dl_info info;
if (dladdr(addr, &info) && info.dli_sname) {
symbol = info.dli_sname;
}
int status = -3;
char * demangledName = abi::__cxa_demangle(symbol, 0, 0, &status);
os << "#" << idx << ": " << addr << " " << (status == 0 ? demangledName : symbol) << "\n";
free(demangledName);
}
}
void signalHandler(int sig, siginfo_t *siginfo, void *uctx)
{
ucontext * context = (ucontext*)uctx;
unsigned long PC = context->uc_mcontext.arm_pc;
unsigned long SP = context->uc_mcontext.arm_sp;
Logger() << __PRETTY_FUNCTION__ << "Fatal signal:" << sig;
const size_t maxNumAddresses = 50;
void* addresses[maxNumAddresses];
std::ostringstream oss;
const size_t actualNumAddresses = captureBacktrace(addresses, maxNumAddresses, PC);
dumpBacktrace(oss, addresses, actualNumAddresses);
Logger() << oss.str();
exit(EXIT_FAILURE);
}
问题:如果我在unwindCallback
中调用_Unwind_GetIP(context)
来获取PC寄存器,我会得到完整的跟踪记录,但这是一个单独的堆栈,显然不是我想要的。因此,我尝试提供从信号处理程序中的ucontext
获取的PC,结果很奇怪:我只得到一条堆栈记录,它是正确的记录——导致出现信号的函数。但是它被记录了两次(即使地址相同,所以这不是符号名称查找错误)。显然,这还不够好——我需要整个堆栈。我想知道这个结果是否仅仅是偶然的(即它通常不起作用)。现在,我读到我还需要提供堆栈指针,我显然可以从
ucontext
中获取,和PC一样。但我不知道该怎么做。我必须手动展开而不是使用_Unwind_Backtrace
吗?如果是这样,请给我一个示例代码?我已经搜索了一天,仍然找不到可以复制粘贴到我的项目中的任何东西。就这个问题而言,这里是包含
_Unwind_Backtrace
定义的libunwind源代码。如果我看到了源代码,我想我可以找到一些解决方法,但它比我预期的要复杂得多。