在ESP8266(Arduino SDK)上实现接近6Mbps的高速IO

TL; DR 代码见GitHub Gist: Jamesits/high-frequency-square-wave-generator-esp8266.ino

在Arduino UNO上实现了高频方波发生器之后,我把魔爪伸向了便宜量足的ESP8266。它能不能产生符合要求的高频波形呢?

初测

分析ESP8266 Arduino SDK的 digitalWrite() 实现后发现,ESP8266的IO分为两组:0-15号,由同一组寄存器控制;16号独立控制。要实现对0-15号IO口的控制,只需要向 GPOS (置1)和 GPOC (置0)寄存器的相应位置写数据即可。

(为了获得尽可能快的结果,代码省去了几乎所有非必要的功能和函数调用。)

测试结果:

  • 80MHz:约5.88MHz
  • 160MHz CPU:约6.25MHz

看门狗问题

但是这样做还有个问题:ESP8266会以每六秒钟多一点一次的频率重启。(如果你的ESP模块的PIN 4上焊有LED,重启时LED会短暂闪亮一下。)研究发现,ESP8266由于在软件上实现了Wi-Fi和TCP stack,故意设计了双重watchdog:软件watchdog会在各种系统函数里面被调用,用来执行网络操作,SDK提供了一个禁用/启用的函数;硬件watchdog是强制开启的,没有提供禁用方法,喂狗间隔6.7s左右,会在软件watchdog调用时喂。由于大部分人写的代码或多或少会频繁调用一些系统函数,因此不会感知到这两个watchdog的存在;而现在的这份代码由于死循环且循环内不调用任何函数,一开机就会触发watchdog重启。

为了测试喂狗造成的时间问题,我在上面的程序里加入了喂狗函数,然后重新测试。

测试结果:

现在的确不重启了。不幸的是,如图所示,喂狗大约需要430ns,这的确是一笔不小的开销。有没有办法绕过硬件watchdog呢?

经过一番研究,我发现Mongoose OS在ESP8266 SDK中禁用了硬件watchdog。他们是怎么实现的呢?其实,硬件watchdog有一个开关,只不过ESP8266 Arduino SDK开发者认为关掉它会影响基础功能,没有实现。Mongoose OS的 esp_hw_wdt_disable() 函数实现了关闭硬件watchdog功能,那么我们把它抄过来。

测试结果:

结果是非常完美的。

实现固定频率方波发生器

接下来,我们把上一篇文章实现的高频方波发生器移植过来。

测试结果:

设定方波频率实测方波频率误差备注
>250KHz333.27KHz-341.82KHz极限频率以上,误差过大无法使用
250KHz249.95KHz0.02%
100KHz99.98KHz0.02%
50KHz49.99KHz0.02%
1000Hz999.80Hz0.02%
  • 得益于高速CPU,ESP8266在方波发生器程序上的表现远好于Arduino UNO
  • CPU频率设置到80MHz或160MHz不影响测试结果
  • 250KHz处有一个间断点;超过此频率即出现较大误差,且脉宽设定也会失效
  • 其余测试数据上的0.02%误差应该是梦源实验室DSView软件浮点数计算导致的系统误差
  • 如果你设定的方波频率过高,ESP8266可能因为固件bug而死机;因为我们禁用了watchdog,它无法自行重启。如果用于生产环境,请务必注意。

参考:

发表评论