基于STM32的OTA远程升级

嵌入式大杂烩 2022-02-20 21:30


上次发过 干货 | 嵌入式OTA升级实现原理,这次就分享一下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        {                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        {                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作者小叶三千原创撰写

温馨提示

由于微信公众号近期改变了推送规则,如果您想经常看到我们的文章,可以在每次阅读后,在页面下方点一个「赞」或「在看」,这样每次推送的文章才会第一时间出现在您的订阅列表里。

免责声明:本文来源网络,免费传达知识,版权归原作者所有。如涉及作品版权问题,请联系我进行删除。


往期推荐:

分享一份嵌入式软件工具清单!

易懂 | 手把手教你编写你的第一个上位机

花了两天时间整理了STM32中的一些C语言知识点!

干货 | 嵌入式OTA升级实现原理

分享一个实用的、可应用于单片机的内存管理模块

C语言、嵌入式位操作精华技巧大汇总

嵌入式大杂烩周记 | 第 1 期

C语言、嵌入式中几个非常实用的宏技巧

分享一个自用的、极简的log模块!

分享一个很酷的IDE!软工必备

在公众号聊天界面回复1024,可获取嵌入式资源;回复 ,可查看文章汇总。

点击阅读原文,查看更多分享。

嵌入式大杂烩 专注于嵌入式技术,包括但不限于C/C++、嵌入式、物联网、Linux等编程学习笔记,同时,内包含大量的学习资源。欢迎关注,一同交流学习,共同进步!
评论 (0)
  • 随着智能驾驶时代到来,汽车正转变为移动计算平台。车载AI技术对存储器提出新挑战:既要高性能,又需低功耗和车规级可靠性。贞光科技代理的紫光国芯车规级LPDDR4存储器,以其卓越性能成为国产芯片产业链中的关键一环,为智能汽车提供坚实的"记忆力"支持。作为官方授权代理商,贞光科技通过专业技术团队和完善供应链,让这款国产存储器更好地服务国内汽车厂商。本文将探讨车载AI算力需求现状及贞光科技如何通过紫光国芯LPDDR4产品满足市场需求。 车载AI算力需求激增的背景与挑战智能驾驶推动算力需求爆发式
    贞光科技 2025-05-07 16:54 229浏览
  • 温度传感器的工作原理依据其类型可分为以下几种主要形式:一、热电阻温度传感器利用金属或半导体材料的电阻值随温度变化的特性实现测温:l ‌金属热电阻‌(如铂电阻 Pt100、Pt1000):高温下电阻值呈线性增长,稳定性高,适用于工业精密测温。l ‌热敏电阻‌(NTC/PTC):NTC 热敏电阻阻值随温度升高而下降,PTC 则相反;灵敏度高但线性范围较窄,常用于电子设备温控。二、热电偶传感器基于‌塞贝克效应‌(Seebeck effect):两种不同
    锦正茂科技 2025-05-09 13:31 245浏览
  • 硅二极管温度传感器是一种基于硅半导体材料特性的测温装置,其核心原理是利用硅二极管的电学参数(如正向压降或电阻)随温度变化的特性实现温度检测。以下是其工作原理、技术特点及典型应用:一、工作原理1、‌PN结温度特性‌硅二极管由PN结构成,当温度变化时,其正向电压 VF与温度呈线性负相关关系。例如,温度每升高1℃,VF约下降2 mV。2、‌电压—温度关系‌通过jing确测量正向电压的微小变化,可推算出环境温度值。部分型号(如SI410)在宽温域内(如1.4 K至475 K)仍能保持高线性度。
    锦正茂科技 2025-05-09 13:52 257浏览
  • 这款无线入耳式蓝牙耳机是长这个样子的,如下图。侧面特写,如下图。充电接口来个特写,用的是卡座卡在PCB板子上的,上下夹紧PCB的正负极,如下图。撬开耳机喇叭盖子,如下图。精致的喇叭(HY),如下图。喇叭是由电学产生声学的,具体结构如下图。电池包(AFS 451012  21 12),用黄色耐高温胶带进行包裹(安规需求),加强隔离绝缘的,如下图。451012是电池包的型号,聚合物锂电池+3.7V 35mAh,详细如下图。电路板是怎么拿出来的呢,剪断喇叭和电池包的连接线,底部抽出PCB板子
    liweicheng 2025-05-06 22:58 641浏览
  • UNISOC Miracle Gaming奇迹手游引擎亮点:• 高帧稳帧:支持《王者荣耀》等主流手游90帧高画质模式,连续丢帧率最高降低85%;• 丝滑操控:游戏冷启动速度提升50%,《和平精英》开镜开枪操作延迟降低80%;• 极速网络:专属游戏网络引擎,使《王者荣耀》平均延迟降低80%;• 智感语音:与腾讯GVoice联合,弱网环境仍能保持清晰通话;• 超高画质:游戏画质增强、超级HDR画质、游戏超分技术,优化游戏视效。全球手游市场规模日益壮大,游戏玩家对极致体验的追求愈发苛刻。紫光展锐全新U
    紫光展锐 2025-05-07 17:07 344浏览
  • 后摄像头是长这个样子,如下图。5孔(D-,D+,5V,12V,GND),说的是连接线的个数,如下图。4LED,+12V驱动4颗LED灯珠,给摄像头补光用的,如下图。打开后盖,发现里面有透明白胶(防水)和白色硬胶(固定),用合适的工具,清理其中的胶状物。BOT层,AN3860,Panasonic Semiconductor (松下电器)制造的,Cylinder Motor Driver IC for Video Camera,如下图。TOP层,感光芯片和广角聚焦镜头组合,如下图。感光芯片,看着是玻
    liweicheng 2025-05-07 23:55 462浏览
  • 在过去的很长一段时间里,外卖市场呈现出美团和饿了么双寡头垄断的局面。美团凭借先发优势、强大的地推团队以及精细化的运营策略,在市场份额上长期占据领先地位。数据显示,截至2024年上半年,美团外卖以68.2%的市场份额领跑外卖行业,成为当之无愧的行业老大。其业务广泛覆盖,从一线城市的繁华商圈到二三线城市的大街小巷,几乎无处不在,为无数消费者提供便捷的外卖服务。饿了么作为阿里本地生活服务的重要一环,依托阿里强大的资金和技术支持,也在市场中站稳脚跟,以25.4%的份额位居第二。尽管市场份额上与美团有一定
    用户1742991715177 2025-05-06 19:43 116浏览
  • 二位半 5线数码管的驱动方法这个2位半的7段数码管只用5个管脚驱动。如果用常规的7段+共阳/阴则需要用10个管脚。如果把每个段看成独立的灯。5个管脚来点亮,任选其中一个作为COM端时,另外4条线可以单独各控制一个灯。所以实际上最多能驱动5*4 = 20个段。但是这里会有一个小问题。如果想点亮B1,可以让第3条线(P3)置高,P4 置低,其它阳极连P3的灯对应阴极P2 P1都应置高,此时会发现C1也会点亮。实际操作时,可以把COM端线P3设置为PP输出,其它线为OD输出。就可以单独控制了。实际的驱
    southcreek 2025-05-07 15:06 559浏览
  • 文/郭楚妤编辑/cc孙聪颖‍相较于一众措辞谨慎、毫无掌舵者个人风格的上市公司财报,利亚德的财报显得尤为另类。利亚德光电集团成立于1995年,是一家以LED显示、液晶显示产品设计、生产、销售及服务为主业的高新技术企业。自2016年年报起,无论业绩优劣,董事长李军每年都会在财报末尾附上一首七言打油诗,抒发其对公司当年业绩的感悟。从“三年翻番顺大势”“智能显示我第一”“披荆斩棘幸从容”等词句中,不难窥见李军的雄心壮志。2012年,利亚德(300296.SZ)在深交所创业板上市。成立以来,该公司在细分领
    华尔街科技眼 2025-05-07 19:25 448浏览
  • Matter协议是一个由Amazon Alexa、Apple HomeKit、Google Home和Samsung SmartThings等全球科技巨头与CSA联盟共同制定的开放性标准,它就像一份“共生契约”,能让原本相互独立的家居生态在应用层上握手共存,同时它并非另起炉灶,而是以IP(互联网协议)为基础框架,将不同通信协议下的家居设备统一到同一套“语义规则”之下。作为应用层上的互通标准,Matter协议正在重新定义智能家居行业的运行逻辑,它不仅能向下屏蔽家居设备制造商的生态和系统,让设备、平
    华普微HOPERF 2025-05-08 11:40 396浏览
  • 飞凌嵌入式作为龙芯合作伙伴,隆重推出FET-2K0300i-S全国产自主可控工业级核心板!FET-2K0300i-S核心板基于龙芯2K0300i工业级处理器开发设计,集成1个64位LA264处理器,主频1GHz,提供高效的计算能力;支持硬件ECC;2K0300i还具备丰富的连接接口USB、SDIO、UART、SPI、CAN-FD、Ethernet、ADC等一应俱全,龙芯2K0300i支持四路CAN-FD接口,具备良好的可靠性、实时性和灵活性,可满足用户多路CAN需求。除性价比超高的国产处理器外,
    飞凌嵌入式 2025-05-07 11:54 97浏览
我要评论
0
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦