學習目標:
● 了解交叉編譯工具鏈
● 理解分步構建交叉編譯工具鏈的方法
● 學會使用Crosstool工具構建交叉編譯工具鏈
2.1??交叉編譯工具鏈介紹
讀者可能會有疑問,為什么要用交叉編譯器?交叉編譯通俗地講就是在一種平臺上編譯出能運行在體系結構不同的另一種平臺上的程式,比如在PC平臺(X86 CPU)上編譯出能運行在以ARM為內核的CPU平臺上的程式,編譯得到的程式在X86 CPU平臺上是不能運行的,必須放到ARM CPU平臺上才能運行,雖然兩個平臺用的都是Linux系統。這種方法在異平臺移植和嵌入式研發時非常有用。相對和交叉編譯,平常做的編譯叫本地編譯,也就是在當前平臺編譯,編譯得到的程式也是在本地執行。用來編譯這種跨平臺程式的編譯器就叫交叉編譯器,相對來說,用來做本地編譯的工具就叫本地編譯器。所以要生成在目標機上運行的程式,必須要用交叉編譯工具鏈來完成。在裁減和制定linux內核用于嵌入式系統之前,由于一般嵌入式研發系統存儲大小有限,通常都要在性能優越的PC上建立一個用于目標機的交叉編譯工具鏈,用該交叉編譯工具鏈在PC上編譯目標機上要運行的程式。交叉編譯工具鏈是個由編譯器、連接器和解釋器組成的綜合研發環境,交叉編譯工具鏈主要由binutils、gcc和glibc 3個部分組成。有時出于減小 libc 庫大小的考慮,也能用別的 c 庫來代替 glibc,例如 uClibc、dietlibc 和 newlib。建立交叉編譯工具鏈是個相當復雜的過程,如果不想自己經歷復雜繁瑣的編譯過程,網上有一些編譯好的可用的交叉編譯工具鏈能下載,但就以學習為目的來說讀者有必要學習自己制作一個交叉編譯工具鏈。本章通過具體的實例講述基于ARM的嵌入式Linux交叉編譯工具鏈的制作過程。
構建交叉編譯器的第一個步驟就是確定目標平臺。在GNU系統中,每個目標平臺都有一個明確的格式,這些信息用于在構建過程中識別要使用的不同工具的正確版本。因此,當在一個特定目標機下運行GCC時,GCC便在目錄路徑中查找包含該目標規范的應用程式路徑。GNU的目標規范格式為CPU-PLATFORM-OS。例如x86/i386 目標機名為i686-pc-linux-gnu。本章的目的是講述建立基于ARM平臺的交叉工具鏈,所以目標平臺名為arm-linux-gnu。
通常構建交叉工具鏈有3種方法。
方法一??分步編譯和安裝交叉編譯工具鏈所需要的庫和原始碼,最終生成交叉編譯工具鏈。該方法相對比較困難,適合想深入學習構建交叉工具鏈的讀者。如果只是想使用交叉工具鏈,建議使用方法二或方法三構建交叉工具鏈。
方法二??通過Crosstool腳本工具來實現一次編譯生成交叉編譯工具鏈,該方法相對于方法一要簡單許多,并且出錯的機會也非常少,建議大多數情況下使用該方法構建交叉編譯工具鏈。
方法三??直接通過網上(ftp.arm.kernel.org.uk)下載已制作好的交叉編譯工具鏈。該方法的好處不用多說,當然是簡單省事,但和此同時該方法有一定的弊端就是局限性太大,因為畢竟是別人構建好的,也就是固定的沒有靈活性,所以構建所用的庫及編譯器的版本也許并不適合你要編譯的程式,同時也許會在使用時出現許多莫名的錯誤,建議讀者慎用此方法。
為了讓讀者真正的學習交叉編譯工具鏈的構建,下面將重點周詳地介紹前兩種構建ARM Linux交叉編譯工具鏈的方法。
2.2.1??分步構建交叉編譯鏈
分步構建,顧名思義就是一步一步地建立交叉編譯鏈,不同于2.2.2節中講述的Crosstool腳本工具一次編譯生成的方法,該方法適合那些希望深入學習了解構建交叉編譯工具鏈的讀者。該方法相對來說難度較大,通常情況下困難重重,猶如唐僧西天取經,不過本文會盡可能周詳地介紹構建的每一個步驟,讀者完萬能根據本節的內容自己獨立實踐,構建自己的交叉工具鏈。該過程所需的時間較長,希望讀者有較強的耐心和毅力去學習和實踐他,通過實踐能使讀者更加清晰交叉編譯器的構建過程及各個工具包的作用。該方法所需資源如表2.1所示。
表2.1??所需資源
安裝包
下載地址
安裝包
下載地址
linux-2.6.10.tar.gz
ftp.kernel.org
glibc-2.3.2.tar.gz
ftp.gnu.org
binutils-2.15.tar.bz2
ftp.gnu.org
glibc-linuxthreads-2.3.2.tar.gz
ftp.gnu.org
gcc-3.3.6.tar.gz
ftp.gnu.org
通過相關站點下載以上資源后,就能開始建立交叉編譯工具鏈了。
1.建立工作目錄
首先建立工作目錄,工作目錄就是在什么目錄下構建交叉工具鏈,目錄的構建一般沒有特別的需求,能根據個人喜好建立。以下所建立的目錄是作者自定義的,當前的用戶定義為mike,因此用戶目錄為/home/mike,在用戶目錄下首先建立一個工作目錄(armlinux),建立工作目錄的命令行操作如下:
# cd /home/mike
# mkdir armlinux
再在這個工作目錄armlinux下建立3個目錄 build-tools、kernel 和 tools。具體操作如下:
# cd armlinux
# mkdir build-tools kernel tools
其中各目錄的作用如下。
● build-tools??用來存放下載的binutils、gcc、glibc等原始碼和用來編譯這些原始碼的目錄;
● kernel??用來存放內核原始碼;
● tools??用來存放編譯好的交叉編譯工具和庫文件。
2.建立環境變量
該步驟的目的是為了方便重復輸入路徑,因為重復操作每件相同的事情總會讓人覺得非常麻煩,如果讀者不習慣使用環境變量就能略過該步,直接輸入絕對路徑就能。聲明以下環境變量的目的是在之后編譯工具庫的時候會用到,非常方便輸入,尤其是能降低輸錯路徑的風險。
# export PRJROOT=/home/mike/armlinux
# export TARGET=arm-linux
# export PREFIX=$PRJROOT/tools
# export TARGET_PREFIX=$PREFIX/$TARGET
# export PATH=$PREFIX/bin:$PATH
注意,用export聲明的變量是臨時的變量,也就是當注銷或更換了控制臺,這些環境變量就消失了,如果還需要使用這些環境變量就必須重復export操作,所以有時會非常麻煩。值得慶幸的是,環境變量也能定義在bashrc文件中,這樣當注銷或更換控制臺時,這些變量就一直有效,就不用老是export這些變量了。
3.編譯、安裝Binutils
Binutils是GNU工具之一,他包括連接器、匯編器和其他用于目標文件和檔案的工具,他是二進制代碼的處理維護工具。安裝Binutils工具包含的程式有addr2line、ar、as、C++filt、gprof、ld、nm、objcopy、objdump、ranlib、readelf、size、strings、strip、libiberty、libbfd和libopcodes。對這些程式的簡單解釋如下。
● addr2line??把程式地址轉換為文件名和行號。在命令行中給他一個地址和一個可執行文件名,他就會使用這個可執行文件的調試信息指出在給出的地址上是哪個文件及行號。
● ar??建立、修改、提取歸檔文件。歸檔文件是包含多個文件內容的一個大文件,其結構確保了能恢復原始文件內容。
● as??主要用來編譯GNU C編譯器gcc輸出的匯編文件,產生的目標文件由連接器ld連接。
●?c++filt??連接器使用他來過濾 C++ 和?Java?符號,防止重載函數沖突。
● gprof??顯示程式調用段的各種數據。
● ld??是連接器,他把一些目標和歸檔文件結合在一起,重定位數據,并連接符號引用。通常,建立一個新編譯程式的最后一步就是調用ld。
● nm??列出目標文件中的符號。
● objcopy??把一種目標文件中的內容復制到另一種類型的目標文件中。
● objdump??顯示一個或更多目標文件的信息。使用選項來控制其顯示的信息,他所顯示的信息通常只有編寫編譯工具的人才感興趣。
● ranlib??產生歸檔文件索引,并將其保存到這個歸檔文件中。在索引中列出了歸檔文件各成員所定義的可重分配目標文件。
● readelf??顯示elf格式可執行文件的信息。
● size??列出目標文件每一段的大小及總體的大小。默認情況下,對于每個目標文件或一個歸檔文件中的每個模塊只產生一行輸出。
● strings??打印某個文件的可打印字符串,這些字符串最少4個字符長,也能使用選項-n設置字符串的最小長度。默認情況下,他只打印目標文件初始化和可加載段中的可打印字符;對于其他類型的文件他打印整個文件的可打印字符。這個程式對于了解非文本文件的內容非常有幫助。
● strip??丟棄目標文件中的全部或特定符號。
● libiberty??包含許多GNU程式都會用到的函數,這些程式有getopt、obstack、strerror、strtol和strtoul。
● libbfd??二進制文件描述庫。
● libopcode??用來處理opcodes的庫,在生成一些應用程式的時候也會用到他。
Binutils工具安裝依賴于Bash、Coreutils、Diffutils、GCC、Gettext、Glibc、Grep、Make、Perl、Sed、Texinfo等工具。
介紹完Binutils工具后,下面將分步介紹安裝binutils-2.15的過程。
首先解壓binutils-2.15.tar.bz2包,命令如下:
# cd $PRJROOT/build-tools
# tar -xjvf binutils-2.15.tar.bz2
接著設置Binutils工具,建議建立一個新的目錄用來存放設置和編譯文件,這樣能使源文件和編譯文件獨立開,具體操作如下:
# cd $PRJROOT/build-tools
# mkdir build-binutils
# cd build-binutils
# ../ binutils-2.15/configure --target=$TARGET --prefix=$PREFIX
其中選項?target的意思是制定生成的是 arm-linux 的工具,--prefix 是指出可執行文件安裝的位置。執行上述操作會出現非常多check信息,最后產生 Makefile 文件。接下來執行make和安裝操作,命令如下:
# make
# make install
該編譯過程較慢,需要數十分鐘,安裝完成后查看/home/mike/armlinux/tools/bin目錄下的文件,如果查看結果如下,表明此時Binutils工具已安裝結束。
# ls $PREFIX/bin
arm-linux-addr2line? ?arm-linux-ld? ?? ???arm-linux-ranlib? ? arm-linux-strip
arm-linux-ar? ?? ?? ?? ? arm-linux-nm? ?? ?? ?arm-linux-readelf
arm-linux-as? ?? ?? ?? ? arm-linux-objcopy??arm-linux-size
arm-linux-c++filt? ?? ???arm-linux-objdump??arm-linux-strings
4.獲得內核頭文件
編譯器需要通過系統內核的頭文件來獲得目標平臺所支持的系統函數調用所需要的信息。對于Linux內核,最佳的方法是下載一個合適的內核,然后復制獲得頭文件。需要對內核做一個基本的設置來生成正確的頭文件;不過,不必編譯內核。對于本例中的目標arm-linux,需要以下步驟。
(1)在kernel目錄下解壓linux-2.6.10.tar.gz內核包,執行命令如下:
# cd $PRJROOT/kernel
# tar -xv*** linux-2.6.10.tar.gz
(2)接下來設置編譯內核使其生成正確的頭文件,執行命令如下:
# cd linux-2.6.10
# make ARCH=arm CROSS_COMPILE=arm-linux- menuconfig
其中ARCH=arm表示是以arm為體系結構,CROSS_COMPILE=arm-linux-表示是以arm-linux-為前綴的交叉編譯器。也能用config和xconfig來代替menuconfig,推薦用make menuconfig,這也是內核研發人員用的最多的設置方法。注意在設置時一定要選擇處理器的類型,這里選擇三星的S3C2410(System Type->ARM System Type->/Samsung S3C2410)。設置完退出并保存,檢查一下內核目錄中的include/linux/version.h和include/linux/autoconf.h文件是不是生成了,這是編譯glibc時要用到的,如果version.h 和 autoconf.h 文件存在,說明生成了正確的頭文件。
復制頭文件到交叉編譯工具鏈的目錄,首先需要在/home/mike/armlinux/tools/arm-linux目錄下建立工具的頭文件目錄inlcude,然后復制內核頭文件到此目錄下,具體操作如下:
# mkdir -p $TARGET_PREFIX/include
# cp -r $PRJROOT/kernel/linux-2.6.10/include/linux $TARGET_PREFIX/include
# cp -r $PRJROOT/kernel/linux-2.6.10/include/asm-arm $TARGET_PREFIX/include/asm
5.編譯安裝boot-trap gcc
這一步的目的主要是建立arm-linux-gcc工具,注意這個gcc沒有glibc庫的支持,所以只能用于編譯內核、BootLoader等不必C庫支持的程式,后面創建C庫也要用到這個編譯器,所以創建他主要是為創建C庫做準備,如果只想編譯內核和BootLoader,那么安裝完這個就能到此結束。安裝命令如下:
# cd $PRJROOT/build-tools
# tar -xv*** gcc-3.3.6.tar.gz
# mkdir build-gcc? ???
# cd gcc-3.3.6
# vi gcc/config/arm/t-linux
由于是第一次安裝ARM交叉編譯工具,沒有支持libc庫的頭文件,所以在gcc/config/arm/t- linux文件中給變量TARGET_LIBGCC2_CFLAGS增加操作參數選項-Dinhibit_libc??-D__gthr_ posix_h來屏蔽使用頭文件,否則一般默認會使用/usr/inlcude頭文件。
將TARGET_LIBGCC2-CFLAGS = -fomit-frame-pointer ?fPIC改為TARGET_LIBGCC2- CFLAGS=-fomit-frame-pointer-fPIC -Dinhibit_libc -D__gthr_posix_h
修改完t-linux文件后保存,緊接著執行設置操作,如下命令:
# cd build-gcc
# ../ build-gcc /configure --target=$TARGET --prefix=$PREFIX --enable-languages=c
--disable-threads --disable-shared
其中選項--enable-languages=c表示只支持C語言,--disable-threads表示去掉thread功能,這個功能需要glibc的支持。--disable-shared表示只進行靜態庫編譯,不支持共享庫編譯。
接下來執行編譯和安裝操作,命令如下:
# make
# make install
安裝完成后,在/home/mike/armlinux/tools/bin下查看,如果arm-linux-gcc等工具已生成,表示boot-trap gcc工具已安裝成功。
6.建立glibc庫
glibc是GUN C庫,他是編譯Linux系統程式非常重要的組成部分。安裝glibc-2.3.2版本之前推薦先安裝以下的工具:
● GNU make 3.79或更新;
● GCC 3.2或更新;
● GNU binutils 2.13或更新。
首先解壓glibc-2.2.3.tar.gz和glibc-linuxthreads-2.2.3.tar.gz原始碼,操作如下:
# cd $PRJROOT/build-tools
# tar -xv*** glibc-2.2.3.tar.gz
# tar -xzvf glibc-linuxthreads-2.2.3.tar.gz --directory=glibc-2.2.3
然后進行編譯設置,glibc-2.2.3設置前必須新建一個編譯目錄,否則在glibc-2.2.3目錄下不允許進行設置操作,此處在$PRJROOT/build-tools目錄下建立名為build-glibc的目錄,設置操作? ? 如下:
# cd $PRJROOT/build-tools
# mkdir build-glibc
# cd build-glibc
# CC=arm-linux-gcc ../glibc-2.2.3 /configure --host=$TARGET --prefix="/usr"
--enable-add-ons --with-headers=$TARGET_PREFIX/include
選項CC=arm-linux-gcc是把CC(Cross Compiler)變量設成剛編譯完的gcc,用他來編譯glibc。--prefix="/usr"定義了一個目錄用于安裝一些和目標機器無關的數據文件,默認情況下是/usr/local目錄。--enable-add-ons是告訴glibc用linuxthreads包,在上面已將他放入glibc原始碼目錄中,這個選項等價于-enable-add-ons=linuxthreads。--with-headers告訴glibc linux內核頭文件的目錄? ? 位置。
設置完后就能編譯和安裝 glibc了,具體操作如下:
# make
# make install
7.編譯安裝完整的gcc
由于第一次安裝的gcc沒有交叉glibc的支持,目前已安裝了glibc,所以需要重新編譯來支持交叉glibc。并且上面的gcc也只支持c語言,目前能讓他同時支持C語言還要和C++語言。具體操作如下:
# cd $PRJROOT/build-tools/gcc-2.3.6
# ./configure --target=arm-linux --enable-languages=c,c++ --prefix=$PREFIX
# make
# make install
安裝完成后會發目前$PREFIX/bin目錄下又多了arm-linux-g++ 、arm-linux-c++等文件。
# ls $PREFIX/bin
arm-linux-addr2line arm-linux-g77? ?? ???arm-linux-gnatbind arm-linux-ranlib
arm-linux-ar? ?? ?? ?arm-linux-gcc? ?? ? arm-linux-jcf-dump??arm-linux-readelf
arm-linux-as? ?? ?? ?arm-linux-gcc-3.3.6 arm-linux-jv-scan? ?arm-linux-size
arm-linux-c++? ?? ???arm-linux-gccbug? ?arm-linux-ld? ?? ?? ?arm-linux-strings
arm-linux-c++filt??arm-linux-gcj? ?? ? arm-linux-nm? ?? ?? ?arm-linux-strip
arm-linux-cpp? ?? ???arm-linux-gcjh? ?? ?arm-linux-objcopy? ?grepjar
arm-linux-g++? ?? ???arm-linux-gcov? ?? ? arm-linux-objdump? ?jar
8.測試交叉編譯工具鏈
到此為止,已介紹完了用分步構建的方法建立交叉編譯工具鏈。下面通過一個簡單的程式測試剛剛建立的交叉編譯工具鏈看是否能夠正常工作。寫一個最簡單的hello.c源文件,內容如下:
#include
int main( )
{
printf(“Hello,world!\n”);
return 0;
}
通過以下命令進行編譯,編譯后生成名為hello的可執行文件,通過file命令能查看文件的類型。當顯示以下信息時表明交叉工具鏈正常安裝了,通過編譯生成了ARM體系可執行的文件。注意,通過該交叉編譯鏈編譯的可執行文件只能在ARM體系下執行,不能在基于X86的普通PC上執行。
# arm-linux-gcc -o hello hello.c
# file hello
hello: ELF 32-bit LSB executable, ARM, version 1 (ARM), for GNU/Linux 2.4.3,
dynamically linked (uses shared libs), not stripped
2.2.2??用Crosstool工具構建交叉工具鏈
Crosstool是一組腳本工具集,可構建和測試不同版本的gcc和glibc,用于那些支持glibc的體系結構。他也是個開源項目,下載地址是http://kegel.com/crosstool。用Crosstool構建交叉工具鏈要比上述的分步編譯容易得多,并且也方便許多,對于僅僅為了工作需要構建交叉編譯工具鏈的讀者建議使用此方法。用Crosstool工具構建所需資源如表2.2所示。
表2.2??所需資源
安裝包
下載地址
crosstool-0.42.tar.gz
http://kegel.com/crosstool
linux-2.6.10.tar.gz
ftp.kernel.org
binutils-2.15.tar.bz2
ftp.gnu.org
gcc-3.3.6.tar.gz
ftp.gnu.org
glibc-2.3.2.tar.gz
ftp.gnu.org
glibc-linuxthreads-2.3.2.tar.gz
ftp.gnu.org
linux-libc-headers-2.6.12.0.tar.bz2
ftp.gnu.org
1.準備資源文件
首先從網上下載所需資源文件linux-2.6.10.tar.gz、binutils-2.15.tar.bz2、gcc-3.3.6.tar.gz、glibc- 2.3.2.tar.gz、glibc-linuxthreads-2.3.2.tar.gz和linux-libc-headers-2.6.12.0.tar.bz2。然后將這些工具包文件放在新建的/home/mike/downloads目錄下,最后在/home/mike目錄下解壓crosstool-0.42.tar.gz,命令如下:
# cd /home/mike
# tar ?xv*** crosstool-0.42.tar.gz
2.建立腳本文件
接著需要建立自己的編譯腳本,起名為arm.sh,為了簡化編寫arm.sh,尋找一個最接近的腳本文件demo-arm.sh作為模板,然后將該腳本的內容復制到arm.sh,修改arm.sh腳本,具體操作如下:
# cd crosstool-0.42
# cp demo-arm.sh arm.sh
# vi arm.sh
修改后的arm.sh腳本內容如下:
#!/bin/sh
set -ex
TARBALLS_DIR=/home/mike/downloads? ?# 定義工具鏈源碼所存放位置。
RESULT_TOP=/opt/crosstool? ?? ?? ?? ?# 定義工具鏈的安裝目錄
export TARBALLS_DIR RESULT_TOP
GCC_LANGUAGES="c,c++"? ?? ?? ?? ?? ? # 定義支持C, C++語言
export GCC_LANGUAGES
# 創建/opt/crosstool目錄
mkdir -p $RESULT_TOP
# 編譯工具鏈,該過程需要數小時完成。
eval ’cat arm.dat gcc-3.3.6-glibc-2.3.2.dat’??sh all.sh --notest
echo Done.
3.建立設置文件
在arm.sh腳本文件中需要注意arm.dat和gcc-3.3.6-glibc-2.3.2.dat兩個文件,這兩個文件是作為Crosstool的編譯的設置文件。其中arm.dat文件內容如下,主要用于定義設置文件、定義生成編譯工具鏈的名稱及定義編譯選項等。
KERNELCONFIG=’pwd’/arm.config??# 內核的設置
TARGET=arm-linux-? ?? ?? ?? ?? ? # 編譯生成的工具鏈名稱
TARGET_CFLAGS="-O"? ?? ?? ?? ?? ? # 編譯選項
gcc-3.3.6-glibc-2.3.2.dat文件內容如下,該文件主要定義編譯過程中所需要的庫及他定義的版本,如果在編譯過程中發現有些庫不存在時,Crosstool會自動在相關網站上下載,該工具在這點上相對比較智能,也非常有用。
BINUTILS_DIR=binutils-2.15
GCC_DIR=gcc-3.3.6
GLIBC_DIR=glibc-2.3.2
GLIBCTHREADS_FILENAME=glibc-linuxthreads-2.3.2
LINUX_DIR=linux-2.6.10
LINUX_SANITIZED_HEADER_DIR=linux-libc-headers-2.6.12.0
4.執行腳本
將Crosstool的腳本文件和設置文件準備好之后,開始執行arm.sh腳本來編譯交叉編譯工具。具體執行命令如下:
# cd crosstool-0.42
# ./arm.sh??
經過數小時的漫長編譯之后,會在/opt/crosstool目錄下生成新的交叉編譯工具,其中包括以下內容:
arm-linux-addr2line arm-linux-g++? ?? ???arm-linux-ld? ?? ?? ?arm-linux-size
arm-linux-ar? ?? ?? ?arm-linux-gcc? ?? ???arm-linux-nm? ?? ?? ?arm-linux-strings
arm-linux-as? ?? ?? ?arm-linux-gcc-3.3.6 arm-linux-objcopy? ?arm-linux-strip
arm-linux-c++? ?? ???arm-linux-gccbug? ? arm-linux-objdump? ?fix-embedded-paths
arm-linux-c++filt? ?arm-linux-gcov? ?? ? arm-linux-ranlib
arm-linux-cpp? ?? ???arm-linux-gprof? ?? ?arm-linux-readelf
5.添加環境變量
然后將生成的編譯工具鏈路徑添加到環境變量PATH上去,添加的方法是在系統/etc/ bashrc文件的最后添加下面一行。
export PATH=/opt/crosstool/gcc-3.3.6-glibc-2.3.2/arm-linux/bin:$PATH ?
設置完環境變量,也就意味著交叉編譯工具鏈已構建完成,然后就能用2.2.1.8節中的方法進行測試剛剛建立的工具鏈,此處就不用再贅述。
2.3??本章小結
本章講述的內容非常有實用價值,因為交叉編譯工具鏈的構建是嵌入式系統研發必不可少的一部分,也是嵌入式系統研發的基礎。本章首先對交叉編譯工具鏈進行了大體的介紹,然后分別介紹了兩種構建交叉編譯工具鏈的方法:分步構建法和Crosstool工具構建法。這兩種構建交叉編譯工具鏈的方法在實際應用中非常廣泛,相信讀者通過學習本章的內容能構建一套自己的交叉編譯工具鏈。第3章將介紹嵌入式系統的啟動程式??BootLoader。
2.4??常見問題
問題1:編譯boot-trap gcc時出現錯誤,提示:crti.o: No such file: No such file or directory collect2: ld returned 1 exit status,為什么會出現這樣的錯誤?
參考答案:
由于在設置時沒有選擇--disable-shared 選項,該選項的意思是只編譯靜態庫。默認選項為--enable-shared,而libf2c 和libiberty 不支持共享庫。
問題2:Glibc里靜態庫和共享庫有什么差別?
參考答案:
應用程式在鏈接靜態庫時,會把引用到的數據和代碼放到生成的可執行文件中,程式運行時就不再需要庫了。應用程式鏈接共享庫時,連接器不會把引用到的數據和代碼放到可執行文件中,而僅做一個標記,當程式運行時,系統會去加載相應的共享庫。鏈接共享庫時,可執行文件的大小會小一些,但運行時依賴于共享庫。啟動靜態庫和共享庫的方法分別是在設置時用 --disable-shared和--enable-shared選項。
問題3:本地編譯器和交叉編譯器的作用。
參考答案:
編譯器能生成用來在和編譯器本身所在的計算機和操作系統(平臺)相同的環境下運行的目標代碼,這種編譯器叫做本地編譯器。另外,編譯器也能生成用來在其他平臺上運行的目標代碼,這種編譯器叫做交叉編譯器。
?
評論
查看更多