ARM標(biāo)準(zhǔn)庫(kù)
ADS提供了ANSI C和C 標(biāo)準(zhǔn)庫(kù),本文僅討論ANSI C庫(kù),該庫(kù)包含下面幾個(gè)部分:
◇IS0 C庫(kù)標(biāo)準(zhǔn)所定義的函數(shù);
◇在Semihosted 環(huán)境下用來(lái)實(shí)現(xiàn)C庫(kù)函數(shù)和目標(biāo)相關(guān)的函數(shù);
◇C和C 編譯器要使用的helper函數(shù)。
該庫(kù)提供的諸如文檔輸入輸出之類(lèi)的設(shè)備,使用了標(biāo)準(zhǔn)的ARM semihosted執(zhí)行環(huán)境(semihosting是針對(duì)ARM目標(biāo)機(jī)的一種機(jī)制,他能夠根據(jù)應(yīng)用程式代碼的輸入/輸出請(qǐng)求,和運(yùn)行有調(diào)度功能的主機(jī)通信,這種技術(shù)允許主機(jī)為通常沒(méi)有輸入和輸出功能的目標(biāo)硬件提供主機(jī)資源)。ARMulator、Angel和Multi-lCE都支持這個(gè)環(huán)境,能夠使用ADs中提供的研發(fā)工具研發(fā)應(yīng)用程式,然后在ARMulator或是研發(fā)板上運(yùn)行和調(diào)試該程式。假如要使應(yīng)用系統(tǒng)單獨(dú)于這個(gè)環(huán)境,則必須重新實(shí)現(xiàn)C庫(kù)中依賴(lài)于這個(gè)環(huán)境的相關(guān)函數(shù),根據(jù)用戶(hù)系統(tǒng)的運(yùn)行環(huán)境對(duì)C庫(kù)進(jìn)行適當(dāng)?shù)牟脺p。 ..
使用ANSI標(biāo)準(zhǔn)C庫(kù)進(jìn)行程式研發(fā),不但能夠提高研發(fā)效率而且能夠增強(qiáng)程式的可移植性。在程式中使用庫(kù)函數(shù),必須先建立一個(gè)庫(kù)函數(shù)能夠執(zhí)行的環(huán)境,這些工作都由庫(kù)中的函數(shù)完成。當(dāng)應(yīng)用程式鏈接了C庫(kù)中的函數(shù)時(shí),C庫(kù)中的函數(shù)將完成:
◇創(chuàng)建C程式所需的執(zhí)行環(huán)境(建立棧,假如需要?jiǎng)?chuàng)建一個(gè)堆,初始化程式使用的部分庫(kù));
◇調(diào)用main()函數(shù)開(kāi)始執(zhí)行C程式;
◇支持程式使用的Is0定義的函數(shù);
◇捕獲運(yùn)行時(shí)的錯(cuò)誤和信號(hào),假如需要,根據(jù)錯(cuò)誤終止執(zhí)行或程式退出。版權(quán)申明:本站文章均來(lái)自網(wǎng)絡(luò),本站所有轉(zhuǎn)載文章言論不代表本站觀點(diǎn)
2 裁減ARM標(biāo)準(zhǔn)C函數(shù)庫(kù)
標(biāo)準(zhǔn)庫(kù)中包含了部分依賴(lài)于ARM semihosted執(zhí)行環(huán)境的函數(shù),這部分函數(shù)的函數(shù)名中包含有單個(gè)或兩個(gè)下劃線(xiàn)“-”,需要重新實(shí)現(xiàn)這部分函數(shù)。假如在程式中定義這些函數(shù),則編譯器就會(huì)使用新定義的函數(shù),這個(gè)過(guò)程稱(chēng)為庫(kù)函數(shù)的裁減。一般情況下,只需要重新定義很少的幾個(gè)函數(shù)就能夠使用C庫(kù)。
ARM應(yīng)用系統(tǒng)開(kāi)始執(zhí)行用戶(hù)應(yīng)用程式,必須先將應(yīng)用程式加載到執(zhí)行域,建立應(yīng)用程式的執(zhí)行環(huán)境。使用C庫(kù)時(shí),這些繁瑣的工作就大部分由c函數(shù)來(lái)完成了。匯編程式完成系統(tǒng)初始化后,跳轉(zhuǎn)到C程式的人口_main () (注意:不是main(),當(dāng)C程式中定義了main()主函數(shù)時(shí),編譯器就會(huì)生成_main 代碼)。由_main()引導(dǎo)庫(kù)函數(shù)完成C執(zhí)行環(huán)境的初始化,具體過(guò)程如下:
◇將非啟動(dòng)代碼的RO和RW執(zhí)行域代碼從加載域地址復(fù)制到執(zhí)行域地址;
◇將ZI段 清零;
◇跳轉(zhuǎn)到_rt_entry。
◇調(diào)用_rt_stackheap_init()建立堆和棧;
◇調(diào)用_rt_lib_init()初始化引用的庫(kù)函數(shù);假如需要,建立main()函數(shù)的參數(shù)argc和argv等;
◇調(diào)用main()函數(shù),執(zhí)行應(yīng)用程式,能夠應(yīng)用庫(kù)函數(shù);
◇用main()函數(shù)的返回值作參數(shù)調(diào)用exit()。
_rt_entry并不是C函數(shù),他是用ARM C庫(kù)編程的起始點(diǎn)。_rt_entry不能用C語(yǔ)言宴現(xiàn),因?yàn)檫@時(shí)候堆棧還沒(méi)有建立,堆棧由_ rt_stackheap_init()來(lái)建立。
上面簡(jiǎn)單介紹了C程式使用庫(kù)函數(shù)時(shí)的調(diào)用過(guò)程,由_rt—stackheap_init()建立C庫(kù)使用的內(nèi)存模型--堆和棧。因?yàn)锳RM庫(kù)是建立在 semihosted執(zhí)行環(huán)境的,他實(shí)現(xiàn)的內(nèi)存模型是基于這個(gè)環(huán)境的,所以必須修改這個(gè)內(nèi)存模型建立機(jī)制。表1列出了需要重新實(shí)現(xiàn)的函數(shù),實(shí)現(xiàn)了這些函數(shù),應(yīng)用程式就能夠脫離宿主機(jī)環(huán)境單獨(dú)運(yùn)行了。其中,必須重新實(shí)現(xiàn)的是_user initial_stackheap(),因?yàn)槟J(rèn)的實(shí)現(xiàn)是基于semihosted執(zhí)行環(huán)境的,該函數(shù)被_n_stackheap_init()調(diào)用創(chuàng)建內(nèi)存模型,其他兩個(gè)函數(shù)沒(méi)有默認(rèn)的實(shí)現(xiàn)。 .
表1
函數(shù)
__user_initial_stackheap()
__user_heap_extend()
__user_stack_slop()
實(shí)現(xiàn)該函數(shù),必須滿(mǎn)足下面的條件:
◇使用不超過(guò)96字節(jié)的棧空間;
◇除了R12(ip)外不要污染其他寄存器;
◇將堆基址、?;?、堆邊界和棧邊界分別存在RO~R3作為返回參數(shù);
◇堆必須保持8個(gè)字節(jié)對(duì)齊。
實(shí)現(xiàn)例程如下:
#include
__value_in-regs struct __initial_stackheap __user_initial-sta ckheap(
{
}
為了提高應(yīng)用程式研發(fā)效率和可移植性,希望在目標(biāo)系統(tǒng)上使用ARM庫(kù)提供的標(biāo)準(zhǔn)輸人輸出庫(kù)函數(shù)。
高層輸入輸出函數(shù)是不依賴(lài)于目標(biāo)系統(tǒng)環(huán)境的,但是高層輸入輸出函數(shù)必須調(diào)用依賴(lài)于目標(biāo)系統(tǒng)的底層函數(shù),才能實(shí)現(xiàn)應(yīng)用系統(tǒng)的輸入輸出。依據(jù)目標(biāo)系統(tǒng)硬件環(huán)境重新定義這些底層函數(shù),就能夠使用庫(kù)提供的標(biāo)準(zhǔn)input/output庫(kù)函數(shù)了。下面以裁減ARM標(biāo)準(zhǔn)庫(kù)提供的printf系列輸出函數(shù)為例來(lái)作說(shuō)明。
標(biāo)準(zhǔn)I/O庫(kù)中最常用的是printf系列函數(shù),包括_printf()、printf()、_fprintf()、fprintf()、 vprintf()和vfprintf()。任何這些函數(shù)非透明地使用_FILE,并且僅依賴(lài)于fputc()和ferror()兩個(gè)函數(shù)。函數(shù) _printf()和_fprintf()和printf()和fprintf()的區(qū)別僅在于前兩個(gè)函數(shù)不能格式化浮點(diǎn)值。只要定義了自己的_FILE 版本和fputc()、ferror()函數(shù),外加定義一個(gè)具備FILE類(lèi)型的_stdout變量,就能夠不作任何修改地使用printf系列、 fwrite()、fputs()和puts()函數(shù)了。
評(píng)論