3

我正在编写一些代码来存储每个内存分配的回溯。然后,我将这些列表写入文件以进行离线分析。在 win32 中,我使用_AddressOfReturnAddress然后手动创建回溯。由于地址对于我使用的每次运行都是随机的,GetModuleInformation因此lpBaseOfDll可以获取基地址。这在没有 FPO 的 x86 上效果很好,这对我来说已经足够了。然后我使用 win32 API 加载 PDB 以将地址转换为函数名称。

我将如何在 linux 上执行此操作?我目前的方法是使用__builtin_return_address(x)addr2line离线来获得相同的结果。问题是每次运行时地址都是随机的,所以addr2line不理解它们。__executable_start不起作用,因为它每次运行都返回相同的值。有没有办法在运行时获取我的可执行文件的基地址?

一次运行给了我这个:

__executable_start: 0x8048000
backtrace:
0x9cce628
0x9cce2b8
0x9cce260
0x9cce1f8
0x9cce138
0x9cce0c8
0x9cce060
0x9cce008

接下来:

__executable_start: 0x8048000
backtrace:
0x8db6628
0x8db62b8
0x8db6260
0x8db61f8
0x8db6138
0x8db60c8
0x8db6060
0x8db6008

等等。

4

1 回答 1

2

您可以使用dl_iterate_phdr()Linux 上的 来确定每个动态加载的对象的加载地址:

#define _GNU_SOURCE
#include <stdio.h>
#include <link.h>

int callback(struct dl_phdr_info *info, size_t size, void *data)
{
    printf("%s @ %#lx\n", info->dlpi_name, (unsigned long)info->dlpi_addr);
    return 0;
}

int main()
{
    dl_iterate_phdr(&callback, NULL);
    return 0;
}
于 2013-06-30T12:47:44.090 回答