基于STM32的BootLoader的OTA远程升级

嵌入式大杂烩 2021-08-31 21:51

关注「嵌入式大杂烩」,选择「星标公众号」一起进步!

上次发过SD卡的Bootloader离线升级后,应大家的要求,这次就讲一下STM32的OTA远程升级。

OTA又叫空中下载技术,是通过移动通信的空中接口实现对移动终端设备数据进行远程管理的技术,还能提供移动化的新业务下载功能。

要实现OTA功能,至少需要两块设备,分别是服务器与客户端。服务器只有一个,客户端可有多个。服务器通过串口与PC机连接,需要下载的镜像文件存放于PC机,命令执行器给服务器发命令及镜像文件。首先命令执行器控制服务器广播当前可用的镜像文件信息,客户端收到信息后进行对比,若有与自身相匹配的镜像,则向服务器请求数据。服务器收到请求后向命令执行器索取固定大小的块,再点对点传送给客户端。镜像传输完毕后,客户端进行校验,完成后发送终止信号。

一. 升级方式的对比

OTA升级与平时用到的SD卡升级、串口升级等等大体原理上是一样的,都是对MCU的Flash进行操作而已。

收到升级指令——>MCU复位或者跳转到Boot程序区——>擦除对应的Flash区域——>获取APP数据——>写入FLASH数据——>校验——>跳转到APP应用程序区

OTA与其他本地升级的区别就是:获取数据的方式不同。比如串口升级,就是通过上位机传输到MCU串口上的数据;SD卡升级,就是通过读取SD卡,把程序通过SPI传输到MCU上;而OTA升级,就是通过带无线传输的模块,把程序传输到MCU上。例如:蓝牙、Wifi、GSM等等。不过大部分的无线模块,通过串口把数据传输到MCU上的,只是服务端不再是PC端了,而是网络服务器。

二. 硬件选择

MCU我这里选用的是STM32F030F4P6的芯片,16K的Flash,应该是ST产品中Flash空间比较小的一种,为的就是体现一下小容量的单片机也可以进行OTA升级。

无线模块我使用的是ESP-8266,WIfi传输方式,应该也是比较大众化的一款模组。(TTL串口连接MCU)

OTA相关的硬件没有了,剩下的无所谓,都是其他功能的,最好有个LED灯,可以明显的看出是否升级成功。





三. 网络服务器的选择

网络服务器多种多样,常用的有阿里云、百度云、腾讯云、移动云等等,有条件的,还可以使用自己的服务器。总之需要实现:网络服务器可以与我们的无线模块进行大数据通信

我这里选用的是OneNet移动云(OTA服务之前是免费,现在是前100个设备免费,之后每增加一个设备1元钱永久),我感觉OneNet相对于阿里云较为简单,没有阿里云那么繁琐,不过阿里云还是比OneNet更专业一点(个人见解),其他的没有用过,大家都可以去试试。

四. 网络服务器的传输方式

我这里使用的是OneNet的服务器,它的OTA服务是通过Http协议进行传输的,有对应的API,我们可以通过OneNet释放的API去访问OTA服务。

五. OTA升级流程

OneNet的OTA升级流程主要为6步:


1. 上报版本号---客户端(MCU)上报当前的一个版本号
2. 检测升级任务---检查服务器是否有待升级的版本
3. 检测Token有效性---检查Token密钥,可省略
4. 下载固件---应用程序传输
5. 上报升级状态---上报服务端升级是否成功,不成功有对应的响应码



六. OneNet服务端配置

1.首先注册OneNet的账号,进入开发者中心,在导航栏选择全部产品->远程升级OTA板块。




2.进入远程升级OTA界面,选择需要升级的模块;然后点击右上角的添加升级包按钮。FOTA升级:对设备中的模组进行升级。SOTA升级:对设备中的应用程序进行升级,我这里选用的是SOTA,因为我要对MCU的应用程序升级。



3.在添加升级包对话框中,输入固件信息,上传固件包文件。产品选你要升级的设备,全部设备也可以;厂商名称选其他,主要是与之后发的对应上即可;模组型号同理;目标版本是你要更新到的版本号,比如你现在是V01,你这里添加的固件是V02的,这个版本号就要填V02;然后上传升级包,只支持Bin和压缩包格式的。



4.点击验证升级按钮,选择验证类型(完整包或者差分包),选择进行测试升级的设备,进行验证。一般跳过验证就行,我这里选的是整包,差分包原理一样。


 

5.单击升级设备列表,进入升级队列模块,在右上角单击添加升级设备按钮,新增设备升级任务。在添加待升级设备对话框中输入对应参数值。初始版本:就是升级前的版本,也是上次升级的版本;升级范围就是你需要给哪些设备升级;升级时机:就是立即升级或是定时在什么时段升级;重试策略:不重试就是如果升级失败就完事了,重试那就失败了还能重试;信号强度和剩余电量只是一个信息的接口,有需要的可以读取来用。

 


6.上述完成后,会出现“待升级”的设备,服务器这边就算配置完了,后续要我们M客户端进行操作了。



七.客户端(MCU)API访问服务端进行OTA升级

无线模组用的是ESP8266,由于OneNet的OTA服务用的是HTTP协议,但是ESP8266没有HTTP协议,所以我使用TCP协议,封装成HTTP的报文格式。

1.ESP8266初始化;连接Wifi,AP_SSID,AP_PASS是WiFi的账号和密码;SERVER_IP和SERVER_PORT是OneNet的Ip和端口号。


#define SERVER_IP "183.230.40.50"#define SERVER_PORT 80uint8_t pro = 0;uint8_t ESP8266_Init(void){        switch(pro)        {                case 0 :                         //printf("+++");                        Uart2_Send("+++");                        Delay_S(2);                        if(ESP8266_SoftReset(50) == 0)                                pro = 1;                        break;                case 1 :                         if(ESP8266_AT_Send("ATE0\r\n",10) == 0)                                pro = 2;                        break;                case 2 :                        if(ESP8266_AT_Send("AT+CWMODE=1\r\n",50) == 0)                //设置8266为STA模式                                pro = 3;                        break;                case 3 :                        if(ESP8266_ConnectionAP(AP_SSID,AP_PASS,200) == 0)                //8266连接AP                                pro = 4;                        break;                case 4 :                        if(ESP8266_AT_Send("AT+CIPMODE=1\r\n",50) == 0)                //8266开启透传模式                                pro = 5;                        break;                case 5 :                        if(ESP8266_Connect_Server(SERVER_IP,SERVER_PORT,50) == 0)        //8266连接TCP服务器                        {                                pro = 0;                                //USART1_Clear();                        //清除串口数据                                return 1;                        }                                                        break;        }        return 0;}


2.上报版本号;dev_id是设备ID,authorization是鉴权参数,ver要上报的版本号,timeout发送超时时间。



//上报版本号uint8_t Report_Version(char *dev_id,char *authorization,char *ver,uint16_t timeout){        uint16_t time=0;        char send_buf[296];        USART1_Clear();                        //清除串口数据                snprintf(send_buf, sizeof(send_buf), "POST /ota/device/version?dev_id=%s HTTP/1.1\r\n"        "Authorization:%s\r\n"        "Host:ota.heclouds.com\r\n"        "Content-Type:application/json\r\n"        "Content-Length:%d\r\n\r\n"        "{\"s_version\":\"%s\"}",        dev_id, authorization, strlen(ver) + 16, ver);              Uart2_Send(send_buf);                while(time<timeout)        {                if(strstr( (const char *)usart_info.buf , (const char *)"\"errno\":0"))                        break;                Delay_Ms(100);                time++;              }        if(time>=timeout)                return 1;                       else                 return 0;            }


3.检查升级任务;dev_id是设备ID,authorization是鉴权参数,cur_version是当前的版本号,timeout发送超时时间



//检查升级任务uint8_t Detect_Task(char *dev_id,char *cur_version,char *authorization,uint16_t timeout){        uint16_t time=0;        char send_buf[280];        USART1_Clear();                        //清除串口数据                snprintf(send_buf, sizeof(send_buf), "GET /ota/south/check?"        "dev_id=%s&manuf=100&model=10001&type=2&version=%s&cdn=false HTTP/1.1\r\n"        "Authorization:%s\r\n"        "Host:ota.heclouds.com\r\n\r\n",        dev_id, cur_version,authorization);             Uart2_Send(send_buf);        while(time<timeout)        {                if(strstr( (const char *)usart_info.buf , (const char *)"\"errno\":0"))                        break;                Delay_Ms(100);                time++;              }        if(time>=timeout)                return 1;                       else                 return 0;            }


3.下载资源(我省略了"检查token有效"步骤);ctoken是上一步“检查升级任务”返回的Token,这个每次请求都不一样,所以注意要记录;size:平台返回的固件大小(字节);bytes_range:分片大小(字节)



/**************************************************************        函数名称:OTA_Download_Range**        函数功能:分片下载固件**        入口参数:token:平台返回的Token*                                                size:平台返回的固件大小(字节)*                                                bytes_range:分片大小(字节)**        返回参数:0-成功        其他-失败**        说明:*************************************************************/uint8_t Download_Task(char *ctoken,unsigned int size, const unsigned short bytes_range,uint16_t timeout){        MD5_CTX md5_ctx;                                                                                        //MD5相关变量        unsigned char md5_t[16];        char md5_t1[16];        char md5_result[40];        uint16_t time=0;        char *data_ptr = NULL;        char send_buf[256];        unsigned char flash_buf[OTA_BUFFER_SIZE];                        //flash读写缓存        unsigned int bytes = 0;        MD5_Init(&md5_ctx);        Flash_cashu();        while(bytes < size)        {                time = 0;                memset(send_buf, 0, sizeof(send_buf));                USART1_Clear();                        //清除串口数据                          snprintf(send_buf, sizeof(send_buf), "GET /ota/south/download/"                "%s HTTP/1.1\r\n"                "Range:bytes=%d-%d\r\n"                "Host:ota.heclouds.com\r\n\r\n",                ctoken, bytes, bytes + bytes_range - 1);                      Uart2_Send(send_buf);                //----------------------------------------------------等待数据---------------------------------------------------------------------                while(time < 30)                {                        if(usart_info.buf[0] != 0)                                break;                        Delay_Ms(100);                        time++;                }                                if(time <= 29)                {                        Delay_Ms(500);                        //----------------------------------------------------跳过HTTP报文头、找到固件数据--------------------------------------------------                        data_ptr = strstr( (const char *)usart_info.buf, "Range");                        data_ptr = strstr(data_ptr, "\r\n");                        data_ptr += 4;                                                //----------------------------------------------------将固件数据写入缓存和闪存-----------------------------------------------------                        if(data_ptr != NULL)                        {                                if((size - bytes) >= OTA_BUFFER_SIZE)                                {                                        memcpy(flash_buf + (bytes % OTA_BUFFER_SIZE), data_ptr, bytes_range);                                        STMFLASH_Write_NoCheck(FLASH_APP1_ADDR + bytes,(uint16_t *)flash_buf,OTA_BUFFER_SIZE / 2);                                        bytes = bytes + OTA_BUFFER_SIZE;                                                                                MD5_Update(&md5_ctx, (unsigned char *)data_ptr, bytes_range);                                }                                else                                {                                        memcpy(flash_buf + (bytes % OTA_BUFFER_SIZE), data_ptr, size - bytes);                                        STMFLASH_Write_NoCheck(FLASH_APP1_ADDR + bytes , (uint16_t *)flash_buf , (size % OTA_BUFFER_SIZE) / 2);                                                                                                MD5_Update(&md5_ctx, (unsigned char *)data_ptr, size - bytes);                                                                                bytes = size;                                }                        }                }        }        //----------------------------------------------------MD校验比对------------------------------------------------------------------        memset(md5_result, 0, sizeof(md5_result));        MD5_Final(&md5_ctx, md5_t);        for(int i = 0; i < 16; i++)        {                if(md5_t[i] <= 0x0f)                        sprintf(md5_t1, "0%x", md5_t[i]);                else                        sprintf(md5_t1, "%x", md5_t[i]);                                strcat(md5_result, md5_t1);        }        if(strcmp(md5_result, ota_info.md5) == 0)                                return 0;        else                return 1; }


4.上报升级状态;这一步由于时间问题,我也省略了,总之程序已经下载到MCU上了,只是没有通知服务器而已,大家最好还是加上这一步。





5.main函数循环;


char rrr;                char dev_id[] = {"640600857"};          char Authorization[] = {"version=2018-10-31&res=products%2F378414&et=1735660800&method=sha1&sign=9EgY%2Bk4r%2BlvCooIGf1ghtQFC0%2Bc%3D"};
char Version[] = {"V10"};


while(1)        {                switch(pro)                {                        case 1 :        //上报版本                                if(Report_Version(dev_id,Authorization,Version,10) == 0)                                        pro++;                                break;                        case 2 :        //检查任务                                if(Detect_Task(dev_id,Version,Authorization,50) == 0)                                        pro++;                                break;                        case 3 :        //接收token、size、md5信息                                rrr = json_get_value((char *)usart_info.buf,"token",ota_info.token);                                rrr = json_get_value((char *)usart_info.buf,"size",ota_info.csize);                                                        rrr = json_get_value((char *)usart_info.buf,"md5",ota_info.md5);                                ota_info.size = atoi(ota_info.csize);                                        pro++;                        break;                        case 4 :        //进行下载                                res = Download_Task(ota_info.token,ota_info.size,OTA_BUFFER_SIZE,10);                                if(res == 0)        //校验成功                                {                                        pro++;                                }                                else if(res == 1)                //校验失败                                {                                        pro = 1;                                }                                                break;                        case 5 :        //Flash写入升级完成的标志位                                USART1_Clear();                                STMFLASH_Unlock();                                STMFLASH_WriteHalfWord(FLASH_APP1_ADDR - 0x64, 0xFF02);//写入数据                                STMFLASH_Lock();                                pro++;                        break;                        case 6 :        //复位或者跳转到APP                                Sys_Soft_Reset();                                //iap_load_app(FLASH_APP1_ADDR);                        break;                }        }


下图是我升级的历史




八.注意事项


1.鉴权参数是需要自己去算的,具体算法请见我之前写的帖子和附件(https://bbs.21ic.com/icview-3144666-1-1.html


2.由于用的是STM32F030F4P6,RAM也非常小,所以局部变量和全局变量的数组不要超过4K,堆栈大小有改动。当前用内存管理的话就不用了。



3.OTA校验用的是MD5,需要把MD5的算法移植一下。
4.别的想不到了,太长时间了。

总结:


OTA的方法只是我个人的理解,可能有的地方不正确,欢迎大家指点。BootLoader代码也是很早之前写过的一个Demo,最简化的,传输协议、加密、升级失败的操作、回滚等等都没有涉及,只是一个OTA演示的例子,代码水平有点差,大家将就的看,参考一下就可以了哈,感谢!


相关附件请点击“阅读原文”内下载


END


本文系21ic论坛蓝V作者小叶三千原创撰写


往期干货:

往期推荐



干货 | 浅析程序开机自启动

程序如何运行?编译、链接、装入?

串口通信 | 简单明了的基础知识

一种无OS的MCU实用软件框架



嵌入式大杂烩 专注于嵌入式技术,包括但不限于C/C++、嵌入式、物联网、Linux等编程学习笔记,同时,内包含大量的学习资源。欢迎关注,一同交流学习,共同进步!
评论 (0)
  • 在CAN总线分析软件领域,当CANoe不再是唯一选择时,虹科PCAN-Explorer 6软件成为了一个有竞争力的解决方案。在现代工业控制和汽车领域,CAN总线分析软件的重要性不言而喻。随着技术的进步和市场需求的多样化,单一的解决方案已无法满足所有用户的需求。正是在这样的背景下,虹科PCAN-Explorer 6软件以其独特的模块化设计和灵活的功能扩展,为CAN总线分析领域带来了新的选择和可能性。本文将深入探讨虹科PCAN-Explorer 6软件如何以其创新的模块化插件策略,提供定制化的功能选
    虹科汽车智能互联 2025-04-28 16:00 181浏览
  • 网约车,真的“饱和”了?近日,网约车市场的 “饱和” 话题再度引发热议。多地陆续发布网约车风险预警,提醒从业者谨慎入局,这背后究竟隐藏着怎样的市场现状呢?从数据来看,网约车市场的“过剩”现象已愈发明显。以东莞为例,截至2024年12月底,全市网约车数量超过5.77万辆,考取网约车驾驶员证的人数更是超过13.48万人。随着司机数量的不断攀升,订单量却未能同步增长,导致单车日均接单量和营收双双下降。2024年下半年,东莞网约出租车单车日均订单量约10.5单,而单车日均营收也不容乐
    用户1742991715177 2025-04-29 18:28 214浏览
  • 在智能硬件设备趋向微型化的背景下,语音芯片方案厂商针对小体积设备开发了多款超小型语音芯片方案,其中WTV系列和WT2003H系列凭借其QFN封装设计、高性能与高集成度,成为微型设备语音方案的理想选择。以下从封装特性、功能优势及典型应用场景三个方面进行详细介绍。一、超小体积封装:QFN技术的核心优势WTV系列与WT2003H系列均提供QFN封装(如QFN32,尺寸为4×4mm),这种封装形式具有以下特点:体积紧凑:QFN封装通过减少引脚间距和优化内部结构,显著缩小芯片体积,适用于智能门铃、穿戴设备
    广州唯创电子 2025-04-30 09:02 211浏览
  • 随着电子元器件的快速发展,导致各种常见的贴片电阻元器件也越来越小,给我们分辨也就变得越来越难,下面就由smt贴片加工厂_安徽英特丽就来告诉大家如何分辨的SMT贴片元器件。先来看看贴片电感和贴片电容的区分:(1)看颜色(黑色)——一般黑色都是贴片电感。贴片电容只有勇于精密设备中的贴片钽电容才是黑色的,其他普通贴片电容基本都不是黑色的。(2)看型号标码——贴片电感以L开头,贴片电容以C开头。从外形是圆形初步判断应为电感,测量两端电阻为零点几欧,则为电感。(3)检测——贴片电感一般阻值小,更没有“充放
    贴片加工小安 2025-04-29 14:59 209浏览
  • 文/郭楚妤编辑/cc孙聪颖‍越来越多的企业开始蚕食动力电池市场,行业“去宁王化”态势逐渐明显。随着这种趋势的加强,打开新的市场对于宁德时代而言至关重要。“我们不希望被定义为电池的制造者,而是希望把自己称作新能源产业的开拓者。”4月21日,在宁德时代举行的“超级科技日”发布会上,宁德时代掌门人曾毓群如是说。随着宁德时代核心新品骁遥双核电池的发布,其搭载的“电电增程”技术也走进业界视野。除此之外,经过近3年试水,宁德时代在换电业务上重资加码。曾毓群认为换电是一个重资产、高投入、长周期的产业,涉及的利
    华尔街科技眼 2025-04-28 21:55 157浏览
  • 文/Leon编辑/cc孙聪颖‍2023年,厨电行业在相对平稳的市场环境中迎来温和复苏,看似为行业增长积蓄势能。带着对市场向好的预期,2024 年初,老板电器副董事长兼总经理任富佳为企业定下双位数增长目标。然而现实与预期相悖,过去一年,这家老牌厨电企业不仅未能达成业绩目标,曾提出的“三年再造一个老板电器”愿景,也因市场下行压力面临落空风险。作为“企二代”管理者,任富佳在掌舵企业穿越市场周期的过程中,正面临着前所未有的挑战。4月29日,老板电器(002508.SZ)发布了2024年年度报告及2025
    华尔街科技眼 2025-04-30 12:40 204浏览
  • 贞光科技代理品牌紫光国芯的车规级LPDDR4内存正成为智能驾驶舱的核心选择。在汽车电子国产化浪潮中,其产品以宽温域稳定工作能力、优异电磁兼容性和超长使用寿命赢得市场认可。紫光国芯不仅确保供应链安全可控,还提供专业本地技术支持。面向未来,紫光国芯正研发LPDDR5车规级产品,将以更高带宽、更低功耗支持汽车智能化发展。随着智能网联汽车的迅猛发展,智能驾驶舱作为人机交互的核心载体,对处理器和存储器的性能与可靠性提出了更高要求。在汽车电子国产化浪潮中,贞光科技代理品牌紫光国芯的车规级LPDDR4内存凭借
    贞光科技 2025-04-28 16:52 241浏览
  • 浪潮之上:智能时代的觉醒    近日参加了一场课题的答辩,这是医疗人工智能揭榜挂帅的国家项目的地区考场,参与者众多,围绕着医疗健康的主题,八仙过海各显神通,百花齐放。   中国大地正在发生着激动人心的场景:深圳前海深港人工智能算力中心高速运转的液冷服务器,武汉马路上自动驾驶出租车穿行的智慧道路,机器人参与北京的马拉松竞赛。从中央到地方,人工智能相关政策和消息如雨后春笋般不断出台,数字中国的建设图景正在智能浪潮中徐徐展开,战略布局如同围棋
    广州铁金刚 2025-04-30 15:24 180浏览
  • 一、gao效冷却与控温机制‌1、‌冷媒流动设计‌采用低压液氮(或液氦)通过毛细管路导入蒸发器,蒸汽喷射至样品腔实现快速冷却,冷却效率高(室温至80K约20分钟,至4.2K约30分钟)。通过控温仪动态调节蒸发器加热功率,结合温度传感器(如PT100铂电阻或Cernox磁场不敏感传感器),实现±0.01K的高精度温度稳定性。2、‌宽温区覆盖与扩展性‌标准温区为80K-325K,通过降压选件可将下限延伸至65K(液氮模式)或4K(液氦模式)。可选配475K高温模块,满足材料在ji端温度下的性能测试需求
    锦正茂科技 2025-04-30 13:08 213浏览
  • 你是不是也有在公共场合被偷看手机或笔电的经验呢?科技时代下,不少现代人的各式机密数据都在手机、平板或是笔电等可携式的3C产品上处理,若是经常性地需要在公共场合使用,不管是工作上的机密文件,或是重要的个人信息等,民众都有防窃防盗意识,为了避免他人窥探内容,都会选择使用「防窥保护贴片」,以防止数据外泄。现今市面上「防窥保护贴」、「防窥片」、「屏幕防窥膜」等产品就是这种目的下产物 (以下简称防窥片)!防窥片功能与常见问题解析首先,防窥片最主要的功能就是用来防止他人窥视屏幕上的隐私信息,它是利用百叶窗的
    百佳泰测试实验室 2025-04-30 13:28 290浏览
我要评论
0
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦