引用 ARM開(kāi)發(fā),對(duì)entry point的含義
本文引用地址:http://cafeforensic.com/article/201611/317248.htm 從事了這么久的ARM開(kāi)發(fā),對(duì)entry point的含義一知半解,今日再次拿出一本寫(xiě)得還算不錯(cuò)的ARM教材來(lái)翻,書(shū)中這一段僅僅是把英文手冊(cè)單純翻譯成中文,并沒(méi)有講到真正的實(shí)質(zhì)。不是ARM公司手冊(cè)寫(xiě)得不詳細(xì),而是多數(shù)中文教材的編寫(xiě)采用的方式是: english--華文 mapping,但英文手冊(cè)是有系列組織的suite,而書(shū)本往往翻譯自系列中的一篇,容易使得讀者管中窺豹。 摘要:ARM link的時(shí)候-entry ‘行標(biāo)號(hào)’ 選項(xiàng)用于指定程序的入口地址,其實(shí)就是ICE把代碼load完之后自動(dòng)跳入的地址,最后的程序中這個(gè)工作是由bootloader來(lái)完成的,當(dāng)然bootloader也可以跳到別的入口地址去。幾乎所有的成品ARM芯片都有內(nèi)置的bootloader,因此開(kāi)發(fā)者看到的程序入口點(diǎn)就是entry point。對(duì)于FPGA上的ARM系統(tǒng)調(diào)試,ARM復(fù)位后跳轉(zhuǎn)到一個(gè)固化地址,就需要自己寫(xiě)bootloader代碼并通過(guò)scatter加載到rom中(我們的EAS片上存儲(chǔ)器包括rom在內(nèi)全部用FPGA的ram實(shí)現(xiàn),tape out的時(shí)候bootloader代碼會(huì)被固化到chip的rom中) 1.entry point在編譯中的指定 看了make file,entry point的指定是在link的時(shí)候用命令行參數(shù)的方式告訴編譯器的, armlink ... -Entry reset_handler 如果采用RVDS或ADS,工程屬性中有entry point選項(xiàng)可供設(shè)置,但最終還是作為參數(shù)傳遞給armlink。 這樣, 在連接ICE并loadimage之后,ICE會(huì)通過(guò)JTAG把代碼映像搬移到內(nèi)存中,然后把當(dāng)前PC指針指向reset_handler,但引出了一個(gè)問(wèn)題:在最終的產(chǎn)品中不存在ICE和JTAG,ARM上電復(fù)位后跳向固化的復(fù)位地址,如0xFFFF0000,那么由誰(shuí)來(lái)讓PC指針指向我們指定的entry point呢?答案是:由bootloader代碼來(lái)完成,進(jìn)一步引出2個(gè)問(wèn)題:1,誰(shuí)來(lái)生成bootloader?2,怎么把bootloader定位到硬件復(fù)位地址? 2. 自己寫(xiě)bootloader并通過(guò)scatter文件將之定位到硬件復(fù)位地址。 或許某些開(kāi)發(fā)工具會(huì)自動(dòng)的為我們生成bootloader,這里介紹的是自力更生的方法。 首先介紹bootloader,這里僅實(shí)現(xiàn)中斷向跳轉(zhuǎn): AREA Vect, CO ENTRY ;這個(gè)entry和本文中的entry point是沒(méi)有關(guān)系的,它表示的是匯編程序代碼部分的開(kāi)始 LDR PC, Reset_Addr LDR PC, Undefined_Addr LDR PC, SWI_Addr LDR PC, Prefetch_Addr LDR PC, Abort_Addr NOP ; Reserved vector LDR PC, IRQ_Addr LDR PC, FIQ_Addr Reset_Addr DCD Reset_Handler Undefined_Addr DCD Undefined_Handler SWI_Addr DCD SWI_Handler Prefetch_Addr DCD Prefetch_Handler Abort_Addr DCD Abort_Handler IRQ_Addr DCD IRQ_Handler FIQ_Addr DCD FIQ_Handler Reset_Handler LDR PC, =0x00000000; 然后是scatter文件: ROM_LOAD 0x00000000 { ROM_EXEC 0x00000000 { * (InRoot$$Sections); 實(shí)現(xiàn)搬移scatter的代碼,編譯器自動(dòng)生成 *(+RO) ; 除了bootloader的所有代碼段放這里 } D-TCM 0x00400000 0x003FFFFF { * (+RW,+ZI);這里放所有的RW段(char ch=4;)和ZI段(char ch; ch會(huì)被初始化零)。 } } ROM2 0xFFFF0000 { BOOTLOAD 0xFFFF0000 { bootloader.o(+RO) ;bootloader的程序段放這里 } } scatter也是在link的時(shí)候通過(guò)參數(shù)‘-scatter 文件名’傳遞的。 3.總結(jié) 在以上條件下,比較使用ICE和最終產(chǎn)品中的啟動(dòng)順序:當(dāng)使用ICE調(diào)試時(shí),ICE讀入映像文件,找到其中的scatter信息,通過(guò)JTAG把代碼段和數(shù)據(jù)段放到指定的位置,然后把PC設(shè)為-entry所指定的標(biāo)號(hào),這就是為什么當(dāng)我們loadimage之后看到的是在entry point中指定的行號(hào)所在的源代碼。當(dāng)脫離ICE運(yùn)行時(shí),ARM上電后跳轉(zhuǎn)到硬件復(fù)位地址0xFFF0000 (ROM),我們已事先在那里燒入了我們的bootloader,bootloder從外設(shè)中讀入加載映像文件(如片外nand,I2C,UART等)并放入scatter的加載域中(0x00000000),然后跳轉(zhuǎn)到0x000000,由于我們已在scatter中將InRoot$$Sections定位在那里,所以它被執(zhí)行,它負(fù)責(zé)把映像文件從加載域搬移到執(zhí)行域中,然后跳轉(zhuǎn)到__main,__main是arm編譯器自動(dòng)生成的,它負(fù)責(zé)清零ZI段和初始化C庫(kù),最后__main跳向C代碼的入口main() |
評(píng)論