歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
您现在的位置: Linux教程網 >> UnixLinux >  >> Linux基礎 >> Linux技術

linux;ar,gcc,ldd,nm命令詳解

轉自:http://blog.csdn.net/stone548534/article/details/8111764
ar
功能說明:建立或修改備存文件,或是從備存文件中抽取文件。
語  法:ar[-dmpqrtx][cfosSuvV][a<成員文件>][b<成員文件>][i<成員文件>][備存文件][成員文件]
補充說明:ar可讓您集合許多文件,成為單一的備存文件。在備存文件中,所有成員文件皆保有原來的屬性與權限。
參  數:
 指令參數
 -d  刪除備存文件中的成員文件。
 -m  變更成員文件在備存文件中的次序。
 -p  顯示備存文件中的成員文件內容。
 -q  將問家附加在備存文件末端。
 -r  將文件插入備存文件中。
 -t  顯示備存文件中所包含的文件。
 -x  自備存文件中取出成員文件。
 選項參數
 a<成員文件>  將文件插入備存文件中指定的成員文件之後。
 b<成員文件>  將文件插入備存文件中指定的成員文件之前。
 c  建立備存文件。
 f  為避免過長的文件名不兼容於其他系統的ar指令指令,因此可利用此參數,截掉要放入備存文件中過長的成員文件名稱。
 i<成員文件>  將問家插入備存文件中指定的成員文件之前。
 o  保留備存文件中文件的日期。
 s  若備存文件中包含了對象模式,可利用此參數建立備存文件的符號表。
 S  不產生符號表。
 u  只將日期較新文件插入備存文件中。
 v  程序執行時顯示詳細的信息。
 V  顯示版本信息。
gcc
基本用法:
在使用Gcc編譯器的時候,我們必須給出一系列必要的調用參數和文件名稱。GCC編譯器的調用參數大約有100多個,其中多數參數我們可能根本就用不到,這裡只介紹其中最基本、最常用的參數。
  GCC最基本的用法是∶gcc[options] [filenames]
  其中options就是編譯器所需要的參數,filenames給出相關的文件名稱。
  -c,只編譯,不連接成為可執行文件,編譯器只是由輸入的.c等源代碼文件生成.o為後綴的目標文件,通常用於編譯不包含主程序的子程序文件。
  -ooutput_filename,確定輸出文件的名稱為output_filename,同時這個名稱不能和源文件同名。如果不給出這個選項,gcc就給出預設的可執行文件a.out。
  -g,產生符號調試工具(GNU的gdb)所必要的符號資訊,要想對源代碼進行調試,我們就必須加入這個選項。
  -O,對程序進行優化編譯、連接,采用這個選項,整個源代碼會在編譯、連接過程中進行優化處理,這樣產生的可執行文件的執行效率可以提高,但是,編譯、連接的速度就相應地要慢一些。
  -O2,比-O更好的優化編譯、連接,當然整個編譯、連接過程會更慢。
  -Idirname,將dirname所指出的目錄加入到程序頭文件目錄列表中,是在預編譯過程中使用的參數。C程序中的頭文件包含兩種情況∶
  A)#include
  B)#include“myinc.h”
  其中,A類使用尖括號(<>),B類使用雙引號(“ ”)。對於A類,預處理程序cpp在系統預設包含文件目錄(如/usr/include)中搜尋相應的文件,而B類,預處理程序在目標文件的文件夾內搜索相應文件。
ldd
1、首先ldd不是一個可執行程序,而只是一個shell腳本
  2、ldd能夠顯示可執行模塊的dependency,其原理是通過設置一系列的環境變量,如下:LD_TRACE_LOADED_OBJECTS、LD_WARN、LD_BIND_NOW、LD_LIBRARY_VERSION、LD_VERBOSE等。當LD_TRACE_LOADED_OBJECTS環境變量不為空時,任何可執行程序在運行時,它都會只顯示模塊的dependency,而程序並不真正執行。要不你可以在shell終端測試一下,如下:
  (1) export LD_TRACE_LOADED_OBJECTS=1
  (2) 再執行任何的程序,如ls等,看看程序的運行結果
  3、ldd顯示可執行模塊的dependency的工作原理,其實質是通過ld-linux.so(elf動態庫的裝載器)來實現的。我們知道,ld-linux.so模塊會先於executable模塊程序工作,並獲得控制權,因此當上述的那些環境變量被設置時,ld-linux.so選擇了顯示可執行模塊的dependency。
  4、實際上可以直接執行ld-linux.so模塊,如:/lib/ld-linux.so.2 --listprogram(這相當於ldd program)ldd命令使用方法(摘自ldd --help)
  名稱 ldd - 打印共享庫的依賴關系
  大綱 ldd [選項]... 文件...
  描述 ldd 輸出在命令行上指定的每個程序或共享庫需要的共享庫。
  選項
  --version
  打印ldd的版本號
  -v --verbose
  打印所有信息,例如包括符號的版本信息
  -d --data-relocs
  執行符號重部署,並報告缺少的目標對象(只對ELF格式適用)
  -r --function-relocs
  對目標對象和函數執行重新部署,並報告缺少的目標對象和函數(只對ELF格式適用)
  --help 用法信息
  ldd的標准版本與glibc2一起提供。Libc5與老版本以前提供,在一些系統中還存在。在libc5版本中長選項不支持。另一方面,glibc2版本不支持-V選項,只提供等價的--version選項。
  如果命令行中給定的庫名字包含'/',這個程序的libc5版本將使用它作為庫名字;否則它將在標准位置搜索庫。運行一個當前目錄下的共享庫,加前綴"./"。
  ldd不能工作在a.out格式的共享庫上。
  ldd不能工作在一些非常老的a.out程序上,這些程序在支持ldd的編譯器發行前已經創建。如果你在這種類型的程序上使用ldd,程序將嘗試argc= 0的運行方式,其結果不可預知。
nm
linux中,nm用來列出目標文件的符號清單。
  下面是nm命令的格式: nm [-a|--debug-syms] [-g|--extern-only][-B][-C|--demangle] [-D|--dynamic][-s|--print-armap][-o|--print-file-name][-n|--numeric-sort][-p|--no-sort] [-r|--reverse-sort][--size-sort][-u|--undefined-only]
[-l|--line-numbers][--help][--version] [-t radix|--radix=radix][-P|--portability] [-fformat|--format=format][--target=bfdname] [objfile...]
  如果沒有為nm命令指出目標文件,則nm假定目標文件是a.out。下面列出該命令的任選項,大部分支持“-”開頭的短格式和“—“開頭的長格式。
  -A、-o或--print-file-name:在找到的各個符號的名字前加上文件名,而不是在此文件的所有符號前只出現文件名一次。
  例如nmlibtest.a的輸出如下:
  CPThread.o:
  00000068T Main__8CPThreadPv
  00000038T Start__8CPThread
  00000014T _._8CPThread
  00000000T __8CPThread
  00000000? __FRAME_BEGIN__
  …………………………………
  則nm–A 的輸出如下:
  libtest.a:CPThread.o:00000068T Main__8CPThreadPv
  libtest.a:CPThread.o:00000038T Start__8CPThread
  libtest.a:CPThread.o:00000014T _._8CPThread
  libtest.a:CPThread.o:00000000T __8CPThread
  libtest.a:CPThread.o:00000000? __FRAME_BEGIN__
  …………………………………………………………
  -a或--debug-syms:顯示調試符號。
  -B:等同於--format=bsd,用來兼容MIPS的nm。
  -C或--demangle:將低級符號名解碼(demangle)成用戶級名字。這樣可以使得C++函數名具有可讀性。
  -D或--dynamic:顯示動態符號。該任選項僅對於動態目標(例如特定類型的共享庫)有意義。
  -fformat:使用format格式輸出。format可以選取bsd、sysv或posix,該選項在GNU的nm中有用。默認為bsd。
  -g或--extern-only:僅顯示外部符號。
  -n、-v或--numeric-sort:按符號對應地址的順序排序,而非按符號名的字符順序。
  -p或--no-sort:按目標文件中遇到的符號順序顯示,不排序。
  -P或--portability:使用POSIX.2標准輸出格式代替默認的輸出格式。等同於使用任選項-fposix。
  -s或--print-armap:當列出庫中成員的符號時,包含索引。索引的內容包含:哪些模塊包含哪些名字的映射。
  -r或--reverse-sort:反轉排序的順序(例如,升序變為降序)。
  --size-sort:按大小排列符號順序。該大小是按照一個符號的值與它下一個符號的值進行計算的。
  -tradix或--radix=radix:使用radix進制顯示符號值。radix只能為“d”表示十進制、“o”表示八進制或“x”表示十六進制。
  --target=bfdname:指定一個目標代碼的格式,而非使用系統的默認格式。
  -u或--undefined-only:僅顯示沒有定義的符號(那些外部符號)。
  -l或--line-numbers:對每個符號,使用調試信息來試圖找到文件名和行號。對於已定義的符號,查找符號地址的行號。對於未定義符號,查找指向符號重定位入口的行號。如果可以找到行號信息,顯示在符號信息之後。
  -V或--version:顯示nm的版本號。
  --help:顯示nm的任選項。
  …………………………………………
  輸出符號類型說明
  A
  該符號的值是絕對的,在以後的鏈接過程中,不允許進行改變。這樣的符號值,常常出現在中斷向量表中,例如用符號來表示各個中斷向量函數在中斷向量表中的位置。
  B
  該符號的值出現在非初始化數據段(bss)中。例如,在一個文件中定義全局staticint
test。則該符號test的類型為b,位於bsssection中。其值表示該符號在bss段中的偏移。一般而言,bss段分配於RAM中
  C
  該符號為common。commonsymbol是未初始話數據段。該符號沒有包含於一個普通section中。只有在鏈接過程中才進行分配。符號的值表示該符號需要的字節數。例如在一個c文件中,定義int test,並且該符號在別的地方會被引用,則該符號類型即為C。否則其類型為B。
  D
  該符號位於初始話數據段中。一般來說,分配到datasection中。例如定義全局int baud_table[5] = {9600, 19200, 38400, 57600,115200},則會分配於初始化數據段中。
  G
  該符號也位於初始化數據段中。主要用於smallobject提高訪問small data object的一種方式。
  I
  該符號是對另一個符號的間接引用。
  N
  該符號是一個debugging符號。
  R
  該符號位於只讀數據區。例如定義全局constint test[] = {123,123};則test就是一個只讀數據區的符號。注意在cygwin下如果使用gcc直接編譯成MZ格式時,源文件中的test對應_test,並且其符號類型為D,即初始化數據段中。但是如果使用m6812-elf-gcc這樣的交叉編譯工具,源文件中的test對應目標文件的test,即沒有添加下劃線,並且其符號類型為R。一般而言,位於rodatasection。值得注意的是,如果在一個函數中定義const
char *test = “abc”, const chartest_int = 3。使用nm都不會得到符號信息,但是字符串“abc”分配於只讀存儲器中,test在rodatasection中,大小為4。
  S
  符號位於非初始化數據區,用於smallobject。
  T
  該符號位於代碼區textsection。
  U
  該符號在當前文件中是未定義的,即該符號的定義在別的文件中。例如,當前文件調用另一個文件中定義的函數,在這個被調用的函數在當前就是未定義的;但是在定義它的文件中類型是T。但是對於全局變量來說,在定義它的文件中,其符號類型為C,在使用它的文件中,其類型為U。
  V
  該符號是一個weakobject。
  W
  Thesymbol is a weak symbol that has not been specifically tagged as aweak object symbol.
  -
  該符號是a.out格式文件中的stabssymbol。
  ?
  該符號類型沒有定義
Copyright © Linux教程網 All Rights Reserved