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中。
後面是不太常用的:
-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