智宇物聯(lián) 專注于提供高穩(wěn)定、高速率的三網(wǎng)物聯(lián)網(wǎng)卡
婁底物聯(lián)網(wǎng)設(shè)備的固件遠(yuǎn)程升級方案以及軟件代碼
- 作者:智宇物聯(lián)
- 發(fā)表時(shí)間:2022年10月13日
- 來源:智宇物聯(lián)
作為通用的物聯(lián)網(wǎng)設(shè)備,由于用戶需求各不相同,不少用戶有一些個(gè)性化的定制要求;
設(shè)備所對接的傳感器協(xié)議也多種多樣,比如Modbus讀寫參數(shù)的數(shù)據(jù)類型,某些物理量需要特殊的數(shù)據(jù)類型;
甚至可能還存在意想不到的bug。
因此,遠(yuǎn)程升級的功能對于設(shè)備來說必不可少。
遠(yuǎn)程固件升級需要解決以下問題:
1)設(shè)備的遠(yuǎn)程訪問
當(dāng)設(shè)備被安裝于局域網(wǎng)內(nèi)部時(shí),位于遠(yuǎn)程的固件升級軟件工具無法穿透路由器訪問設(shè)備。
2)固件的分包以及傳送
由于設(shè)備的處理器資源有限,無法移植開源的http、FTP等協(xié)議棧,無法通過http、FTP等協(xié)議從服務(wù)器上下載固件,而需要自己實(shí)現(xiàn)代碼,采用TCP協(xié)議進(jìn)行固件包的發(fā)送;
而且對于幾十k甚至上百k的固件,需要將固件拆成幾百個(gè)字節(jié)的數(shù)據(jù)包,逐一發(fā)給設(shè)備;
4)固件的有效性檢驗(yàn)
固件在傳輸過程中,難免會出現(xiàn)錯(cuò)誤。
比如WiFi模塊,或者是ethernet模塊將數(shù)據(jù)通過uart轉(zhuǎn)發(fā)給處理器時(shí),如果有干擾、數(shù)據(jù)可能被破壞;
或者是處理器太忙,來不及接收數(shù)據(jù),導(dǎo)致固件包丟失數(shù)據(jù);
如果不對固件進(jìn)行有效性檢驗(yàn),將被破壞的固件升級進(jìn)控制器,會導(dǎo)致設(shè)備變磚而無法使用;
5)bootloader程序
bootloader程序需要下載固件的有效性檢驗(yàn),程序的擦除、固件數(shù)據(jù)從備份區(qū)到程序區(qū)的搬移。
6)處理器的固件升級軟件實(shí)現(xiàn)
軟件需要實(shí)現(xiàn)數(shù)據(jù)包接收,固件有效性驗(yàn)證、存儲,數(shù)據(jù)應(yīng)答等。
遠(yuǎn)程固件升級系統(tǒng)架構(gòu)

遠(yuǎn)程固件升級系統(tǒng)架構(gòu)
設(shè)備作為TCP客戶端連接到云服務(wù)器上的TCP服務(wù)端,定時(shí)發(fā)送心跳,維護(hù)連接,從而實(shí)現(xiàn)TCP的長鏈接。
在PC電腦上開發(fā)遠(yuǎn)程升級工具,作為TCP客戶端與云服務(wù)器上的TCP服務(wù)器建立連接;
當(dāng)需要遠(yuǎn)程升級時(shí),通過PC工具向云服務(wù)器發(fā)送消息,所發(fā)消息中包括了遠(yuǎn)程設(shè)備的設(shè)備編號,以及PC工具的設(shè)備編號;
服務(wù)器收到消息之后,根據(jù)消息中的目標(biāo)設(shè)備編號,從其維護(hù)的長鏈接中找到與該編號對應(yīng)的鏈接,通過該鏈接向設(shè)備轉(zhuǎn)發(fā)該消息;
設(shè)備收到消息之后,對消息中的固件包進(jìn)行有效性驗(yàn)證,如果有效,則寫入到固件暫存區(qū),并回復(fù)成功,否則回復(fù)失敗。
一些設(shè)計(jì)要點(diǎn)
處理器的存儲空間安排:
以STM32F103RCT6為例,該處理器有256KByte的FLASH空間;
4KByte的空間用于bootloader程序。
52KByte用于存儲用戶數(shù)據(jù);
剩余的FLASH空間一半作為程序存儲區(qū),一半作為固件暫存區(qū),程序必須小于100KByte。
固件的生成與分包
在Keil中,將程序的memory的起始地址設(shè)置為0x8001000,大小設(shè)置為0x19000。
同時(shí),設(shè)置運(yùn)行fromelf.exe,使得編譯程序時(shí)自動(dòng)生成用于固件升級的bin文件。

Keil設(shè)置
通過delphi將生成出來的bin文件讀入,并采用下述代碼進(jìn)行發(fā)包,加上協(xié)議頭以及CRC32的校驗(yàn)值。
pkgs := stream.Size div perpage;
rem := stream.Size mod perpage;
addr := 0;
if(rem > 0) then
begin
pkgs := pkgs + 1;
end;
strcrc := '';
for i:= 0 to (pkgs - 1) do
begin
curlen := perpage;
if((i + 1) * perpage > bytecount) then
begin
curlen := bytecount - (i * perpage);
end;
payload := inttohex(i* perpage, 8)+inttohex(curlen, 8); //
stream.Position := i * perpage;
k := 0;
tmpstr := '';
for j:= 0 to (curlen - 1) do
begin
stream.Read(val, 1);
if((k and 1) = 0) then
begin
tmpstr := inttohex(val, 2);
end
else
begin
tmpstr := inttohex(val, 2) + tmpstr;
end;
inc(k);
if((k and 1) = 0) then
begin
payload := payload + tmpstr;
end;
end;
tempcrc := crc(payload);
payload := tempcrc +payload;
payload := inttohex((2 + 4 + 4 + perpage) * i, 8) + payload;
payload := '01'+payload;
strcrc := strcrc + tempcrc;
payload := payload + crc(payload);;
str := header+'&msgid='+inttostr(msgid)+'&length='+inttostr(1 + 2 + 4 + 4 + 4 + 2+ curlen)+'&cmd='+payload;
inc(msgid);
strcommands.Add(str);
end;
固件的有效性驗(yàn)證以及升級的可靠性保證
整個(gè)固件包根據(jù)處理器的資源拆分為500個(gè)byte一個(gè)數(shù)據(jù)包;
每一個(gè)數(shù)據(jù)包都計(jì)算CRC32的數(shù)值并加入數(shù)據(jù)包中;
所以CRC32的數(shù)值再計(jì)算一遍CRC32數(shù)值并放入開始升級的命令之中;
控制器收到固件之后,重新計(jì)算500個(gè)byte的CRC32的計(jì)算值并與收到的CRC32值進(jìn)行比對,只有兩者相等時(shí)才存入暫存區(qū);
當(dāng)收到所有數(shù)據(jù)包時(shí),從暫存區(qū)中按包讀出固件,計(jì)算CRC32值與同時(shí)存儲的CRC32值比對,同時(shí)計(jì)算所有CRC32數(shù)值的CRC32數(shù)值,與開始升級的命令中所攜帶的數(shù)值比對。
只有所有CRC32的數(shù)值都相同的情況下,應(yīng)用程序才將升級程序的標(biāo)志位寫入到FLASH中,并重啟處理器進(jìn)入bootloader程序。
bootloader程序從FLASH中讀取到升級程序的標(biāo)志,則從暫存區(qū)中按包讀取數(shù)據(jù),進(jìn)行同樣的CRC32的驗(yàn)證過程,確保無誤的情況下,將暫存區(qū)中的固件搬移到程序區(qū)。
全部程序搬移完之后,再逐個(gè)字節(jié)比較暫存區(qū)以及程序區(qū)的內(nèi)容。
比對時(shí),再檢驗(yàn)CRC32是否正確。
只有CRC32數(shù)值正確并且與程序區(qū)的數(shù)據(jù)都相等的情況下,才清空升級程序的標(biāo)志,完成升級過程。
升級程序的步驟及代碼
步驟1:PC工具發(fā)送清空暫存區(qū)的命令,將暫存區(qū)的內(nèi)存都擦寫成0xff。
步驟2:PC工具發(fā)送寫固件數(shù)據(jù)包的命令,處理器收到之后,進(jìn)行有效性驗(yàn)證,并寫入暫存區(qū),重復(fù)該過程,完成整個(gè)固件的發(fā)送。
步驟3:PC工具發(fā)送開始升級的命令,處理器收到之后,再進(jìn)行一次有效性驗(yàn)證,并重啟,進(jìn)入bootloader程序。
步驟4:bootloader程序進(jìn)行有效性驗(yàn)證之后,將暫存區(qū)的固件搬移至程序區(qū),完成升級;
代碼如下:
U32 data, value, dataB;
U8 res = FALSE;
U8 flag;
U16 pointer;
U16 len;
U8 *ins = lins + AP_ID_HEX_BYTE;
if(fnCRC16_Check(lins, llen)){
len = 0;
if(llen >= AP_ID_HEX_BYTE){
len = llen - AP_ID_HEX_BYTE;
}
if(inscode == FM_OPERATECODE_START){
if(fmups.m_uchState == FM_STATE_IDLE){
if(len == (1 + 4 + FM_STARTCODE_LEN + 2 )){
if(fnFM_IsStartStopValid(&ins[1 + 4])){
data = (U32)ins[1] << 24;
data |= (U32)ins[2] << 16;
data |= (U32)ins[3] << 8;
data |= (U32)ins[4];
if(data < FLASH_ROM_SIZE_FIRMWARE){
fmups.m_uchState = FM_STATE_INIT;
fmups.m_ulLen = data;
fmups.m_uiTimer = FM_STATE_TIME;
res = TRUE;
}
}
}
}
}
else if(inscode == FM_OPERATECODE_DOWNLOAD){
if(fmups.m_uchState == FM_STATE_DOWNLOAD){
if((len > (1 + 4 + 2 + 4 + 4 + 2))
&& (len <= (1 + 4 + 2 + 4 + 4 + 2 + FM_DOWNLOAD_EVERYMSG))){
data = (U32)ins[1] << 24;
data |= (U32)ins[2] << 16;
data |= (U32)ins[3] << 8;
data |= (U32)ins[4];
value = (U32)ins[7] << 24;
value |= (U32)ins[8] << 16;
value |= (U32)ins[9] << 8;
value |= (U32)ins[10];
dataB = (U32)ins[11] << 24;
dataB |= (U32)ins[12] << 16;
dataB |= (U32)ins[13] << 8;
dataB |= (U32)ins[14];
flag = TRUE;
if(value != fmups.m_uchPointer){
flag = FALSE;
if((value + dataB) == fmups.m_uchPointer){
res = TRUE;
}
}
if(data >= (FLASH_ROM_SIZE_FIRMWARE - (FM_DOWNLOAD_EVERYMSG + 4+ 4 + 2))){
flag = FALSE;
}
if(dataB > FM_DOWNLOAD_EVERYMSG){
flag = FALSE;
}
if((fmups.m_uchPointer + dataB) > fmups.m_ulLen){
flag = FALSE;
}
if(flag){
if(value == fmups.m_uchPointer){
res = fnFL_WriteBytesAndCheck(data + FLASH_ROM_ADDR_FIRMWARE, (2 + 4 + 4 + dataB), &ins[5]);
if(res){
fmups.m_uchPointer += dataB;
}
}else{
res = TRUE;
}
}
}
}
}
else if(inscode == FM_OPERATECODE_STOP){
if(len == (1 + 2 + 2 + FM_STARTCODE_LEN)){
if(fnFM_IsStartStopValid(&ins[3])){
if(fmups.m_uchState == FM_STATE_COMPLETE){
if(fnFM_Check(ins[1], ins[2])){
res = fnFM_ProCon(ins[1], ins[2]);
if(res){
fmups.m_uchReStartTimer = 10;
}
}
}
}
}
}
else if(inscode == FM_OPERATECODE_RESET){
if(len == (1 + 2 + FM_STARTCODE_LEN)){
if(fnFM_IsStartStopValid(&ins[1])){
res = TRUE;
fmups.m_uchState = FM_STATE_IDLE;
fmups.m_uiTimer = 0;
}
}
}
}
ack[0] = inscode | 0x80;
ack[1] = res;
ack[2] = 0 ;
pointer = 3;
return(pointer);

- 婁底物聯(lián)網(wǎng)卡嵌入式(什么是物聯(lián)網(wǎng)卡,有什么用)
- 婁底南京物聯(lián)網(wǎng)卡限制(如何看待南京大眾書局的聯(lián)名卡事件)
- 婁底自動(dòng)售貨機(jī)物聯(lián)卡的核心功能與優(yōu)勢
- 婁底物聯(lián)卡:解鎖自動(dòng)售貨機(jī)的智能進(jìn)化密碼
- 婁底物聯(lián)卡:解鎖自動(dòng)售貨機(jī)智能化的關(guān)鍵密鑰
- 婁底重構(gòu)無人零售的智慧引擎(物聯(lián)卡)
- 婁底智能終端革命:解碼自動(dòng)售貨機(jī)重構(gòu)零售業(yè)的底層邏輯
- 婁底物聯(lián)卡賦能自動(dòng)售貨機(jī):智能化升級與價(jià)值釋放
- 婁底自動(dòng)售貨機(jī)物聯(lián)卡:智能化、場景化與生態(tài)化演進(jìn)?
- 婁底自動(dòng)售貨機(jī)的數(shù)字化躍遷引擎?
- 婁底哪些公司要用物聯(lián)網(wǎng)卡(什么是物聯(lián)網(wǎng)卡,有什么用)
- 婁底華為物聯(lián)網(wǎng)卡網(wǎng)速慢(華為手機(jī)流量網(wǎng)速慢怎么辦)
- 婁底醫(yī)院需要物聯(lián)網(wǎng)卡嗎(什么是物聯(lián)網(wǎng)卡,與手機(jī)卡有什么區(qū)別,能用在手機(jī)上嗎)
- 婁底智能穿戴物聯(lián)卡行業(yè)綜合解決方案
- 婁底安防監(jiān)控物聯(lián)卡行業(yè)綜合解決方案:智能化時(shí)代的“安全守護(hù)者”
- 婁底物聯(lián)卡是如何盤活自動(dòng)售貨機(jī)市場的?
- 婁底車聯(lián)網(wǎng)物聯(lián)卡行業(yè)綜合解決方案:驅(qū)動(dòng)智慧出行的“數(shù)字紐帶”
- 婁底POS機(jī)物聯(lián)卡行業(yè)綜合解決方案:驅(qū)動(dòng)智能支付場景的數(shù)字化轉(zhuǎn)型
- 婁底智慧林業(yè):構(gòu)建全鏈路數(shù)字化生態(tài)系統(tǒng)的創(chuàng)新實(shí)踐
- 婁底智慧農(nóng)業(yè):重塑現(xiàn)代農(nóng)業(yè)的數(shù)字化革命?