* 本系列博客学习由非官方人员 半颗心脏 <http://blog.csdn.net/xh870189248>
潜心所力所写,仅仅做个人技术交流分享,不做任何商业用途。如有不对之处,请留言,本人及时更改。
1、 Esp8266之 搭建开发环境,开始一个“hellow world”串口打印。
<http://blog.csdn.net/xh870189248/article/details/77985541>
2、 Esp8266之 利用GPIO开始使用按钮点亮你的“第一盏灯”。
<http://blog.csdn.net/xh870189248/article/details/78126689>
3、 Esp8266之 利用 "软件定时器 " 定时0.5秒闪烁点亮一盏LED。
<http://blog.csdn.net/xh870189248/article/details/78155357>
4 、Esp8266之 了解PWM,更为深入地用PWM控制一盏LED的亮度变化。
<http://blog.csdn.net/xh870189248/article/details/78202224>
5 、Esp8266之 原生乐鑫SDK高级使用之封装Post与Get请求云端,拿到“天气预报信息”。
<http://blog.csdn.net/xh870189248/article/details/78656563>
6 、Esp8266之 了解 SmartConfig与Airkiss一键配网,给8266配网上云端。无需把wifi名字密码写在固件里。
<http://blog.csdn.net/xh870189248/article/details/78677952>
7 、Esp8266之 了解 softAP热点配网模式原理,仿“机智云”定义自己的热点配网模式协议。
<http://blog.csdn.net/xh870189248/article/details/78703905>
8、 Esp8266之 你要找的8266作为UDP、TCP客户端或服务端的角色通讯,都在这了。
<http://blog.csdn.net/xh870189248/article/details/78739311>
9、 Esp8266进阶之路第1篇: [小实战上篇]Windows系统搭建8266的本地Mqtt服务器,局域网点亮一盏LED灯。
<http://blog.csdn.net/xh870189248/article/details/78761948>
10、 Esp8266进阶之路第2篇: [小实战下篇]Windows系统搭建8266的本地Mqtt服务器,局域网点亮一盏LED灯。
<http://blog.csdn.net/xh870189248/article/details/78777371>
11、 Esp8266进阶之路第3篇: 8266接入阿里智能,点亮一盏LED灯,期待天猫精灵语音控制的不约而至!
<http://blog.csdn.net/xh870189248/article/details/78807018>
12、 Esp8266进阶之路第4篇: 图文并茂学习阿里云主机搭建8266MQTT服务器,实现移动网络远程控制一盏LED。
<http://blog.csdn.net/xh870189248/article/details/78867173>
13、 Esp8266进阶之路第5篇: 动手做个8266毕设小案例,smartConfig + MQTT协议轻松实现远程控制一盏LED。
<http://blog.csdn.net/xh870189248/article/details/79052347>
14、 Esp8266进阶之路第6篇: esp8266的 FreeRtos系统学习的正确姿势 ------ 环境搭建、烧录。
<http://blog.csdn.net/xh870189248/article/details/79103373>
15、 Esp8266进阶之路第7篇: esp8266的 物联网又一股清流,8266接入阿里云平台非阿里智能的SDS服务,点亮一盏LED灯。
<http://blog.csdn.net/xh870189248/article/details/79197459>
16、 Esp8266进阶之路第8篇: esp8266的 基于Nonos移植红外线H1838,实现红外遥控器配网,远程控制一盏灯。
<http://blog.csdn.net/xh870189248/article/details/79486075>
17、 Esp8266进阶之路第9篇: esp8266自研的快速上电开关五次 (开-关为一次) ,无需按键触发则8266进去一键配网模式。
<https://blog.csdn.net/xh870189248/article/details/80027961>
18、 Esp8266进阶之路第10篇: esp8266 基于NONOS 实现 OTA 远程升级,实现无线“ 热修复 ”升级固件程序。
<https://blog.csdn.net/xh870189248/article/details/80095139>
19、 Esp8266进阶之路第11篇: esp8266驱动 ds18b20、dht11 温湿度传感器,采集温湿度传感器到服务器。
<https://blog.csdn.net/xh870189248/article/details/80284827>
20、 Esp8266进阶之路第12篇: 深入学习esp8266的esp now模式,仿机智云做一个小网关,实现无需网络下轻松彼此连接通讯交互数据。
<https://blog.csdn.net/xh870189248/article/details/80631739>
21、 Esp8266进阶之路第13篇: 浅谈 esp8266 如何在本地局域网网络情况下实现最大效率地和前端实现数据交互。
<https://blog.csdn.net/xh870189248/article/details/80859347>
22、 Esp8266进阶之路第14篇: esp8266的工程如何添加第三方静态库文件以及如何自定义文件夹,聊聊那些makeFile的事。。
<https://blog.csdn.net/xh870189248/article/details/80909216>
23、 Esp8266进阶之路第15篇: 再来一波 esp8266 基于 freeRtos系统连接自己私有的服务器实现OTA远程升级,接触下
lwip的基本知识。。 <https://blog.csdn.net/xh870189248/article/details/80924538>
24、 Esp8266进阶之路第16篇: 渗透学习回顾下esp8266的外置spi芯片25q系列,熟悉8266代码块在其的分布,得心应手放置图片或其他资料。
<https://blog.csdn.net/xh870189248/article/details/81017735>
25、 Esp8266进阶之路第17篇: 深聊下esp8266的串口 Uart 通讯中断编程,为您准备好了 NONOS 版本 和 RTOS
系统的串口驱动文件。 <https://blog.csdn.net/xh870189248/article/details/81146532>
26、 Esp8266进阶之路第18篇: RTOS分析 MQTT 实现过程,实现移植 MQTT协议在 esp8266 rtos实时系统,可断线重连。
<https://blog.csdn.net/xh870189248/article/details/81181707>
27、 Esp8266进阶之路第19篇: 跟紧脚步,用VisualStudio Code开发 esp8266 rtos SDK v3.0版本,全新的 idf
框架,节省内存模块化开发。 <https://blog.csdn.net/xh870189248/article/details/81382279>
28、 Esp8266进阶之路第20篇: 教你轻松自如使用cJson在乐鑫 esp8266 如何解析一段json数据以及如何生成一段json数据。
<https://blog.csdn.net/xh870189248/article/details/83149619>
29、 Esp8266进阶之路第21篇:
百万条消息免费之乐鑫esp8266使用TCP直连模式MQTT协议接入阿里云物联网平台,支持私家服务器对接,支持阿里云规则引擎。
<https://blog.csdn.net/xh870189248/article/details/83149619>
30、 Esp8266进阶之路第22篇: 乐鑫esp8266 SDK编程使用 IIC总线驱动 0.96寸的OLED显示屏,显示天气预报信息。
<https://blog.csdn.net/xh870189248/article/details/83443614>
31、 Esp8266进阶之路第23篇: 当esp8266遇到
Html,该怎么内置网页控制设备,理清内置网页的实现过程,实现无需路由器手机也可以控制esp8266。
<https://blog.csdn.net/xh870189248/article/details/83543997>

* 很多人怎么联系我一起学习进步,下面打个小小公告:
玩转esp8266带你飞、加群付费QQ群,提高门槛,不喜的朋友勿喷勿加:434878850
esp8266源代码免费学习汇总(持续更新,欢迎star):https://github.com/xuhongv/StudyInEsp8266
<https://github.com/xuhongv/StudyInEsp8266>
esp32源代码免费学习汇总(持续更新,欢迎star):https://github.com/xuhongv/StudyInEsp32
<https://github.com/xuhongv/StudyInEsp32>



文章目录

* 一、前言; <https://blog.csdn.net/xh870189248/article/details/80924538#_45>
* 二、回顾下`OTA`的流程;
<https://blog.csdn.net/xh870189248/article/details/80924538#OTA_60>
* 三、`lwip`网络框架的知识的使用;
<https://blog.csdn.net/xh870189248/article/details/80924538#lwip_83>
* 四、如何处理服务器返回的数据?
<https://blog.csdn.net/xh870189248/article/details/80924538#_132>
* 五、扇区的擦除和烧写?
<https://blog.csdn.net/xh870189248/article/details/80924538#_218>
* 六、如何调用? <https://blog.csdn.net/xh870189248/article/details/80924538#_236>
* 七、好好享用吧! <https://blog.csdn.net/xh870189248/article/details/80924538#_315>
* 八、下载: <https://blog.csdn.net/xh870189248/article/details/80924538#_329>
* 九、注意事项: <https://blog.csdn.net/xh870189248/article/details/80924538#_346>


<>一、前言;

* esp8266实现OTA已经不是什么奇怪的事情了,主要esp8266还有支持裸跑非系统和rtos
实时系统之分,导致现在有2个版本的代码,前面我已经料到了在NONOS怎么实现远程升级我们的固件,那么这篇的话,而是基于freeRtos实时系统的。
* 主要涉及到知识有以下:
* lwip网络框架的基本使用,域名和DNS解析的使用。
* freeRtos的基本认识。
* http请求的协议认识和原理。
* esp8266的内存划分的认识。
* 实现的功能特色有以下:
* 支持本地服务器和远程广域网服务器。(通过DNS解析)
* 支持显示下载固件进度。
<>二、回顾下OTA的流程;

* 我们已经很清楚,esp8266实现远程升级的原理过程如下: Created with Raphaël 2.2.0
开始请求服务器资源比如:user2.4096.new.6.bin网络良好,服务器响应正确。准备下载... ...下载完毕,确认是否重启?重启成功!根据
boot 信息,执行user2.4096.new.6.bin。yes
*
私有服务的放置其实也是一样,只需要在阿里云或者京东云或者腾讯云这些平台搭建服务器,把编译好的固件放上去,提供链接即可,和本地一样。

*
注意放上去的2个固件的配置信息必须一致。

<>三、lwip网络框架的知识的使用;

* ①:lwip使用DNS来解析网址域名。
因为esp8266这个实时系统支持lwip,所以直接调用即可;首先要打开DNS功能,在lwipopts.h头文件有个宏定义LWIP_DNS
置为1即可。下面就是调用代码了:
//dns begin dns_init(); //初始化 struct ip_addr addr; //你要解析的域名,比如 www.baidu.com
char name0[] = "www.baidu.com"; //调用netconn_gethostbyname err_t err =
netconn_gethostbyname((char*) name0, &addr); //定义字符串来接收IP char* ipAdress; if
(err == ERR_OK) { //格式化IP ipAdress = ip_ntoa(&addr); //打印下 printf("dns ok , the
%s ip is:%s",name0, ipAdress); }else{ return; } //dns over
* ②:先通过socket连接服务器的端口,一般地,默认是80!其中http请求的格式内容如下:注意这个Host的内容可以是域名或者IP
,这个要根据服务器来配置。 GET /api/download/user2.4096.new.6.bin HTTP/1.0 Host:
www.baidu.com Connection: keep-alive Cache-Control: no-cache User-Agent:
Mozilla/5.0 (Windows NT 5.1) AppleWebKit/537.36 (KHTML, like Gecko)
Chrome/30.0.1599.101 Safari/537.36 Accept: */* Accept-Encoding:
gzip,deflate,sdch Accept-Language: zh-CN,zh;q=0.8
* ③:如果正确响应的话,也是要遵循一定的规则的,我们可以用postMan工具来看看,这显示下载的大小是296412,单位是Byte
;那么下面的话,我们根据这个大小去判断我们下载的进度啦!


<>四、如何处理服务器返回的数据?

* 我们是采用http请求服务器,那么服务器响应正确的话,也是按照一定的协议返回来给我们的 。小徐在尝试了几个服务器,一个是Tomcat,一个是用php
搭建的服务器,返回的内容格式有所差别,最大的不同点是:一个是Accept-Length:指定内容大小,一个是Content-Length:
指定大小,为此我这次封装的方法文件,是完全可以兼容起来的。详情可以看下面的部分代码。 //判断 "Accept-Length" 或者
"Context-Length" bool isAccept = false; if((ptr = (char *)strstr(pusrdata,
"Accept-Length")) != NULL) isAccept = true; else isAccept = false; if
(totallength == 0&& (ptr = (char *)strstr(pusrdata, "\r\n\r\n")) != NULL) { ptr
= (char *) strstr(pusrdata, "\r\n\r\n"); length -= ptr - pusrdata; length -= 4;
printf("ota_start_download:upgrade file download pusrdata:%s\n\n", pusrdata);
//返回"Accept-Length: "在pusrdata中首次出现的地址 if (isAccept) ptr = (char *)
strstr(pusrdata, "Accept-Length: "); else ptr = (char *) strstr(pusrdata,
"Content-Length: "); if (ptr != NULL) { //注意这个"Accept-Length" 或者
"Context-Length"占用的字节不同的 if (isAccept) ptr += 15; else ptr += 16; ptmp2 = (char
*) strstr(ptr, "\r\n"); if (ptmp2 != NULL) { //打印下收到的数据头 //printf("ptr = %s
\n", ptr); //清空lengthbuffer memset(lengthbuffer, 0, sizeof(lengthbuffer));
memcpy(lengthbuffer, ptr, ptmp2 - ptr); sumlength = atoi(lengthbuffer); if
(sumlength > 0) { if (false == system_upgrade(pusrdata, sumlength)) {
system_upgrade_flag_set(UPGRADE_FLAG_IDLE); goto ota_recycle; } flash_erased =
true; ptr = (char *) strstr(pusrdata, "\r\n\r\n"); if (false ==
system_upgrade(ptr + 4, length)) { system_upgrade_flag_set(UPGRADE_FLAG_IDLE);
goto ota_recycle; } totallength += length; return; } } else {
system_upgrade_flag_set(UPGRADE_FLAG_IDLE); goto ota_recycle; } }
* 上面我们拿到了这个总数据,那么我们下载固件时候又是怎么样判断是否拉取全部呢?其实通过我们拉取的内容大小和这个总数相比较,就可以得到是否下载完毕。
一般我们都是下载的数据量除于总数据量就是百分比了!由于esp8266不支持浮点类型,也就是不支持小数。
但是我们又要显示进度咋办?yiban其实很简单,算法如下:

* sumlength为总数据量,totallength是已下载的数据量! int part =sumlength/100; int projress =
(totallength / part); printf("已经下载: %d , 总数据: %d , 进度: %d\% \n", totallength,
sumlength,projress);
<>五、扇区的擦除和烧写?

*
要弄明白esp8266的OTA,必须要对其的扇区明明白白。我们现在最常用的是esp8266-12f,其的容量大小是32Mbit,也就是最大代码是1M
,2个分区:1024+1024!而大家玩的esp8266-01是8Mbit, 也即是512+512。

*
从下面官方给的图可以看到,一个支持OTA的扇区分布是对称的,如果你的是8Mbit的话,那么最大的支持就是512K,而32Mbit的就是最大支持1M
,其他空间用户数据!说白了,加载的只是这2个空间的任意一个固件!



* 下面我再弄一个各个容量大小分布的:


<>六、如何调用?

* 因为个人小徐是个喜欢封装代码降低耦合的程序员,做项目都要明确划分功能。那么这次的封装也是基于面对对象思想做的,用回调实现进度显示:
* 判断当前是处于哪个分区,如果是分区一那么我们拉取的就是分区二的代码,反而也是这样; //注册回调函数
system_ota_register_callBack(ota_call_back); //如果当前是处于 user1.bin,那么就拉取云端
user2.bin 升级 if(system_upgrade_userbin_check()==UPGRADE_FW_BIN1){ printf("now
is user1.bin\n");
system_ota_config_start("www.baidu.com","api/v1/emq/download/iot?bin2.bin",80,true,true);
//如果当前是处于 user2.bin,那么就拉取云端 user1.bin 升级 }else
if(system_upgrade_userbin_check()==UPGRADE_FW_BIN2){ printf("now is
user2.bin.\n");
system_ota_config_start("www.baidu.com","api/v1/emq/download/iot?bin1.bin",80,true,true);
}else{ //不支持云端升级 printf("now is not support FOTA..\n"); }
* 看看我们刚刚注册的那个回调函数: void ota_call_back(int progress, ota_code code) {
//下载中,可看到进度数值。 if (code == OTA_DOWNLOADING) { printf("ota_DowmLoading: %d%%
\n",progress); //下载完毕 } else if (code == OTA_SUCCEED) { printf("OTA_SUCCEED!
\n"); system_upgrade_deinit(); // 看你需要,下载完毕是否重启系统 system_upgrade_reboot(); } }
* 这是小徐专心封装的函数和方法,直接调用即可: typedef enum { OTA_FAIL = 0, //下载失败,可能是网速问题
OTA_SUCCEED , //下载并且烧录成功,请重启系统 OTA_DOWNLOADING , //下载中,可看进度条 OTA_INIT_ERROR ,
//初始化失败,网址有误。 OTA_INIT_OK , //初始化成功。 }ota_code; //progress是进度条,code是状态码 typedef
void (*ota_download_CallBack)(int progress,ota_code code);
//注册下载服务器bin文件的进度百分比回调函数:0到100% (整型); void
system_ota_register_callBack(ota_download_CallBack callBack);
/******************************************************************************
* FunctionName : system_ota_config_start * Description : 传入网址,并且开始下载 *
Parameters : char *domainName : 域名或ip地址 * char* requestResource : 请求的资源地址 * int
port : 端口号 * bool isDNS : 是否要DNS解析 * bool isGet : get请求或者post提交 * Returns :
是否成功
*******************************************************************************/
bool system_ota_config_start(char *domainName, char* requestResource,int port,
bool isDNS, bool isGet);
<>七、好好享用吧!

* 下面是我串口打印截图:




<>八、下载:

*
这个是小徐根据官方的代码。适配了私有服务器、适配了返回数据内容的,而且还支持下载进度显示,真是老费力了!
所以代码收点小费,也就是一包烟的事情。这个比某宝上的方案代码便宜了几十倍。

*
这个demo需要注意的是:如果你的服务器地址直接是IP的,就不用DNS解析;如果是域名,那么就需要DNS
的;还有需要啰嗦的一点就是:注意你的网络良好、服务器是否访问、还有最为重要的分区说明。

*
好了,如果你确保上面问题,那么这个demo发生有趣的问题,就是不断来回升级,在两个分区跳转。。呵呵。

*
代码工程:http://www.demodashi.com/demo/13533.html
<http://www.demodashi.com/demo/13533.html>

*
esp8266学习汇总:https://github.com/xuhongv/StudyInEsp8266
<https://github.com/xuhongv/StudyInEsp8266>

*
esp32学习汇总:https://github.com/xuhongv/StudyInEsp32
<https://github.com/xuhongv/StudyInEsp32>

*
小徐QQ群,少走弯路,玩转8266SDK编程:434878850

<>九、注意事项:

* 根据不同的服务器不一样,有可能会出现多余的body内容返回,导致下载成功,但是校验不通过的情况,从而升级失败!,感谢群里的一个小伙伴的解决方法提供如下:
* 修改下接受的校验和处理即可!!