0

使用 backtrace 和 backtrace_symbols,我可以获得一个感兴趣的错位函数名称(我们将其称为属于 libfsw.so 的 funcA_mangledName。

我的目标是获取定义它的源文件和行号。我可以对未在库文件中定义的函数执行此操作,如下所示。stacktrace 保存回溯。文件名 = S_main_executable 在一般情况下。

sprintf(syscom[jj], "addr2line %p -e %s", stacktrace[jj], filename);
system(syscom[jj]);

但是,当函数是库的一部分时,这不起作用,即文件名 = libfsw.so。

向后工作我可以在 linux 终端上执行此操作:

nm libfsw.so | grep funcA_mangledName

得到:000000000020cbea T funcA_mangledName

然后当我进入linux终端时:

addr2line 0x000000000020cbea -e libfsw.so

我得到了正确的源文件和行号。

从一开始到正确的文件偏移量,我缺少什么?

4

1 回答 1

0

我想到了!通过减去库文件的虚拟基地址,我能够将回溯给出的“虚拟地址”转换为文件偏移量。我假设只要库被加载到连续内存中,这应该总是有效的。

for (jj=2; jj < trace_size; ++jj) {
   void *handle;
   struct link_map *map;

   handle = dlopen(filename, RTLD_LAZY);
   dlinfo(handle, RTLD_DI_LINKMAP, &map);

   // Override the address given by backtrace (only needed for library files)
   stacktrace[jj] = stacktrace[jj] - map->l_addr;

   sprintf(syscom[jj], "addr2line %p -e %s", stacktrace[jj], filename);
   system(syscom[jj]);
   dlclose(handle);
}
于 2021-01-14T06:53:58.440 回答