gcc、arm-linux-gcc和arm-elf-gcc的關係?

2021-02-13 strongerHuang
二、GCC的組成結構GCC 內部結構主要由 Binutils、gcc-core、Glibc 等軟體包組成。

舉例描述下上面 3 個包是如何進行運作的。有一個 c 源文件 test.c 源碼如下:

#include<stdio.h>
int main(int argc, char *argv[])
{
printf("Hello Linux!!\n");
return 0;
}

編譯命令為:gcc -o test test.c 編譯生成 test 可執行文件。gcc 編譯流程分為四個步驟:預處理、編譯 、彙編、連結。個人認為預處理和編譯主要由 gcc-core 來完成,彙編和連結主要由 Binutils 來完成。

那麼何時用到 glibc 呢?看到源碼中的 printf 函數沒有,這個函數在 GCC 中是以庫函數的形式存在,這個庫函數在 glibc 庫中,在 stdio.h 頭文件中被聲明。
總的來說,如果真正了解了上面 3 個軟體包的作用,自然就明白 GCC 是如何工作的。

三、交叉編譯

交叉編譯(或交叉建立)是這樣一種過程,它在一種機器結構下編譯的軟體將在另一種完全不同的機器結構下執行。一個常見的例子是在 PC 機上為運行在基於 ARM、PowerPC或 MIPS 的目標機的編譯軟體。

幸運的是,GCC 使得這一過程所面臨的困難要比聽起來小得多。GCC 中的一般工具通常都是通過在命令行上調用命令(如 gcc)來執行的。在使用交叉編譯的情況下,這些工具將根據它編譯的目標而命名。

例如,要使用交叉工具鏈為 ARM 機器編譯簡單的 Hello World 程序,你可以運行如下所示的命令:使用如下命令編譯並測試這個代碼:arm-linux-gcc -o hello hello.c。

四、arm-linux-gcc

arm-linux-gcc 是基於 ARM 目標機的交叉編譯軟體, arm-linux-gcc 跟 GCC 所需的安裝包不同:

x86 跟 ARM 所使用的指令集是不一樣的,所以所需要的 binutils 肯定不一樣;上面提到過 gcc-core 是依賴於 binutils 的,自然 ARM 跟 x86 所使用的 gcc-core 包也不一樣;glibc 一個 c 庫,最終是以庫的形式存在於編譯器中,自然 ARM 所使用的 glibc 庫跟 x86 同樣也不一樣,其它的依此類推。

五、arm-elf-gcc

arm-elf-gcc 跟 arm-linux-gcc 一樣,也是是基於 ARM 目標機的交叉編譯軟體。但是它們不是同一個交叉編譯軟體,兩者是有區別的,兩者區別主要在於使用不同的 C 庫文件。

arm-linux-gcc 使用 GNU 的 Glibc,而 arm-elf-gcc 一般使用 uClibc/uC-libc 或者使用 RedHat專門為嵌入式系統的開發的C庫newlib。只是所應用的領域不同而已,Glibc是針對PC開發的,uClibc/uC-libc是與Glibc API兼容的小型化C語言庫,實現了Glibc部分功能。

六、uClibc/uC-libc 

uClinux有兩個經常使用的libc庫:uC-libc和uClibc。雖然兩者名字很相似,其實有差別,下面就簡單的介紹一下二者的不同之處。uC -libc是最早為uClinux開發的庫,是Jeff Dionne和Kenneth Albanowski為在EKLs項目中支持m68000在Linux-8086 C庫源碼上移植的。

uC-libc是一個完全的libc實現,但其中有一些api是非標準的,有些libc的標準也沒有實現。uC-libc穩定地支持 m68000,ColdFire和沒有MMU的ARM。其主要設計目標是「小」、「輕」,並儘量與標準一致,雖然它的API和很多libc兼容,但是似乎並不像它期望的那樣和所有標準一致。

uClibc就是為了解決這個問題從uC-libc中發展出來的。它的所有API都是標準的(正確的返回類型,參數等等),它彌補了uC-libc中沒有實現的libc標準,現在已經被移植到多種架構中。一般來講,它儘量兼容glibc以便使應用程式用uClibc改寫變的容易。

uClibc能夠在標準的 VM linux和uClinux上面使用。為了應用程式的簡潔,它甚至可以在許多支持MMU的平臺上被編譯成共享庫。Erik Anderson在uClibc背後做了很多的工作。uClibc支持許多系列的處理器:m68000,Coldfire,ARM,MIPS,v850, x86,i960,Sparc,SuperH,Alpha,PowerPC和Hitachi 8。

不斷增加的平臺支持顯示uClibc能夠很容易的適應新的架構。uClinux發行版提供了環境能夠讓你選擇使用uC-libc或是uClibc編譯。對於m68000和Coldfire平臺來說,選擇uC-libc還是稍微好一點,因為它支持共享庫,而共享庫是這些cpu經常使用的 libc。uClibc也幾乎和所有的平臺都能很好的工作。

newlib 是一個用於嵌入式系統的開放原始碼的C語言程序庫,由libc和libm兩個庫組成,特點是輕量級,速度快,可移植到很多CPU結構上。newlib實現了許多複雜的功能,包括字符串支持,浮點運算,內存分配(如malloc)和I/O流函數(printf,fprinf()等等)。其中libc提供了c 語言庫的實現,而libm提供了浮點運算支持。

七、C語言庫的選擇

在為ARM交叉編譯gcc編譯器時,對gcc指定不同的配置選項時,使用的C語言庫就不同,gcc編譯器默認使用Glibc,也可以使用 uClibc/uC-libc(基本兼容Glibc API),當使用--with-newlib時,gcc編譯器不使用Glibc。當沒有交叉編譯Glibc時,可以使用--with-newlib禁止連接Glibc而編譯bootstrap gcc編譯器。

從gcc源目錄下的config/arm中的t-linux和t-arm-elf中可以看出,不同的--target也影響gcc連接C語言庫,t-linux(--target=arm-linux)默認使用Glibc,-arm-elf(--target=arm-elf)使用- Dinhibit_libc禁止連接Glibc,這時我們就可以使用newlib等其他C語言庫編譯GCC工具鏈。

雖然GCC工具鏈配置了不同的的C語言庫,但由於這些C語言庫都可以用來支持GCC,它們對核心數據的處理上不存在較大出入。因而arm-linux-* 和 arm-elf-*區別主要表現在C語言庫的實現上,例如不同系統調用,不同的函數集實現,不同的ABI/啟動代碼以及不同系統特性等微小的差別。

arm-linux-*和 arm-elf-*的使用沒有一個絕對的標準,排除不同庫實現的差異,gcc可以編譯任何系統。arm-linux-*和 arm-elf-*都可以用來編譯裸機程序和作業系統,只是在遵循下面的描述時系統程序顯得更加協調:

arm-linux-*針對運行linux的ARM機器,其依賴於指定的C語言庫Glibc,因為同樣使用Glibc的linux而使得arm-linux-*在運行linux的ARM機器上編譯顯得更加和諧。

arm-elf-*則是一個獨立的編譯體系,不依賴於指定的C語言庫Glibc,可以使用newlib等其他C語言庫,不要求作業系統支持,當其使用為嵌入式系統而設計的一些輕巧的C語言庫時編譯裸機程序(沒有linux等大型作業系統的程序),如監控程序,bootloader等能使得系統程序更加小巧快捷。

相關焦點

  • ARM-linux-gcc交叉編譯工具提示arm-linux-gcc can not find
    在Ubuntu12.04下安裝了arm-linux-gcc後,編譯Linux內核發現提示arm-linux-gcc can not find,查找了相關的網絡資料找到了答案
  • CMake設置arm-linux-gcc交叉編譯器
    主機:Ubuntu10.04交叉編譯器:EABI-4.3.3本文引用地址:http://www.eepw.com.cn/article/201611/319327.htmCMake在ubuntu系統下默認使用系統的gcc
  • 一次搞定 Arm Linux 交叉編譯
    為了讓這個流程變得簡單,開發者們為不同的晶片開發了不同的編譯器,比如針對 Arm 平臺的 arm-linux-gcc,針對 mips 平臺的 mips-linux-gnu-gcc,這些編譯器都是基於 GCC 針對具體的架構指令集進行對應配置,所以它們在運行的時候就就會生成和該目標平臺對應的可執行文件。
  • Linux GCC常用命令
    假設有一個由test1.c和 test2.c兩個源文件組成的程序,為了對它們進行編譯,並最終生成可執行程序test,可以使用下面這條命令: gcc test1.c test2.c -o test 如果同時處理的文件不止一個,GCC仍然會按照預處理、編譯和連結的過程依次進行。
  • arm-linux-ld命令 ld連結腳本
    我們看它是如何編譯的,注意我們這裡使用的不是arm-linux-gcc而是arm-elf-gcc,二者之間沒有什麼比較大的區別,arm-linux-gcc可能包含更多的庫文件,在命令行的編譯上面是沒有區別。
  • OpenCV ffmpeg移植到ARM平臺
    4 修改生成的Makefile文件:# The name of your C compiler:CC= gcc 該成 CC= /root/arm-none-linux-gnueabi/bin/arm-none-linux-gnueabi-gcc(根據你自己交叉編譯器的位置修改)# library (.a) file creation commandAR
  • Linux系統下多線程程序編譯makefile文件
    在Linux下,編譯一個多線程的程序命令為:本文引用地址:http://www.eepw.com.cn/article/201609/305140.htmgcc -lpthread -o file
  • ARM Linux根文件系統Root Filesystem的製作
    說Busybox和arm-linux-gcc有兼容性問題,不過我覺得那是比較低版本的時代問題了,我用Busybox 1.8.2和arm-linux-gcc 3.4.1/3.3.2都可以。解壓縮以後找到Makefile裡面的ARCH和CROSS_COMPILE,改成:ARCH ?= armCROSS_COMPILE ?
  • c編譯器so easy,gcc c編譯器生成、使用動靜態庫
    第一章程序開發人員大多接觸過c編譯器,請注意,不要將c編譯器和編輯器弄混淆哦。本文對c編譯器的講解,同樣基於gcc c編譯器,本文主要目的在於對linux環境下gcc c編譯器生成和使用靜態庫和動態庫予以介紹。此外,本文為系列教程第一篇——基本概念篇,之後將帶來另外兩篇。
  • 移植Lua到ARM平臺
    交叉編譯器:arm-linux-gcc 4.3.2版本(如果你不知道怎麼搭建交叉編譯環境,可以參考我的另一篇文章:http://hi.baidu.com/hqwfreefly/blog/item/8364fed7b3f58dc950da4b5f.html)3.Lua5.1版本源碼移植步驟:1下載Lua原始碼:http://www.lua.org
  • Linux下C編程基礎之:gcc編譯器
    本文引用地址:http://www.eepw.com.cn/article/257151.htmGNUCC(簡稱為gcc)是GNU項目中符合ANSIC標準的編譯系統,能夠編譯用C、C++和ObjectC等語言編寫的程序。
  • 嵌入式ARM-Linux平臺上的編譯、配置和運行使用
    /configure配置編譯環境;執行make進行編譯export ARCH=armexport CROSS_COMPILE=arm-linux-gnueabi-.執行make進行編譯,完成後執行make install,編譯好的openssl庫和頭文件等被安裝在目錄/usr/local/ssl下export ARCH=armexport CROSS_COMPILE=arm-linux-gnueabi-.
  • Notepad++編譯和運行C語言(GCC)
    我們在學習C語言的時候,實際上只需要編譯器和編輯器就能開搞了。(初學者過早接觸IDE不利於理解程序構建的過程)在看這篇文章的時候,假設你已經知道如何把GCC配置到環境變量,並且會在命令行/終端下使用gcc 編譯C代碼文件 並運行了。
  • ARM/uClinux應用程式的開發
    對於以前沒用過linux的人來說(比如我),開發程序前應該花3,4天時間熟悉linux環境,尤其是它的編輯器,用慣集成編譯環境的人有時連編譯器和編輯器的概念都模糊了,所以一般是直接進入集成編譯環境,連寫帶編一氣呵成,殊不知有些集成編譯器提供的編輯器弱智的一塌胡塗,如果用熟了linux下的emacs,你就會發現他們之間的差距大概……要像我和蓋茨那麼大吧。
  • 你真的了解Linux下gcc編譯器的工作過程嗎?
    我們以Linux作業系統gcc編譯器為例,來看下對C程序的編譯處理過程。小豆丁:嗯嗯,快講吧,我已經準備好記錄了!老張:計算機只認識二進位0和1這種文件,而我們寫的C程序顯然不是二進位文件,這就需要編譯器對我們寫好的程序進行編譯,使之變成計算機可以識別的二進位文件,也就是你剛剛所問的可執行程序。
  • 第1個ARM裸板程序及引申
    0x56000054 */    ldr r1, =0x56000054    ldr r0, =0  /* mov r0, #0 */    str r0, [r1]     /* 死循環 */halt:    b halt 將代碼上傳到伺服器, 先編譯: arm-linux-gcc
  • Linux下diff的操作詳解
    _lyn/arch/arm/boot/compressed/ashldi3.o and linux-4.19-rc3/arch/arm/boot/compressed/ashldi3.o differdiff -ruN linux-4.19-rc3_lyn/arch/arm/boot/compressed/.ashldi3.o.cmd linux-4.19-rc3/arch/arm/boot
  • 以Linux系統上的gcc為例,解密C語言編譯背後的全過程!
    以Linux系統上的gcc為例,通常我們編譯一個源文件都是用下面的命令: $gcc hello.c –o hello $gcc -save-temps hello.c –o hello $ls hello hello.c hello.i hello.o hello.s C編譯器的編譯過程主要分成四步:
  • 交叉編譯場景分析(arm-linux)(三)--編譯zlib
    基本信息: 軟體名稱 zlib 功能簡述 zlib 一個壓縮解壓程序庫 下載地址 http://www.zlib.net/ 軟體版本 zlib-1.2.3.tar.gz 依賴關係 默認 前置條件 源文件位置:$(WORK_DIR)/zlib