GCC參數(shù)的分別是哪些,相信很多沒有經(jīng)驗(yàn)的人對此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個問題。
站在用戶的角度思考問題,與客戶深入溝通,找到廣饒網(wǎng)站設(shè)計(jì)與廣饒網(wǎng)站推廣的解決方案,憑借多年的經(jīng)驗(yàn),讓設(shè)計(jì)與互聯(lián)網(wǎng)技術(shù)結(jié)合,創(chuàng)造個性化、用戶體驗(yàn)好的作品,建站類型包括:成都網(wǎng)站設(shè)計(jì)、成都網(wǎng)站制作、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣、域名注冊、網(wǎng)站空間、企業(yè)郵箱。業(yè)務(wù)覆蓋廣饒地區(qū)。
大多數(shù)程序和庫在編譯時默認(rèn)的優(yōu)化級別是"2"(使用gcc選項(xiàng):"-O2")并且在Intel/AMD平臺上默認(rèn)按照i386處理器來編譯。
如果你只想讓編譯出來的程序運(yùn)行在特定的平臺上,就需要執(zhí)行更高級的編譯器優(yōu)化選項(xiàng),以產(chǎn)生只能運(yùn)行于特定平臺的代碼。
一種方法是修改每個源碼包中的Makefile文件,在其中尋找CFLAGS和CXXFLAGS變量(C和C++編譯器的編譯選項(xiàng))并修改它的值。
一些源碼包比如binutils, gcc, glibc等等,在每個子文件夾中都有Makefile文件,這樣修改起來就太累了!
另一種簡易做法是設(shè)置CFLAGS和CXXFLAGS環(huán)境變量。大多數(shù)configure腳本會使用這兩個環(huán)境變量代替Makefile文件中的值。
但是少數(shù)configure腳本并不這樣做,他們必須需要手動編輯才行。
為了設(shè)置CFLAGS和CXXFLAGS環(huán)境變量,你可以在bash中執(zhí)行如下命令(也可以寫進(jìn).bashrc以成為默認(rèn)值):
export CFLAGS="-O3 -march=
這是一個確保能夠在幾乎所有平臺上都能正常工作的最小設(shè)置。
"-march"選項(xiàng)表示為特定的cpu類型編譯二進(jìn)制代碼(不能在更低級別的cpu上運(yùn)行),
Intel通常是:pentium2, pentium3, pentium3m, pentium4, pentium4m, pentium-m, prescott, nocona
說明:pentium3m/pentium4m是筆記本用的移動P3/P4;pentium-m是迅馳I/II代筆記本的cpu;
prescott是帶SSE3的P4(以滾燙到可以煎雞蛋而聞名);nocona則是最新的帶有EMT64(64位)的P4(同樣可以煎雞蛋)
AMD通常是:k6, k6-2, k6-3, athlon, athlon-tbird, athlon-xp, athlon-mp, opteron, athlon64, athlon-fx
用AMD的一般都是DIYer,就不必解釋了吧。
如果編譯時沒有抱怨"segmentation fault, core dumped",那么你設(shè)定的"-O"優(yōu)化參數(shù)一般就沒什么問題。
否則請降低優(yōu)化級別("-O3" -> "-O2" -> "-O1" -> 取消)。
個人意見:不鼓勵使用過多的自定義優(yōu)化選項(xiàng),其實(shí)他們之間沒什么明顯的速度差異(有時"-O3"反而更慢)。
編譯器對硬件非常敏感,特別是在使用較高的優(yōu)化級別的時候,一丁點(diǎn)的內(nèi)存錯誤都可能導(dǎo)致致命的失敗。
所以在編譯時請千萬不要超頻你的電腦(我編譯關(guān)鍵程序時總是先降頻然的)。
注意:選項(xiàng)的順序很重要,如果有兩個選項(xiàng)互相沖突,則以后一個為準(zhǔn)。
比如"-O3"將打開-finline-functions選項(xiàng),但是可以用"-O3 -fno-inline-functions"既使用-O3的功能又關(guān)閉函數(shù)內(nèi)嵌功能。
更多的優(yōu)化選項(xiàng)請參見:
http://gcc.gnu.org/onlinedocs/gcc-3.4.4/gcc/Optimize-Options.html
http://gcc.gnu.org/onlinedocs/gcc-3.4.4/gcc/i386-and-x86_002d64-Options.html
http://gcc.gnu.org/onlinedocs/gcc-4.0.2/gcc/Optimize-Options.html
http://gcc.gnu.org/onlinedocs/gcc-4.0.2/gcc/i386-and-x86_002d64-Options.html
所有GCC選項(xiàng)完整列表參見:
http://gcc.gnu.org/onlinedocs/gcc-3.4.4/gcc/Option-Summary.html
http://gcc.gnu.org/onlinedocs/gcc-4.0.2/gcc/Option-Summary.html
有兩個頁面值的參考:
(對于gentoo-1.4)比較安全的優(yōu)化選項(xiàng)
http://www.freehackers.org/gentoo/gccflags/flag_gcc3.html
(對于gentoo-1.4)進(jìn)階優(yōu)化選項(xiàng)
http://www.freehackers.org/gentoo/gccflags/flag_gcc3opt.html
*******************************************************************
哦,忘了說一聲,"-O2"已經(jīng)啟用絕大多數(shù)安全的優(yōu)化選項(xiàng)了,所以其實(shí)你不必對那一堆選項(xiàng)發(fā)愁。
先說說"-O3"在"-O2"基礎(chǔ)上增加的幾項(xiàng),你可以按需添加(還算比較安全):
[gcc-3.4.4]
-finline-functions 允許編譯器選擇某些簡單的函數(shù)在其被調(diào)用處展開
-fweb 為每個web結(jié)構(gòu)體分配一個偽寄存器
-frename-registers 試圖驅(qū)除代碼中的假依賴關(guān)系,這個選項(xiàng)對具有大量寄存器的機(jī)器很有效。
[gcc-4.0.2]
-finline-functions 說明如上
-funswitch-loops 將循環(huán)體中不改變值的變量移動到循環(huán)體之外
-fgcse-after-reload **不太明白它的含義**[哪位大峽知道給小弟講解一下,先行謝過 ]
說完"-O3"再說說在嵌入式系統(tǒng)上常用的"-Os"選項(xiàng),這個選項(xiàng)其實(shí)也很重要,它的含義是對生成的二進(jìn)制代碼進(jìn)行尺寸上的優(yōu)化,它打開了所有"-O2"打開的選項(xiàng),因此通常認(rèn)為的"-Os"生成的二進(jìn)制代碼執(zhí)行效率低的潛在意識是錯誤的!當(dāng)然該選項(xiàng)與"-O2"的不同之處在于它在"-O2"的基礎(chǔ)上禁止了所有為了對齊而插入的空間,也就是將所有"-falign-*"系列的選項(xiàng)禁用了。這種禁用究竟是否一定降低了代碼的執(zhí)行效率,依據(jù)程序的不同而不同,據(jù)說某些情況下"-Os"的效率比"-O3"還要高14%!請兄弟們在實(shí)踐中自己摸索吧...
---------------------------------------------
下面選擇我認(rèn)為比較重要的幾項(xiàng)簡單介紹一下[gcc-3.4.4],GCC選項(xiàng)完整列表太長了!精力有限。
[注意]這里列出的都是非默認(rèn)的選項(xiàng),你只需要添加你所需要的選項(xiàng)即可
-w 禁止輸出警告消息
-Werror 將所有警告轉(zhuǎn)換為錯誤
-Wall 顯示所有的警告消息
-v 顯示編譯程序的當(dāng)前版本號
-V
-ansi 按照ANSI標(biāo)準(zhǔn)編譯程序,但并不限制與標(biāo)準(zhǔn)并不沖突的GNU擴(kuò)展(一般不用該選項(xiàng))
-pedantic 如果要限制代碼必須嚴(yán)格符合ISO標(biāo)準(zhǔn),就在"-ansi"的基礎(chǔ)上同時啟用這個選項(xiàng)(很少使用)
-std=
-static 連接器將忽略動態(tài)連接庫,同時通過將靜態(tài)目標(biāo)文件直接包含到結(jié)果目標(biāo)文件完成對所有引用的解析。
-shared 連接器將生成共享目標(biāo)代碼,該共享庫可在運(yùn)行時動態(tài)連接到程序形成完整的可執(zhí)行體。
如果使用gcc命令創(chuàng)建共享庫作為其輸出,該選項(xiàng)可以防止連接器將缺失main()方法視為錯誤。
為了可以正確的工作,應(yīng)該一致的使用選項(xiàng)"-fpic"以及目標(biāo)平臺選項(xiàng)編譯構(gòu)成同一個庫的所有共享目標(biāo)模塊。
-shared-libgcc 該選項(xiàng)指定使用共享版本的libgcc,在沒有共享版本的libgcc的機(jī)器上該選項(xiàng)無效。
-specs=
該選項(xiàng)可以通過指定配置文件來覆蓋默認(rèn)配置,指定的文件將在默認(rèn)配置文件讀取后進(jìn)行處理以修改默認(rèn)配置。
-pipe 使用管道而不是臨時文件一個階段到另一個階段交換輸出的方式,可以加快編譯速度。建議使用。
-o
--help 顯示gcc的命令行選項(xiàng)列表;與"-v"一起使用時還將顯示gcc調(diào)用的各個進(jìn)程所接受的選項(xiàng)。
--target-help 顯示目標(biāo)機(jī)器相關(guān)的命令行選項(xiàng)列表
-b
目標(biāo)機(jī)通過指定包含編譯程序的目錄來確定,通常為/usr/local/lib/gcc-lib/
-B
這個前綴可以是用冒號分割的多個路徑,環(huán)境變量GCC_EXEC_PREFIX和這個選項(xiàng)有相同的效果。
-I
-dumpmachine 顯示該程序的目標(biāo)機(jī)名字,不做其他任何動作
-dumpspecs 顯示構(gòu)件編譯程序的規(guī)范信息,包括用來編譯、匯編和連接gcc編譯程序自身用到的所有選項(xiàng),不做其他任何動作。
-dumpversion 顯示編譯程序自身的版本號,不做其他任何動作
-falign-functions=N 將所有函數(shù)的起始地址在N(N=1,2,4,8,16...)的邊界上對齊,默認(rèn)為機(jī)器自身的默認(rèn)值,指定為1表示禁止對齊。
-falign-jumps=N 將分支目標(biāo)在N(N=1,2,4,8,16...)的邊界上對齊,默認(rèn)為機(jī)器自身的默認(rèn)值,指定為1表示禁止對齊。
-fno-align-labels 建議使用它,以保證不和-falign-jumps("-O2"默認(rèn)啟用的選項(xiàng))沖突
-fno-align-loops 建議使用它,以確保不會在分支目標(biāo)前插入多余的空指令。
-fbranch-probabilities 在使用"-fprofile-arcs"選項(xiàng)編譯程序并執(zhí)行它來創(chuàng)建包含每個代碼塊執(zhí)行次數(shù)的文件之后,程序可以利用這一選項(xiàng)再次編譯,
文件中所產(chǎn)生的信息將被用來優(yōu)化那些經(jīng)常發(fā)生的分支代碼。如果沒有這些信息,gcc將猜測那一分支可能經(jīng)常發(fā)生并進(jìn)行優(yōu)化。
這類優(yōu)化信息將會存放在一個以源文件為名字的并以".da"為后綴的文件中。
-fno-guess-branch-probability 默認(rèn)情況下gcc將使用隨機(jī)模型進(jìn)行猜測哪個分支更可能被經(jīng)常執(zhí)行,并以此來優(yōu)化代碼,該選項(xiàng)關(guān)閉它。
-fprofile-arcs 在使用這一選項(xiàng)編譯程序并運(yùn)行它以創(chuàng)建包含每個代碼塊的執(zhí)行次數(shù)的文件后,程序可以再次使用"-fbranch-probabilities"編譯,
文件中的信息可以用來優(yōu)化那些經(jīng)常選取的分支。如果沒有這些信息,gcc將猜測哪個分支將被經(jīng)常運(yùn)行以進(jìn)行優(yōu)化。
這類優(yōu)化信息將會存放在一個以源文件為名字的并以".da"為后綴的文件中。
-fforce-addr 必須將地址復(fù)制到寄存器中才能對他們進(jìn)行運(yùn)算。由于所需地址通常在前面已經(jīng)加載到寄存器中了,所以這個選項(xiàng)可以改進(jìn)代碼。
-fforce-mem 必須將數(shù)值復(fù)制到寄存器中才能對他們進(jìn)行運(yùn)算。由于所需數(shù)值通常在前面已經(jīng)加載到寄存器中了,所以這個選項(xiàng)可以改進(jìn)代碼。
-ffreestanding 所編譯的程序能夠在獨(dú)立的環(huán)境中運(yùn)行,該環(huán)境可以沒有標(biāo)準(zhǔn)庫,而且可以不從main()函數(shù)開始運(yùn)行。
該選項(xiàng)將設(shè)置"-fno-builtin",且等同于"-fno-hosted"。
-fhosted 所編譯的程序需要運(yùn)行在宿主環(huán)境中,其中需要有完整的標(biāo)準(zhǔn)庫,而且main()函數(shù)具有int型的返回值。
-fno-builtin 除非利用"__builtin_"進(jìn)行引用,否則不識別所有內(nèi)建函數(shù)。
-fmerge-all-constants 試圖將跨編譯單元的所有常量值和數(shù)組合并在一個副本中。但是標(biāo)準(zhǔn)C/C++要求每個變量都必須有不同的存儲位置。
-fmove-all-movables 將所有不變的表達(dá)式移動到循環(huán)體之外,這種做法的好壞取決于源代碼中的循環(huán)結(jié)構(gòu)。
-fnon-call-exceptions 產(chǎn)生的代碼可供陷阱指令(如非法浮點(diǎn)運(yùn)算和非法內(nèi)存尋址)拋出異常,需要相關(guān)平臺的運(yùn)行時支持,并不普遍有效。
-fomit-frame-pointer 對于不需要棧指針的函數(shù)就不在寄存器中保存指針,因此可以忽略存儲和檢索地址的代碼,并將寄存器用于普通用途。
所有"-O"級別都打開著一選項(xiàng),但僅在調(diào)試器可以不依靠棧指針運(yùn)行時才有效。建議不需要調(diào)試的情況下顯式的設(shè)置它。
-fno-optional-diags 禁止輸出診斷消息,C++標(biāo)準(zhǔn)并不需要這些消息。
-fpermissive 將代碼中與標(biāo)準(zhǔn)不符合的診斷消息作為警告而不是錯誤輸出。
-fpic 生成可用于共享庫的位置獨(dú)立代碼(PIC),所有的內(nèi)存尋址均通過全局偏移表(GOT)完成。該選項(xiàng)并非在所有的機(jī)器上都有效。
要確定一個地址,需要將代碼自身的內(nèi)存位置作為表中的一項(xiàng)插入。該選項(xiàng)可以產(chǎn)生在共享庫中存放并從中加載的目標(biāo)模塊。
-fprefetch-loop-arrays 生成數(shù)組預(yù)讀取指令,對于使用巨大數(shù)組的程序可以加快代碼執(zhí)行速度,適合數(shù)據(jù)庫相關(guān)的大型軟件等。
-freg-struct-return 生成用寄存器返回短結(jié)構(gòu)的代碼,如果寄存器無法榮納將使用內(nèi)存。
-fstack-check 為防止程序棧溢出而進(jìn)行必要的檢測,在多線程環(huán)境中運(yùn)行時才可能需要它。
-ftime-report 編譯完成后顯示編譯耗時的統(tǒng)計(jì)信息
-funroll-loops 如果在編譯時可以確定迭代的次數(shù)非常少而且循環(huán)中的指令也非常少,可以使用該選項(xiàng)進(jìn)行循環(huán)展開,以驅(qū)除循環(huán)和復(fù)制指令。
-finline-limit=
--param
名字 解釋
max-delay-slot-insn-search 較大的數(shù)目可以生成更優(yōu)化的代碼,但是會降低編譯速度,默認(rèn)為100
max-delay-slot-live-search 較大的數(shù)目可以生成更優(yōu)化的代碼,但是會降低編譯速度,默認(rèn)為333
max-gcse-memory 執(zhí)行GCSE優(yōu)化使用的最大內(nèi)存量,太小將使該優(yōu)化無法進(jìn)行,默認(rèn)為50M
max-gcse-passes 執(zhí)行GCSE優(yōu)化的最大迭代次數(shù),默認(rèn)為1
*******************************************************************
說完了命令行選項(xiàng),下面來說說與硬件體系結(jié)構(gòu)(主要是cpu)相關(guān)的設(shè)置[僅針對i386/x86_64]
最大名鼎鼎的"-march"上面已經(jīng)說過了,下面講講別的(僅挑些實(shí)用的)
-mfpmath=sse P3和athlon-tbird以上級別的cpu支持
-masm=
-mieee-fp 指定編譯器使用IEEE浮點(diǎn)比較,這樣將會正確的處理比較結(jié)果為無序的情況。
-malign-double 將double, long double, long long對齊于雙字節(jié)邊界上;有助于生成更高速的代碼,但是程序的尺寸會變大。
-m128bit-long-double 指定long double為128位,pentium以上的cpu更喜歡這種標(biāo)準(zhǔn)。
-mregparm=N 指定用于傳遞整數(shù)參數(shù)的寄存器數(shù)目(默認(rèn)不使用寄存器)。0<=N<=3 ;注意:當(dāng)N>0時你必須使用同一參數(shù)重新構(gòu)建所有的模塊,包括所有的庫。
-mmmx
-mno-mmx
-msse
-mno-sse
-msse2
-mno-sse2
-msse3
-mno-sse3
-m3dnow
-mno-3dnow
上面的這些不用解釋了,一看就明白,根據(jù)自己的CPU決定吧
-maccumulate-outgoing-args 指定在函數(shù)引導(dǎo)段中計(jì)算輸出參數(shù)所需最大空間,這在大部分現(xiàn)代cpu中是較快的方法;缺點(diǎn)是會增加代碼尺寸。
-mthreads 支持Mingw32的線程安全異常處理。對于依賴于線程安全異常處理的程序,必須啟用這個選項(xiàng)。
使用這個選項(xiàng)時會定義"-D_MT",它將包含使用選項(xiàng)"-lmingwthrd"連接的一個特殊的線程輔助庫,用于為每個線程清理異常處理數(shù)據(jù)。
-minline-all-stringops 嵌入所有的字符串操作。可以提高字符串操作的性能,但是會增加代碼尺寸。
-momit-leaf-frame-pointer 不為葉子函數(shù)在寄存器中保存棧指針,這樣可以節(jié)省寄存器,但是將會是調(diào)試變的困難。參見"-fomit-frame-pointer"。
下面這幾個僅用于x86_64環(huán)境:
-m64 生成專門運(yùn)行于64位環(huán)境的代碼,不能運(yùn)行于32位環(huán)境
-mcmodel=small [默認(rèn)值]程序和它的符號必須位于2GB以下的地址空間。指針仍然是64位。程序可以靜態(tài)連接也可以動態(tài)連接。
-mcmodel=kernel 內(nèi)核運(yùn)行于2GB地址空間之外。在編譯linux內(nèi)核時必須使用該選項(xiàng)!
-mcmodel=medium 程序必須位于2GB以下的地址空間,但是它的符號可以位于任何地址空間。程序可以靜態(tài)連接也可以動態(tài)連接。
注意:共享庫不能使用這個選項(xiàng)編譯!
-mcmodel=large 對地址空間沒有任何限制,這個選項(xiàng)的功能目前尚未實(shí)現(xiàn)。
==============================
既然已經(jīng)講了這么多了索性再講講gcc使用的一些環(huán)境變量
除了大名鼎鼎的CFLAGS和CXXFLAGS以外(其實(shí)是Autoconf的環(huán)境變量),再挑幾個說說:
所有的PATH類環(huán)境變量(除LD_RUN_PATH外)都是用冒號分割的目錄列表。
C_INCLUDE_PATH 編譯C程序時使用的環(huán)境變量,用于查找頭文件。
CPLUS_INCLUDE_PATH 編譯C++程序時使用的環(huán)境變量,用于查找頭文件。
OBJC_INCLUDE_PATH 編譯Obj-C程序時使用的環(huán)境變量,用于查找頭文件。
CPATH 編譯C/C++/Obj-C程序時使用的環(huán)境變量,用于查找頭文件。
COMPILER_PATH 如果沒有用GCC_EXEC_PREFIX定位子程序,編譯程序?qū)诖瞬檎宜淖映绦颉?/p>
LIBRARY_PATH 連接程序?qū)⒃谶@些目錄中尋找特殊的連接程序文件。
LD_LIBRARY_PATH 該環(huán)境變量不影響編譯程序,但是程序運(yùn)行的時候會有影響:程序會查找該目錄列表以尋找共享庫。
當(dāng)不能夠在編譯程序的目錄中找到共享庫的時候,執(zhí)行程序必須設(shè)置該環(huán)境變量。
LD_RUN_PATH 該環(huán)境變量不影響編譯程序,但是程序運(yùn)行的時候會有影響:它在運(yùn)行時指出了文件的名字,運(yùn)行的程序可以由此得到它的符號名字和地址。
由于地址不會重新載入,因而可能符號應(yīng)用其他文件中的絕對地址。這個和ld工具使用的"-R"選項(xiàng)完全一樣。
GCC_EXEC_PREFIX 編譯程序執(zhí)行所有子程序的名字的前綴,默認(rèn)值是"
其中的
LANG 指定編譯程序使用的字符集,可用于創(chuàng)建寬字符文件、串文字、注釋;默認(rèn)為英文。[目前只支持日文"C-JIS,C-SJIS,C-EUCJP",不支持中文]
LC_ALL 指定多字節(jié)字符的字符分類,主要用于確定字符串的字符邊界以及編譯程序使用何種語言發(fā)出診斷消息;默認(rèn)設(shè)置與LANG相同。
中文相關(guān)的幾項(xiàng):"zh_CN.GB2312 , zh_CN.GB18030 , zh_CN.GBK , zh_CN.UTF-8 , zh_TW.BIG5"
TMPDIR 編譯程序存放臨時工作文件的臨時目錄,這些臨時文件通常在編譯結(jié)束時被刪除。
看完上述內(nèi)容,你們掌握GCC參數(shù)的分別是哪些的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝各位的閱讀!