<ul id="i4ug0"></ul>
  • <ul id="i4ug0"></ul>
    <ul id="i4ug0"></ul> <ul id="i4ug0"><sup id="i4ug0"></sup></ul><ul id="i4ug0"></ul>
  • 智宇物聯(lián) 專(zhuān)注于提供高穩(wěn)定、高速率的三網(wǎng)物聯(lián)網(wǎng)卡

    秀山物聯(lián)網(wǎng)設(shè)備的固件遠(yuǎn)程升級(jí)方案以及軟件代碼

    • 作者:智宇物聯(lián)
    • 發(fā)表時(shí)間:2022年10月13日
    • 來(lái)源:智宇物聯(lián)

    作為通用的物聯(lián)網(wǎng)設(shè)備,由于用戶需求各不相同,不少用戶有一些個(gè)性化的定制要求;

    設(shè)備所對(duì)接的傳感器協(xié)議也多種多樣,比如Modbus讀寫(xiě)參數(shù)的數(shù)據(jù)類(lèi)型,某些物理量需要特殊的數(shù)據(jù)類(lèi)型;

    甚至可能還存在意想不到的bug。

    因此,遠(yuǎn)程升級(jí)的功能對(duì)于設(shè)備來(lái)說(shuō)必不可少。

    遠(yuǎn)程固件升級(jí)需要解決以下問(wèn)題:

    1)設(shè)備的遠(yuǎn)程訪問(wèn)

    當(dāng)設(shè)備被安裝于局域網(wǎng)內(nèi)部時(shí),位于遠(yuǎn)程的固件升級(jí)軟件工具無(wú)法穿透路由器訪問(wèn)設(shè)備。

    2)固件的分包以及傳送

    由于設(shè)備的處理器資源有限,無(wú)法移植開(kāi)源的http、FTP等協(xié)議棧,無(wú)法通過(guò)http、FTP等協(xié)議從服務(wù)器上下載固件,而需要自己實(shí)現(xiàn)代碼,采用TCP協(xié)議進(jìn)行固件包的發(fā)送;

    而且對(duì)于幾十k甚至上百k的固件,需要將固件拆成幾百個(gè)字節(jié)的數(shù)據(jù)包,逐一發(fā)給設(shè)備;

    4)固件的有效性檢驗(yàn)

    固件在傳輸過(guò)程中,難免會(huì)出現(xiàn)錯(cuò)誤。

    比如WiFi模塊,或者是ethernet模塊將數(shù)據(jù)通過(guò)uart轉(zhuǎn)發(fā)給處理器時(shí),如果有干擾、數(shù)據(jù)可能被破壞;

    或者是處理器太忙,來(lái)不及接收數(shù)據(jù),導(dǎo)致固件包丟失數(shù)據(jù);

    如果不對(duì)固件進(jìn)行有效性檢驗(yàn),將被破壞的固件升級(jí)進(jìn)控制器,會(huì)導(dǎo)致設(shè)備變磚而無(wú)法使用;

    5)bootloader程序

    bootloader程序需要下載固件的有效性檢驗(yàn),程序的擦除、固件數(shù)據(jù)從備份區(qū)到程序區(qū)的搬移。

    6)處理器的固件升級(jí)軟件實(shí)現(xiàn)

    軟件需要實(shí)現(xiàn)數(shù)據(jù)包接收,固件有效性驗(yàn)證、存儲(chǔ),數(shù)據(jù)應(yīng)答等。

    遠(yuǎn)程固件升級(jí)系統(tǒng)架構(gòu)

    物聯(lián)網(wǎng)設(shè)備的固件遠(yuǎn)程升級(jí)方案以及軟件代碼

    遠(yuǎn)程固件升級(jí)系統(tǒng)架構(gòu)

    設(shè)備作為T(mén)CP客戶端連接到云服務(wù)器上的TCP服務(wù)端,定時(shí)發(fā)送心跳,維護(hù)連接,從而實(shí)現(xiàn)TCP的長(zhǎng)鏈接。

    在PC電腦上開(kāi)發(fā)遠(yuǎn)程升級(jí)工具,作為T(mén)CP客戶端與云服務(wù)器上的TCP服務(wù)器建立連接;

    當(dāng)需要遠(yuǎn)程升級(jí)時(shí),通過(guò)PC工具向云服務(wù)器發(fā)送消息,所發(fā)消息中包括了遠(yuǎn)程設(shè)備的設(shè)備編號(hào),以及PC工具的設(shè)備編號(hào);

    服務(wù)器收到消息之后,根據(jù)消息中的目標(biāo)設(shè)備編號(hào),從其維護(hù)的長(zhǎng)鏈接中找到與該編號(hào)對(duì)應(yīng)的鏈接,通過(guò)該鏈接向設(shè)備轉(zhuǎn)發(fā)該消息;

    設(shè)備收到消息之后,對(duì)消息中的固件包進(jìn)行有效性驗(yàn)證,如果有效,則寫(xiě)入到固件暫存區(qū),并回復(fù)成功,否則回復(fù)失敗。

    一些設(shè)計(jì)要點(diǎn)

    處理器的存儲(chǔ)空間安排:

    以STM32F103RCT6為例,該處理器有256KByte的FLASH空間;

    4KByte的空間用于bootloader程序。

    52KByte用于存儲(chǔ)用戶數(shù)據(jù);

    剩余的FLASH空間一半作為程序存儲(chǔ)區(qū),一半作為固件暫存區(qū),程序必須小于100KByte。

    固件的生成與分包

    在Keil中,將程序的memory的起始地址設(shè)置為0x8001000,大小設(shè)置為0x19000。

    同時(shí),設(shè)置運(yùn)行fromelf.exe,使得編譯程序時(shí)自動(dòng)生成用于固件升級(jí)的bin文件。

    物聯(lián)網(wǎng)設(shè)備的固件遠(yuǎn)程升級(jí)方案以及軟件代碼

    Keil設(shè)置

    通過(guò)delphi將生成出來(lái)的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)證以及升級(jí)的可靠性保證

    整個(gè)固件包根據(jù)處理器的資源拆分為500個(gè)byte一個(gè)數(shù)據(jù)包;

    每一個(gè)數(shù)據(jù)包都計(jì)算CRC32的數(shù)值并加入數(shù)據(jù)包中;

    所以CRC32的數(shù)值再計(jì)算一遍CRC32數(shù)值并放入開(kāi)始升級(jí)的命令之中;

    控制器收到固件之后,重新計(jì)算500個(gè)byte的CRC32的計(jì)算值并與收到的CRC32值進(jìn)行比對(duì),只有兩者相等時(shí)才存入暫存區(qū);

    當(dāng)收到所有數(shù)據(jù)包時(shí),從暫存區(qū)中按包讀出固件,計(jì)算CRC32值與同時(shí)存儲(chǔ)的CRC32值比對(duì),同時(shí)計(jì)算所有CRC32數(shù)值的CRC32數(shù)值,與開(kāi)始升級(jí)的命令中所攜帶的數(shù)值比對(duì)。

    只有所有CRC32的數(shù)值都相同的情況下,應(yīng)用程序才將升級(jí)程序的標(biāo)志位寫(xiě)入到FLASH中,并重啟處理器進(jìn)入bootloader程序。

    bootloader程序從FLASH中讀取到升級(jí)程序的標(biāo)志,則從暫存區(qū)中按包讀取數(shù)據(jù),進(jìn)行同樣的CRC32的驗(yàn)證過(guò)程,確保無(wú)誤的情況下,將暫存區(qū)中的固件搬移到程序區(qū)。

    全部程序搬移完之后,再逐個(gè)字節(jié)比較暫存區(qū)以及程序區(qū)的內(nèi)容。

    比對(duì)時(shí),再檢驗(yàn)CRC32是否正確。

    只有CRC32數(shù)值正確并且與程序區(qū)的數(shù)據(jù)都相等的情況下,才清空升級(jí)程序的標(biāo)志,完成升級(jí)過(guò)程。

    升級(jí)程序的步驟及代碼

    步驟1:PC工具發(fā)送清空暫存區(qū)的命令,將暫存區(qū)的內(nèi)存都擦寫(xiě)成0xff。

    步驟2:PC工具發(fā)送寫(xiě)固件數(shù)據(jù)包的命令,處理器收到之后,進(jìn)行有效性驗(yàn)證,并寫(xiě)入暫存區(qū),重復(fù)該過(guò)程,完成整個(gè)固件的發(fā)送。

    步驟3:PC工具發(fā)送開(kāi)始升級(jí)的命令,處理器收到之后,再進(jìn)行一次有效性驗(yàn)證,并重啟,進(jìn)入bootloader程序。

    步驟4:bootloader程序進(jìn)行有效性驗(yàn)證之后,將暫存區(qū)的固件搬移至程序區(qū),完成升級(jí);

    代碼如下:

    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)設(shè)備的固件遠(yuǎn)程升級(jí)方案以及軟件代碼

    最新資訊
    最熱資訊
    主站蜘蛛池模板: 久久久久成人精品一区二区 | 成人免费午夜视频| 爽爽爽爽爽爽爽成人免费观看| 韩国免费一级成人毛片| 欧美成人久久久| 国产成人av大片大片在线播放| 99久久精品国产一区二区成人 | 日韩精品成人一区二区三区| 成人欧美日韩一区二区三区| 成人在线播放av| 色老头成人免费综合视频| 国产午夜成人AV在线播放| 亚洲第一成人在线| 九九九国产精品成人免费视频| 美国成人a免费毛片| 国内外成人在线视频| 国产成人在线网站| 日韩精品无码成人专区| 久久成人无码国产免费播放| 成人av在线一区二区三区| 中文国产成人精品久久一区| 国产成人精品97| 国产成人教育视频在线观看| 欧美成人免费在线| 精品国产成人亚洲午夜福利| 亚洲成人在线网| 国产成人免费片在线视频观看| 成人小视频在线观看| 欧美xxxx成人免费网站| 91成人精品视频| 久久久久亚洲精品成人网小说| 四虎在线成人免费网站| 国产成人愉拍精品| 亚洲精品成人网久久久久久| 国产成人免费网站app下载| 国产成人综合久久亚洲精品| 成人综合激情另类小说| 成人午夜福利视频| 国产成人无码一二三区视频| 国产成人精品一区二区秒拍| 国产成人最新毛片基地|