我正在编译 c++ 代码,我正在尝试添加 -rdynamic 选项,这样我就可以打印出有意义的堆栈跟踪来调试我的 c++ 程序,但是 clang 会抛出一个警告说“编译期间未使用的参数:'-rdynamic' ”。
作为测试,在我的系统上,我尝试编写一个简单的 c++ 程序并使用 -rdynamic 编译它,它没有问题,但是对于这个项目,它似乎没有成功。
任何建议都非常适用
我正在编译 c++ 代码,我正在尝试添加 -rdynamic 选项,这样我就可以打印出有意义的堆栈跟踪来调试我的 c++ 程序,但是 clang 会抛出一个警告说“编译期间未使用的参数:'-rdynamic' ”。
作为测试,在我的系统上,我尝试编写一个简单的 c++ 程序并使用 -rdynamic 编译它,它没有问题,但是对于这个项目,它似乎没有成功。
任何建议都非常适用
-rdynamic
当您只是编译源代码而不是链接它时,您可能会使用该标志。它是链接器的标志,因此仅在链接时才需要它。某些版本的 clang 可能无法识别它,在这种情况下,您可以指示 clang 将正确的选项传递给链接器,通常是:
-Wl,--export-dynamic
所以,例如
clang++ -rdynamic test.cpp
或者
clang++ --Wl,--export-dynamic test.cpp
但是如果是单独编译链接,只能在链接阶段使用:
clang++ -c test.cpp
clang++ --Wl,--export-dynamic test.o
(或作为最后一步clang++ -rdynamic test.o
:)
答案是对的nos's
,对我帮助很大。
一个小提示,--Wl,--export-dynamic
应该是-Wl,--export-dynamic
并且有一些方法可以确保-rdynamic
有效。
用于readelf -s
查看 ELF 符号:
例如
$ cat t.c
#include <stdio.h>
void bar() {}
void baz() {}
void foo() {}
int main() { foo(); printf("test"); return 0; }
$ clang -O0 -o test t.c
$ readelf -s test >test.elf
Symbol table '.dynsym' contains 7 entries:
Num: Value Size Type Bind Vis Ndx Name
0: 0000000000000000 0 NOTYPE LOCAL DEFAULT UND
1: 0000000000000000 0 NOTYPE WEAK DEFAULT UND _ITM_deregisterTMCloneTab
2: 0000000000000000 0 FUNC GLOBAL DEFAULT UND __libc_start_main@GLIBC_2.17 (2)
3: 0000000000000000 0 NOTYPE WEAK DEFAULT UND __gmon_start__
4: 0000000000000000 0 FUNC GLOBAL DEFAULT UND abort@GLIBC_2.17 (2)
5: 0000000000000000 0 NOTYPE WEAK DEFAULT UND _ITM_registerTMCloneTable
6: 0000000000000000 0 FUNC GLOBAL DEFAULT UND printf@GLIBC_2.17 (2)
$ clang -rdynamic -O0 -o test1 t.c
$ readelf -s test1 >test1.elf
Symbol table '.dynsym' contains 24 entries:
Num: Value Size Type Bind Vis Ndx Name
0: 0000000000000000 0 NOTYPE LOCAL DEFAULT UND
1: 0000000000000000 0 NOTYPE WEAK DEFAULT UND _ITM_deregisterTMCloneTab
2: 0000000000000000 0 FUNC GLOBAL DEFAULT UND __libc_start_main@GLIBC_2.17 (2)
3: 0000000000000000 0 NOTYPE WEAK DEFAULT UND __gmon_start__
4: 0000000000000000 0 FUNC GLOBAL DEFAULT UND abort@GLIBC_2.17 (2)
5: 0000000000000000 0 NOTYPE WEAK DEFAULT UND _ITM_registerTMCloneTable
6: 0000000000000000 0 FUNC GLOBAL DEFAULT UND printf@GLIBC_2.17 (2)
7: 0000000000420038 0 NOTYPE GLOBAL DEFAULT 25 _bss_end__
8: 00000000004009a0 68 FUNC GLOBAL DEFAULT 14 main
9: 0000000000420030 0 NOTYPE GLOBAL DEFAULT 25 __bss_start__
10: 0000000000420030 0 NOTYPE GLOBAL DEFAULT 25 __bss_start
11: 0000000000400994 4 FUNC GLOBAL DEFAULT 14 bar
12: 0000000000400a7c 4 OBJECT GLOBAL DEFAULT 16 _IO_stdin_used
13: 0000000000420038 0 NOTYPE GLOBAL DEFAULT 25 _end
14: 0000000000420038 0 NOTYPE GLOBAL DEFAULT 25 __end__
15: 0000000000420020 0 NOTYPE GLOBAL DEFAULT 24 __data_start
16: 0000000000420030 0 NOTYPE GLOBAL DEFAULT 24 _edata
17: 0000000000400a68 4 FUNC GLOBAL DEFAULT 14 __libc_csu_fini
18: 000000000040099c 4 FUNC GLOBAL DEFAULT 14 foo
19: 00000000004009e8 128 FUNC GLOBAL DEFAULT 14 __libc_csu_init
20: 00000000004008a0 0 FUNC GLOBAL DEFAULT 14 _start
21: 0000000000420020 0 NOTYPE WEAK DEFAULT 24 data_start
22: 0000000000400998 4 FUNC GLOBAL DEFAULT 14 baz
23: 0000000000420038 0 NOTYPE GLOBAL DEFAULT 25 __bss_end__
您将在 中看到所有符号.dynsym
,而不仅仅是使用过的符号。
还有一些关于标志strip
影响的有趣测试-rdynamic
:
gcc debug symbols (-g flag) vs linker's -rdynamic option