Posted on 2019-04-10 14:28
Prayer 閱讀(627)
評(píng)論(0) 編輯 收藏 引用 所屬分類:
LINUX/UNIX/AIX
nm命令介紹的很多,但大多不介紹其函數(shù)符號(hào)標(biāo)志的含義。
最近在調(diào)試動(dòng)態(tài)庫(kù)時(shí)常用到,其中用的最多的用法:
nm -A * |grep “aaa” | c++filt // -A 為了顯示文件, c++filt轉(zhuǎn)換為可讀風(fēng)格,好像有個(gè)參數(shù)也能實(shí)現(xiàn)類似功能
其他內(nèi)容整理如下(原作者未知):
nm用于列出目標(biāo)文件的符號(hào)清單,如果沒(méi)有指定目標(biāo)文件,則默認(rèn)為“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...]
對(duì)于每一個(gè)符號(hào),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的輸出,對(duì)于cleanup_before_linux這個(gè)符號(hào)來(lái)說(shuō),00000024是以16進(jìn)制顯示的其值,T為其類型,而cleanup_before_linux是其名字。可以看出,上面顯示的cleanup_before_linux這個(gè)symbol的值實(shí)際上是該函數(shù)在text section中的偏移。但是,每個(gè)符號(hào)的值的具體含義依其類型而異。當(dāng)然,對(duì)于每個(gè)符號(hào)的值,其類型、其值以及它們所屬的section是密切相關(guān)的。
下面說(shuō)明符號(hào)類型:對(duì)于每一個(gè)符號(hào)來(lái)說(shuō),其類型如果是小寫(xiě)的,則表明該符號(hào)是local的;大寫(xiě)則表明該符號(hào)是global(external)的。
符號(hào) 類型 | 說(shuō)明 |
A | 該符號(hào)的值是絕對(duì)的,在以后的鏈接過(guò)程中,不允許進(jìn)行改變。這樣的符號(hào)值,常常出現(xiàn)在中斷向量表中,例如用符號(hào)來(lái)表示各個(gè)中斷向量函數(shù)在中斷向量表中的位置。 |
B | 該符號(hào)的值出現(xiàn)在非初始化數(shù)據(jù)段(bss)中。例如,在一個(gè)文件中定義全局static int test。則該符號(hào)test的類型為b,位于bss section中。其值表示該符號(hào)在bss段中的偏移。一般而言,bss段分配于RAM中 |
C | 該符號(hào)為common。common symbol是未初始話數(shù)據(jù)段。該符號(hào)沒(méi)有包含于一個(gè)普通section中。只有在鏈接過(guò)程中才進(jìn)行分配。符號(hào)的值表示該符號(hào)需要的字節(jié)數(shù)。例如在一個(gè)c文件中,定義int test,并且該符號(hào)在別的地方會(huì)被引用,則該符號(hào)類型即為C。否則其類型為B。 |
D | 該符號(hào)位于初始話數(shù)據(jù)段中。一般來(lái)說(shuō),分配到data section中。例如定義全局int baud_table[5] = {9600, 19200, 38400, 57600, 115200},則會(huì)分配于初始化數(shù)據(jù)段中。 |
G | 該符號(hào)也位于初始化數(shù)據(jù)段中。主要用于small object提高訪問(wèn)small data object的一種方式。 |
I | 該符號(hào)是對(duì)另一個(gè)符號(hào)的間接引用。 |
N | 該符號(hào)是一個(gè)debugging符號(hào)。 |
R | 該符號(hào)位于只讀數(shù)據(jù)區(qū)。例如定義全局const int test[] = {123, 123};則test就是一個(gè)只讀數(shù)據(jù)區(qū)的符號(hào)。注意在cygwin下如果使用gcc直接編譯成MZ格式時(shí),源文件中的test對(duì)應(yīng)_test,并且其符號(hào)類型為D,即初始化數(shù)據(jù)段中。但是如果使用m6812-elf-gcc這樣的交叉編譯工具,源文件中的test對(duì)應(yīng)目標(biāo)文件的test,即沒(méi)有添加下劃線,并且其符號(hào)類型為R。一般而言,位于rodata section。值得注意的是,如果在一個(gè)函數(shù)中定義const char *test = “abc”, const char test_int = 3。使用nm都不會(huì)得到符號(hào)信息,但是字符串“abc”分配于只讀存儲(chǔ)器中,test在rodata section中,大小為4。 |
S | 符號(hào)位于非初始化數(shù)據(jù)區(qū),用于small object。 |
T | 該符號(hào)位于代碼區(qū)text section。 |
U | 該符號(hào)在當(dāng)前文件中是未定義的,即該符號(hào)的定義在別的文件中。例如,當(dāng)前文件調(diào)用另一個(gè)文件中定義的函數(shù),在這個(gè)被調(diào)用的函數(shù)在當(dāng)前就是未定義的;但是在定義它的文件中類型是T。但是對(duì)于全局變量來(lái)說(shuō),在定義它的文件中,其符號(hào)類型為C,在使用它的文件中,其類型為U。 |
V | 該符號(hào)是一個(gè)weak object。 |
W | The symbol is a weak symbol that has not been specifically tagged as a weak object symbol. |
- | 該符號(hào)是a.out格式文件中的stabs symbol。 |
| 該符號(hào)類型沒(méi)有定義 |