![]() |
|
嵌入式Linux編譯器GCC編譯流程 |
|
作為自由軟件的旗艦項目,Richard Stallman在十多年前剛開始寫作GCC的時候,還只是僅僅把它當作一個C程序語言的編譯器,GCC的意思也只是GNU C Compiler而已。 經過了這么多年的發展,嵌入式Linux編譯器GCC已經不僅僅能支持C語言,它現在還支持Ada語言、C++語言、Java語言、Objective C語言、PASCAL語言、COBOL語言,并支持函數式編程和邏輯編程的Mercury語言等。而GCC也不再單指GNU C語言編譯器的意思了,而是變成了GNU編譯器家族了。 GCC的編譯流程分為了4個步驟,分別如下。 ·預處理(Pre-Processing)。 編譯器通過程序的擴展名可分辨編寫原始程序碼所用的語言,由于不同的程序所需要執行編譯的步驟是不同的,因此GCC根據不同的后綴名對它們進行分別處理,表1.1指出了不同后綴名的處理方式。 GCC所支持后綴名解釋 GCC使用的基本語法為: 這里的option是GCC使用時的一些選項,通過指定不同的選項GCC可以實現其強大的功能。這里的filename則是GCC要編譯的文件,GCC會根據用戶所指定的編譯選項以及所識別的文件后綴名來對編譯文件進行相應的處理。 本節從編譯流程的角度講解GCC的常見使用方法。 首先,這里有一段簡單的C語言程序,該程序由兩個文件組成,其中“hello.h”為頭文件,在“hello.c”中包含了“hello.h”,其源文件如下所示。 /*hello.h*/ 1.預處理階段 GCC的選項“-E”可以使編譯器在預處理結束時就停止編譯,選項“-o”是指定GCC輸出的結果,其命令格式為如下所示。 表2.6指出后綴名為“.i”的文件是經過預處理的C原始程序。要注意,“hello.h”文件是不能進行編譯的,因此,使編譯器在預處理后停止的命令如下所示。 在此處,選項“-o”是指目標文件,由表2.6可知,“.i”文件為已經過預處理的C原始程序。以下列出了hello.i文件的部分內容。 由此可見,GCC確實進行了預處理,它把“hello.h”的內容插入到hello.i文件中了。 2.編譯階段 編譯器在預處理結束之后,GCC首先要檢查代碼的規范性、是否有語法錯誤等,以確定代碼的實際要做的工作,在檢查無誤后,就開始把代碼翻譯成匯編語言,GCC的選項“-S”能使編譯器在進行完匯編之前就停止。由表1.1可知,“.s”是匯編語言原始程序,因此,此處的目標文件就可設為“.s”類型。 以下列出了hello.s的內容,可見GCC已經將其轉化為匯編了,感興趣的讀者可以分析一下這一行簡單的C語言小程序用匯編代碼是如何實現的。 .file "hello.c" 可以看到,這一小段C語言的程序在匯編中已經復雜很多了,這也是C語言作為中級語言的優勢所在。 3.匯編階段 匯編階段是把編譯階段生成的“.s”文件生成目標文件,讀者在此使用選項“-c”就可看到匯編代碼已轉化為“.o”的二進制目標代碼了,如下所示。 4.鏈接階段 在成功編譯之后,就進入了鏈接階段。在這里涉及一個重要的概念——函數庫。 在這個程序中并沒有定義“printf”的函數實現,在預編譯中包含進的“stdio.h”中也只有該函數的聲明,而沒有定義函數的實現,那么,是在哪里實現“printf”函數的呢? 后的答案是:系統把這些函數實現都已經被放入名為libc.so.6的庫文件中去了,在沒有特別指定時,GCC會到系統默認的搜索路徑“/usr/lib”下進行查找,也就是鏈接到libc.so.6庫函數中去,這樣就能實現函數“printf”了,而這也就是鏈接的作用。 完成了鏈接之后,GCC就可以生成可執行文件,其命令如下所示。 運行該可執行文件,出現正確的結果。 熱點鏈接:
1、WindowsAndroid:在PC上運行Android
|