看着这个问题和这个问题,我可以看到backtrace_symbols()
要工作,必须使用-rdynamic
标志进行编译。
我已经在测试程序中尝试过它并且它可以工作,但是我正在编写一个也被编译的程序,-static
并且这个页面说当传递给编译器/链接器时它backtrace_symbols()
不起作用。-static
是否有任何快速的解决方法,或者我的静态链接程序中永远不会有人类可读的回溯功能?
答案就在眼前:它在我在问题中链接的同一页面中。最后,我成功使用了libunwind
.
#include <libunwind.h>
#include <stdio.h>
void do_backtrace()
{
unw_cursor_t cursor;
unw_context_t context;
unw_getcontext(&context);
unw_init_local(&cursor, &context);
while (unw_step(&cursor) > 0)
{
unw_word_t offset, pc;
char fname[64];
unw_get_reg(&cursor, UNW_REG_IP, &pc);
fname[0] = '\0';
(void) unw_get_proc_name(&cursor, fname, sizeof(fname), &offset);
printf ("%p : (%s+0x%x) [%p]\n", pc, fname, offset, pc);
}
}
int main()
{
do_backtrace();
return 0;
}
我遇到了链接错误,因为我(再次)忘记在命令行末尾放置链接器选项。我真的不明白为什么g++
/gcc
在忽略命令行选项时至少不发出警告。正确的编译命令行是(-g
不需要):
g++ -static unwind.cpp -o unwind -lunwind -lunwind-x86
如果您绝对必须将程序编译为静态,您仍然可以使用backtrace()
来查找函数的地址,然后通过解析调试信息来查找函数名称,例如使用libdwarf 。
但这不是一项简单的任务,所以我建议使用-rdynamic
flag。