1.命令簡(jiǎn)介
nm 命令是 GNU Binutils 二進(jìn)制工具集的一員,用于顯示目標(biāo)文件中的符號(hào)。如果沒有為 nm 命令指明目標(biāo)文件,則 nm 假定目標(biāo)文件是 a.out。
nm 命令顯示的符號(hào)類型,至少使用以下類型,其他類型取決于目標(biāo)文件格式。符號(hào)類型如果是小寫,符號(hào)通常是本地的;如果是大寫,符號(hào)是全局的(外部的)。但是,有一些小寫符號(hào)類型表示特殊的全局符號(hào),例如 u、v 和 w。
A
該符號(hào)的值是絕對(duì)的,在以后的鏈接過程中,不允許進(jìn)行改變。這樣的符號(hào)值,常常出現(xiàn)在中斷向量表中,例如用符號(hào)來表示各個(gè)中斷向量函數(shù)在中斷向量表中的位置。
b,B
該符號(hào)的值出現(xiàn)在非初始化數(shù)據(jù)段(BSS)中。例如,在一個(gè)文件中定義全局static int test。則該符號(hào)test的類型為b,位于bss section中。其值表示該符號(hào)在BSS段中的偏移。
C
該符號(hào)為common。common symbol是未初始化的數(shù)據(jù)。該符號(hào)沒有包含在一個(gè)普通section中,只有在鏈接過程中才進(jìn)行分配。符號(hào)的值表示該符號(hào)需要的字節(jié)數(shù)。例如在一個(gè)C文件中,定義int test,并且該符號(hào)在別的地方會(huì)被引用,則該符號(hào)類型即為C,否則其類型為B。
d,D
該符號(hào)位于初始化數(shù)據(jù)段(data section)。例如定義全局變量 int baud_table[5]={9600, 19200, 38400, 57600, 115200},則會(huì)被分配在初始化數(shù)據(jù)段中。
g,G
該符號(hào)也位于初始化數(shù)據(jù)段中。主要用于small object提高訪問small data object的一種方式
i
這是對(duì)標(biāo)準(zhǔn)ELF符號(hào)類型集的GNU擴(kuò)展。它表示一個(gè)符號(hào)如果被重定位引用,不會(huì)計(jì)算該符號(hào)的地址,而是必須在運(yùn)行時(shí)計(jì)算
N
該符號(hào)是一個(gè)debugging符號(hào)。
p
該符號(hào)在stackunwindsection
r,R
該符號(hào)位于只讀數(shù)據(jù)段(readonly data section)。例如定義全局const int test[]={123,123};則test就是一個(gè)只讀數(shù)據(jù)段的符號(hào)。
s,S
符號(hào)位于非初始化數(shù)據(jù)區(qū),用于small object。
t,T
該符號(hào)位于代碼段(text section)。
u
符號(hào)是唯一的全局符號(hào)。這是GNU對(duì)標(biāo)準(zhǔn)ELF符號(hào)綁定集的擴(kuò)展。對(duì)于這樣的符號(hào),動(dòng)態(tài)鏈接器將確保在整個(gè)過程中只有一個(gè)使用此名稱和類型的符號(hào)。
U
該符號(hào)在當(dāng)前文件中是未定義的,即該符號(hào)定義在別的文件中。例如,當(dāng)前文件調(diào)用另一個(gè)文件中定義的函數(shù),這個(gè)被調(diào)用的函數(shù)在當(dāng)前文件就是未定義的,但是在定義它的文件中類型是T。對(duì)于全局變量來說,在定義它的文件中,其符號(hào)類型為B或D,在使用它的文件中,其類型為U。
v,V
該符號(hào)是一個(gè)弱符號(hào)。當(dāng)弱定義符號(hào)與正常定義符號(hào)鏈接時(shí),使用正常定義符號(hào)時(shí)不會(huì)出錯(cuò)。當(dāng)鏈接未定義的弱定義符號(hào),弱符號(hào)的值將變?yōu)榱悖覜]有錯(cuò)誤。在某些系統(tǒng)上,大寫表示已指定默認(rèn)值
w,W
該符號(hào)是一個(gè)弱符號(hào),未專門標(biāo)記為弱對(duì)象符號(hào)。當(dāng)弱定義符號(hào)與正常定義符號(hào)鏈接時(shí),使用正常定義符號(hào)時(shí)不會(huì)出錯(cuò)。當(dāng)鏈接未定義的弱未定義符號(hào)時(shí),該符號(hào)的值將以系統(tǒng)特定的方式確定,且不會(huì)出錯(cuò)。在某些系統(tǒng)上,大寫表示已指定默認(rèn)值
-
該符號(hào)是a.out格式文件中的stabs symbol。在這種情況下,打印的下一個(gè)值是stabs other字段、stabs desc字段和stab類型。stabs符號(hào)用于保存調(diào)試信息
?
該符號(hào)類型沒有定義
2.命令格式
nm[-A|-o|--print-file-name][-a|--debug-syms]
[-B|--format=bsd][-C|--demangle[=style]]
[-D|--dynamic][-f|--format=]
[-g|--extern-only][-h|--help]
[-l|--line-numbers][-n|-v|--numeric-sort]
[-P|--portability][-p|--no-sort]
[-r|--reverse-sort][-S|--print-size]
[-s|--print-armap][-t|--radix=]
[-u|--undefined-only][-V|--version]
[-X32_64][--defined-only][--no-demangle]
[--plugin][--size-sort][--special-syms]
[--synthetic][--target=bfdname]
[objfile...]
3.選項(xiàng)說明
-A,-o,--print-file-name
在找到的各個(gè)符號(hào)的名字前加上文件名,而不是在此文件的所有符號(hào)前只出現(xiàn)文件名一次
-a,--debug-syms
顯示調(diào)試符號(hào)
-B,--format=bsd
用來兼容MIPS的nm
-C,--demangle[=STYLE]
將低級(jí)符號(hào)名解碼(demangle)成用戶級(jí)名字,比如去除編譯時(shí)添加的前置下劃線,這樣可以使得 C++函數(shù)名具有可讀性。不同的編譯器符號(hào)修飾風(fēng)格不同,可以使用=STYLE 參數(shù)來選擇合適的解碼風(fēng)格
-D, --dynamic:顯示動(dòng)態(tài)符號(hào)。該任選項(xiàng)僅對(duì)于動(dòng)態(tài)目標(biāo)(例如特定類型的共享庫)有意義
-f,--format=FORMAT
FORMAT可取值bsd、sysv或posix,該選項(xiàng)在GNUnm中有用,默認(rèn)為bsd
-g,--extern-only
僅顯示外部符號(hào)
-h,--help
顯示幫助信息
-l,--line-numbers
對(duì)每個(gè)符號(hào),使用調(diào)試信息來試圖找到文件名和行號(hào)。對(duì)于已定義的符號(hào),查找符號(hào)地址的行號(hào)。對(duì)于未定義符號(hào),查找指向符號(hào)重定位入口的行號(hào)。如果可以找到行號(hào)信息,顯示在符號(hào)信息之后
-n,-v,--numeric-sort
按符號(hào)對(duì)應(yīng)地址的順序排序,而非按符號(hào)名的字符順序
-P,--portability
使用 POSIX.2 標(biāo)準(zhǔn)輸出格式代替默認(rèn)的輸出格式。等同于-f posix
-p,--no-sort
按目標(biāo)文件中遇到的符號(hào)順序顯示,不排序
-r,--reverse-sort
逆序排序。例如,升序變?yōu)榻敌?-S,--print-size
以 BSD 輸出樣式輸出已定義符號(hào)的值和大小。對(duì)于不記錄符號(hào)大小的目標(biāo)文件格式,此選項(xiàng)不起作用,除非使用了--size sort,在這種情況下,將顯示計(jì)算的大小
-s,--print-armap
當(dāng)列出庫中成員的符號(hào)時(shí),同時(shí)列出索引。索引的內(nèi)容包含:模塊與其包含的名字的定義之間的映射
-t,--radix=RADIX
使用基數(shù) radix 進(jìn)制顯示符號(hào)值。radix 只能為 d(十進(jìn)制)、o(八進(jìn)制)或 x(十六進(jìn)制)
-u,--undefined-only
僅顯示沒有定義的符號(hào)
-V,--version
顯示版本信息并退出
-X
為了與 AIX 版本的 nm 兼容,選項(xiàng)-X 將被忽略。它可接受一個(gè)參數(shù),該參數(shù)必須是字符串32_64。AIX nm 的默認(rèn)模式對(duì)應(yīng)于-X 32,GNU nm 不支持模式-X 32
--defined-only
僅顯示有定義的符號(hào)
--no-demangle
不解碼低級(jí)符號(hào)名,這是默認(rèn)選項(xiàng)
--pluginNAME
加載名為 name 的插件以添加對(duì)額外目標(biāo)類型的支持。只有在啟用插件支持的情況下構(gòu)建了工具鏈時(shí),此選項(xiàng)才可用
--size-sort
按符號(hào)大小排列
--special-syms
顯示目標(biāo)相關(guān)的具體特殊含義的符號(hào)。這些符號(hào)通常被特定目標(biāo)文件用于某些特殊處理,當(dāng)包含在正常符號(hào)列表中時(shí)通常不起作用。例如,對(duì)于ARM目標(biāo),此選項(xiàng)將跳過用于標(biāo)記ARM代碼、Thumb代碼和數(shù)據(jù)之間轉(zhuǎn)換的映射符號(hào)
--synthetic
輸出合成符號(hào)。合成符號(hào)是鏈接器為各種目的創(chuàng)建的特殊符號(hào),默認(rèn)情況下不會(huì)顯示它們,因?yàn)樗鼈儾皇嵌M(jìn)制文件源代碼的一部分
--target=BFDNAME
指定系統(tǒng)默認(rèn)格式以外的目標(biāo)文件格式
4.常用示例
首先給出后面大部分示例所基于的源代碼以及編譯指令。涉及兩個(gè) C++ 源文件。
test.cpp:
#include
intdUnInitialized;
intdInitialized=1;
charsTest[]="good";
voidprint()
{
std::cout<<"dUnInitialized="<"dInitialized="<"sTest="<
main.cpp:
使用-g選項(xiàng)加入調(diào)試信息,分別編譯生成目標(biāo)文件objdump.o與main.o。
g++-c-gtest.cpp-otest.o
g++-c-gmain.cpp-omain.o
然后通過ar命令將兩個(gè)目標(biāo)文件打包成靜態(tài)庫libobjdump.a。
arcrvlibnm.amain.otest.o
(1)列出目標(biāo)文件test.o與main.o的符號(hào)清單。
nm-C*.o
main.o:
U__cxa_atexit
U__dso_handle
UdUnInitialized
0000000000000057t_GLOBAL__sub_I_main
0000000000000000Tmain
000000000000001at__static_initialization_and_destruction_0(int,int)
Uprint()
Ustd::Init()
Ustd::~Init()
0000000000000000bstd::__ioinit
test.o:
U__cxa_atexit
0000000000000000DdInitialized
U__dso_handle
0000000000000000BdUnInitialized
00000000000000aet_GLOBAL__sub_I_dUnInitialized
0000000000000004DsTest
0000000000000071t__static_initialization_and_destruction_0(int,int)
0000000000000000Tprint()
Ustd::operator<<(int)
?????????????????U?std::operator<<(std::ostream&?(*)(std::ostream&))
?????????????????U?std::Init()
?????????????????U?std::~Init()
?????????????????U?std::cout
?????????????????U?std::basic_ostream>&std::endl>(std::basic_ostream>&)
0000000000000004bstd::__ioinit
Ustd::basic_ostream>&std::operator<>(std::basic_ostream>&,charconst*)
使用 -C 選項(xiàng)將符號(hào)解碼成可讀形式,從 test.o 的輸出結(jié)果可以看出,已初始化的全局變量 dInitialized 與 sTest,符號(hào)類型是 D,所以其位于初始化的 Data 段。未初始化的全局變量 dUnInitialized 符號(hào)類型是 B,所以其位于未初始化的 BSS 段。函數(shù)print() 的符號(hào)類型是 T,說明其位于代碼段(Text Section)。
從 main.o 的輸出結(jié)果可以看出,變量 dUnInitialized 與函數(shù) print() 因?yàn)榫x在其它文件中,所以符號(hào)類型是 U,表示該符號(hào)在當(dāng)前文件中是未定義的。值得注意的是,變量 localVar 因?yàn)槭蔷植孔兞浚?nm 無法獲取其符號(hào)。
參考文獻(xiàn)
[1] nm manual
[2] GNU Binutils
[3] Linux命令手冊(cè).nm
[4] linux中的nm命令簡(jiǎn)介
責(zé)任編輯:xj
原文標(biāo)題:每天一個(gè) Linux 命令(101):nm 命令
文章出處:【微信公眾號(hào):Linux愛好者】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
-
Linux
+關(guān)注
關(guān)注
87文章
11304瀏覽量
209535 -
命令
+關(guān)注
關(guān)注
5文章
684瀏覽量
22027 -
GNU
+關(guān)注
關(guān)注
0文章
143瀏覽量
17494
原文標(biāo)題:每天一個(gè) Linux 命令(101):nm 命令
文章出處:【微信號(hào):LinuxHub,微信公眾號(hào):Linux愛好者】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論