17.deepsleep深睡眠模式

一. 关于UPL协处理器的概念

ESP32 有强大的超低功耗协处理器 (ULP co-processor)


ULP 协处理器是一个功耗极低的协处理器设备,无论主 CPU 是处于正常运行模式还是 Deep-sleep 模式,ULP 协处理器都可以独立运行。超低功耗协处理器的补充使得 ESP32 能够胜任一些对低功耗要求较高的应用场合。

ULP 协处理器的主要特性有:

采用 8 MHz 频率和 8 KB 内存
内建 ADC 和 I2C 接口
支持正常模式和 Deep-sleep 模式
可唤醒主 CPU 或向主 CPU 发送中断
能够访问主 CPU 的外围设备、内部传感器及 RTC 寄存器

鉴于以上的特性,ULP 协处理器能够在消耗较低电流的情况下,完成 ADC 采样,进行 I2C Sensor 的读写,驱动 RTC GPIO 口动作,可以在某些超低功耗场景中完全替代主 CPU。

重要的: ULP是ESP32做出优秀低功耗产品的关键

但是致命的: ULP只能用汇编😂😂😂😂

参考文档: https://blog.csdn.net/espressif/article/details/79131076

二. 理解了ULP之后, 让我们看一下ESP32的工作模式


可以看到ESP32在深睡眠模式下可以启动或停止ULP协处理器

三. ESP32 deepsleep模式唤醒方式及关键API

唤醒方式:
定时器唤醒
两种引脚唤醒方式
触摸按键唤醒
ULP唤醒
1. 开始进入深睡眠: esp_deep_sleep_start();
esp_deep_sleep_start();


2. 获取esp32被唤醒的原因 esp_deep_sleep_get_wakeup_cause();

这是一个ESP-IDF的原生方法, 如果我们想用, 需要引入头文件
#include <esp_sleep.h>
注意:#include <esp_deep_sleep.h>即将被弃用, 所以不要再用这个头文件了

Serial.println(esp_deep_sleep_get_wakeup_cause());
返回: 被唤醒原因码:

原因码 对应原因 说明
0 ESP_SLEEP_WAKEUP_UNDEFINED 没有定义被唤醒的原因(第一次启动时会报)
2 ESP_SLEEP_WAKEUP_EXT0 被RTC_GPIO唤醒
3 ESP_SLEEP_WAKEUP_EXT1 被RTC_CNTL引脚集合的变化唤醒
4 ESP_SLEEP_WAKEUP_TIMER 被ESP的定时器唤醒
5 ESP_SLEEP_WAKEUP_TOUCHPAD 被触摸唤醒
6 ESP_SLEEP_WAKEUP_ULP 被ULP唤醒
7 ESP_SLEEP_WAKEUP_GPIO 被GPIO唤醒(仅限轻睡眠模式light sleep)
8 ESP_SLEEP_WAKEUP_UART 被串口唤醒(仅限轻睡眠模式light sleep)


3. 设置具体的唤醒源请看下面的相关章节


四. RTC memory

ESP32有8KB的RTC存储器
在RTC memory里的变量不会因为deepsleep被清除, 创建方法:

RTC_DATA_ATTR int bootCount = 0;
注意, RTC memory会被硬件reset清除

五. 进入休眠后定时器唤醒

esp_sleep_enable_timer_wakeup(20000000);
参数:

定时时间,单位μ秒, 类型uint64_t, 所以定时时间要在584942年以内😂😂😂
例子:
#include <Arduino.h>
#include <esp_sleep.h>
 
RTC_DATA_ATTR int bootCount = 0;
 
void setup()
{
  Serial.begin(115200);
  Serial.printf("ESP32 is restart now! It's the %d time\r\n", ++bootCount);
  delay(5000);
  esp_sleep_enable_timer_wakeup(20000000);
  Serial.println(esp_sleep_get_wakeup_cause());
}
 
void loop()
{
  Serial.println("ESP32 will sleep now!");
  delay(100);
  esp_deep_sleep_start();
}



六. 进入休眠后被RTC_GPIO唤醒 (引脚唤醒方式一)

首先,并不是每个GPIO都是RTC_GPIO, 详见下表



注意: 我们填写的GPIO引脚号是真正的引脚号 不是其RTC_GPIO编号
esp_sleep_enable_ext0_wakeup(GPIO_NUM_35, 0);


#include <Arduino.h>
#include <esp_sleep.h>
 
RTC_DATA_ATTR int bootCount = 0;
 
void setup()
{
  Serial.begin(115200);
  Serial.printf("ESP32 is restart now! It's the %d time\r\n", ++bootCount);
  esp_sleep_enable_ext0_wakeup(GPIO_NUM_35, 0);
  Serial.printf("the wakeup reason is :%d\r\n", esp_sleep_get_wakeup_cause());
}
 
void loop()
{
  delay(3000);
  Serial.println("ESP32 will sleep now!");
  delay(100);
  esp_deep_sleep_start();
}


七. 进入休眠后被RTC_CNTL唤醒 (引脚唤醒方式二)

思考一个问题, 如果我们有8个GPIO引脚想唤醒ESP32, 难道要用上一节的方法操作8遍吗?

当然不是, 我们可以直接操作引脚集合, (我们用一个mask片选想操作的引脚,然后这些引脚都具有了唤醒ESP32的能力)

我们可以设置这些引脚是 每个都能触发(每个葫芦娃都能自己去救爷爷), 还是一起共同发力才能触发(集齐七龙珠??)

esp_sleep_enable_ext1_wakeup(uint64_t mask, esp_sleep_ext1_wakeup_mode_t mode);
参数:
mask :

如: 我们想让 32 33 35 39触发, 这样计算mask

注意: 我们不要使用 37 38

mode: 触发方式 可选:
ESP_EXT1_WAKEUP_ALL_LOW : 全都置低时触发唤醒
ESP_EXT1_WAKEUP_ANY_HIGH : 任意置高时触发唤醒

八. 进入休眠后被触摸按键唤醒

值得注意的是,

触摸按键唤醒所需deepsleep电流要大于 按键和定时器
必须写触摸回调函数, 否则无用
#include <Arduino.h>
#include <esp_sleep.h>
 
RTC_DATA_ATTR int bootCount = 0;
RTC_DATA_ATTR int BTN_Pin_BITMASK = 0;
 
 
void callbackPin2()
{
  Serial.println("T2 weak ESP32 up");
}
 
void setup()
{
  Serial.begin(115200);
  Serial.printf("ESP32 is restart now! It's the %d time\r\n", ++bootCount);
  esp_sleep_enable_touchpad_wakeup();
  Serial.printf("the wakeup reason is :%d\r\n", esp_sleep_get_wakeup_cause());
  touchAttachInterrupt(2,callbackPin2,40);
}
 
void loop()
{
  delay(3000);
  Serial.println("ESP32 will sleep now!");
  delay(100);
  esp_deep_sleep_start();
}

九. 被ULP唤醒
这个要做一个专门的ULP专题
————————————————
链接:https://blog.csdn.net/finedayforu/article/details/108547038

 


评论:
相关文章
MimiClaw应用与开发教程1:部署和测试

MimiClaw‌ 是一款基于 ‌ESP32-S3‌ 芯片的超轻量级AI助手,适合嵌入式AI与物联网开发者快速部署本地化AI代理。本系列教程基于MimiClaw的Arduino移植版本进行讲解,小节主要讲解部署和测试。


ESP32扫描wifi 热点列表

就像我们用手机打开WiFi功能后可以浏览附近的可用WiFi。要将手机连接到热点,通常需要打开Wi-Fi设置应用程序,列出可用的网络,然后选择所需的热点。然后输入密码(或不输入密码),可以使用ESP32进行相同的操作。


MimiClaw 配置飞书机器人和添加硬件控制技能

本文本介绍配置飞书机器人为MimiClaw的一个输入/输出端,和添加一个控制WS2812与LED的控制技能。


ESP32-S3 部署 MimicLaw 完整教程:从零到成功调用 DeepSeek

一块 30 块钱的开发板 + 一个大模型 API,就能做出可以听懂人话的智能硬件。 本文记录完整安装过程和踩坑经验,确保你跟着做就能跑通。


MimiClaw 架构全解析,把 “智能龙虾” 跑在 ESP32 上

本文将从手绘架构图入手,逐层拆解 MimiClaw 的分层设计、核心模块、数据流转与底层实现,带你解剖这只“智能虾”的技术骨架,看懂在 C 语言加持下,AI 智能体如何以可穿戴设备的形态,在你身边稳稳运行、离线服务、主动响应。


如何用 platform.local.txt 深度定制 ESP32 编译流程?

本文介绍如何在不脱离 ArduinoIDE 可视化开发的前提下,通过一个名为 platform.local.txt 的小文件,实现对 ESP32 编译流程的精准控制。


优化Arduino-ESP32程序体积

本文将系统分析程序体积增长的五大根源,并提供经过验证的优化方案,帮助减小固件大小。


开发ESP32大模型AI语音助手-从软件到硬件

本文所DIY的语音助手设备端使用的是MicroPython、服务端是Python,对于很多开发者来说MicroPython入门没难度。


【ESP32 C++教程】Unit10-2:音频录制

本小节使用音频开发框架实现一个音频录制到文件的示例。


ESP32 I2S 接口深度解析:从时序、格式到 ESP-IDF 驱动实战

I2S协议通过BCLK、LRCLK和DATA三线精准传输音频数据,但时序边沿、帧格式、时钟源等细节常引发噪声或断连。本文详解ESP32的I2S实现,从协议原理到ESP-IDF v5.x代码配置,助你避开常见陷阱,确保音频稳定传输。

玩转 ESP32 + Arduino

作者:finedayforu@CSDN   共24讲

玩转 ESP32 + Arduino