Addr2line 工具(它是标准的 GNU Binutils 中的一部分)是一个可以将指令的地址和可执行映像转换成文件名、函数名和源代码行数的工具。这种功能对于将跟踪地址转换成更有意义的内容来说简直是太棒了。
要了解这个过程是怎样工作的,我们可以试验一个简单的交互式的例子。这个示例 C 文件(main.c)是通过 cat 一个简单的应用程序实现的(也就是说,将标准输出的文本重定向到一个文件中)。然后使用 gcc 来编译这个文件,它会传递一些特殊的选项。首先,要(使用 -Wl 选项)通知链接器生成一个映像文件,并(使用 -g 选项)通知编译器生成调试符号。最终生成可执行文件 test。得到新的可执行应用程序之后,您就可以使用grep 工具在映像文件中查找 main 来寻找它的地址了。使用这个地址和 Addr2line 工具,就可以判断出函数名(main)、源文件(/work/main.c)以及它在源文件中的行号(4)。
要了解这个过程是怎样工作的,我们可以试验一个简单的交互式的例子。这个示例 C 文件(main.c)是通过 cat 一个简单的应用程序实现的(也就是说,将标准输出的文本重定向到一个文件中)。然后使用 gcc 来编译这个文件,它会传递一些特殊的选项。首先,要(使用 -Wl 选项)通知链接器生成一个映像文件,并(使用 -g 选项)通知编译器生成调试符号。最终生成可执行文件 test。得到新的可执行应用程序之后,您就可以使用grep 工具在映像文件中查找 main 来寻找它的地址了。使用这个地址和 Addr2line 工具,就可以判断出函数名(main)、源文件(/work/main.c)以及它在源文件中的行号(4)。
在调用 Addr2line 工具时,要使用 -e 选项来指定可执行映像是 test。通过使用 -f 选项,可以告诉工具输出函数名。
[root@localhost work]# cat main.c
#include <stdio.h>
int main(void)
{
char *str = "hello";
str[0] = 'h';
return 0;
}
[root@localhost work]# gcc -Wl,-Map=main.map -g -o main main.c(编译生成可执行文件及main.map)
[root@localhost work]# ls
1.txt main main.c main.map
[root@localhost work]# grep main main.map
0x00000000080482c4 __libc_start_main@@GLIBC_2.0
0x0000000008048394 main
OUTPUT(main elf32-i386)
[root@localhost work]# addr2line 0x0000000008048394 -e main -f
main
/work/main.c:4