Esp8266进阶之路18 esp8266 基于NONOS实现 OTA 远程升级,实现无线“热修复”升级固件程序。(附带Demo)


  • 本系列博客学习由非官方人员 半颗心脏 潜心所力所写,不做开发板。仅仅做个人技术交流分享,不做任何商业用途。如有不对之处,请留言,本人及时更改。
序号 SDK版本 内容 链接
1 nonos2.0 搭建开发环境,开始一个“hellow world”串口打印。 点我访问
2 nonos2.0 利用GPIO开始使用按钮点亮你的“第一盏灯。 点我访问
3 nonos2.0 利用 "软件定时器 " 定时0.5秒闪烁点亮一盏LED。 点我访问
4 nonos2.0 用PWM控制一盏LED的亮度变化。 点我访问
5 nonos2.0 SDK高级使用之封装Post与Get请求云端,拿到“天气预报信息”。 点我访问
6 nonos2.0 了解 SmartConfig与Airkiss一键配网,给8266配网上云端。无需把wifi名字密码写在固件里。 点我访问
7 nonos2.0 了解 softAP热点配网模式原理,仿“机智云”定义自己的热点配网模式协议。 点我访问
8 nonos2.0 你要找的8266作为UDP、TCP客户端或服务端的角色通讯,都在这了。 点我访问
9 nonos2.0 [小实战上篇]Windows系统搭建8266的本地Mqtt服务器,局域网点亮一盏LED灯。 点我访问
10 nonos2.0 [小实战下篇]Windows系统搭建8266的本地Mqtt服务器,局域网点亮一盏LED灯。 点我访问
11 rtos2.0 接入阿里智能,点亮一盏LED灯,期待天猫精灵语音控制的不约而至! 点我访问
12 nonos2.0 图文并茂学习阿里云主机搭建8266MQTT服务器,实现移动网络远程控制一盏LED。 点我访问
13 nonos2.0 动手做个8266毕设小案例,smartConfig + MQTT协议轻松实现远程控制一盏LED。 点我访问
14 rtos2.0 FreeRtos系统学习的正确姿势 ------ 环境搭建、烧录。 点我访问
15 rtos2.0 接入阿里云平台非阿里智能的SDS服务,点亮一盏LED灯。 点我访问
16 nonos2.0 基于Nonos移植红外线H1838,实现红外遥控器配网,远程控制一盏灯。 点我访问
17 nonos2.0 esp8266自研的快速上电开关五次 (开-关为一次) ,无需按键触发则8266进去一键配网模式。 点我访问
18 nonos2.0 基于NONOS 实现 OTA 远程升级,实现无线“ 热修复 ”升级固件程序。 点我访问
19 nonos2.0 驱动 ds18b20、dht11 温湿度传感器,采集温湿度传感器到服务器。 点我访问
20 nonos2.0 深入学习esp8266的esp now模式,仿机智云做一个小网关,实现无需网络下轻松彼此连接通讯交互数据。 点我访问
21 nonos2.0 浅谈 esp8266 如何在本地局域网网络情况下实现最大效率地和前端实现数据交互。 点我访问
22 nonos2.0 esp8266的工程如何添加第三方静态库文件以及如何自定义文件夹,聊聊那些makeFile的事 点我访问
23 nonos2.0 再来一波 esp8266 基于 freeRtos系统连接自己私有的服务器实现OTA远程升级,接触下 lwip的基本知识。 点我访问
24 nonos2.0 渗透学习回顾下esp8266的外置spi芯片25q系列,熟悉8266代码块在其的分布,得心应手放置图片或其他资料。 点我访问
25 rtos2.0 深聊下esp8266的串口 Uart 通讯中断编程,为您准备好了 NONOS 版本 和 RTOS 系统的串口驱动文件。 点我访问
26 nonos2.0 RTOS分析 MQTT 实现过程,实现移植 MQTT协议在 esp8266 rtos实时系统,可断线重连。 点我访问
27 rtos3.0 跟紧脚步,用VisualStudio Code开发 esp8266 rtos SDK v3.0版本,全新的 idf 框架,节省内存模块化开发。 点我访问
28 rtos3.0 教你轻松自如使用cJson在乐鑫 esp8266 如何解析一段json数据以及如何生成一段json数据。 点我访问
29 rtos3.0 百万条消息免费之使用TCP直连模式MQTT协议接入阿里云物联网平台,支持私家服务器对接支持阿里云规则引擎。 点我访问
30 rtos3.0 SDK编程使用 IIC总线驱动 0.96寸的OLED显示屏,显示天气预报信息。 点我访问
31 rtos3.0 当esp8266遇到 Html,该怎么内置网页控制设备,理清内置网页的实现过程,实现无需路由器手机也可以控制esp8266。 点我访问
32 rtos3.0 细聊HmacMD5的加密方法带来的安全性,并实践在esp8266上,最大保障传输的过程的信息的安全性。 点我访问
33 rtos3.0 如何优雅地像乐鑫原厂封装esp8266底层寄存器的逻辑思维,做成自己的静态库库文件,让第三方人使用? 点我访问
34 nonos3.0 编程使用 SPI 驱动基于Max7219芯片的八位数码管,显示日期信息。 点我访问
35 nonos3.0 借助机智云平台做一个商业化的七彩RGB灯泡可调整体方案项目,炫彩夺目高大尚。 点我访问
36 rtos3.0 认识Rtos 3.0 sdk 工程结构,如何向esp-idf工程靠近的,如何自定义头文件编译? 点我访问
37 rtos3.0 你要找的基本外设功能都在这里了,包括Gpio、Pwm 和 Uart 接口使用。 点我访问
38 rtos3.0 一篇文章带你搞掂存储技术 NVS 的认识和使用,如何利用NVS保存整型、字符串、数组以及结构体。 点我访问
39 rtos3.0 带你捋一捋市面上的微信公众号配网智能设备 esp8266 并绑定设备的过程,移植并成功实现在 esp8266 rtos3.1 sdk。 点我访问
40 rtos3.0 基于乐鑫idf框架,研究出超稳定、掉线重连、解决内存泄露问题的Mqtt框架!支持esp8266和esp32! 点我访问
41 rtos3.0 esp8266-12模块基于rtos3.1版本ota功能远程空中升级固件,官网基础之上增加dns域名解析! 点我访问
42 rtos3.0 我又来了,基于rtos3.0版本 SDK编程 SPI 驱动 ws2812b 七彩灯,代码全部开源奉献给你们! 点我访问
43 rtos3.0 基于rtos3.0版本扫描周围获取附近可用的 Wi-Fi 热点路由器信息,同样适合esp32。 点我访问
44 rtos3.0 整理分享那些我在项目中常用的esp8266 rtos3.0版本的常见驱动,Button按键长短按、PWM平滑调光等。 点我访问
45 rtos3.0 内置仅1M的Esp8285,如何攻破最棘手的OTA问题,大大节省资源成本开发产品 点我访问
46 rtos3.0 详细分析Esp8266上电信息打印的数据,如何做到串口通讯上电不乱码打印; 点我访问
47 rtos3.0 无需外网,如何实现在本地局域网与控制端做数据交换的一些开发经验; 点我访问
48 rtos3.0 迟来的1024程序员祝福,开源分享一个驱动 ds18b20 获取温度的工程。 点我访问
49 rtos3.2 aliyun sdk 直连接入阿里云物联网平台,实现天猫精灵找队友零配网功能和语音控制。 点我访问
持续更新,欢迎关注我,QQ群讨论群:434878850

  • 很多人怎么联系我一起学习进步,下面打个小小公告和干货无偿分享:

玩转esp8266带你飞、加群付费QQ群,提高门槛,不喜的朋友勿喷勿加:434878850
esp8266源代码免费学习汇总(持续更新,欢迎star):https://github.com/xuhongv/StudyInEsp8266
esp32源代码免费学习汇总(持续更新,欢迎star):https://github.com/xuhongv/StudyInEsp32



一、前言;


写了这么多的8266博文,一直以满意100%的心态去敲写代码固件烧录,以致很少出现 bug , 目前呢?浏览了网上很多8266的OTA的demo,还特意在某宝上搜了一把,竟然卖的价格为“399”元一个8266的OTA升级方案!天呐!代码这么值钱么?那我之前开源了那么多的工程源码带博文,岂不是无价之宝~~

其实开心就好,那么本文的话,就带大家共睹8266的OTA的远程升级的整个流程!记住我是小徐,大家有什么8266疑问,尽管在博文留言!


二、OTA的认识;


在Android开发领域里,有一种叫“热修复”的功能,能够不通过下载APK重新安装它,即可实现修复了Bug。8266的OTA升级也是如此,官网称之为FOTA ,对这个理念的东西,我也是不太懂!反正可以实现无线烧固件就可以啦!

  • ①、8266的在线远程升级的总体流程怎么样的?

这里写图片描述


  • ②、升级时候,内部运行是怎么样的?

1.从官网提供的图片可以看到,内存大小分为四块,分别是 user1.bin、user2.bin、boot.bin 启动、master_device.key 用户数据 ,初始化系统参数的2个 blanck.bin和esp_init_data_default.bin和预留空间;

2.由下面的理解可知,8266要运行哪个固件,取决于 boot.bin文件的信息,每次升级前,都会查询一次当前运行的是 user1.bin 还是 user2.bin ,进而从云端拉取对应相反的那一个bin;

这里写图片描述


  • ③、市面上那么对的8266模块,应该怎么配置烧录地址?

很简单问题,只需要修改工程目录下的makefile文件即可,翻开它在 25行左右,可以看到下面信息,这个是乐鑫没有OTA在线升级的配置,可以看到 BOOT?=none,好!那么从这里开始修改!

BOOT?=none
APP?=0
SPI_SPEED?=40
SPI_MODE?=QIO
SPI_SIZE_MAP?=0

我这里详细地说下这里包含了啥信息!
由下面可得,不同的8266模块,其外部falsh大小决定了编译时候的user.bin路径,所以大家在此OTA升级,必须要摸清楚你买的模块是哪个falsh大小的,注意1M = 8Mbit!!!安信可的 32Mbit 其实就是 4M的falsh,类似 25Q32这样的存储芯片。


字段 数值 含义 备注
BOOT none 表示不执行 boot.bin文件,即没有boot.bin文件
new 表示执行 boot.bin文件,每次执行将要执行boot.bin文件
APP 1 表示生成 user1.bin文件 注意这个仅当上面的BOOT=new 才有效
2 表示生成 user2.bin文件 注意这个仅当上面的BOOT=new 才有效
SPI_SPEED 40 表示烧录时候的频率选择,这里一般为40,对应烧录工具默认值即可
SPI_MODE QIO 这个主要看8266模块制作商,比如现在常见的是安信可的8266-12F,就是QIO。
SPI_SIZE_MAP 1 flash=256,user1地址0x1000,user2地址0x41000 制作对应不同的falsh大小固件
2 flash=1024,user1地址0x1000,user2地址0x81000 制作对应不同的falsh大小固件
3 flash=2048,user1地址0x1000,user2地址0x81000 制作对应不同的falsh大小固件
4 flash=4096,user1地址0x1000,user2地址0x81000 制作对应不同的falsh大小固件
5 flash=2048,user1地址0x1000,user2地址0x101000 制作对应不同的falsh大小固件
6 flash=4096,user1地址0x1000,user2地址0x101000 制作对应不同的falsh大小固件
8 flash=8192,user1地址0x1000,user2地址0x101000 制作对应不同的falsh大小固件
9 flash=16384,user1地址0x1000,user2地址0x101000 制作对应不同的falsh大小固件
其它数值 flash=256,user1地址0x1000,user2地址0x41000 制作对应不同的falsh大小固件

综合上面的表格,我们才有了官网给的下面的图片说明!(注意看blank.binesp_init_data_default.bin地址)


这里写图片描述


三、固件生成注意事项;


  • ①:生成user1.bin文件:需要注意您的8266模块大小(后面我会列出安信可的所有8266模块的falsh大小),配置SPI_SIZE_MAP?=6,然后配置我们上述所说的makefile文件配置!下面是我用安信可的8266-12F的配置:
              BOOT?=new
              APP?=1 //此刻生成的是 user1.bin文件
              SPI_SPEED?=40
              SPI_MODE?=QIO
              SPI_SIZE_MAP?=6

于是乎,上面的配置就生成如下(boot.bin文件版本支持1.4或以上):

这里写图片描述


  • 生成user2.bin文件:修改您的bug之后,就修改APP?=2即可!
              BOOT?=new
              APP?=2 //此刻生成的是 user2.bin文件
              SPI_SPEED?=40
              SPI_MODE?=QIO
              SPI_SIZE_MAP?=6

于是乎,上面的配置就生成如下,注意此刻的地址是与上面不同的:

这里写图片描述


四、本地局域网的服务器搭建;


  • 我们必须要把user1.bin文件和user2.bin文件放在服务器,让8266自己判断当前运行的是哪个文件,然后自己去拉取! 这里我就用 tomcat 来搭建本地服务器,怎么搭建的看我之前写的文章点我查看

确保以上步骤完毕之后,我在解压后的目录的webapps下新建一个xuhong文件夹,再新建一个8266NewBin文件夹,把2个bin文件(注意他们都在我们的工程目录的 bin/upgrade下面,后缀名是 bin ,)放在这里!路径注意不要带中文字符!


这里写图片描述


  • 我们放置好的文件在服务器了:

这里写图片描述

放好之后,我们在电脑测试下路径是否有误?打开 http://localhost:8080/xuhong/8266NewBin/user1.4096.new.6.bin ,发现浏览器会弹窗出现是否下载此文件!

同时,用同一个路由器连接的手机打开电脑的网关分配的局域网IP地址!这个见下图了,同时确认电脑已经开启了8080对外端口,否则无法访问!),注意不要混乱!我这获取的电脑IP是192.168.1.105,我们在手机打开http://192.168.1.105:8080/xuhong/8266NewBin/user1.4096.new.6.bin ,发现也提示下载文件了!

恭喜!本地部署成功!


这里写图片描述


-这里写图片描述


四、硬件代码;


  • 注意要修改 结构体的upgrade_server_infoURL申请的内存大小,下面是 4M * 1024 =4096 !

/**
 * server_ip: 服务器地址
 * port:服务器端口
 * path:文件路径
 */
void ICACHE_FLASH_ATTR ota_start_Upgrade(const char *server_ip, uint16_t port,const char *path) {
	const char* file;
	//获取系统的目前加载的是哪个bin文件
	uint8_t userBin = system_upgrade_userbin_check();

	switch (userBin) {

	//如果检查当前的是处于user1的加载文件,那么拉取的就是user2.bin
	case UPGRADE_FW_BIN1:
		file = "user2.4096.new.6.bin";
		break;

		//如果检查当前的是处于user2的加载文件,那么拉取的就是user1.bin
	case UPGRADE_FW_BIN2:
		file = "user1.4096.new.6.bin";
		break;

		//如果检查都不是,可能此刻不是OTA的bin固件
	default:
		os_printf("Fail read system_upgrade_userbin_check! \n\n");
		return;
	}

	struct upgrade_server_info* update =
			(struct upgrade_server_info *) os_zalloc(
					sizeof(struct upgrade_server_info));
	update->pespconn = (struct espconn *) os_zalloc(sizeof(struct espconn));
	//设置服务器地址
	os_memcpy(update->ip, server_ip, 4);
	//设置服务器端口
	update->port = port;
	//设置OTA回调函数
	update->check_cb = ota_finished_callback;
	//设置定时回调时间
	update->check_times = 10000;
	//从 4M *1024 =4096申请内存
	update->url = (uint8 *)os_zalloc(4096);

	//打印下請求地址
	os_printf("Http Server Address:%d.%d.%d.%d ,port: %d,filePath: %s,fileName: %s \n",
			IP2STR(update->ip), update->port, path, file);

	//拼接完整的 URL去请求服务器
	os_sprintf((char*) update->url, "GET /%s%s HTTP/1.1\r\n"
			"Host: "IPSTR":%d\r\n"
	"Connection: keep-alive\r\n"
	"\r\n", path, file, IP2STR(update->ip), update->port);

	if (system_upgrade_start(update) == false) {
		os_printf(" Could not start upgrade\n");
		//释放资源
		os_free(update->pespconn);
		os_free(update->url);
		os_free(update);
	} else {
		os_printf(" Upgrading...\n");
	}
}

②、在回调函数我们写如下,表示检测是否已经升级完成,如果返回true,则重启复位模块!

    struct upgrade_server_info *update = arg;
	if (update->upgrade_flag == true){
		os_printf("OTA  Success ! rebooting!\n");
		system_upgrade_reboot();
	}else{
		os_printf("OTA failed!\n");
	}

  • 部分截图分享:

这里写图片描述


  • 升级完成:

这里写图片描述



⑤、后记;


此代码参考了网上的许多文章,感谢各位的支持!特别是 gitHub上阿秀的开源!

代码亲测可用,我花了2小时特此写了此博文,免得以后的小伙伴入坑,所以源码下载的话给点赏金支持下!谢谢那个热爱技术、知道敲代码入坑的你!

  • 注意:下载此工程的伙伴注意了:烧录成功后,每次启动模块,模块会自动进去smartConfig配网模式,配合手机配网成功后,模块将自动请求服务器在线升级的哦!

  • 源码下载:http://www.demodashi.com/demo/12994.html



发布了152 篇原创文章 · 获赞 787 · 访问量 80万+
展开阅读全文

没有更多推荐了,返回首页

©️2019 CSDN 皮肤主题: 代码科技 设计师: Amelia_0503

分享到微信朋友圈

×

扫一扫,手机浏览