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