我如何列出从一个.so文件导出的符号?如果可能的话,我还想知道它们的来源(例如,如果它们是从静态库中提取的)。

我使用的是gcc 4.0.2,如果有区别的话。


当前回答

Nm -g列出了extern变量,这不是必要的导出符号。 任何非静态文件作用域变量(在C语言中)都是extern变量。

nm -D将在动态表中列出该符号,您可以通过dlsym找到它的地址。

nm -版本

GNU nm 2.17.50.0.6-12.el5 20061020

其他回答

对于共享库libNAME。所以-D开关是必要的,以查看符号在我的Linux

nm -D libNAME.so

并为静态库如别人所报道

nm -g libNAME.a

您可以使用binutils工具链中的nm -g工具。然而,它们的来源并不总是现成的。我甚至不确定这些信息是否总能被检索到。也许objcopy揭示了更多的信息。

/EDIT:这个工具的名字当然是nm。标志-g只用于显示导出的符号。

列出符号的标准工具是nm,你可以这样简单地使用它:

nm -gD yourLib.so

如果你想看到c++库的符号,添加“-C”选项,它需要这些符号(它的可读性要强得多)。

nm -gDC yourLib.so

如果你的.so文件是elf格式的,你有两个选择:

objdump (-C也可用于调用c++):

$ objdump -TC libz.so

libz.so:     file format elf64-x86-64

DYNAMIC SYMBOL TABLE:
0000000000002010 l    d  .init  0000000000000000              .init
0000000000000000      DF *UND*  0000000000000000  GLIBC_2.2.5 free
0000000000000000      DF *UND*  0000000000000000  GLIBC_2.2.5 __errno_location
0000000000000000  w   D  *UND*  0000000000000000              _ITM_deregisterTMCloneTable

或者使用readelf:

$ readelf -Ws libz.so
Symbol table '.dynsym' contains 112 entries:
   Num:    Value          Size Type    Bind   Vis      Ndx Name
     0: 0000000000000000     0 NOTYPE  LOCAL  DEFAULT  UND
     1: 0000000000002010     0 SECTION LOCAL  DEFAULT   10
     2: 0000000000000000     0 FUNC    GLOBAL DEFAULT  UND free@GLIBC_2.2.5 (14)
     3: 0000000000000000     0 FUNC    GLOBAL DEFAULT  UND __errno_location@GLIBC_2.2.5 (14)
     4: 0000000000000000     0 NOTYPE  WEAK   DEFAULT  UND _ITM_deregisterTMCloneTable

我一直在想为什么-fvisibility=hidden和#pragma GCC可见性似乎没有任何影响,因为所有的符号都是用nm可见的-直到我发现这篇文章指向readelf和objdump,这让我意识到似乎实际上有两个符号表:

你可以用nm列出来 你可以用readelf和objdump列出

我认为前者包含可以用strip或-s开关剥离的调试符号,可以给链接器或安装命令。即使nm不再列出任何东西,导出的符号仍然会被导出,因为它们在ELF“动态符号表”中,也就是后者。

对于Android .so文件,NDK工具链附带了其他答案中提到的所需工具:readelf, objdump和nm。