Select one of the symbols to view example projects that use it.
 
Outline
#include <stddef.h>
#include <string.h>
#include <sys/lock.h>
#include <sys/param.h>
#include "esp_attr.h"
#include "esp_sleep.h"
#include "esp_log.h"
#include "esp_memory_utils.h"
#include "soc/soc_caps.h"
#include "sdkconfig.h"
#include "driver/gpio.h"
#include "hal/gpio_hal.h"
#include "hal/rtc_io_hal.h"
#include "soc/rtc_io_periph.h"
#include "hal/rtc_hal.h"
#include "esp_private/gpio.h"
#include "esp_private/sleep_gpio.h"
#include "esp_private/spi_flash_os.h"
#include "esp_private/startup_internal.h"
#include "bootloader_flash.h"
TAG
esp_sleep_config_gpio_isolate()
esp_sleep_enable_gpio_switch(bool)
esp_sleep_isolate_digital_gpio()
esp_deep_sleep_wakeup_io_reset()
Files
loading...
SourceVuESP-IDF Framework and ExamplesESP-IDFcomponents/esp_hw_support/sleep_gpio.c
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
/* * SPDX-FileCopyrightText: 2015-2024 Espressif Systems (Shanghai) CO LTD * * SPDX-License-Identifier: Apache-2.0 *//* ... */ #include <stddef.h> #include <string.h> #include <sys/lock.h> #include <sys/param.h> #include "esp_attr.h" #include "esp_sleep.h" #include "esp_log.h" #include "esp_memory_utils.h" #include "soc/soc_caps.h" #include "sdkconfig.h" #include "driver/gpio.h" #include "hal/gpio_hal.h" #include "hal/rtc_io_hal.h" #include "soc/rtc_io_periph.h" #include "hal/rtc_hal.h" #include "esp_private/gpio.h" #include "esp_private/sleep_gpio.h" #include "esp_private/spi_flash_os.h" #include "esp_private/startup_internal.h" #include "bootloader_flash.h"20 includes static const char *TAG = "sleep_gpio"; #if CONFIG_GPIO_ESP32_SUPPORT_SWITCH_SLP_PULL void gpio_sleep_mode_config_apply(void) { for (gpio_num_t gpio_num = GPIO_NUM_0; gpio_num < GPIO_NUM_MAX; gpio_num++) { if (GPIO_IS_VALID_GPIO(gpio_num)) { gpio_sleep_pupd_config_apply(gpio_num); }{...} }{...} }{...} IRAM_ATTR void gpio_sleep_mode_config_unapply(void) { for (gpio_num_t gpio_num = GPIO_NUM_0; gpio_num < GPIO_NUM_MAX; gpio_num++) { if (GPIO_IS_VALID_GPIO(gpio_num)) { gpio_sleep_pupd_config_unapply(gpio_num); }{...} }{...} }{...} /* ... */#endif void esp_sleep_config_gpio_isolate(void) { ESP_EARLY_LOGI(TAG, "Configure to isolate all GPIO pins in sleep state"); for (gpio_num_t gpio_num = GPIO_NUM_0; gpio_num < GPIO_NUM_MAX; gpio_num++) { if (GPIO_IS_VALID_GPIO(gpio_num)) { gpio_sleep_set_direction(gpio_num, GPIO_MODE_DISABLE); gpio_sleep_set_pull_mode(gpio_num, GPIO_FLOATING); }{...} }{...} #if CONFIG_ESP_SLEEP_MSPI_NEED_ALL_IO_PU gpio_sleep_set_pull_mode(esp_mspi_get_io(ESP_MSPI_IO_CLK), GPIO_PULLUP_ONLY); gpio_sleep_set_pull_mode(esp_mspi_get_io(ESP_MSPI_IO_Q), GPIO_PULLUP_ONLY); gpio_sleep_set_pull_mode(esp_mspi_get_io(ESP_MSPI_IO_D), GPIO_PULLUP_ONLY); gpio_sleep_set_pull_mode(esp_mspi_get_io(ESP_MSPI_IO_HD), GPIO_PULLUP_ONLY); gpio_sleep_set_pull_mode(esp_mspi_get_io(ESP_MSPI_IO_WP), GPIO_PULLUP_ONLY); #if SOC_SPI_MEM_SUPPORT_OPI_MODE bool octal_mspi_required = bootloader_flash_is_octal_mode_enabled(); #if CONFIG_SPIRAM_MODE_OCT octal_mspi_required |= true; #endif // CONFIG_SPIRAM_MODE_OCT if (octal_mspi_required) { gpio_sleep_set_pull_mode(esp_mspi_get_io(ESP_MSPI_IO_DQS), GPIO_PULLUP_ONLY); gpio_sleep_set_pull_mode(esp_mspi_get_io(ESP_MSPI_IO_D4), GPIO_PULLUP_ONLY); gpio_sleep_set_pull_mode(esp_mspi_get_io(ESP_MSPI_IO_D5), GPIO_PULLUP_ONLY); gpio_sleep_set_pull_mode(esp_mspi_get_io(ESP_MSPI_IO_D6), GPIO_PULLUP_ONLY); gpio_sleep_set_pull_mode(esp_mspi_get_io(ESP_MSPI_IO_D7), GPIO_PULLUP_ONLY); }{...} #endif/* ... */ // SOC_SPI_MEM_SUPPORT_OPI_MODE/* ... */ #endif // CONFIG_ESP_SLEEP_MSPI_NEED_ALL_IO_PU }{ ... } void esp_sleep_enable_gpio_switch(bool enable) { ESP_EARLY_LOGI(TAG, "%s automatic switching of GPIO sleep configuration", enable ? "Enable" : "Disable"); for (gpio_num_t gpio_num = GPIO_NUM_0; gpio_num < GPIO_NUM_MAX; gpio_num++) { if (GPIO_IS_VALID_GPIO(gpio_num)) { /* If the PSRAM is disable in ESP32xx chips equipped with PSRAM, there will be a large current leakage. */ #if CONFIG_ESP_SLEEP_PSRAM_LEAKAGE_WORKAROUND && CONFIG_SPIRAM if (gpio_num == esp_mspi_get_io(ESP_MSPI_IO_CS1)) { gpio_sleep_sel_dis(gpio_num); continue; }{...} #endif/* ... */ // CONFIG_ESP_SLEEP_PSRAM_LEAKAGE_WORKAROUND && CONFIG_SPIRAM #if CONFIG_ESP_SLEEP_FLASH_LEAKAGE_WORKAROUND if (gpio_num == esp_mspi_get_io(ESP_MSPI_IO_CS0)) { gpio_sleep_sel_dis(gpio_num); continue; }{...} #endif/* ... */ // CONFIG_ESP_SLEEP_FLASH_LEAKAGE_WORKAROUND if (enable) { gpio_sleep_sel_en(gpio_num); }{...} else { gpio_sleep_sel_dis(gpio_num); }{...} }{...} }{...} }{ ... } #if SOC_GPIO_SUPPORT_HOLD_IO_IN_DSLP && !SOC_GPIO_SUPPORT_HOLD_SINGLE_IO_IN_DSLP IRAM_ATTR void esp_sleep_isolate_digital_gpio(void) { gpio_hal_context_t gpio_hal = { .dev = GPIO_HAL_GET_HW(GPIO_PORT_0) }{...}; /* no need to do isolate if digital IOs are not being held in deep sleep */ if (!gpio_hal_deep_sleep_hold_is_en(&gpio_hal)) { return; }{...} /** * there is a situation where we cannot isolate digital IO before deep sleep: * - task stack is located in external ram(mspi ram), since we will isolate mspi io * * assert here instead of returning directly, because if digital IO is not isolated, * the bottom current of deep sleep will be higher than light sleep, and there is no * reason to use deep sleep at this time. *//* ... */ assert(esp_ptr_internal(&gpio_hal) && "If hold digital IO, the stack of the task calling esp_deep_sleep_start must be in internal ram!"); /* isolate digital IO that is not held(keep the configuration of digital IOs held by users) */ for (gpio_num_t gpio_num = GPIO_NUM_0; gpio_num < GPIO_NUM_MAX; gpio_num++) { if (GPIO_IS_VALID_DIGITAL_IO_PAD(gpio_num) && !gpio_hal_is_digital_io_hold(&gpio_hal, gpio_num)) { /* disable I/O */ gpio_hal_input_disable(&gpio_hal, gpio_num); gpio_hal_output_disable(&gpio_hal, gpio_num); /* disable pull up/down */ gpio_hal_pullup_dis(&gpio_hal, gpio_num); gpio_hal_pulldown_dis(&gpio_hal, gpio_num); /* make pad work as gpio(otherwise, deep sleep bottom current will rise) */ gpio_hal_func_sel(&gpio_hal, gpio_num, PIN_FUNC_GPIO); }{...} }{...} }{ ... } /* ... */#endif //SOC_GPIO_SUPPORT_HOLD_IO_IN_DSLP && !SOC_GPIO_SUPPORT_HOLD_SINGLE_IO_IN_DSLP #if SOC_DEEP_SLEEP_SUPPORTED void esp_deep_sleep_wakeup_io_reset(void) { #if SOC_PM_SUPPORT_EXT1_WAKEUP uint32_t rtc_io_mask = rtc_hal_ext1_get_wakeup_pins(); // Disable ext1 wakeup before releasing hold, such that wakeup status can reflect the correct wakeup pin rtc_hal_ext1_clear_wakeup_pins(); for (int gpio_num = 0; gpio_num < SOC_GPIO_PIN_COUNT && rtc_io_mask != 0; ++gpio_num) { int rtcio_num = rtc_io_num_map[gpio_num]; if ((rtc_io_mask & BIT(rtcio_num)) == 0) { continue; }{...} rtcio_hal_hold_disable(rtcio_num); rtc_io_mask &= ~BIT(rtcio_num); }{...} #endif/* ... */ #if SOC_GPIO_SUPPORT_DEEPSLEEP_WAKEUP uint32_t dl_io_mask = SOC_GPIO_DEEP_SLEEP_WAKE_VALID_GPIO_MASK; gpio_hal_context_t gpio_hal = { .dev = GPIO_HAL_GET_HW(GPIO_PORT_0) }{...}; while (dl_io_mask) { int gpio_num = __builtin_ffs(dl_io_mask) - 1; bool wakeup_io_enabled = gpio_hal_deepsleep_wakeup_is_enabled(&gpio_hal, gpio_num); if (wakeup_io_enabled) { // Disable the wakeup before releasing hold, such that wakeup status can reflect the correct wakeup pin gpio_hal_deepsleep_wakeup_disable(&gpio_hal, gpio_num); gpio_hal_hold_dis(&gpio_hal, gpio_num); }{...} dl_io_mask &= ~BIT(gpio_num); }{...} #endif/* ... */ }{ ... } /* ... */#endif #if CONFIG_ESP_SLEEP_GPIO_RESET_WORKAROUND || CONFIG_PM_SLP_DISABLE_GPIO ESP_SYSTEM_INIT_FN(esp_sleep_startup_init, SECONDARY, BIT(0), 105) { // Configure to isolate (disable the Input/Output/Pullup/Pulldown // function of the pin) all GPIO pins in sleep state esp_sleep_config_gpio_isolate(); // Enable automatic switching of GPIO configuration esp_sleep_enable_gpio_switch(true); return ESP_OK; }{...} void esp_sleep_gpio_include(void) { // Linker hook function, exists to make the linker examine this file }{...} /* ... */#endif
Details