当前位置:   article > 正文

【Linux】symbol lookup error: undefined symbol + nm指令定位错误

symbol lookup error

一、undefined symbol错误

今天在运行模块执行文件时,出现了如下报错 "symbol lookup error""undefined symbol",提示 cos_getfile_mcd 可执行文件在加载 .so 文件时,出现了无法找到符号的错误,并给出了具体错误:_ZN20CCosGetfileTimerInfoC2Ev 符号未定义。

那么如何定位该错误呢?一般可以先使用 ldd指令 去查看一下可执行文件的链接库,但是我的可执行文件是在加载调用.so文件的过程中出现报错,ldd指令并没有解决我的问题,因此要用的nm指令来定位错误源。那么接着请往下看看 nm指令 介绍。

二、nm指令

1、nm指令的作用

nm命令主要是用来列出某些文件中的符号(说白了就是一些函数和全局变量等),后端程序员一般需要掌握该指令。

2、nm指令的参数

nm指令的参数如下:

  1. # 指令使用格式
  2. nm -xx [objfile...]
  3. # 指令中的-xx即为如下参数列表
  4. [-a|--debug-syms]
  5. [-g|--extern-only] //只显示外部符号.
  6. [-B]
  7. [-C|--demangle[=style]]
  8. [-D|--dynamic]
  9. [-S|--print-size]
  10. [-s|--print-armap]
  11. [-A|-o|--print-file-name]
  12. [-n|-v|--numeric-sort]
  13. [-p|--no-sort]
  14. [-r|--reverse-sort] [--size-sort]
  15. [-u|--undefined-only] //只显示未定义的符号.
  16. [-t radix|--radix=radix] //符号值得进制。d 十进制, o 八进制, x 十六进制.
  17. [-P|--portability]
  18. [--target=bfdname]
  19. [-fformat|--format=format] //输出的格式,有"bsd","sysv""posix"可选。默认是“bsd”
  20. [--defined-only] //只显示已定义的符号.
  21. [-l|--line-numbers] //对每一个符号,使用调试信息去查找文件名和行号
  22. [--no-demangle]
  23. [-V|--version]
  24. [-X 32_64]
  25. [--help] //显示帮助

3、nm指令输出

nm命令的输出包含三个部分:1)符号值,默认显示十六进制,也可以指定; 2 )符号类型,小写表示本地符号,大写表示全局符号(external),符号类型介绍如下所示;3)符号名称,符号名称前后分别会加上一段拓展名,代表不同的符号类型,例如后面的扩展名D1EV是指的C++析构函数。

输出示例如下。

  • A:符号值是绝对的。在进一步的连接中,不会被改变
  • B:符号位于未初始化数据段(BSS)
  • C:共用(common)符号. 共用符号是未初始化的数据。在连接时,多个共用符号可能采用一个同样的名字,如果这个符号在某个地方被定义,共用符号被认为是未定义的引用
  • D:已初始化数据段的符号
  • G:已初始化数据段中的小目标(small objective)符号. 一些目标文件格式允许更有效的访问小目标数据,比如一个全局的int变量相对于一个大的全局数组
  • I:其他符号的直接应用,这是GNU扩展的,很少用
  • N:调试符号
  • R:只读数据段符号
  • S:未初始化数据段中的小目标(small object)符号
  • T:代码段的符号
  • U:未定义符号
  • V:弱对象(weak object)符号. 当一个已定义的弱符号被连接到一个普通定义符号,普通定义符号可以正常使用,当一个未定义的弱对象被连接到一个未定义的符号,弱符号的值为0
  • W:一个没有被指定一个弱对象符号的弱符号(weak symbol)
  • -:a.out目标文件中的刺符号(stabs symbol). 这种情况下,打印的下一个值是其他字段,描述字段,和类型。刺符号用于保留调试信息
  • ?:未知符号类型,或者目标文件特有的符号类型

三、问题定位及解决步骤

1、使用nm指令定位未定义的符号

定位指令如下所示,可以看到结果中红框标注的符号,其类型即为未定义的 U。那么这里就可以确定代码中未定义的符号即为CCosGetfileTimerInfo。

nm -g ../../bin/cos_getfile.so | grep CCosGetfileTime

 2、使用grep命令在源码中定位符号

使用grep指令定位源码中CCosGetfileTimerInfo符号的位置,并定位到cos_getfile_mcd中的位置,之后去源码中查看相关定义使用的具体位置,并进行输出测试(在使用处前后加上测试输出,定位是否在此处出错),接着就可以确定出未定义符号的位置。

grep CCosGetfileTimerInfo *

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/很楠不爱3/article/detail/185129
推荐阅读
相关标签
  

闽ICP备14008679号