13、ESP32 深度睡眠

1、深度睡眠

        ESP32 可以在不同的电源模式之间切换:

  • 活动模式
  • 调制解调器睡眠模式
  • 浅睡眠模式
  • 深度睡眠模式
  • 休眠模式

         在深度睡眠模式下,CPU 或 Wi-Fi 活动都不会发生,但超低功耗 (ULP) 协处理器仍可打开电源,RTC 内存也会保持通电状态,因此我们可以为 ULP 编写程序并将其存储在 RTC 内存中,以访问外围设备、内部定时器和内部传感器。

        如果需要通过外部事件、计时器唤醒主 CPU,同时保持最小的功耗,则此操作模式非常有用。

        在深度睡眠期间,ULP 协处理器可以使用部分 ESP32 引脚,即 RTC_GPIO 引脚和 Touch 引脚。

        深度睡眠设置步骤:

  1. 首先需要配置唤醒源,可以使用一个或组合多个唤醒源。
  2. 可以决定在深度睡眠期间关闭或保持打开哪些外围设备。默认情况下,ESP32 会自动关闭定义的唤醒源不需要的外设。
  3. 最后,使用 esp_deep_sleep_start() 将 ESP32 置于深度睡眠模式的功能。

2、唤醒源

        将 ESP32 置于深度睡眠模式后,有几种方法可以唤醒它:

  • 使用计时器,使用预定义的时间段唤醒 ESP32;
  • 使用触摸引脚;
  • 使用外部唤醒:可以使用一种外部唤醒,也可以使用几种不同的外部唤醒;
  • 使用 ULP 协处理器唤醒。

       

        进入睡眠函数:esp_deep_sleep_start();

        返回唤醒原因函数:esp_sleep_get_wakeup_cause();


3、定时器唤醒

        ESP32 RTC 控制器具有内置定时器,可用于在预定义的时间后唤醒 ESP32。

        ESP32 可以进入深度睡眠模式,然后在预定义的时间段唤醒。如果正在运行需要时间戳或日常任务的项目,同时保持低功耗,则此功能特别有用。
        启用定时器唤醒只需在以下函数中指定休眠时间(单位微秒):esp_sleep_enable_timer_wakeup(time_in_us);

#include <Arduino.h>


// 使用定时器唤醒时,通电的部分是 RTC 控制器、RTC 外设和 RTC 存储器。

#define uS_TO_S_FACTOR 1000000  /* 微秒到秒的转换系数 */
#define TIME_TO_SLEEP  5        /* ESP32休眠时间(单位秒) */

RTC_DATA_ATTR int bootCount = 0;    // RTC 内存上定义深度睡眠中唤醒的次数 bootCount


// 唤醒原因
void print_wakeup_reason()
{
    esp_sleep_wakeup_cause_t wakeup_reason;

    wakeup_reason = esp_sleep_get_wakeup_cause();

    switch(wakeup_reason)
    {
        case ESP_SLEEP_WAKEUP_EXT0 : Serial.println("Wakeup caused by external signal using RTC_IO"); break;
        case ESP_SLEEP_WAKEUP_EXT1 : Serial.println("Wakeup caused by external signal using RTC_CNTL"); break;
        case ESP_SLEEP_WAKEUP_TIMER : Serial.println("Wakeup caused by timer"); break;
        case ESP_SLEEP_WAKEUP_TOUCHPAD : Serial.println("Wakeup caused by touchpad"); break;
        case ESP_SLEEP_WAKEUP_ULP : Serial.println("Wakeup caused by ULP program"); break;
        default : Serial.printf("Wakeup was not caused by deep sleep: %d\n",wakeup_reason); break;
    }
}


void setup()
{
    Serial.begin(115200);
    delay(2000);        // 打开串行监视器

    
    ++bootCount;    // 每次重新启动时计数
    Serial.println("Boot number: " + String(bootCount));

    print_wakeup_reason();      // 打印ESP32的唤醒原因

    esp_sleep_enable_timer_wakeup(TIME_TO_SLEEP * uS_TO_S_FACTOR);      // 配置唤醒源(微秒单位)
    Serial.println("Setup ESP32 to sleep for every " + String(TIME_TO_SLEEP) +" Seconds");


    Serial.println("Going to sleep now");
    delay(1000);
    Serial.flush(); 
    esp_deep_sleep_start();     // 进入睡眠
    Serial.println("This will never be printed");
}

void loop()
{
}

4、触摸唤醒

        启用触摸唤醒只需调用函数:esp_sleep_enable_touchpad_wakeup();

#include <Arduino.h>


// 触摸引脚设置阈值
#if CONFIG_IDF_TARGET_ESP32
    #define THRESHOLD     40        /* 值越大,灵敏度越高 */
#else       // ESP32-S2 and ESP32-S3 + default for other chips (to be adjusted) */
    #define THRESHOLD     5000      /* 值越小,灵敏度越高 */
#endif

RTC_DATA_ATTR int bootCount = 0;    // RTC 内存上定义深度睡眠中唤醒的次数 bootCount
touch_pad_t touchPin;


// 唤醒原因
void print_wakeup_reason()
{
    esp_sleep_wakeup_cause_t wakeup_reason;

    wakeup_reason = esp_sleep_get_wakeup_cause();

    switch(wakeup_reason)
    {
        case ESP_SLEEP_WAKEUP_EXT0 : Serial.println("Wakeup caused by external signal using RTC_IO"); break;
        case ESP_SLEEP_WAKEUP_EXT1 : Serial.println("Wakeup caused by external signal using RTC_CNTL"); break;
        case ESP_SLEEP_WAKEUP_TIMER : Serial.println("Wakeup caused by timer"); break;
        case ESP_SLEEP_WAKEUP_TOUCHPAD : Serial.println("Wakeup caused by touchpad"); break;
        case ESP_SLEEP_WAKEUP_ULP : Serial.println("Wakeup caused by ULP program"); break;
        default : Serial.printf("Wakeup was not caused by deep sleep: %d\n",wakeup_reason); break;
    }
}


// 被唤醒的触摸 IO
void print_wakeup_touchpad()
{
    touchPin = esp_sleep_get_touchpad_wakeup_status();

    #if CONFIG_IDF_TARGET_ESP32
    switch(touchPin)
    {
        case 0  : Serial.println("Touch detected on GPIO 4"); break;
        case 1  : Serial.println("Touch detected on GPIO 0"); break;
        case 2  : Serial.println("Touch detected on GPIO 2"); break;
        case 3  : Serial.println("Touch detected on GPIO 15"); break;
        case 4  : Serial.println("Touch detected on GPIO 13"); break;
        case 5  : Serial.println("Touch detected on GPIO 12"); break;
        case 6  : Serial.println("Touch detected on GPIO 14"); break;
        case 7  : Serial.println("Touch detected on GPIO 27"); break;
        case 8  : Serial.println("Touch detected on GPIO 33"); break;
        case 9  : Serial.println("Touch detected on GPIO 32"); break;
        default : Serial.println("Wakeup not by touchpad"); break;
    }
    #else
        if(touchPin < TOUCH_PAD_MAX)
        {
            Serial.printf("Touch detected on GPIO %d\n", touchPin); 
        }
        else
        {
            Serial.println("Wakeup not by touchpad");
        }
    #endif
}


void setup()
{
    Serial.begin(115200);

    ++bootCount;    // 每次重新启动时计数
    Serial.println("Boot number: " + String(bootCount));

    print_wakeup_reason();      // 打印唤醒原因
    print_wakeup_touchpad();    // 打印触摸 IO

    #if CONFIG_IDF_TARGET_ESP32 
        touchSleepWakeUpEnable(T3, THRESHOLD);    // 触摸 T3 GPIO15 上设置中断
        touchSleepWakeUpEnable(T7, THRESHOLD);    // 触摸 T7 GPIO27 上设置中断
    #else //ESP32-S2 + ESP32-S3
        touchSleepWakeUpEnable(T3, THRESHOLD);
    #endif

    Serial.println("Going to sleep now");
    esp_deep_sleep_start();     // 进入睡眠
    Serial.println("This will never be printed");
}

void loop()
{
}


5、外部唤醒

        除了定时器和触摸,我们还可以通过切换针脚上的信号值(例如按下按钮)将 ESP32 从深度睡眠中唤醒。这称为外部唤醒。外部唤醒有两种可能性:ext0 和 ext1。

外部唤醒 (ext0)

        ext0 允许使用 RTC GPIO 引脚唤醒 ESP32,如果请求此唤醒源,RTC 外围设备将在深度睡眠期间保持打开状态。

        使用函数:esp_sleep_enable_ext0_wakeup(GPIO_NUM_X, level);

  • GPIO_NUM_X:使用引脚, X 表示该引脚的 GPIO 编号
  • level:表示将触发唤醒的 GPIO 的状态,可以是 1 或 0
#include <Arduino.h>

RTC_DATA_ATTR int bootCount = 0;    // RTC 内存上定义深度睡眠中唤醒的次数 bootCount

// 打印唤醒源
void print_wakeup_reason()
{
    esp_sleep_wakeup_cause_t wakeup_reason;

    wakeup_reason = esp_sleep_get_wakeup_cause();

    switch(wakeup_reason)
    {
        case ESP_SLEEP_WAKEUP_EXT0 : Serial.println("Wakeup caused by external signal using RTC_IO"); break;
        case ESP_SLEEP_WAKEUP_EXT1 : Serial.println("Wakeup caused by external signal using RTC_CNTL"); break;
        case ESP_SLEEP_WAKEUP_TIMER : Serial.println("Wakeup caused by timer"); break;
        case ESP_SLEEP_WAKEUP_TOUCHPAD : Serial.println("Wakeup caused by touchpad"); break;
        case ESP_SLEEP_WAKEUP_ULP : Serial.println("Wakeup caused by ULP program"); break;
        default : Serial.printf("Wakeup was not caused by deep sleep: %d\n",wakeup_reason); break;
    }
}

void setup()
{
    Serial.begin(115200);
    delay(1000);

    ++bootCount;    // 每次重新启动时计数
    Serial.println("Boot number: " + String(bootCount));

    print_wakeup_reason();      // 打印唤醒原因

    esp_sleep_enable_ext0_wakeup(GPIO_NUM_33, 1);     // GPIO 33 高电平触发,ext0 只能使用一个 IO 作为唤醒源

    Serial.println("Going to sleep now");
    delay(1000);
    esp_deep_sleep_start();     // 睡眠
    Serial.println("This will never be printed");
}

void loop()
{
}
外部唤醒 (ext1)

        此唤醒源允许使用多个 RTC GPIO,可以使用两种不同的逻辑函数:

  • 选择的引脚都为高电平,唤醒 ESP32;
  • 选择的引脚都为低电平,唤醒 ESP32。

        此唤醒源由 RTC 控制器实现。因此,RTC 外设和 RTC 存储器可以在此模式下关闭。

        使用函数:esp_sleep_enable_ext1_wakeup(bitmask, mode);

  • bitmask:将导致唤醒的 GPIO 编号的位掩码;
    • 1个引脚位掩码:假如想用 GPIO 33 引脚,首先计算 2^33 的十进制数,然后把十进制转换为 16 进制就是我们需要的 GPIO 33 的位掩码
    • 多个引脚位掩码:假如想用 GPIO 2 和 GPIO 15 作为唤醒源,首先计算 2^2 + 2^15 得到一个十进制数,再转换为十六进制就是我们需要的位掩码
  • mode:唤醒 ESP32 的逻辑。
    • ESP_EXT1_WAKEUP_ALL_LOW:当所有 GPIO 变为低电平时唤醒
    • ESP_EXT1_WAKEUP_ANY_HIGH:如果任何 GPIO 变为高电平唤醒

        当使用多个引脚唤醒 ESP32 时,知道哪个引脚导致唤醒很有用。为此可以使用以下函数:esp_sleep_get_ext1_wakeup_status();,函数会返回以 2 为基数的数字 value,计算可以获得GPIO = log(value) / log(2)

#include <Arduino.h>

#define BUTTON_PIN_BITMASK 0x8004 // GPIOs 2 and 15

RTC_DATA_ATTR int bootCount = 0;    // RTC 内存上定义深度睡眠中唤醒的次数 bootCount

// 打印唤醒源
void print_wakeup_reason()
{
    esp_sleep_wakeup_cause_t wakeup_reason;

    wakeup_reason = esp_sleep_get_wakeup_cause();

    switch(wakeup_reason)
    {
        case ESP_SLEEP_WAKEUP_EXT0 : Serial.println("Wakeup caused by external signal using RTC_IO"); break;
        case ESP_SLEEP_WAKEUP_EXT1 : Serial.println("Wakeup caused by external signal using RTC_CNTL"); break;
        case ESP_SLEEP_WAKEUP_TIMER : Serial.println("Wakeup caused by timer"); break;
        case ESP_SLEEP_WAKEUP_TOUCHPAD : Serial.println("Wakeup caused by touchpad"); break;
        case ESP_SLEEP_WAKEUP_ULP : Serial.println("Wakeup caused by ULP program"); break;
        default : Serial.printf("Wakeup was not caused by deep sleep: %d\n",wakeup_reason); break;
    }
}

void print_GPIO_wake_up()
{
    uint64_t GPIO_reason = esp_sleep_get_ext1_wakeup_status();      // 获取引起唤醒的 gpio 位掩码,如果唤醒由其他源引起返回0
    Serial.print("GPIO that triggered the wake up: GPIO ");
    Serial.println((log(GPIO_reason))/log(2), 0);
}

void setup()
{
    Serial.begin(115200);
    delay(1000);

    ++bootCount;    // 每次重新启动时计数
    Serial.println("Boot number: " + String(bootCount));

    print_wakeup_reason();      // 打印唤醒原因
    print_GPIO_wake_up();       // 打印唤醒 IO

    esp_sleep_enable_ext1_wakeup(BUTTON_PIN_BITMASK, ESP_EXT1_WAKEUP_ANY_HIGH);     // GPIO 2 / 15 高电平唤醒

    Serial.println("Going to sleep now");
    delay(1000);
    esp_deep_sleep_start();     // 睡眠
    Serial.println("This will never be printed");
}

void loop()
{
}

 区别:

        ext0:

                使用 RTC_IO 唤醒,因此需要 RTC 外设,

                只能使用一个 GPIO 作为唤醒源,执行特定任务。
        ext1:

                使用 RTC 控制器,所以不需要上电的外设,

                允许使用不同的按钮唤醒 ESP32,并根据按下的按钮执行不同的任务。
        注意,使用内部下拉 / 下拉也需要打开 RTC 外围设备。

  • 4
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

无敌暴龙战士朵拉

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值