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