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