还不会OTA升级?手把手教你基于STM32的BootLoader的OTA远程升级

原创 嵌入式ARM 2021-08-02 18:00


上次发过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作者小叶三千原创撰写


推荐阅读
基于STM32F103的SD卡Bootloader离线升级
单片机的Bootloader,可以实现用户轻松升级程序
工程师实录:记一则bootloader的cache问题

→点关注,不迷路←
嵌入式ARM 关注这个时代最火的嵌入式ARM,你想知道的都在这里。
评论
  • PLC组态方式主要有三种,每种都有其独特的特点和适用场景。下面来简单说说: 1. 硬件组态   定义:硬件组态指的是选择适合的PLC型号、I/O模块、通信模块等硬件组件,并按照实际需求进行连接和配置。    灵活性:这种方式允许用户根据项目需求自由搭配硬件组件,具有较高的灵活性。    成本:可能需要额外的硬件购买成本,适用于对系统性能和扩展性有较高要求的场合。 2. 软件组态   定义:软件组态主要是通过PLC
    丙丁先生 2025-01-06 09:23 86浏览
  • By Toradex 秦海1). 简介嵌入式平台设备基于Yocto Linux 在开发后期量产前期,为了安全以及提高启动速度等考虑,希望将 ARM 处理器平台的 Debug Console 输出关闭,本文就基于 NXP i.MX8MP ARM 处理器平台来演示相关流程。 本文所示例的平台来自于 Toradex Verdin i.MX8MP 嵌入式平台。  2. 准备a). Verdin i.MX8MP ARM核心版配合Dahlia载板并
    hai.qin_651820742 2025-01-07 14:52 58浏览
  • 本文介绍Linux系统更换开机logo方法教程,通用RK3566、RK3568、RK3588、RK3576等开发板,触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。制作图片开机logo图片制作注意事项(1)图片必须为bmp格式;(2)图片大小不能大于4MB;(3)BMP位深最大是32,建议设置为8;(4)图片名称为logo.bmp和logo_kernel.bmp;开机
    Industio_触觉智能 2025-01-06 10:43 87浏览
  • 这篇内容主要讨论三个基本问题,硅电容是什么,为什么要使用硅电容,如何正确使用硅电容?1.  硅电容是什么首先我们需要了解电容是什么?物理学上电容的概念指的是给定电位差下自由电荷的储藏量,记为C,单位是F,指的是容纳电荷的能力,C=εS/d=ε0εrS/4πkd(真空)=Q/U。百度百科上电容器的概念指的是两个相互靠近的导体,中间夹一层不导电的绝缘介质。通过观察电容本身的定义公式中可以看到,在各个变量中比较能够改变的就是εr,S和d,也就是介质的介电常数,金属板有效相对面积以及距离。当前
    知白 2025-01-06 12:04 180浏览
  • 彼得·德鲁克被誉为“现代管理学之父”,他的管理思想影响了无数企业和管理者。然而,关于他的书籍分类,一种流行的说法令人感到困惑:德鲁克一生写了39本书,其中15本是关于管理的,而其中“专门写工商企业或为企业管理者写的”只有两本——《为成果而管理》和《创新与企业家精神》。这样的表述广为流传,但深入探讨后却发现并不完全准确。让我们一起重新审视这一说法,解析其中的矛盾与根源,进而重新认识德鲁克的管理思想及其著作的真正价值。从《创新与企业家精神》看德鲁克的视角《创新与企业家精神》通常被认为是一本专为企业管
    优思学院 2025-01-06 12:03 134浏览
  • 大模型的赋能是指利用大型机器学习模型(如深度学习模型)来增强或改进各种应用和服务。这种技术在许多领域都显示出了巨大的潜力,包括但不限于以下几个方面: 1. 企业服务:大模型可以用于构建智能客服系统、知识库问答系统等,提升企业的服务质量和运营效率。 2. 教育服务:在教育领域,大模型被应用于个性化学习、智能辅导、作业批改等,帮助教师减轻工作负担,提高教学质量。 3. 工业智能化:大模型有助于解决工业领域的复杂性和不确定性问题,尽管在认知能力方面尚未完全具备专家级的复杂决策能力。 4. 消费
    丙丁先生 2025-01-07 09:25 87浏览
  • 在智能家居领域中,Wi-Fi、蓝牙、Zigbee、Thread与Z-Wave等无线通信协议是构建短距物联局域网的关键手段,它们常在实际应用中交叉运用,以满足智能家居生态系统多样化的功能需求。然而,这些协议之间并未遵循统一的互通标准,缺乏直接的互操作性,在进行组网时需要引入额外的网关作为“翻译桥梁”,极大地增加了系统的复杂性。 同时,Apple HomeKit、SamSung SmartThings、Amazon Alexa、Google Home等主流智能家居平台为了提升市占率与消费者
    华普微HOPERF 2025-01-06 17:23 151浏览
  • 每日可见的315MHz和433MHz遥控模块,你能分清楚吗?众所周知,一套遥控设备主要由发射部分和接收部分组成,发射器可以将控制者的控制按键经过编码,调制到射频信号上面,然后经天线发射出无线信号。而接收器是将天线接收到的无线信号进行解码,从而得到与控制按键相对应的信号,然后再去控制相应的设备工作。当前,常见的遥控设备主要分为红外遥控与无线电遥控两大类,其主要区别为所采用的载波频率及其应用场景不一致。红外遥控设备所采用的射频信号频率一般为38kHz,通常应用在电视、投影仪等设备中;而无线电遥控设备
    华普微HOPERF 2025-01-06 15:29 137浏览
  • 根据环洋市场咨询(Global Info Research)项目团队最新调研,预计2030年全球无人机锂电池产值达到2457百万美元,2024-2030年期间年复合增长率CAGR为9.6%。 无人机锂电池是无人机动力系统中存储并释放能量的部分。无人机使用的动力电池,大多数是锂聚合物电池,相较其他电池,锂聚合物电池具有较高的能量密度,较长寿命,同时也具有良好的放电特性和安全性。 全球无人机锂电池核心厂商有宁德新能源科技、欣旺达、鹏辉能源、深圳格瑞普和EaglePicher等,前五大厂商占有全球
    GIRtina 2025-01-07 11:02 80浏览
  • 根据Global Info Research项目团队最新调研,预计2030年全球封闭式电机产值达到1425百万美元,2024-2030年期间年复合增长率CAGR为3.4%。 封闭式电机是一种电动机,其外壳设计为密闭结构,通常用于要求较高的防护等级的应用场合。封闭式电机可以有效防止外部灰尘、水分和其他污染物进入内部,从而保护电机的内部组件,延长其使用寿命。 环洋市场咨询机构出版的调研分析报告【全球封闭式电机行业总体规模、主要厂商及IPO上市调研报告,2025-2031】研究全球封闭式电机总体规
    GIRtina 2025-01-06 11:10 109浏览
  • 村田是目前全球量产硅电容的领先企业,其在2016年收购了法国IPDiA头部硅电容器公司,并于2023年6月宣布投资约100亿日元将硅电容产能提升两倍。以下内容主要来自村田官网信息整理,村田高密度硅电容器采用半导体MOS工艺开发,并使用3D结构来大幅增加电极表面,因此在给定的占位面积内增加了静电容量。村田的硅技术以嵌入非结晶基板的单片结构为基础(单层MIM和多层MIM—MIM是指金属 / 绝缘体/ 金属) 村田硅电容采用先进3D拓扑结构在100um内,使开发的有效静电容量面积相当于80个
    知白 2025-01-07 15:02 87浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦