首页
登录 | 注册

Bug:编译链接,发现找不到符号,用nm工具查看,发现符号属性有两个分别是U和T

Bug:编译链接,发现找不到符号,用nm工具查看,发现符号属性有两个分别是U和T。

U代表是未定义符号,而T表示的是符号定义在Text段

nm -l 库名 | grep -rns "符号名" | c++filt

查看发现出现了如下情况:

符号名  U  src/File1.cpp:行号1

符号名 T   src/Src2/File1.cpp:行号2

具体查看,发现U的地方是调用的T的实现,并不是要求暴露的,只是在库中就已经先调用了。而T的地方是真正实现的。

 

既然符号已经暴露了,但是仍然不能链接是为什么?

解决的办法是:将库的链接顺序调整下,就ok了

附带:nm的相关命令资料

摘自:http://www.cnblogs.com/itech/archive/2012/09/16/2687423.html

 

Linux的nm查看动态和静态库中的符号


功能

列出.o .a .so中的符号信息,包括诸如符号的值,符号类型及符号名称等。所谓符号,通常指定义出的函数,全局变量等等。


使用

nm [option(s)] [file(s)]

有用的options:

  • -A 在每个符号信息的前面打印所在对象文件名称;
  • -C 输出demangle过了的符号名称;
  • -D 打印动态符号;
  • -l 使用对象文件中的调试信息打印出所在源文件及行号;
  • -n 按照地址/符号值来排序;
  • -u 打印出那些未定义的符号;

常见的符号类型:

  • A 该符号的值在今后的链接中将不再改变;
  • B 该符号放在BSS段中,通常是那些未初始化的全局变量;
  • D 该符号放在普通的数据段中,通常是那些已经初始化的全局变量;
  • T 该符号放在代码段中,通常是那些全局非静态函数;
  • U 该符号未定义过,需要自其他对象文件中链接进来;
  • W 未明确指定的弱链接符号;同链接的其他对象文件中有它的定义就用上,否则就用一个系统特别指定的默认值。

注意几点:

  • -C 总是适用于c++编译出来的对象文件。还记得c++中有重载么?为了区分重载函数,c++编译器会将函数返回值/参数等信息附加到函数名称中去形成一个mangle过的符号,那用这个选项列出符号的时候,做一个逆操作,输出那些原始的、我们可理解的符号名称。
  • 使用 -l 时,必须保证你的对象文件中带有符号调式信息,这一般要求你在编译的时候指定一个 -g 选项,见 Linux:Gcc
  • 使用nm前,最好先用Linux:File查看对象文件所属处理器架构,然后再用相应交叉版本的nm工具。


举例

更详细的内容见man page。这里举例说明:

nm -u hello.o
显示hello.o 中的未定义符号,需要和其他对象文件进行链接.
nm -A /usr/lib/* 2>/dev/null | grep "T memset"

在 /usr/lib/ 目录下找出哪个库文件定义了memset函数. 

 



2020 jeepxie.net webmaster#jeepxie.net
10 q. 0.007 s.
京ICP备10005923号