基于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等编程学习笔记,同时,内包含大量的学习资源。欢迎关注,一同交流学习,共同进步!
评论
  • 2024年是很平淡的一年,能保住饭碗就是万幸了,公司业绩不好,跳槽又不敢跳,还有一个原因就是老板对我们这些员工还是很好的,碍于人情也不能在公司困难时去雪上加霜。在工作其间遇到的大问题没有,小问题还是有不少,这里就举一两个来说一下。第一个就是,先看下下面的这个封装,你能猜出它的引脚间距是多少吗?这种排线座比较常规的是0.6mm间距(即排线是0.3mm间距)的,而这个规格也是我们用得最多的,所以我们按惯性思维来看的话,就会认为这个座子就是0.6mm间距的,这样往往就不会去细看规格书了,所以这次的运气
    wuliangu 2025-01-21 00:15 204浏览
  • 数字隔离芯片是一种实现电气隔离功能的集成电路,在工业自动化、汽车电子、光伏储能与电力通信等领域的电气系统中发挥着至关重要的作用。其不仅可令高、低压系统之间相互独立,提高低压系统的抗干扰能力,同时还可确保高、低压系统之间的安全交互,使系统稳定工作,并避免操作者遭受来自高压系统的电击伤害。典型数字隔离芯片的简化原理图值得一提的是,数字隔离芯片历经多年发展,其应用范围已十分广泛,凡涉及到在高、低压系统之间进行信号传输的场景中基本都需要应用到此种芯片。那么,电气工程师在进行电路设计时到底该如何评估选择一
    华普微HOPERF 2025-01-20 16:50 78浏览
  • 现在为止,我们已经完成了Purple Pi OH主板的串口调试和部分配件的连接,接下来,让我们趁热打铁,完成剩余配件的连接!注:配件连接前请断开主板所有供电,避免敏感电路损坏!1.1 耳机接口主板有一路OTMP 标准四节耳机座J6,具备进行音频输出及录音功能,接入耳机后声音将优先从耳机输出,如下图所示:1.21.2 相机接口MIPI CSI 接口如上图所示,支持OV5648 和OV8858 摄像头模组。接入摄像头模组后,使用系统相机软件打开相机拍照和录像,如下图所示:1.3 以太网接口主板有一路
    Industio_触觉智能 2025-01-20 11:04 166浏览
  • 高速先生成员--黄刚这不马上就要过年了嘛,高速先生就不打算给大家上难度了,整一篇简单但很实用的文章给大伙瞧瞧好了。相信这个标题一出来,尤其对于PCB设计工程师来说,心就立马凉了半截。他们辛辛苦苦进行PCB的过孔设计,高速先生居然说设计多大的过孔他们不关心!另外估计这时候就跳出很多“挑刺”的粉丝了哈,因为翻看很多以往的文章,高速先生都表达了过孔孔径对高速性能的影响是很大的哦!咋滴,今天居然说孔径不关心了?别,别急哈,听高速先生在这篇文章中娓娓道来。首先还是要对各位设计工程师的设计表示肯定,毕竟像我
    一博科技 2025-01-21 16:17 108浏览
  • 临近春节,各方社交及应酬也变得多起来了,甚至一月份就排满了各式约见。有的是关系好的专业朋友的周末“恳谈会”,基本是关于2025年经济预判的话题,以及如何稳定工作等话题;但更多的预约是来自几个客户老板及副总裁们的见面,他们为今年的经济预判与企业发展焦虑而来。在聊天过程中,我发现今年的聊天有个很有意思的“点”,挺多人尤其关心我到底是怎么成长成现在的多领域风格的,还能掌握一些经济趋势的分析能力,到底学过哪些专业、在企业管过哪些具体事情?单单就这个一个月内,我就重复了数次“为什么”,再辅以我上次写的:《
    牛言喵语 2025-01-22 17:10 70浏览
  •     IPC-2581是基于ODB++标准、结合PCB行业特点而指定的PCB加工文件规范。    IPC-2581旨在替代CAM350格式,成为PCB加工行业的新的工业规范。    有一些免费软件,可以查看(不可修改)IPC-2581数据文件。这些软件典型用途是工艺校核。    1. Vu2581        出品:Downstream     
    电子知识打边炉 2025-01-22 11:12 82浏览
  • 本文介绍瑞芯微开发板/主板Android配置APK默认开启性能模式方法,开启性能模式后,APK的CPU使用优先级会有所提高。触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。源码修改修改源码根目录下文件device/rockchip/rk3562/package_performance.xml并添加以下内容,注意"+"号为添加内容,"com.tencent.mm"为AP
    Industio_触觉智能 2025-01-17 14:09 173浏览
  • 日前,商务部等部门办公厅印发《手机、平板、智能手表(手环)购新补贴实施方案》明确,个人消费者购买手机、平板、智能手表(手环)3类数码产品(单件销售价格不超过6000元),可享受购新补贴。每人每类可补贴1件,每件补贴比例为减去生产、流通环节及移动运营商所有优惠后最终销售价格的15%,每件最高不超过500元。目前,京东已经做好了承接手机、平板等数码产品国补优惠的落地准备工作,未来随着各省市关于手机、平板等品类的国补开启,京东将第一时间率先上线,满足消费者的换新升级需求。为保障国补的真实有效发放,基于
    华尔街科技眼 2025-01-17 10:44 221浏览
  • 随着消费者对汽车驾乘体验的要求不断攀升,汽车照明系统作为确保道路安全、提升驾驶体验以及实现车辆与环境交互的重要组成,日益受到业界的高度重视。近日,2024 DVN(上海)国际汽车照明研讨会圆满落幕。作为照明与传感创新的全球领导者,艾迈斯欧司朗受邀参与主题演讲,并现场展示了其多项前沿技术。本届研讨会汇聚来自全球各地400余名汽车、照明、光源及Tier 2供应商的专业人士及专家共聚一堂。在研讨会第一环节中,艾迈斯欧司朗系统解决方案工程副总裁 Joachim Reill以深厚的专业素养,主持该环节多位
    艾迈斯欧司朗 2025-01-16 20:51 202浏览
  •  光伏及击穿,都可视之为 复合的逆过程,但是,复合、光伏与击穿,不单是进程的方向相反,偏置状态也不一样,复合的工况,是正偏,光伏是零偏,击穿与漂移则是反偏,光伏的能源是外来的,而击穿消耗的是结区自身和电源的能量,漂移的载流子是 客席载流子,须借外延层才能引入,客席载流子 不受反偏PN结的空乏区阻碍,能漂不能漂,只取决于反偏PN结是否处于外延层的「射程」范围,而穿通的成因,则是因耗尽层的过度扩张,致使跟 端子、外延层或其他空乏区 碰触,当耗尽层融通,耐压 (反向阻断能力) 即告彻底丧失,
    MrCU204 2025-01-17 11:30 188浏览
  •  万万没想到!科幻电影中的人形机器人,正在一步步走进我们人类的日常生活中来了。1月17日,乐聚将第100台全尺寸人形机器人交付北汽越野车,再次吹响了人形机器人疯狂进厂打工的号角。无独有尔,银河通用机器人作为一家成立不到两年时间的创业公司,在短短一年多时间内推出革命性的第一代产品Galbot G1,这是一款轮式、双臂、身体可折叠的人形机器人,得到了美团战投、经纬创投、IDG资本等众多投资方的认可。作为一家成立仅仅只有两年多时间的企业,智元机器人也把机器人从梦想带进了现实。2024年8月1
    刘旷 2025-01-21 11:15 556浏览
  • Ubuntu20.04默认情况下为root账号自动登录,本文介绍如何取消root账号自动登录,改为通过输入账号密码登录,使用触觉智能EVB3568鸿蒙开发板演示,搭载瑞芯微RK3568,四核A55处理器,主频2.0Ghz,1T算力NPU;支持OpenHarmony5.0及Linux、Android等操作系统,接口丰富,开发评估快人一步!添加新账号1、使用adduser命令来添加新用户,用户名以industio为例,系统会提示设置密码以及其他信息,您可以根据需要填写或跳过,命令如下:root@id
    Industio_触觉智能 2025-01-17 14:14 126浏览
  • 嘿,咱来聊聊RISC-V MCU技术哈。 这RISC-V MCU技术呢,简单来说就是基于一个叫RISC-V的指令集架构做出的微控制器技术。RISC-V这个啊,2010年的时候,是加州大学伯克利分校的研究团队弄出来的,目的就是想搞个新的、开放的指令集架构,能跟上现代计算的需要。到了2015年,专门成立了个RISC-V基金会,让这个架构更标准,也更好地推广开了。这几年啊,这个RISC-V的生态系统发展得可快了,好多公司和机构都加入了RISC-V International,还推出了不少RISC-V
    丙丁先生 2025-01-21 12:10 171浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦