Makefile 中 gcc 參數小結

gcc/g++在執行編譯工作的時候,總共需要4步
  1.預處理,生成.i的文件[預處理器cpp](把"include"的文件
拷貝到要編譯的源文件中,用實際值替代"define"的文本,在調用宏的地方進行宏替換)
  2.將預處理後的文件不轉換成彙編語言,生成文件.s[編譯器egcs]
  3.有彙編變爲目標代碼(機器代碼)生成.o的文件[彙編器as]
  4.連接目標代碼,生成可執行程序[鏈接器ld]

   gcc最基本的用法是 :gcc [options][filenames]
  其中,options就是編譯器所需要的參數,filenames給出相關的文件名稱。

  使用gcc編譯器時,必須給出一系列必要的調用參數和文件名稱。不同參數的先後順序對執行結果沒有影響,只有在使用同類參數時的先後順序才需要考慮。如果使用了多個 -L 的參數來定義庫目錄,gcc會根據多個 -L 參數的先後順序來執行相應的庫目錄。
 

-l –L 參數:

   -l用力指定程序要鏈接的庫,-l緊跟着庫名,注意庫名和真正的庫文件名不同,拿數學庫來說,庫名是m,但庫文件名是libm.so

   放在/lib,/usr/lib下的庫直接用-l就可以了,但方在其他路徑下的,就得指明路徑了,就用-L參數了,比如我們把libtest.so放在/aaa/bbb/ccc下面,就用-L/aaa/bbb/ccc –ltest了
注意libxxx.so只是一個鏈接


-I:

   指定頭文件目錄,如-I/myinclude

 

-c:
  只激活預處理,編譯,和彙編,也就是他只把程序做成obj文件
  例子用法:
  gcc -c hello.c
  他將生成.o的obj文件

-o:
  制定目標名稱,缺省的時候,gcc 編譯出來的文件是a.out,很難聽,如果
  你和我有同感,改掉它,哈哈
  例子用法
  gcc -o hello.exe hello.c (哦,windows用習慣了)
  gcc -o hello.asm -S hello.c

-static:
  此選項將禁止使用動態庫,所以,編譯出來的東西,一般都很大,也不需要什麼
  動態連接庫,就可以運行.

-share:
  此選項將盡量使用動態庫,所以生成文件比較小,但是需要系統由動態庫.

 

-fPIC:

   表示編譯爲位置獨立的代碼,用於編譯共享庫。目標文件需要創建成位置無關碼,概念上就是在可執行程序裝載它們的時候,它們可以放在可執行程序的內存裏的任何地方。

 

-g:

   選項告訴GCC產生能被GUN調速器使用的調試信息以使gdb調試你的程序

 

-Wall:

   啓用警告檢測

 

-Wl:

   表示後面的參數將傳給link程序ld(因爲gcc可能會自動調用ld)

  -Wl.option
  此選項傳遞option給連接程序;如果option中間有逗號,就將option分成多個選
  項,然後傳遞給會連接程序.

   在通過gcc/g++間接調用鏈接程序ld時,所有的ld選項前必須加上“-Wl,”,因爲-Map是ld的選項。所以,要讓g++生成mapfile,需要增加編譯參數“-Wl,-Map,mapfile”。
    例:gcc -ohelloworld helloworld.c -Wl,-Map,helloworld.map

 

   爲什麼 gcc 頑固地提示某些符號無法解決呢?
   把其中的庫列表用
  -Wl,--start-group 和 -Wl,--end-group 括起來.

 

-EL小端編譯

-EB大端編譯:
   gcc默認爲-EL小端,而mips-linux-gnu-gcc默認使用大端,當然mips-linux-gnu-ld也要加入-EL的參數


--gc-sections:

   會使ld刪除沒有被使用的section。

   -ffunction-sections, -fdata-sections會使compiler爲每個function和dataitem分配獨立的section。
 
  鏈接操作以section作爲最小的處理單元,只要一個section中有某個符號被引用,該section就會被放入output中。

   這些選項一起使用會從最終的輸出文件中刪除所有未被使用的function和data, 只包含用到的unction和data。

 

後面是不太常用的:

 

 -x language filename

  設定文件所使用的語言,使後綴名無效,對以後的多個有效.也就是根
  據約定C語言的後綴名稱是.c的,而C++的後綴名是.C或者.cpp,如果
  你很個性,決定你的C代碼文件的後綴名是.pig 哈哈,那你就要用這
  個參數,這個參數對他後面的文件名都起作用,除非到了下一個參數
  的使用。

  看到英文,應該可以理解的。
  例子用法:
  gcc -x c hello.pig
  -x none filename
  關掉上一個選項,也就是讓gcc根據文件名後綴,自動識別文件類型

  例子用法:
  gcc -x c hello.pig -x none hello2.c

  -S
  只激活預處理和編譯,就是指把文件編譯成爲彙編代碼。
  例子用法
  gcc -S hello.c
  他將生成.s的彙編代碼,你可以用文本編輯器察看

  -E
  只激活預處理,這個不生成文件,你需要把它重定向到一個輸出文件裏
  面.

  例子用法:
  gcc -E hello.c > pianoapan.txt
  gcc -E hello.c | more
  慢慢看吧,一個hello word 也要與處理成800行的代碼

  -pipe
  使用管道代替編譯中臨時文件,在使用非gnu彙編工具的時候,可能有些問
  題

  gcc -pipe -o hello.exe hello.c
  -ansi
  關閉gnu c中與ansi c不兼容的特性,激活ansi c的專有特性(包括禁止一
  些asm inline typeof關鍵字,以及UNIX,vax等預處理宏,
  -fno-asm
  此選項實現ansi選項的功能的一部分,它禁止將asm,inline和typeof用作
  關鍵字。

  -fno-strict-prototype
  只對g++起作用,使用這個選項,g++將對不帶參數的函數,都認爲是沒有顯式
  的對參數的個數和類型說明,而不是沒有參數.
  而gcc無論是否使用這個參數,都將對沒有帶參數的函數,認爲城沒有顯式說
  明的類型

  -fthis-is-varialble
  就是向傳統c++看齊,可以使用this當一般變量使用.

  -fcond-mismatch
  允許條件表達式的第二和第三參數類型不匹配,表達式的值將爲void類型

  -funsigned-char
  -fno-signed-char
  -fsigned-char
  -fno-unsigned-char
  這四個參數是對char類型進行設置,決定將char類型設置成unsigned char(前
  兩個參數)或者 signed char(後兩個參數)

  -include file
  包含某個代碼,簡單來說,就是便以某個文件,需要另一個文件的時候,就可以
  用它設定,功能就相當於在代碼中使用#include<filename>

  例子用法:
  gcc hello.c -include /root/pianopan.h
  -imacros file

  將file文件的宏,擴展到gcc/g++的輸入文件,宏定義本身並不出現在輸入文件
  中

  -Dmacro
  相當於C語言中的#define macro

  -Dmacro=defn
 
 相當於C語言中的#define macro=defn

  -Umacro
  相當於C語言中的#undef macro

  -undef
  取消對任何非標準宏的定義

  -Idir
  在你是用#include"file"的時候,gcc/g++會先在當前目錄查找你所制定的頭
  文件,如果沒有找到,他回到缺省的頭文件目錄找,如果使用-I制定了目錄,他
  回先在你所制定的目錄查找,然後再按常規的順序去找.
  對於#include<file>,gcc/g++會到-I制定的目錄查找,查找不到,然後將到系
  統的缺省的頭文件目錄查找

  -I-
  就是取消前一個參數的功能,所以一般在-Idir之後使用

  -idirafter dir
  在-I的目錄裏面查找失敗,講到這個目錄裏面查找.

  -iprefix prefix
  -iwithprefix dir
  一般一起使用,當-I的目錄查找失敗,會到prefix+dir下查找

  -nostdinc
  使編譯器不再系統缺省的頭文件目錄裏面找頭文件,一般和-I聯合使用,明確
  限定頭文件的位置

  -nostdin C++
  規定不在g++指定的標準路經中搜索,但仍在其他路徑中搜索,.此選項在創建
  libg++庫使用

  -C
  在預處理的時候,不刪除註釋信息,一般和-E使用,有時候分析程序,用這個很
  方便的

  -M
  生成文件關聯的信息。包含目標文件所依賴的所有源代碼
  你可以用gcc -M hello.c來測試一下,很簡單。

  -MM
  和上面的那個一樣,但是它將忽略由#include<file>造成的依賴關係。

  -MD
  和-M相同,但是輸出將導入到.d的文件裏面

  -MMD
  和-MM相同,但是輸出將導入到.d的文件裏面

  -Wa,option
  此選項傳遞option給彙編程序;如果option中間有逗號,就將option分成多個選
  項,然後傳遞給會彙編程序

  -llibrary
  制定編譯的時候使用的庫

  例子用法
  gcc -lcurses hello.c
  使用ncurses庫編譯程序

  -Ldir
  制定編譯的時候,搜索庫的路徑。比如你自己的庫,可以用它制定目錄,不然
  編譯器將只在標準庫的目錄找。這個dir就是目錄的名稱。

  -O0
  -O1
  -O2
  -O3
  編譯器的優化選項的4個級別,-O0表示沒有優化,-O1爲缺省值,-O3優化級別最
  高

  -g
  只是編譯器,在編譯的時候,產生條是信息。

  -gstabs
  此選項以stabs格式聲稱調試信息,但是不包括gdb調試信息.

  -gstabs+
  此選項以stabs格式聲稱調試信息,並且包含僅供gdb使用的額外調試信息.

  -ggdb
  此選項將儘可能的生成gdb的可以使用的調試信息.

  -traditional
  試圖讓編譯器支持傳統的C語言特性





幾個簡單的示例:

 CC=g++
 test:
  @$(CC) -g daemon.cpp  sharemem.cpp ReadConfig.cpp GetConfigInfo.cpp  -o testdaemon libn6601.so -lpthread -lrt -lzlog -Wl,-rpath,/root/daemon
  clean:
  @rm main.o testdaemon *.o -rf

其中-Wl,-rpath,/root/daemon   

(1)程序運行時,優先到rpath指定的目錄去尋找依賴庫

(2)程序鏈接時,在指定的目錄中,隱式的鏈接那些動態庫所需要的鏈接庫。

往往我們都熟知第一個功能,忽略第二個功能。而第二個功能正是現在所需要的。

我們將liba.so,libb.so 和 libc.so拷貝的同一個目錄中,然後利用rpath鏈接應用程序,這樣編譯便不需要顯示的去鏈接liba.so所依賴的庫了。



  1 #fz 執行pkg-config --cflags --libs glib-2.0  
  2 #結果爲:-I/usr/include/glib-2.0 -I/usr/lib/glib-2.0/include -lglib-2.0
  3 # 即自動產生glib-2.0 的頭文件和 庫的信息     
  4 #gcc *.c -g `pkg-config --cflags --libs glib-2.0` -lteststring -lpthread -lpcap -otesttrojan
  5 all:  trojan.c
  6   gcc *.c -g `pkg-config --cflags --libs glib-2.0` -lteststring -lpthread -lpcap -otesttrojan













發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章