思路:
常把單片機系統的復位分為冷啟動和熱啟動。所謂冷啟動,也就是一般所說的上電復位,冷啟動后片內外RAM的內容是隨機的;單片機的熱啟動是通過外部電路給運行中的單片機的復位端一復位電平而實現的,也就是所說的按鍵復位或看門狗復位。復位后,RAM的內容都沒有改變。在某些場合,必須區分出設備的重啟是熱重啟還是冷重啟。常用的方法是:確定某內存單位為標志位(如0x40003FF4~0x40003FF7 RAM單元),啟動時首先讀該內存單元的內容,如果它等於一個特定的值(例如為0xAA55AA55),就認為是熱啟動,否則就是冷啟動。
根據以上的設計思路思路定義一個變量:
uint32 unStartFlag;
在程序啟動時判斷:
if(unStartFlag==0xAA55AA55)
{
//熱啟動處理
}
else
{
//冷啟動處理
unStartFlag=0xAA55AA55;
}
然而實際調試中發現,無論是熱啟動還是冷啟動,開機后所有內存單元的值都被復位為0,當然也實現不了熱啟動的要求。通過看keil MDK自帶的啟動代碼Startup.s,在這個啟動代碼中也並沒有發現將整個RAM區域清零的語句。反匯編程序,發現從啟動代碼執行結束到跳轉到main函數過程中,編譯器還執行了很多庫函數,其中__scatterload_zeroinit函數將所有W/R RAM都初始化為0(默認設置下)。為了判斷冷、熱啟動,必須人為控制某些特定RAM在復位時不被編譯器初始化為0。通過查找編譯器手冊,在為處理器的RAM中分出一塊小片RAM,設置為NoInit格式(不對其初始化為0),如下圖:
然后使用__at關鍵字將冷、熱啟動標志位定位到這個NoInit區域:
uint32 unStartFlag __at (0x40003FF4);
這樣,當熱啟動時,變量unStartFlag所在的內存區域就不會被初始化為0,也實現了冷熱啟動的判斷。
定義鐵電0xFF7~0xFF8區域存儲冷啟動次數
0xFF9~0xFFA區域存儲熱啟動次數
0xFFB~0xFFC區域存儲總啟動次數
原文鏈接:https://blog.csdn.net/zhzht19861011/article/details/6249318
原文中使用LPC的工程,那么其他的單片機如果也是cortex 系列,應該大同小異。
其次我們在使用的過程中,其實對keil mdk的這個編譯器工具還有很多需要學習的地方,使用到的僅僅數皮毛啊。