當(dāng)所有的系統(tǒng)初始化工作完成之后,就需要把程序流程轉(zhuǎn)入主應(yīng)用程序,即呼叫主應(yīng)用程序。最簡(jiǎn)單的一種情況是:
IMPORTmain
Bmain
直接從啟動(dòng)代碼跳轉(zhuǎn)到應(yīng)用程序的主函數(shù)入口,當(dāng)然主函數(shù)名字可以由用戶隨便定義。
在ARM ADS環(huán)境中,還另外提供了一套系統(tǒng)級(jí)的呼叫機(jī)制。
IMPORT__main
B__main
__main()是編譯系統(tǒng)提供的一個(gè)函數(shù),負(fù)責(zé)完成庫函數(shù)的初始化和初始化應(yīng)用程序執(zhí)行環(huán)境,最后自動(dòng)跳轉(zhuǎn)到main()。所以說,前者是庫函數(shù),后者就是我們自己編寫的main()主函數(shù);
因此我們用的B __main其實(shí)是執(zhí)行庫函數(shù),然后該庫函數(shù)再調(diào)用我們的main() 函數(shù),因此在單步調(diào)試時(shí)會(huì)看到先要跑一段程序(其實(shí)是庫函數(shù)),然后再單步到我們自己的main函數(shù)(這個(gè)同時(shí)也說明如果有B __main 則就對(duì)應(yīng)必須有main函數(shù),否則編譯出錯(cuò)),如果我們用 B main來進(jìn)入我們的主函數(shù)的話,那在單步調(diào)試時(shí)就看到直接進(jìn)入到我們自己的main函數(shù)了,中間不會(huì)看到其他程序;
那么用B __main和用B main 這兩這進(jìn)入我們的main函數(shù)方式有什么不同呢?
如果采用前者則會(huì)由編譯器加入一段"段拷貝"程序,即我們說的從加載域到執(zhí)行域轉(zhuǎn)化程序;而采用后者就沒有這個(gè)了,因此如果要進(jìn)行 "段拷貝"只能自己動(dòng)手編寫程序來實(shí)現(xiàn)了,完成段拷貝后就可以進(jìn)入我們的主函數(shù)了,當(dāng)然這個(gè)主函數(shù)不一定是叫做main(),可以起個(gè)其他好聽的名字,這 個(gè)有別于使用B __main方式;不管采用哪種方式進(jìn)入我們的程序,都要有一段"段拷貝"程序,跑完了段拷貝后才能可以進(jìn)入我們主程序了!(順便提一 下:startup.s這個(gè)文件并沒有所謂的"段拷貝"功能,再看也無益!)
對(duì)含有啟動(dòng)程序來說,"執(zhí)行地址與加載地址相同"不容易實(shí)現(xiàn):
如果執(zhí)行地址與加載地址相同哪當(dāng)然不需要做"段拷貝",但是個(gè)人理解編譯 器還會(huì)加入"段拷貝"程序(如果用B __main的話),只是因?yàn)闂l件不滿足而不執(zhí)行而已;但是對(duì)含有啟動(dòng)程序來說,"執(zhí)行地址與加載地址相同"就不容易了.因?yàn)閱?dòng)程序是要燒到非易失存儲(chǔ) 器里,用來在上電執(zhí)行的,而這個(gè)程序必定會(huì)有RW段,如果RW放在非易失存儲(chǔ)器,如FLASH,那就不好實(shí)現(xiàn)RW功能了,因此要給RW移動(dòng)到能夠?qū)崿F(xiàn)RW 功能的存儲(chǔ)器,如SRAM等.因此,對(duì)含有啟動(dòng)程序來說,"執(zhí)行地址與加載地址相同"就不容易實(shí)現(xiàn);程序的入口點(diǎn)在C 庫中的__main 處,在該點(diǎn),庫代碼執(zhí)行以下操作:
1. 將非零(只讀和讀寫)運(yùn)行區(qū)域從其載入地址復(fù)制到運(yùn)行地址。
2. 清零ZI 區(qū)域。
3. 跳轉(zhuǎn)到__rt_entry。
評(píng)論