1. 程式人生 > >nm命令 檢視符號檔案

nm命令 檢視符號檔案

nm命令介紹的很多,但大多不介紹其函式符號標誌的含義。
最近在除錯動態庫時常用到,其中用的最多的用法:
nm -A * |grep “aaa” | c++filt // -A 為了顯示檔案, c++filt轉換為可讀風格,好像有個引數也能實現類似功能

其他內容整理如下(原作者未知):

nm用於列出目標檔案的符號清單,如果沒有指定目標檔案,則預設為“a.out”。nm的格式如下:

nm [‘-a’|‘–debug-syms’] [‘-g’|‘–extern-only’]
[‘-B’] [‘-C’|‘–demangle’[=style]] [‘-D’|‘–dynamic’]
[‘-S’|‘–print-size’] [‘-s’|‘–print-armap’]
[‘-A’|‘-o’|‘–print-file-name’][‘–special-syms’]
[‘-n’|‘-v’|‘–numeric-sort’] [‘-p’|‘–no-sort’]
[‘-r’|‘–reverse-sort’] [‘–size-sort’] [‘-u’|‘–undefined-only’]
[‘-t’ radix|‘–radix=’radix] [‘-P’|‘–portability’]
[‘–target=’bfdname] [‘-f’format|‘–format=’format]
[‘–defined-only’] [‘-l’|‘–line-numbers’] [‘–no-demangle’]
[‘-V’|‘–version’] [‘-X 32_64’] [‘–help’] [objfile…]

對於每一個符號,nm列出其值(the symbol value),型別(the symbol type)和其名字(the symbol name)。
如下例:

 00000024 T cleanup_before_linux
 00000018 T cpu_init
 00000060 T dcache_disable
 00000054 T dcache_enable
 0000006c T dcache_status
 00000000 T do_reset
 0000003c T icache_disable
 00000030 T icache_enable
 00000048 T icache_status

上面的顯示是使用nm cpu.o的輸出,對於cleanup_before_linux這個符號來說,00000024是以16進位制顯示的其值,T為其型別,而cleanup_before_linux是其名字。可以看出,上面顯示的cleanup_before_linux這個symbol的值實際上是該函式在text section中的偏移。但是,每個符號的值的具體含義依其型別而異。當然,對於每個符號的值,其型別、其值以及它們所屬的section是密切相關的。
下面說明符號型別:對於每一個符號來說,其型別如果是小寫的,則表明該符號是local的;大寫則表明該符號是global(external)的。

符號
型別 說明
A 該符號的值是絕對的,在以後的連結過程中,不允許進行改變。這樣的符號值,常常出現在中斷向量表中,例如用符號來表示各個中斷向量函式在中斷向量表中的位置。
B 該符號的值出現在非初始化資料段(bss)中。例如,在一個檔案中定義全域性static int test。則該符號test的型別為b,位於bss section中。其值表示該符號在bss段中的偏移。一般而言,bss段分配於RAM中
C 該符號為common。common symbol是未初始話資料段。該符號沒有包含於一個普通section中。只有在連結過程中才進行分配。符號的值表示該符號需要的位元組數。例如在一個c檔案中,定義int test,並且該符號在別的地方會被引用,則該符號型別即為C。否則其型別為B。
D 該符號位於初始話資料段中。一般來說,分配到data section中。例如定義全域性int baud_table[5] = {9600, 19200, 38400, 57600, 115200},則會分配於初始化資料段中。
G 該符號也位於初始化資料段中。主要用於small object提高訪問small data object的一種方式。
I 該符號是對另一個符號的間接引用。
N 該符號是一個debugging符號。
R 該符號位於只讀資料區。例如定義全域性const int test[] = {123, 123};則test就是一個只讀資料區的符號。注意在cygwin下如果使用gcc直接編譯成MZ格式時,原始檔中的test對應_test,並且其符號型別為D,即初始化資料段中。但是如果使用m6812-elf-gcc這樣的交叉編譯工具,原始檔中的test對應目標檔案的test,即沒有新增下劃線,並且其符號型別為R。一般而言,位於rodata section。值得注意的是,如果在一個函式中定義const char *test = “abc”, const char test_int = 3。使用nm都不會得到符號資訊,但是字串“abc”分配於只讀儲存器中,test在rodata section中,大小為4。
S 符號位於非初始化資料區,用於small object。
T 該符號位於程式碼區text section。
U 該符號在當前檔案中是未定義的,即該符號的定義在別的檔案中。例如,當前檔案呼叫另一個檔案中定義的函式,在這個被呼叫的函式在當前就是未定義的;但是在定義它的檔案中型別是T。但是對於全域性變數來說,在定義它的檔案中,其符號型別為C,在使用它的檔案中,其型別為U。
V 該符號是一個weak object。
W The symbol is a weak symbol that has not been specifically tagged as a weak object symbol.
- 該符號是a.out格式檔案中的stabs symbol。
? 該符號型別沒有定義