ESPHome 2026.5.0-dev
Loading...
Searching...
No Matches
deep_sleep_component.h
Go to the documentation of this file.
1#pragma once
2
5#include "esphome/core/hal.h"
7
8#ifdef USE_ESP32
9#include <esp_sleep.h>
10#endif
11
12#ifdef USE_TIME
14#include "esphome/core/time.h"
15#endif
16
17#include <cinttypes>
18
19namespace esphome {
20namespace deep_sleep {
21
22#if defined(USE_ESP32) || defined(USE_BK72XX)
23
36#endif
37
38#if defined(USE_BK72XX)
44#endif // USE_BK72XX
45
46#ifdef USE_ESP32
47#if defined(USE_ESP32) && !defined(USE_ESP32_VARIANT_ESP32C2) && !defined(USE_ESP32_VARIANT_ESP32C3)
48struct Ext1Wakeup {
49 uint64_t mask;
50 esp_sleep_ext1_wakeup_mode_t wakeup_mode;
51};
52#endif
53
55 // Run duration if woken up by timer or any other reason besides those below.
57 // Run duration if woken up by touch pads.
59 // Run duration if woken up by GPIO pins.
61};
62
63#endif // USE_ESP32
64
65template<typename... Ts> class EnterDeepSleepAction;
66
67template<typename... Ts> class PreventDeepSleepAction;
68
76 public:
78 void set_sleep_duration(uint32_t time_ms);
79#if defined(USE_ESP32)
83 void set_wakeup_pin(InternalGPIOPin *pin) { this->wakeup_pin_ = pin; }
84
85 void set_wakeup_pin_mode(WakeupPinMode wakeup_pin_mode);
86#endif // USE_ESP32
87
88#if defined(USE_BK72XX)
89 void init_wakeup_pins_(size_t capacity) { this->wakeup_pins_.init(capacity); }
90 void add_wakeup_pin(InternalGPIOPin *wakeup_pin, WakeupPinMode wakeup_pin_mode) {
91 this->wakeup_pins_.emplace_back(WakeUpPinItem{wakeup_pin, wakeup_pin_mode, !wakeup_pin->is_inverted()});
92 }
93#endif // USE_BK72XX
94
95#if defined(USE_ESP32)
96#if !defined(USE_ESP32_VARIANT_ESP32C2) && !defined(USE_ESP32_VARIANT_ESP32C3)
97 void set_ext1_wakeup(Ext1Wakeup ext1_wakeup);
98#endif
99
100#if !defined(USE_ESP32_VARIANT_ESP32C2) && !defined(USE_ESP32_VARIANT_ESP32C3) && \
101 !defined(USE_ESP32_VARIANT_ESP32C6) && !defined(USE_ESP32_VARIANT_ESP32C61) && !defined(USE_ESP32_VARIANT_ESP32H2)
102 void set_touch_wakeup(bool touch_wakeup);
103#endif
104
105 // Set the duration in ms for how long the code should run before entering
106 // deep sleep mode, according to the cause the ESP32 has woken.
107 void set_run_duration(WakeupCauseToRunDuration wakeup_cause_to_run_duration);
108#endif // USE_ESP32
109
111 void set_run_duration(uint32_t time_ms);
112
113 void setup() override;
114 void dump_config() override;
115 void loop() override;
116 float get_setup_priority() const override;
117
119 void begin_sleep(bool manual = false);
120
121 void prevent_deep_sleep();
122 void allow_deep_sleep();
123
124 protected:
125 // Returns nullopt if no run duration is set. Otherwise, returns the run
126 // duration before entering deep sleep.
127 optional<uint32_t> get_run_duration_() const;
128
130 bool prepare_to_sleep_();
131 void deep_sleep_();
132
133#ifdef USE_BK72XX
134 bool pin_prevents_sleep_(WakeUpPinItem &pinItem) const;
135 bool get_real_pin_state_(InternalGPIOPin &pin) const { return (pin.digital_read() ^ pin.is_inverted()); }
136#endif // USE_BK72XX
137
138 optional<uint64_t> sleep_duration_;
139
140#ifdef USE_BK72XX
142#endif // USE_BK72XX
143
144#ifdef USE_ESP32
147
148#if !defined(USE_ESP32_VARIANT_ESP32C2) && !defined(USE_ESP32_VARIANT_ESP32C3)
149 optional<Ext1Wakeup> ext1_wakeup_;
150#endif
151
152 optional<bool> touch_wakeup_;
153
154 optional<WakeupCauseToRunDuration> wakeup_cause_to_run_duration_;
155#endif // USE_ESP32
156
157 optional<uint32_t> run_duration_;
159 bool prevent_{false};
160};
161
162extern bool global_has_deep_sleep; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
163
164template<typename... Ts> class EnterDeepSleepAction : public Action<Ts...> {
165 public:
167 TEMPLATABLE_VALUE(uint32_t, sleep_duration);
168
169#ifdef USE_TIME
170 void set_until(uint8_t hour, uint8_t minute, uint8_t second) {
171 this->hour_ = hour;
172 this->minute_ = minute;
173 this->second_ = second;
174 }
175
176 void set_time(time::RealTimeClock *time) { this->time_ = time; }
177#endif
178
179 void play(const Ts &...x) override {
180 if (this->sleep_duration_.has_value()) {
181 this->deep_sleep_->set_sleep_duration(this->sleep_duration_.value(x...));
182 }
183#ifdef USE_TIME
184
185 if (this->hour_.has_value()) {
186 auto time = this->time_->now();
187 const uint32_t timestamp_now = time.timestamp;
188
189 bool after_time = false;
190 if (time.hour > this->hour_) {
191 after_time = true;
192 } else {
193 if (time.hour == this->hour_) {
194 if (time.minute > this->minute_) {
195 after_time = true;
196 } else {
197 if (time.minute == this->minute_) {
198 if (time.second > this->second_) {
199 after_time = true;
200 }
201 }
202 }
203 }
204 }
205
206 time.hour = *this->hour_;
207 time.minute = *this->minute_;
208 time.second = *this->second_;
209 time.recalc_timestamp_utc();
210
211 time_t timestamp = time.timestamp; // timestamp in local time zone
212
213 if (after_time)
214 timestamp += 60 * 60 * 24;
215
216 int32_t offset = ESPTime::timezone_offset();
217 timestamp -= offset; // Change timestamp to utc
218 const uint32_t ms_left = (timestamp - timestamp_now) * 1000;
219 this->deep_sleep_->set_sleep_duration(ms_left);
220 }
221#endif
222 this->deep_sleep_->begin_sleep(true);
223 }
224
225 protected:
227#ifdef USE_TIME
228 optional<uint8_t> hour_;
229 optional<uint8_t> minute_;
230 optional<uint8_t> second_;
231
233#endif
234};
235
236template<typename... Ts> class PreventDeepSleepAction : public Action<Ts...>, public Parented<DeepSleepComponent> {
237 public:
238 void play(const Ts &...x) override { this->parent_->prevent_deep_sleep(); }
239};
240
241template<typename... Ts> class AllowDeepSleepAction : public Action<Ts...>, public Parented<DeepSleepComponent> {
242 public:
243 void play(const Ts &...x) override { this->parent_->allow_deep_sleep(); }
244};
245
246} // namespace deep_sleep
247} // namespace esphome
Fixed-capacity vector - allocates once at runtime, never reallocates This avoids std::vector template...
Definition helpers.h:522
virtual bool digital_read()=0
virtual bool is_inverted() const =0
Helper class to easily give an object a parent of type T.
Definition helpers.h:2013
This component allows setting up the node to go into deep sleep mode to conserve battery.
void set_run_duration(WakeupCauseToRunDuration wakeup_cause_to_run_duration)
optional< uint32_t > get_run_duration_() const
bool pin_prevents_sleep_(WakeUpPinItem &pinItem) const
void set_wakeup_pin_mode(WakeupPinMode wakeup_pin_mode)
bool get_real_pin_state_(InternalGPIOPin &pin) const
void add_wakeup_pin(InternalGPIOPin *wakeup_pin, WakeupPinMode wakeup_pin_mode)
void begin_sleep(bool manual=false)
Helper to enter deep sleep mode.
void set_wakeup_pin(InternalGPIOPin *pin)
Set the pin to wake up to on the ESP32 once it's in deep sleep mode.
void set_sleep_duration(uint32_t time_ms)
Set the duration in ms the component should sleep once it's in deep sleep mode.
void set_ext1_wakeup(Ext1Wakeup ext1_wakeup)
optional< WakeupCauseToRunDuration > wakeup_cause_to_run_duration_
EnterDeepSleepAction(DeepSleepComponent *deep_sleep)
TEMPLATABLE_VALUE(uint32_t, sleep_duration)
void set_until(uint8_t hour, uint8_t minute, uint8_t second)
The RealTimeClock class exposes common timekeeping functions via the device's local real-time clock.
ESPTime now()
Get the time in the currently defined timezone.
uint8_t second
uint8_t minute
uint8_t hour
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.
@ WAKEUP_PIN_MODE_IGNORE
Ignore the fact that we will wake up when going into deep sleep.
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
uint16_t uint16_t & capacity
Definition helpers.cpp:25
static void uint32_t
static int32_t timezone_offset()
Definition time.cpp:353
time_t timestamp
unix epoch time (seconds since UTC Midnight January 1, 1970)
Definition time.h:48
esp_sleep_ext1_wakeup_mode_t wakeup_mode
uint16_t x
Definition tt21100.cpp:5
uint16_t timestamp
Definition tt21100.cpp:2