1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/cpu/esp32/periph/gpio.c

500 lines
13 KiB
C

/*
* Copyright (C) 2022 Gunar Schorcht
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @ingroup cpu_esp32
* @ingroup drivers_periph_gpio
* @{
*
* @file
* @brief Low-level GPIO driver implementation for ESP32
*
* @author Gunar Schorcht <gunar@schorcht.net>
* @}
*/
#include <assert.h>
#include <stdbool.h>
#include "log.h"
#include "periph/gpio.h" /* RIOT gpio.h */
#include "hal/gpio_hal.h"
#include "hal/gpio_types.h"
#include "hal/rtc_io_types.h"
#include "esp/common_macros.h"
#include "esp_intr_alloc.h"
#include "rom/ets_sys.h"
#include "soc/gpio_reg.h"
#include "soc/gpio_sig_map.h"
#include "soc/gpio_struct.h"
#include "soc/io_mux_reg.h"
#include "soc/rtc_cntl_reg.h"
#include "soc/rtc_io_periph.h"
#if __xtensa__
#include "xtensa/xtensa_api.h"
#endif
#include "esp_idf_api/gpio.h"
#include "adc_arch.h"
#include "bitfield.h"
#include "esp_common.h"
#include "esp_sleep.h"
#include "gpio_arch.h"
#include "irq_arch.h"
#include "syscalls.h"
#define ENABLE_DEBUG 0
#include "debug.h"
#define ESP_PM_WUP_PINS_ANY_HIGH ESP_EXT1_WAKEUP_ANY_HIGH
#define ESP_PM_WUP_PINS_ALL_LOW ESP_EXT1_WAKEUP_ALL_LOW
#ifndef ESP_PM_WUP_LEVEL
#define ESP_PM_WUP_LEVEL ESP_PM_WUP_PINS_ANY_HIGH
#endif
#define GPIO_PRO_CPU_INTR_ENA (BIT(2))
/* architecture specific tables */
extern gpio_pin_usage_t _gpio_pin_usage [GPIO_PIN_NUMOF];
_Static_assert(ARRAY_SIZE(_gpio_pin_usage) == SOC_GPIO_PIN_COUNT,
"size of _gpio_pin_usage does not match SOC_GPIO_PIN_COUNT");
#if !IS_USED(MODULE_ESP_IDF_GPIO_HAL)
extern const uint32_t _gpio_to_iomux_reg[GPIO_PIN_NUMOF];
_Static_assert(ARRAY_SIZE(_gpio_to_iomux_reg) == SOC_GPIO_PIN_COUNT,
"size of _gpio_to_iomux_reg does not match SOC_GPIO_PIN_COUNT");
#endif
/* String representation of usage types */
const char* _gpio_pin_usage_str[] =
{
"GPIO", "ADC", "CAN", "DAC", "EMAC", "I2C", "PWM", "SPI", "SPI Flash", "UART", "N/A"
};
#ifdef ESP_PM_WUP_PINS
/* for saving the pullup/pulldown settings of wakeup pins in deep sleep mode */
static bool _gpio_pin_pu[GPIO_PIN_NUMOF] = { };
static bool _gpio_pin_pd[GPIO_PIN_NUMOF] = { };
#endif
#if defined(CPU_FAM_ESP32)
#define GPIO_IN_GET(b) (b < 32) ? GPIO.in & BIT(b) : GPIO.in1.val & BIT(b-32)
#define GPIO_OUT_SET(b) if (b < 32) { GPIO.out_w1ts = BIT(b); } else { GPIO.out1_w1ts.val = BIT(b-32); }
#define GPIO_OUT_CLR(b) if (b < 32) { GPIO.out_w1tc = BIT(b); } else { GPIO.out1_w1tc.val = BIT(b-32); }
#define GPIO_OUT_XOR(b) if (b < 32) { GPIO.out ^= BIT(b); } else { GPIO.out1.val ^= BIT(b-32); }
#define GPIO_OUT_GET(b) (b < 32) ? (GPIO.out >> b) & 1 : (GPIO.out1.val >> (b-32)) & 1
#elif defined(CPU_FAM_ESP32C3)
#define GPIO_IN_GET(b) GPIO.in.val & BIT(b)
#define GPIO_OUT_SET(b) GPIO.out_w1ts.val = BIT(b)
#define GPIO_OUT_CLR(b) GPIO.out_w1tc.val = BIT(b)
#define GPIO_OUT_XOR(b) GPIO.out.val ^= BIT(b)
#define GPIO_OUT_GET(b) (GPIO.out.val >> b) & 1
#else
#error "Platform implementation is missing"
#endif
#ifndef NDEBUG
int _gpio_init_mode_check(gpio_t pin, gpio_mode_t mode)
{
assert(pin < GPIO_PIN_NUMOF);
/* check if the pin can be used as GPIO or if it is used for something else */
if (_gpio_pin_usage[pin] != _GPIO) {
LOG_TAG_ERROR("gpio", "GPIO%d is already used as %s signal\n", pin,
_gpio_pin_usage_str[_gpio_pin_usage[pin]]);
return -1;
}
return 0;
}
#endif
int gpio_init(gpio_t pin, gpio_mode_t mode)
{
DEBUG("%s: gpio=%d mode=%d\n", __func__, pin, mode);
assert(_gpio_init_mode_check(pin, mode) == 0);
assert(pin < GPIO_PIN_NUMOF);
gpio_config_t cfg = { };
#if SOC_RTCIO_INPUT_OUTPUT_SUPPORTED
/* if we come from deep sleep, the GPIO is configured as RTC IO */
esp_idf_rtc_gpio_deinit(pin);
#endif
cfg.pin_bit_mask = (1ULL << pin);
switch (mode) {
case GPIO_IN:
case GPIO_IN_PD:
case GPIO_IN_PU:
cfg.mode = GPIO_MODE_DEF_INPUT;
break;
case GPIO_IN_OD:
case GPIO_IN_OD_PU:
cfg.mode = (GPIO_MODE_DEF_INPUT) | (GPIO_MODE_DEF_OUTPUT) | (GPIO_MODE_DEF_OD);
break;
case GPIO_IN_OUT:
cfg.mode = (GPIO_MODE_DEF_INPUT) | (GPIO_MODE_DEF_OUTPUT);
break;
case GPIO_OUT:
cfg.mode = GPIO_MODE_DEF_OUTPUT;
break;
case GPIO_OD:
case GPIO_OD_PU:
cfg.mode = (GPIO_MODE_DEF_OUTPUT) | (GPIO_MODE_DEF_OD);
break;
}
/* digital GPIO configuration */
cfg.pull_up_en = ((mode == GPIO_IN_PU) ||
(mode == GPIO_OD_PU) ||
(mode == GPIO_IN_OD_PU)) ? GPIO_PULLUP_ENABLE
: GPIO_PULLUP_DISABLE;
cfg.pull_down_en = (mode == GPIO_IN_PD) ? GPIO_PULLUP_ENABLE
: GPIO_PULLUP_DISABLE;
cfg.intr_type = GPIO_INTR_DISABLE;
#ifdef ESP_PM_WUP_PINS
/* for saving the pullup/pulldown settings of wakeup pins in deep sleep mode */
_gpio_pin_pu[pin] = cfg.pull_up_en;
_gpio_pin_pd[pin] = cfg.pull_down_en;
#endif
return (esp_idf_gpio_config(&cfg) == ESP_OK) ? 0 : -1;
}
#if MODULE_PERIPH_GPIO_IRQ
/* interrupt enabled state is required for sleep modes */
static bool gpio_int_enabled_table[GPIO_PIN_NUMOF] = { };
static bool _isr_installed = false;
int gpio_init_int(gpio_t pin, gpio_mode_t mode, gpio_flank_t flank,
gpio_cb_t cb, void *arg)
{
DEBUG("%s: gpio=%d mode=%d flank=%d\n", __func__, pin, mode, flank);
assert(_gpio_init_mode_check(pin, mode) == 0);
if (gpio_init(pin, mode) != 0) {
return -1;
}
gpio_int_type_t type = GPIO_INTR_DISABLE;
switch (flank) {
case GPIO_NONE:
type = GPIO_INTR_DISABLE;
break;
case GPIO_RISING:
type = GPIO_INTR_POSEDGE;
break;
case GPIO_FALLING:
type = GPIO_INTR_NEGEDGE;
break;
case GPIO_BOTH:
type = GPIO_INTR_ANYEDGE;
break;
case GPIO_LOW:
type = GPIO_INTR_LOW_LEVEL;
break;
case GPIO_HIGH:
type = GPIO_INTR_HIGH_LEVEL;
break;
}
if (esp_idf_gpio_set_intr_type(pin, type) != ESP_OK) {
return -1;
}
if (!_isr_installed &&
esp_idf_gpio_install_isr_service(ESP_INTR_FLAG_LEVEL1) != ESP_OK) {
return -1;
}
_isr_installed = true;
if (esp_idf_gpio_isr_handler_add(pin, cb, arg) != ESP_OK) {
return -1;
}
gpio_irq_enable(pin);
return (gpio_int_enabled_table[pin]) ? 0 : -1;
}
void gpio_irq_enable(gpio_t pin)
{
DEBUG("%s: gpio=%d\n", __func__, pin);
assert(pin < GPIO_PIN_NUMOF);
if (esp_idf_gpio_intr_enable(pin) == ESP_OK) {
gpio_int_enabled_table[pin] = true;
}
}
void gpio_irq_disable(gpio_t pin)
{
DEBUG("%s: gpio=%d\n", __func__, pin);
assert(pin < GPIO_PIN_NUMOF);
if (esp_idf_gpio_intr_disable(pin) == ESP_OK) {
gpio_int_enabled_table[pin] = false;
}
}
#endif /* MODULE_PERIPH_GPIO_IRQ */
#if IS_USED(MODULE_ESP_IDF_GPIO_HAL)
static gpio_hal_context_t _gpio_hal_ctx = {
.dev = GPIO_HAL_GET_HW(GPIO_PORT_0)
};
/*
* Since `gpio_hal_get_level` returns the current pin level and not the value
* last written to the output, we have to handle the state of each pin in a
* separate static variable.
*/
static BITFIELD(_output, GPIO_PIN_NUMOF);
int gpio_read(gpio_t pin)
{
assert(pin < GPIO_PIN_NUMOF);
/* if the pin is not an input, it alsways returns 0 */
/* TODO: not really clear whether it should return the last written value
* in this case. */
int value = gpio_hal_get_level(&_gpio_hal_ctx, pin);
DEBUG("%s gpio=%u val=%d\n", __func__, pin, value);
return value;
}
void gpio_write(gpio_t pin, int value)
{
DEBUG("%s gpio=%u val=%d\n", __func__, pin, value);
assert(pin < GPIO_PIN_NUMOF);
if (value) {
bf_set(_output, pin);
}
else {
bf_unset(_output, pin);
}
gpio_hal_set_level(&_gpio_hal_ctx, pin, value);
}
void gpio_set(gpio_t pin)
{
DEBUG("%s gpio=%u\n", __func__, pin);
assert(pin < GPIO_PIN_NUMOF);
bf_set(_output, pin);
gpio_hal_set_level(&_gpio_hal_ctx, pin, 1);
}
void gpio_clear(gpio_t pin)
{
DEBUG("%s gpio=%u\n", __func__, pin);
assert(pin < GPIO_PIN_NUMOF);
bf_unset(_output, pin);
gpio_hal_set_level(&_gpio_hal_ctx, pin, 0);
}
void gpio_toggle(gpio_t pin)
{
DEBUG("%s gpio=%u\n", __func__, pin);
assert(pin < GPIO_PIN_NUMOF);
bf_toggle(_output, pin);
gpio_hal_set_level(&_gpio_hal_ctx, pin, bf_isset(_output, pin) ? 1 : 0);
}
#else /* IS_USED(MODULE_ESP_IDF_GPIO_HAL) */
int gpio_read(gpio_t pin)
{
assert(pin < GPIO_PIN_NUMOF);
int value;
if (REG_GET_BIT(_gpio_to_iomux_reg[pin], FUN_IE)) {
/* in case the pin is any kind of input, read from input register */
value = (GPIO_IN_GET(pin)) ? 1 : 0;
}
else {
/* otherwise read the last value written to the output register */
value = GPIO_OUT_GET(pin);
}
DEBUG("%s gpio=%u val=%d\n", __func__, pin, value);
return value;
}
void gpio_write(gpio_t pin, int value)
{
DEBUG("%s gpio=%u val=%d\n", __func__, pin, value);
assert(pin < GPIO_PIN_NUMOF);
if (value) {
GPIO_OUT_SET(pin);
}
else {
GPIO_OUT_CLR(pin);
}
}
void gpio_set(gpio_t pin)
{
DEBUG("%s gpio=%u\n", __func__, pin);
assert(pin < GPIO_PIN_NUMOF);
GPIO_OUT_SET(pin);
}
void gpio_clear(gpio_t pin)
{
DEBUG("%s gpio=%u\n", __func__, pin);
assert(pin < GPIO_PIN_NUMOF);
GPIO_OUT_CLR(pin);
}
void gpio_toggle(gpio_t pin)
{
DEBUG("%s gpio=%u\n", __func__, pin);
assert(pin < GPIO_PIN_NUMOF);
GPIO_OUT_XOR(pin);
}
#endif /* IS_USED(MODULE_ESP_IDF_GPIO_HAL) */
int gpio_set_pin_usage(gpio_t pin, gpio_pin_usage_t usage)
{
assert(pin < GPIO_PIN_NUMOF);
_gpio_pin_usage[pin] = usage;
return 0;
}
gpio_pin_usage_t gpio_get_pin_usage (gpio_t pin)
{
return (pin < GPIO_PIN_NUMOF) ? _gpio_pin_usage[pin] : _NOT_EXIST;
}
const char* gpio_get_pin_usage_str(gpio_t pin)
{
return _gpio_pin_usage_str[_gpio_pin_usage[((pin < GPIO_PIN_NUMOF) ? pin : _NOT_EXIST)]];
}
#if MODULE_PERIPH_GPIO_IRQ
static uint32_t gpio_int_saved_type[GPIO_PIN_NUMOF];
#endif
void gpio_pm_sleep_enter(unsigned mode)
{
if (mode == ESP_PM_DEEP_SLEEP) {
#ifdef ESP_PM_GPIO_HOLD
/*
* Activate the power domain for RTC peripherals when
* ESP_PM_GPIO_HOLD is defined for deep sleep mode.
*/
esp_idf_gpio_deep_sleep_hold();
esp_sleep_pd_config(ESP_PD_DOMAIN_RTC_PERIPH, ESP_PD_OPTION_ON);
#endif
#ifdef ESP_PM_WUP_PINS
static const gpio_t wup_pins[] = { ESP_PM_WUP_PINS };
/*
* Prepare the wake-up pins if a single pin or a comma-separated list of
* pins is defined for wake-up.
*/
uint64_t wup_pin_mask = 0;
for (unsigned i = 0; i < ARRAY_SIZE(wup_pins); i++) {
wup_pin_mask |= 1ULL << wup_pins[i];
#if SOC_RTCIO_INPUT_OUTPUT_SUPPORTED
/* If internal pullups/pulldowns are used, they have to be
activated also in deep sleep mode in RTC power domain */
bool pu_pd_used = false;
assert(rtc_io_num_map[wup_pins[i]] >= 0);
if (_gpio_pin_pu[wup_pins[i]]) {
pu_pd_used = true;
esp_idf_rtc_gpio_pullup_en(wup_pins[i]);
}
else {
esp_idf_rtc_gpio_pullup_dis(wup_pins[i]);
}
if (_gpio_pin_pd[wup_pins[i]]) {
pu_pd_used = true;
esp_idf_rtc_gpio_pulldown_en(wup_pins[i]);
}
else {
esp_idf_rtc_gpio_pulldown_dis(wup_pins[i]);
}
if (pu_pd_used) {
/* If internal pullups/pulldowns are used, the RTC power domain
* must remain active in deep sleep mode */
esp_sleep_pd_config(ESP_PD_DOMAIN_RTC_PERIPH, ESP_PD_OPTION_ON);
}
#endif /* SOC_RTCIO_INPUT_OUTPUT_SUPPORTED */
}
esp_sleep_enable_ext1_wakeup(wup_pin_mask, ESP_PM_WUP_LEVEL);
#endif /* ESP_PM_WUP_PINS */
}
else {
#if MODULE_PERIPH_GPIO_IRQ
for (unsigned i = 0; i < GPIO_PIN_NUMOF; i++) {
if (gpio_int_enabled_table[i] && GPIO.pin[i].int_type) {
gpio_int_saved_type[i] = GPIO.pin[i].int_type;
switch (gpio_int_saved_type[i]) {
case GPIO_LOW:
case GPIO_FALLING:
esp_idf_gpio_wakeup_enable(i, GPIO_INTR_LOW_LEVEL);
DEBUG("%s gpio=%u GPIO_LOW\n", __func__, i);
break;
case GPIO_HIGH:
case GPIO_RISING:
esp_idf_gpio_wakeup_enable(i, GPIO_INTR_HIGH_LEVEL);
DEBUG("%s gpio=%u GPIO_HIGH\n", __func__, i);
break;
case GPIO_BOTH:
DEBUG("%s gpio=%u GPIO_BOTH not supported\n",
__func__, i);
break;
default:
break;
}
}
}
esp_sleep_enable_gpio_wakeup();
#endif
}
}
void gpio_pm_sleep_exit(uint32_t cause)
{
(void)cause;
#if MODULE_PERIPH_GPIO_IRQ
DEBUG("%s\n", __func__);
for (unsigned i = 0; i < GPIO_PIN_NUMOF; i++) {
if (gpio_int_enabled_table[i]) {
GPIO.pin[i].int_type = gpio_int_saved_type[i];
}
}
#endif
}