1. 程式人生 > >GCC引數含義

GCC引數含義

-fPIC :                  該選項用於生成位置無關的程式碼
-shared :              將-fPIC生成的位置無關的程式碼作為動態庫,一般情況下,-fPIC和-shared都是一起使用的。生成SO檔案,共享庫
-static :                 此選項將禁止使用動態庫,所以,編譯出來的東西,一般都很大,也不需要什麼動態連線庫,就可以執行

GNU CC(簡稱gcc)是GNU專案中符合ANSI C標準的編譯系統,能夠編譯用C、C++、Object C、Jave等多種語言編寫的程式。gcc又可以作為交叉編譯工具,它能夠在當前CPU平臺上為多種不同體系結構的硬體平臺開發軟體,非常適合在

嵌入式領域的開發編譯,如常用的arm-Linux-gcc交叉編譯工具

    通常後跟一些選項和檔名來使用 GCC 編譯器。gcc 命令的基本用法如下:

    gcc [options] [filenames]

    選項指定編譯器怎樣進行編譯。

一、gcc 編譯流程

1.預處理-Pre-Processing

gcc  -E  test.c  -o  test.i    //.i檔案

2.編譯-Compiling

gcc  -S  test.i  -o   test.s  //.s檔案

3.彙編-Assembling         //.o檔案

gcc  -c  test.s  -o  test.o

4.連結-Linking           //bin檔案

gcc  test.o  -o  test

二、gcc工程慣用

1.編譯

gcc  -c  test.c          //.o檔案,彙編

gcc  -o  test  test.c  //bin可執行檔案

gcc   test.c             //a.out可執行檔案

如果是c++ 直接將gcc改為g++即可。

2.常用引數

1)-E引數

  -E 選項指示編譯器僅對輸入檔案進行預處理。當這個選項被使用時, 前處理器的輸出被送到標準輸出而不是儲

存在檔案裡.

2)-S引數

-S 編譯選項告訴 GCC 在為 C 程式碼產生了組合語言檔案後停止編譯

。 GCC 產生的組合語言檔案的預設副檔名

是 .s 。

3)-c引數

-c 選項告訴 GCC 僅把原始碼編譯為目的碼。預設時 GCC 建立的目的碼檔案有一個 .o 的副檔名

4)-o引數

        -o 編譯選項來為將產生的可執行檔案用指定的檔名。

5)-O引數

-O 選項告訴 GCC 對原始碼進行基本優化。這些優化在大多數情況下都會使程式執行的更快。 -O2 選項告訴

GCC 產生儘可能小和儘可能快的程式碼。 如-O2,-O3,-On(n 常為0–3);

-O  主要進行跳轉和延遲退棧兩種優化;

-O0 表示不做優化

-O1 為預設優化

-O2 除了完成-O1的優化之外,還進行一些額外的調整工作,如指令調整等。

-O3 則包括迴圈展開和其他一些與處理特性相關的優化工作。

選項將使編譯的速度比使用 -O 時慢, 但通常產生的程式碼執行速度會更快。

如:

[[email protected] test]# gcc test.c -O3

[[email protected] test]# gcc -O3 test.c

[[email protected] test]# gcc -o tt test.c -O2

[[email protected] test]# gcc -O2 -o tt test.c

6)除錯選項-g和-pg

  GCC 支援數種除錯和剖析選項,常用到的是 -g 和 -pg 。

-g 選項告訴 GCC 產生能被 GNU 偵錯程式使用的除錯資訊以便除錯你的程式GCC 提供了一個很多其他 C 編譯器裡沒有的特性, 在 GCC 裡你能使-g 和 -O(產生優化程式碼)聯用

-pg 選項告訴 GCC 在編譯好的程式里加入額外的程式碼。執行程式時, 產生 gprof 用的剖析資訊以顯示你的程式的耗時情況。

7) -l引數和-L引數

      -l引數就是用來指定程式要連結的庫,-l引數緊接著就是庫名,那麼庫名跟真正的庫檔名有什麼關係呢?

      就拿數學庫來說,他的庫名是m,他的庫檔名是libm.so,很容易看出,把庫檔名的頭lib和尾.so去掉就是庫名

了。

如:

       gcc  xxx.c   -lm( 動態數學庫)

                               -lpthread

      好了現在我們知道怎麼得到庫名了,比如我們自已要用到一個第三方提供的庫名字叫libtest.so,那麼我們只要把

libtest.so拷貝到 /usr/lib裡,編譯時加上-ltest引數,我們就能用上libtest.so庫了(當然要用libtest.so庫裡的函式,我們還

需要與 libtest.so配套的標頭檔案)。

      放在/lib和/usr/lib和/usr/local/lib裡的庫直接用-l引數就能連結了,但如果庫檔案沒放在這三個目錄裡,而是放在

其他目錄裡, 這時我們只用-l引數的話,連結還是會出錯,出錯資訊大概是:“/usr/bin/ld: cannot find

-lxxx”,也就是連結 程式ld在那3個目錄裡找不到libxxx.so,這時另外一個引數-L就派上用場了,比如常用的X11的庫

,它放在/usr/X11R6/lib目錄 下,我們編譯時就要用-L/usr/X11R6/lib -lX11引數,-L引數跟著的是庫檔案所在的目錄

名。再比如我們把libtest.so放在/aaa/bbb/ccc目錄下,那連結引數就是-L/aaa/bbb/ccc -ltest

      另外,大部分libxxxx.so只是一個連結,以RH9為例,比如libm.so它連結到/lib/libm.so.x,/lib/libm.so.6 又連結

到/lib/libm-2.3.2.so,如果沒有這樣的連結,還是會出錯,因為ld只會找libxxxx.so,所以如果你要用到xxxx

庫,而只有libxxxx.so.x或者libxxxx-x.x.x.so,做一個連結就可以了ln -s libxxxx-x.x.x.so libxxxx.so

手工來寫連結引數總是很麻煩的,還好很多庫開發包提供了生成連結引數的程式,名字一般叫xxxx-config,一般放在/usr/bin目錄下比如 gtk1.2的連結引數生成程式是gtk-config,執行gtk-config –libs就能得到以下輸出”-L/usr/lib -L/usr/X11R6/lib -lgtk -lgdk -rdynamic -lgmodule -lglib -ldl -lXi -lXext -lX11 -lm”,這就是編譯一個gtk1.2程式所需的gtk連結引數,xxx-config除了–libs引數外還有一個引數是–cflags用來生成標頭檔案包含目錄的,也就是-I引數,在下面我們將會講到。你可以試試執行gtk-config –libs –cflags,看看輸出結果。

      現在的問題就是怎樣用這些輸出結果了,最笨的方法就是複製貼上或者照抄,聰明的辦法是在編譯命令列里加入這個`xxxx-config –libs –cflags`,比如編譯一個gtk程式:gcc gtktest.c `gtk-config –libs –cflags`這樣就差不多了。注意`不是單引號,而是1鍵左邊那個鍵。

      除了xxx-config以外,現在新的開發包一般都用pkg-config來生成連結引數使用方法跟xxx-config類似,但xxx-config是針對特定的開發包,但pkg-config包含很多開發包的連結引數的生成,用pkg-config –list-all命令可以列出所支援的所有開發包,pkg-config的用法就是pkg-config pagName –libs –cflags,其中pagName是包名,是pkg-config–list-all裡列出名單中的一個,比如gtk1.2的名字就是gtk+, pkg-config gtk+ –libs –cflags的作用跟gtk-config –libs –cflags是一樣的。比如:

gcc gtktest.c `pkg-config gtk+ –libs –cflags`。

8) -include和-I引數

      -include用來包含標頭檔案,但一般情況下包含標頭檔案都在原始碼裡用#i nclude xxxxxx實現,-include引數很少用。-I參

數是用來指定標頭檔案目錄,/usr/include目錄一般是不用指定的,gcc知道去那裡找,但 是如果標頭檔案不

在/usr/icnclude裡我們就要用-I引數指定了,比如標頭檔案放在/myinclude目錄裡,那編譯命令列就要加上-I/myinclude

引數了,如果不加你會得到一個”xxxx.h: No such file or directory”的錯誤。-I引數可以用相對路徑,比如標頭檔案在當前

目錄,可以用-I.來指定。上面我們提到的–cflags引數就是用來生成-I引數的。

9)-Wall、-w 和 -v引數

 -Wall 打印出gcc提供的警告資訊

     -w     關閉所有警告資訊

     -v      列出所有編譯步驟

四. 幾個相關的環境變數

PKG_CONFIG_PATH:用來指定pkg-config用到的pc檔案的路徑,預設是/usr/lib/pkgconfig,pc檔案是文字檔案,副檔名是.pc,裡面定義開發包的安裝路徑,Libs引數和Cflags引數等等。

CC:用來指定c編譯器。

CXX:用來指定cxx編譯器。

LIBS:跟上面的–libs作用差不多。

CFLAGS:跟上面的–cflags作用差不多。

CC,CXX,LIBS,CFLAGS手動編譯時一般用不上,在做configure時有時用到,一般情況下不用管。

      環境變數設定方法:export  ENV_NAME=xxxxxxxxxxxxxxxxx

五. 關於交叉編譯

      交叉編譯通俗地講就是在一種平臺上編譯出能執行在體系結構不同的另一種平臺上,比如在我們地PC平臺(X86 CPU)上編譯出能執行在arm CPU平臺上的程式,編譯得到的程式在X86 CPU平臺上是不能執行的,必須放到arm  CPU 平臺上才能執行。當然兩個平臺用的都是linux。這種方法在異平臺移植和嵌入式開發時用得非常普遍。相對與交叉編譯,我們平常做的編譯就叫本地編譯,也 就是在當前平臺編譯,編譯得到的程式也是在本地執行。用來編譯這種程式的編譯器就叫交叉編譯器,相對來說,用來做本地編譯的就叫本地編譯器,一般用的都是gcc,但這種gcc跟本地的gcc編譯器是不一樣的,需要在編譯gcc時用特定的configure引數才能得到支援交叉編譯的gcc。為了不 跟本地編譯器混淆,交叉編譯器的名字一般都有字首,比如armc-xxxx-linux-gnu-gcc,arm-xxxx-linux-gnu- g++ 等等

交叉編譯器的使用方法

      使用方法跟本地的gcc差不多,但有一點特殊的是:必須用-L和-I引數指定編譯器用arm系統的庫和標頭檔案,不能用本地(X86)的庫(標頭檔案有時可以用本地的)。

例子:

arm-xxxx-linux-gnu-gcc test.c -L/path/to/sparcLib -I/path/to/armInclude

六、man gcc 部分

GCC(1)     GNU  GCC(1)

NAME

       gcc - GNU project C and C++ compiler

SYNOPSIS

       gcc [-c | -S | -E] [-std=standard]

           [-g] [-pg] [-Olevel]

           [-Wwarn…] [-pedantic]

           [-Idir…] [-Ldir…]

           [-Dmacro[=defn]…] [-Umacro]

           [-foption…] [-mmachine-option…]

           [-o outfile] infile…

       Only the most useful options are listed here; see below for the remain-

       der.  g++ accepts mostly the same options as gcc.