ESPHome 2025.9.1
Loading...
Searching...
No Matches
deep_sleep_esp32.cpp
Go to the documentation of this file.
1#ifdef USE_ESP32
2#include "soc/soc_caps.h"
3#include "driver/gpio.h"
5#include "esphome/core/log.h"
6
7namespace esphome {
8namespace deep_sleep {
9
10static const char *const TAG = "deep_sleep";
11
13 if (this->wakeup_cause_to_run_duration_.has_value()) {
14 esp_sleep_wakeup_cause_t wakeup_cause = esp_sleep_get_wakeup_cause();
15 switch (wakeup_cause) {
16 case ESP_SLEEP_WAKEUP_EXT0:
17 case ESP_SLEEP_WAKEUP_EXT1:
18 case ESP_SLEEP_WAKEUP_GPIO:
19 return this->wakeup_cause_to_run_duration_->gpio_cause;
20 case ESP_SLEEP_WAKEUP_TOUCHPAD:
21 return this->wakeup_cause_to_run_duration_->touch_cause;
22 default:
23 return this->wakeup_cause_to_run_duration_->default_cause;
24 }
25 }
26 return this->run_duration_;
27}
28
30 this->wakeup_pin_mode_ = wakeup_pin_mode;
31}
32
33#if !defined(USE_ESP32_VARIANT_ESP32C3) && !defined(USE_ESP32_VARIANT_ESP32C6)
34void DeepSleepComponent::set_ext1_wakeup(Ext1Wakeup ext1_wakeup) { this->ext1_wakeup_ = ext1_wakeup; }
35
36#if !defined(USE_ESP32_VARIANT_ESP32H2)
37void DeepSleepComponent::set_touch_wakeup(bool touch_wakeup) { this->touch_wakeup_ = touch_wakeup; }
38#endif
39
40#endif
41
43 wakeup_cause_to_run_duration_ = wakeup_cause_to_run_duration;
44}
45
47 if (wakeup_pin_ != nullptr) {
48 LOG_PIN(" Wakeup Pin: ", this->wakeup_pin_);
49 }
50 if (this->wakeup_cause_to_run_duration_.has_value()) {
51 ESP_LOGCONFIG(TAG,
52 " Default Wakeup Run Duration: %" PRIu32 " ms\n"
53 " Touch Wakeup Run Duration: %" PRIu32 " ms\n"
54 " GPIO Wakeup Run Duration: %" PRIu32 " ms",
55 this->wakeup_cause_to_run_duration_->default_cause, this->wakeup_cause_to_run_duration_->touch_cause,
56 this->wakeup_cause_to_run_duration_->gpio_cause);
57 }
58}
59
61 if (this->wakeup_pin_mode_ == WAKEUP_PIN_MODE_KEEP_AWAKE && this->wakeup_pin_ != nullptr &&
62 this->wakeup_pin_->digital_read()) {
63 // Defer deep sleep until inactive
64 if (!this->next_enter_deep_sleep_) {
65 this->status_set_warning();
66 ESP_LOGW(TAG, "Waiting for wakeup pin state change");
67 }
68 this->next_enter_deep_sleep_ = true;
69 return false;
70 }
71 return true;
72}
73
75#if !defined(USE_ESP32_VARIANT_ESP32C3) && !defined(USE_ESP32_VARIANT_ESP32C6) && !defined(USE_ESP32_VARIANT_ESP32H2)
76 if (this->sleep_duration_.has_value())
77 esp_sleep_enable_timer_wakeup(*this->sleep_duration_);
78 if (this->wakeup_pin_ != nullptr) {
79 const auto gpio_pin = gpio_num_t(this->wakeup_pin_->get_pin());
81 gpio_sleep_set_pull_mode(gpio_pin, GPIO_PULLUP_ONLY);
82 } else if (this->wakeup_pin_->get_flags() & gpio::FLAG_PULLDOWN) {
83 gpio_sleep_set_pull_mode(gpio_pin, GPIO_PULLDOWN_ONLY);
84 }
85 gpio_sleep_set_direction(gpio_pin, GPIO_MODE_INPUT);
86 gpio_hold_en(gpio_pin);
87#if !SOC_GPIO_SUPPORT_HOLD_SINGLE_IO_IN_DSLP
88 // Some ESP32 variants support holding a single GPIO during deep sleep without this function
89 // For those variants, gpio_hold_en() is sufficient to hold the pin state during deep sleep
90 gpio_deep_sleep_hold_en();
91#endif
92 bool level = !this->wakeup_pin_->is_inverted();
94 level = !level;
95 }
96 esp_sleep_enable_ext0_wakeup(gpio_pin, level);
97 }
98 if (this->ext1_wakeup_.has_value()) {
99 esp_sleep_enable_ext1_wakeup(this->ext1_wakeup_->mask, this->ext1_wakeup_->wakeup_mode);
100 }
101
102 if (this->touch_wakeup_.has_value() && *(this->touch_wakeup_)) {
103 esp_sleep_enable_touchpad_wakeup();
104 esp_sleep_pd_config(ESP_PD_DOMAIN_RTC_PERIPH, ESP_PD_OPTION_ON);
105 }
106#endif
107
108#if defined(USE_ESP32_VARIANT_ESP32H2)
109 if (this->sleep_duration_.has_value())
110 esp_sleep_enable_timer_wakeup(*this->sleep_duration_);
111 if (this->ext1_wakeup_.has_value()) {
112 esp_sleep_enable_ext1_wakeup(this->ext1_wakeup_->mask, this->ext1_wakeup_->wakeup_mode);
113 }
114#endif
115
116#if defined(USE_ESP32_VARIANT_ESP32C3) || defined(USE_ESP32_VARIANT_ESP32C6)
117 if (this->sleep_duration_.has_value())
118 esp_sleep_enable_timer_wakeup(*this->sleep_duration_);
119 if (this->wakeup_pin_ != nullptr) {
120 const auto gpio_pin = gpio_num_t(this->wakeup_pin_->get_pin());
121 if (this->wakeup_pin_->get_flags() && gpio::FLAG_PULLUP) {
122 gpio_sleep_set_pull_mode(gpio_pin, GPIO_PULLUP_ONLY);
123 } else if (this->wakeup_pin_->get_flags() && gpio::FLAG_PULLDOWN) {
124 gpio_sleep_set_pull_mode(gpio_pin, GPIO_PULLDOWN_ONLY);
125 }
126 gpio_sleep_set_direction(gpio_pin, GPIO_MODE_INPUT);
127 gpio_hold_en(gpio_pin);
128#if !SOC_GPIO_SUPPORT_HOLD_SINGLE_IO_IN_DSLP
129 // Some ESP32 variants support holding a single GPIO during deep sleep without this function
130 // For those variants, gpio_hold_en() is sufficient to hold the pin state during deep sleep
131 gpio_deep_sleep_hold_en();
132#endif
133 bool level = !this->wakeup_pin_->is_inverted();
135 level = !level;
136 }
137 esp_deep_sleep_enable_gpio_wakeup(1 << this->wakeup_pin_->get_pin(),
138 static_cast<esp_deepsleep_gpio_wake_up_mode_t>(level));
139 }
140#endif
141 esp_deep_sleep_start();
142}
143
144} // namespace deep_sleep
145} // namespace esphome
146#endif
void status_set_warning(const char *message=nullptr)
virtual gpio::Flags get_flags() const =0
Retrieve GPIO pin flags.
virtual bool digital_read()=0
virtual uint8_t get_pin() const =0
virtual bool is_inverted() const =0
void set_run_duration(WakeupCauseToRunDuration wakeup_cause_to_run_duration)
optional< uint32_t > get_run_duration_() const
void set_wakeup_pin_mode(WakeupPinMode wakeup_pin_mode)
void set_ext1_wakeup(Ext1Wakeup ext1_wakeup)
optional< WakeupCauseToRunDuration > wakeup_cause_to_run_duration_
bool has_value() const
Definition optional.h:92
WakeupPinMode
The values of this enum define what should be done if deep sleep is set up with a wakeup pin on the E...
@ WAKEUP_PIN_MODE_KEEP_AWAKE
As long as the wakeup pin is still in the wakeup state, keep awake.
@ WAKEUP_PIN_MODE_INVERT_WAKEUP
Automatically invert the wakeup level.
@ FLAG_PULLUP
Definition gpio.h:21
@ FLAG_PULLDOWN
Definition gpio.h:22
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7