ESPHome 2026.1.3
Loading...
Searching...
No Matches
gpio.cpp
Go to the documentation of this file.
1#ifdef USE_ESP32
2
3#include "gpio.h"
4#include "esphome/core/log.h"
5#include "driver/gpio.h"
6#include "driver/rtc_io.h"
7#include "hal/gpio_hal.h"
8#include "soc/soc_caps.h"
9#include "soc/gpio_periph.h"
10#include <cinttypes>
11
12#if (SOC_RTCIO_PIN_COUNT > 0)
13#include "hal/rtc_io_hal.h"
14#endif
15
16#ifndef SOC_GPIO_SUPPORT_RTC_INDEPENDENT
17#define SOC_GPIO_SUPPORT_RTC_INDEPENDENT 0 // NOLINT
18#endif
19
20namespace esphome {
21namespace esp32 {
22
23static const char *const TAG = "esp32";
24
25static const gpio_hal_context_t GPIO_HAL = {.dev = GPIO_HAL_GET_HW(GPIO_PORT_0)};
26
27bool ESP32InternalGPIOPin::isr_service_installed = false; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
28
29static gpio_mode_t flags_to_mode(gpio::Flags flags) {
32 return GPIO_MODE_INPUT;
34 return GPIO_MODE_OUTPUT;
36 return GPIO_MODE_OUTPUT_OD;
38 return GPIO_MODE_INPUT_OUTPUT_OD;
40 return GPIO_MODE_INPUT_OUTPUT;
41 // unsupported or gpio::FLAG_NONE
42 return GPIO_MODE_DISABLE;
43}
44
45struct ISRPinArg {
46 gpio_num_t pin;
48 bool inverted;
49#if defined(USE_ESP32_VARIANT_ESP32)
50 bool use_rtc;
51 int rtc_pin;
52#endif
53};
54
56 auto *arg = new ISRPinArg{}; // NOLINT(cppcoreguidelines-owning-memory)
57 arg->pin = this->get_pin_num();
58 arg->flags = gpio::FLAG_NONE;
59 arg->inverted = this->pin_flags_.inverted;
60#if defined(USE_ESP32_VARIANT_ESP32)
61 arg->use_rtc = rtc_gpio_is_valid_gpio(this->get_pin_num());
62 if (arg->use_rtc)
63 arg->rtc_pin = rtc_io_number_get(this->get_pin_num());
64#endif
65 return ISRInternalGPIOPin((void *) arg);
66}
67
68void ESP32InternalGPIOPin::attach_interrupt(void (*func)(void *), void *arg, gpio::InterruptType type) const {
69 gpio_int_type_t idf_type = GPIO_INTR_ANYEDGE;
70 switch (type) {
72 idf_type = this->pin_flags_.inverted ? GPIO_INTR_NEGEDGE : GPIO_INTR_POSEDGE;
73 break;
75 idf_type = this->pin_flags_.inverted ? GPIO_INTR_POSEDGE : GPIO_INTR_NEGEDGE;
76 break;
78 idf_type = GPIO_INTR_ANYEDGE;
79 break;
81 idf_type = this->pin_flags_.inverted ? GPIO_INTR_HIGH_LEVEL : GPIO_INTR_LOW_LEVEL;
82 break;
84 idf_type = this->pin_flags_.inverted ? GPIO_INTR_LOW_LEVEL : GPIO_INTR_HIGH_LEVEL;
85 break;
86 }
87 gpio_set_intr_type(this->get_pin_num(), idf_type);
88 gpio_intr_enable(this->get_pin_num());
90 auto res = gpio_install_isr_service(ESP_INTR_FLAG_LEVEL3);
91 if (res != ESP_OK) {
92 ESP_LOGE(TAG, "attach_interrupt(): call to gpio_install_isr_service() failed, error code: %d", res);
93 return;
94 }
96 }
97 gpio_isr_handler_add(this->get_pin_num(), func, arg);
98}
99
100size_t ESP32InternalGPIOPin::dump_summary(char *buffer, size_t len) const {
101 return snprintf(buffer, len, "GPIO%" PRIu32, static_cast<uint32_t>(this->pin_));
102}
103
105 gpio_config_t conf{};
106 conf.pin_bit_mask = 1ULL << static_cast<uint32_t>(this->pin_);
107 conf.mode = flags_to_mode(this->flags_);
108 conf.pull_up_en = this->flags_ & gpio::FLAG_PULLUP ? GPIO_PULLUP_ENABLE : GPIO_PULLUP_DISABLE;
109 conf.pull_down_en = this->flags_ & gpio::FLAG_PULLDOWN ? GPIO_PULLDOWN_ENABLE : GPIO_PULLDOWN_DISABLE;
110 conf.intr_type = GPIO_INTR_DISABLE;
111 gpio_config(&conf);
112 if (this->flags_ & gpio::FLAG_OUTPUT) {
113 gpio_set_drive_capability(this->get_pin_num(), this->get_drive_strength());
114 }
115}
116
118 // can't call gpio_config here because that logs in esp-idf which may cause issues
119 gpio_set_direction(this->get_pin_num(), flags_to_mode(flags));
120 gpio_pull_mode_t pull_mode = GPIO_FLOATING;
122 pull_mode = GPIO_PULLUP_PULLDOWN;
123 } else if (flags & gpio::FLAG_PULLUP) {
124 pull_mode = GPIO_PULLUP_ONLY;
125 } else if (flags & gpio::FLAG_PULLDOWN) {
126 pull_mode = GPIO_PULLDOWN_ONLY;
127 }
128 gpio_set_pull_mode(this->get_pin_num(), pull_mode);
129}
130
132 return bool(gpio_get_level(this->get_pin_num())) != this->pin_flags_.inverted;
133}
135 gpio_set_level(this->get_pin_num(), value != this->pin_flags_.inverted ? 1 : 0);
136}
137void ESP32InternalGPIOPin::detach_interrupt() const { gpio_intr_disable(this->get_pin_num()); }
138
139} // namespace esp32
140
141using namespace esp32;
142
144 auto *arg = reinterpret_cast<ISRPinArg *>(this->arg_);
145 return bool(gpio_hal_get_level(&GPIO_HAL, arg->pin)) != arg->inverted;
146}
147
148void IRAM_ATTR ISRInternalGPIOPin::digital_write(bool value) {
149 auto *arg = reinterpret_cast<ISRPinArg *>(this->arg_);
150 gpio_hal_set_level(&GPIO_HAL, arg->pin, value != arg->inverted);
151}
152
154 // not supported
155}
156
158 auto *arg = reinterpret_cast<ISRPinArg *>(arg_);
159 gpio::Flags diff = (gpio::Flags)(flags ^ arg->flags);
160 if (diff & gpio::FLAG_OUTPUT) {
161 if (flags & gpio::FLAG_OUTPUT) {
162 gpio_hal_output_enable(&GPIO_HAL, arg->pin);
164 gpio_hal_od_enable(&GPIO_HAL, arg->pin);
165 } else {
166 gpio_hal_output_disable(&GPIO_HAL, arg->pin);
167 }
168 }
169 if (diff & gpio::FLAG_INPUT) {
170 if (flags & gpio::FLAG_INPUT) {
171 gpio_hal_input_enable(&GPIO_HAL, arg->pin);
172#if defined(USE_ESP32_VARIANT_ESP32)
173 if (arg->use_rtc) {
174 if (flags & gpio::FLAG_PULLUP) {
175 rtcio_hal_pullup_enable(arg->rtc_pin);
176 } else {
177 rtcio_hal_pullup_disable(arg->rtc_pin);
178 }
180 rtcio_hal_pulldown_enable(arg->rtc_pin);
181 } else {
182 rtcio_hal_pulldown_disable(arg->rtc_pin);
183 }
184 } else
185#endif
186 {
187 if (flags & gpio::FLAG_PULLUP) {
188 gpio_hal_pullup_en(&GPIO_HAL, arg->pin);
189 } else {
190 gpio_hal_pullup_dis(&GPIO_HAL, arg->pin);
191 }
193 gpio_hal_pulldown_en(&GPIO_HAL, arg->pin);
194 } else {
195 gpio_hal_pulldown_dis(&GPIO_HAL, arg->pin);
196 }
197 }
198 } else {
199 gpio_hal_input_disable(&GPIO_HAL, arg->pin);
200 }
201 }
202 arg->flags = flags;
203}
204
205} // namespace esphome
206
207#endif // USE_ESP32
Copy of GPIOPin that is safe to use from ISRs (with no virtual functions)
Definition gpio.h:92
void digital_write(bool value)
Definition gpio.cpp:148
void pin_mode(gpio::Flags flags)
Definition gpio.cpp:157
gpio_num_t get_pin_num() const
Definition gpio.h:33
struct esphome::esp32::ESP32InternalGPIOPin::PinFlags pin_flags_
void attach_interrupt(void(*func)(void *), void *arg, gpio::InterruptType type) const override
Definition gpio.cpp:68
gpio_drive_cap_t get_drive_strength() const
Definition gpio.h:34
void detach_interrupt() const override
Definition gpio.cpp:137
ISRInternalGPIOPin to_isr() const override
Definition gpio.cpp:55
void pin_mode(gpio::Flags flags) override
Definition gpio.cpp:117
void digital_write(bool value) override
Definition gpio.cpp:134
size_t dump_summary(char *buffer, size_t len) const override
Definition gpio.cpp:100
uint16_t type
uint16_t flags
@ INTERRUPT_FALLING_EDGE
Definition gpio.h:51
@ INTERRUPT_RISING_EDGE
Definition gpio.h:50
@ INTERRUPT_HIGH_LEVEL
Definition gpio.h:54
@ INTERRUPT_LOW_LEVEL
Definition gpio.h:53
@ INTERRUPT_ANY_EDGE
Definition gpio.h:52
@ FLAG_OUTPUT
Definition gpio.h:28
@ FLAG_OPEN_DRAIN
Definition gpio.h:29
@ FLAG_NONE
Definition gpio.h:26
@ FLAG_PULLUP
Definition gpio.h:30
@ FLAG_INPUT
Definition gpio.h:27
@ FLAG_PULLDOWN
Definition gpio.h:31
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
std::string size_t len
Definition helpers.h:595