深入談?wù)処AP升級(jí)
掃描二維碼
隨時(shí)隨地手機(jī)看文章
最近做過(guò)51_IAP和ARM_IAP升級(jí),感覺(jué)到他們的相同點(diǎn)和不同點(diǎn),特記錄如下:
共同點(diǎn):
做好IAP的關(guān)鍵都是中斷向量的映射(REMAP)問(wèn)題
一般都分為bootloader區(qū)和用戶區(qū)
不同點(diǎn):
單片機(jī)的中斷向量位置是固定的,位于0x0000的底部。
ARM的中斷向量也一般是在0x0000的底部,但有的ARM可以中斷向量的映射機(jī)制,可以將RAM或其他地址的FLASH映射到0x0000底部
單片機(jī)沒(méi)有中斷向量管理機(jī)制。一般0x0000-0x0002是一個(gè)3個(gè)字節(jié)的LJMP指令(該指令跳轉(zhuǎn)到用戶程序,注意不一定是main函數(shù),可能是包括一些堆棧,寄存器初始化的匯編的初始化部分,然后跳轉(zhuǎn)到main函數(shù)),后面是各種具體中斷的跳轉(zhuǎn)函數(shù)的入口地址,例如串口,I2C等等。
ARM有中斷管理機(jī)制。(cortex-M3內(nèi)核又不一樣)
在ARM體系中,異常中斷向量表的大小為32字節(jié),其中每個(gè)異常中斷占據(jù)4個(gè)字節(jié)大小,保留了4個(gè)字節(jié)空間
0x00 復(fù)位
0x04 未定義的指令
0x08 軟件中斷
0x0C 指令預(yù)取終止
0x10 數(shù)據(jù)訪問(wèn)終止
0x14 保留 未使用
0x18 IRQ模式 這個(gè)就是我們常用各種中斷(串口,i2c等)的一個(gè)總中斷入口,后面再根據(jù)中斷其他寄存器進(jìn)行判斷處理
0x1C FIQ模式
中斷向量從0x0000000--0x00000001C共4*8=32個(gè)字節(jié),還應(yīng)包括后面到0x00000040之前的代碼都應(yīng)該映射,因?yàn)楹竺姘ň唧w跳轉(zhuǎn)的地址。(所以要映射從0x0000000----0x000003F共64個(gè)字節(jié),自己看匯編覺(jué)得)
對(duì)于下載的程序,必須要在KEIL設(shè)置好Ro_Base地址,然后將生成好的bin文件下載到R0_Base地址處。原因是絕對(duì)地址不同造成,也就是說(shuō)對(duì)某個(gè)bin文件并不是想下載到哪個(gè)地方都可以運(yùn)行。
值得注意:
1.MC51設(shè)置R0_Base, 只表示用戶代碼的存放區(qū),R0_Base這個(gè)地址并不存放中斷向量表,也不是main的入口地址
2.ARM設(shè)置R0_Base, R0_Base開(kāi)始處就是中斷向量表
MC51設(shè)置Ro_Base位置在于BL51 Locate--》Code Range
ARM設(shè)置ro_base位置在于target-->IROM
做過(guò)升級(jí)的例子:
(1)C8051F02*將用戶的中斷向量復(fù)制到0x0003之后,同時(shí)0x000-0x0002保留bootloader跳轉(zhuǎn)地址,另外找個(gè)地方保存用戶跳轉(zhuǎn)地址,用于從bootloader區(qū)跳轉(zhuǎn)到用戶區(qū)。值得注意:bootloader區(qū)由于其他中斷向量表被用戶中斷向量表占用,bootloader代碼中不能出現(xiàn)中斷函數(shù)。
(2) STR912 內(nèi)部有bank0,bank1的映射功能,即可將bank0映射到0x0000,亦可將bank1映射到0x0000,這樣就可以將一個(gè)bank作為bootloader ,另一個(gè)作為用戶程序
(3) 2410 中斷向量表位于底部,沒(méi)有重映射機(jī)制。采用將中斷向量二次映射的方法,將中斷向量映射到指定的RAM區(qū),然后在RAM寫(xiě)入用戶的中斷向量表。
(4) ADU7020 中斷向量表位于底部,有重新映射機(jī)制,REMAP寄存器,可以將中斷向量映射到FLASH或RAM選擇,可以將用戶的中斷向量拷貝到RAM區(qū),然后選擇將中斷向量映射到RAM,然后跳轉(zhuǎn)
(5) STM32 支持中斷向量表映射。cortex-M3核有專門的向量偏移寄存器,該寄存器決定:
1. 向量表是位于FLASH還是RAM,向量表的基址
2. 向量表的偏移量
注:
一個(gè)優(yōu)秀的IAP升級(jí)程序,必須做好升級(jí)中出現(xiàn)故障等異常的處理。保證系統(tǒng)不會(huì)崩潰
(1) 接收到升級(jí)命令,準(zhǔn)備升級(jí)
(2) 接收升級(jí)數(shù)據(jù)完成,每個(gè)包最好有包序號(hào)和校驗(yàn)
(3) 整個(gè)數(shù)據(jù)接收完,進(jìn)行總的校驗(yàn)
(4) 試運(yùn)行剛升級(jí)的版本,發(fā)送讀取版本信息或其他指令測(cè)試當(dāng)前版本是否工作正常,如果不正常,外界通過(guò)對(duì)外專門的RESET引腳,恢復(fù)到原版本
(5) 接收到版本確認(rèn)命令,將升級(jí)版本作為當(dāng)前運(yùn)行版本,并將原版本進(jìn)行保存
另外:
(1)要保證升級(jí)過(guò)程中任何時(shí)候掉電,下次上電后,系統(tǒng)能正常工作。例如新的程序正在搬運(yùn)到運(yùn)行區(qū)時(shí)候掉電,下次上電要實(shí)現(xiàn)自動(dòng)搬運(yùn)
(2)正常情況下,系統(tǒng)一般由bootloader區(qū)跳轉(zhuǎn)到用戶區(qū)。如果用戶區(qū)程序有問(wèn)題的話,就玩OVER了,因此一定要在跳轉(zhuǎn)之前加一定延時(shí),可以響應(yīng)升級(jí)命令
(3)執(zhí)行bootloader區(qū)和用戶區(qū)跳轉(zhuǎn)指令,由于只是PC指針的跳轉(zhuǎn),寄存器并未恢復(fù)到默認(rèn)值,因此對(duì)bootloader和用戶區(qū)代碼中的寄存器初始化要小心。
說(shuō)到底最重要的還是bootloader程序一定要寫(xiě)好,這個(gè)畢竟是出廠前固化在FLASH中
給手機(jī)升級(jí),主板升級(jí),其中強(qiáng)調(diào)的就是不能中途掉電,否則就要。。。這個(gè)應(yīng)該就是沒(méi)做好升級(jí)處理吧