使用keil判斷ARM的冷啟動和熱啟動的方法
編譯環(huán)境:Keil MDK V4.10
本文引用地址:http://www.ex-cimer.com/article/201611/319247.htm思路:
常把單片機(jī)系統(tǒng)的復(fù)位分為冷啟動和熱啟動。所謂冷啟動,也就是一般所說的上電復(fù)位,冷啟動后片內(nèi)外RAM的內(nèi)容是隨機(jī)的,通常是0x00或0xFF;單片機(jī)的熱啟動是通過外部電路給運行中的單片機(jī)的復(fù)位端一復(fù)位電平而實現(xiàn)的,也就是所說的按鍵復(fù)位或看門狗復(fù)位。復(fù)位后,RAM的內(nèi)容都沒有改變。在某些場合,必須區(qū)分出設(shè)備的重啟是熱重啟還是冷重啟。常用的方法是:確定某內(nèi)存單位為標(biāo)志位(如0x40003FF4~0x40003FF7 RAM單元),啟動時首先讀該內(nèi)存單元的內(nèi)容,如果它等于一個特定的值(例如為0xAA55AA55),就認(rèn)為是熱啟動,否則就是冷啟動。
根據(jù)以上的設(shè)計思路思路定義一個變量:
uint32
在程序啟動時判斷:
if(unStartFlag==0xAA55AA55)
{
//熱啟動處理
}
else
{
//冷啟動處理
unStartFlag=0xAA55AA55;
}
然而實際調(diào)試中發(fā)現(xiàn),無論是熱啟動還是冷啟動,開機(jī)后所有內(nèi)存單元的值都被復(fù)位為0,當(dāng)然也實現(xiàn)不了熱啟動的要求。通過看keil MDK自帶的啟動代碼Startup.s,在這個啟動代碼中也并沒有發(fā)現(xiàn)將整個RAM區(qū)域清零的語句。反匯編程序,發(fā)現(xiàn)從啟動代碼執(zhí)行結(jié)束到跳轉(zhuǎn)到main函數(shù)過程中,編譯器還執(zhí)行了很多庫函數(shù),其中__scatterload_zeroinit函數(shù)將所有W/R RAM都初始化為0(默認(rèn)設(shè)置下)。為了判斷冷、熱啟動,必須人為控制某些特定RAM在復(fù)位時不被編譯器初始化為0。通過查找編譯器手冊,在為處理器的RAM中分出一塊小片RAM,設(shè)置為NoInit格式(不對其初始化為0),如下圖:
然后使用__at關(guān)鍵字將冷、熱啟動標(biāo)志位定位到這個NoInit區(qū)域:
uint32 unStartFlag __at (0x40003FF4);
這樣,當(dāng)熱啟動時,變量unStartFlag所在的內(nèi)存區(qū)域就不會被初始化為0,也實現(xiàn)了冷熱啟動的判斷。
定義鐵電0xFF7~0xFF8區(qū)域存儲冷啟動次數(shù)
0xFF9~0xFFA區(qū)域存儲熱啟動次數(shù)
0xFFB~0xFFC區(qū)域存儲總啟動次數(shù)
評論