DoubleLi

qq: 517712484 wx: ldbgliet

  博客园 :: 首页 :: 博问 :: 闪存 :: 新随笔 :: 联系 :: 订阅 订阅 :: 管理 ::

一、undefined symbol错误

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

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

二、nm指令

1、nm指令的作用

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

2、nm指令的参数

nm指令的参数如下:

 
 
# 指令使用格式
 
nm -xx [objfile...]
 
# 指令中的-xx即为如下参数列表
 
[-a|--debug-syms]
 
[-g|--extern-only] //只显示外部符号.
 
[-B]
 
[-C|--demangle[=style]]
 
[-D|--dynamic]
 
[-S|--print-size]
 
[-s|--print-armap]
 
[-A|-o|--print-file-name]
 
[-n|-v|--numeric-sort]
 
[-p|--no-sort]
 
[-r|--reverse-sort] [--size-sort]
 
[-u|--undefined-only] //只显示未定义的符号.
 
[-t radix|--radix=radix] //符号值得进制。d 十进制, o 八进制, x 十六进制.
 
[-P|--portability]
 
[--target=bfdname]
 
[-fformat|--format=format] //输出的格式,有"bsd","sysv""posix"可选。默认是“bsd”
 
[--defined-only] //只显示已定义的符号.
 
[-l|--line-numbers] //对每一个符号,使用调试信息去查找文件名和行号
 
[--no-demangle]
 
[-V|--version]
 
[-X 32_64]
 
[--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 *
 

 
posted on 2022-10-31 21:41  DoubleLi  阅读(3865)  评论(0编辑  收藏  举报