/* * Copyright (C) 2014-2015 Freie Universität Berlin * 2015 Hamburg University of Applied Sciences * 2017-2020 Inria * 2017 OTA keys S.A. * * 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_stm32 * @ingroup drivers_periph_gpio * @{ * * @file * @brief Low-level GPIO driver implementation * * @author Hauke Petersen * @author Fabian Nack * @author Alexandre Abadie * @author Katja Kirstein * @author Vincent Dupont * * @} */ #include "cpu.h" #include "bitarithm.h" #include "periph/gpio.h" #include "periph_conf.h" #ifdef MODULE_PERIPH_GPIO_IRQ /** * @brief The STM32F0 family has 16 external interrupt lines */ #define EXTI_NUMOF (16U) #define EXTI_MASK (0xFFFF) /** * @brief Allocate memory for one callback and argument per EXTI channel */ static gpio_isr_ctx_t isr_ctx[EXTI_NUMOF]; #endif /* MODULE_PERIPH_GPIO_IRQ */ #if defined(CPU_FAM_STM32L4) || defined(CPU_FAM_STM32WB) || \ defined(CPU_FAM_STM32G4) || defined(CPU_FAM_STM32G0) || \ defined(CPU_FAM_STM32L5) || defined(CPU_FAM_STM32U5) || \ defined(CPU_FAM_STM32WL) || defined(CPU_FAM_STM32C0) #define EXTI_REG_RTSR (EXTI->RTSR1) #define EXTI_REG_FTSR (EXTI->FTSR1) #define EXTI_REG_PR (EXTI->PR1) #define EXTI_REG_IMR (EXTI->IMR1) #elif defined(CPU_FAM_STM32MP1) #define EXTI_REG_RTSR (EXTI->RTSR1) #define EXTI_REG_FTSR (EXTI->FTSR1) #define EXTI_REG_PR (EXTI->PR1) #define EXTI_REG_IMR (EXTI_C2->IMR1) #else #define EXTI_REG_RTSR (EXTI->RTSR) #define EXTI_REG_FTSR (EXTI->FTSR) #define EXTI_REG_PR (EXTI->PR) #define EXTI_REG_IMR (EXTI->IMR) #endif /** * @brief Extract the port base address from the given pin identifier */ static inline GPIO_TypeDef *_port(gpio_t pin) { return (GPIO_TypeDef *)(pin & ~(0x0f)); } /** * @brief Extract the port number form the given identifier * * The port number is extracted by looking at bits 10, 11, 12, 13 of the base * register addresses. */ static inline int _port_num(gpio_t pin) { #if defined(CPU_FAM_STM32MP1) return (((pin - GPIOA_BASE) >> 12) & 0x0f); #else return ((pin >> 10) & 0x0f); #endif } /** * @brief Extract the pin number from the last 4 bit of the pin identifier */ static inline int _pin_num(gpio_t pin) { return (pin & 0x0f); } static inline void port_init_clock(GPIO_TypeDef *port, gpio_t pin) { (void)port; /* <-- Only used for when port G requires special handling */ #if defined(CPU_FAM_STM32F0) || defined (CPU_FAM_STM32F3) || defined(CPU_FAM_STM32L1) periph_clk_en(AHB, (RCC_AHBENR_GPIOAEN << _port_num(pin))); #elif defined (CPU_FAM_STM32L0) || defined(CPU_FAM_STM32G0) || \ defined(CPU_FAM_STM32C0) periph_clk_en(IOP, (RCC_IOPENR_GPIOAEN << _port_num(pin))); #elif defined(CPU_FAM_STM32L4) || defined(CPU_FAM_STM32WB) || \ defined(CPU_FAM_STM32G4) || defined(CPU_FAM_STM32L5) || \ defined(CPU_FAM_STM32U5) || defined (CPU_FAM_STM32WL) #if defined(CPU_FAM_STM32U5) periph_clk_en(AHB2, (RCC_AHB2ENR1_GPIOAEN << _port_num(pin))); #else periph_clk_en(AHB2, (RCC_AHB2ENR_GPIOAEN << _port_num(pin))); #endif #ifdef PWR_CR2_IOSV if (port == GPIOG) { /* Port G requires external power supply */ periph_clk_en(APB1, RCC_APB1ENR1_PWREN); PWR->CR2 |= PWR_CR2_IOSV; } #endif /* PWR_CR2_IOSV */ #elif defined(CPU_FAM_STM32MP1) periph_clk_en(AHB4, (RCC_MC_AHB4ENSETR_GPIOAEN << _port_num(pin))); #else periph_clk_en(AHB1, (RCC_AHB1ENR_GPIOAEN << _port_num(pin))); #endif } static inline void set_mode(GPIO_TypeDef *port, int pin_num, unsigned mode) { uint32_t tmp = port->MODER; tmp &= ~(0x3 << (2 * pin_num)); tmp |= ((mode & 0x3) << (2 * pin_num)); port->MODER = tmp; } int gpio_init(gpio_t pin, gpio_mode_t mode) { GPIO_TypeDef *port = _port(pin); int pin_num = _pin_num(pin); /* enable clock */ port_init_clock(port, pin); /* set mode */ set_mode(port, pin_num, mode); /* set pull resistor configuration */ port->PUPDR &= ~(0x3 << (2 * pin_num)); port->PUPDR |= (((mode >> 2) & 0x3) << (2 * pin_num)); /* set output mode */ port->OTYPER &= ~(1 << pin_num); port->OTYPER |= (((mode >> 4) & 0x1) << pin_num); /* set pin speed to maximum */ port->OSPEEDR |= (3 << (2 * pin_num)); return 0; } void gpio_init_af(gpio_t pin, gpio_af_t af) { GPIO_TypeDef *port = _port(pin); uint32_t pin_num = _pin_num(pin); /* enable clock */ port_init_clock(port, pin); /* set selected function */ port->AFR[(pin_num > 7) ? 1 : 0] &= ~(0xf << ((pin_num & 0x07) * 4)); port->AFR[(pin_num > 7) ? 1 : 0] |= (af << ((pin_num & 0x07) * 4)); /* set pin to AF mode */ set_mode(port, pin_num, 2); } void gpio_init_analog(gpio_t pin) { /* enable clock, needed as this function can be used without calling * gpio_init first */ #if defined(CPU_FAM_STM32F0) || defined (CPU_FAM_STM32F3) || defined(CPU_FAM_STM32L1) periph_clk_en(AHB, (RCC_AHBENR_GPIOAEN << _port_num(pin))); #elif defined (CPU_FAM_STM32L0) || defined(CPU_FAM_STM32G0) || \ defined(CPU_FAM_STM32C0) periph_clk_en(IOP, (RCC_IOPENR_GPIOAEN << _port_num(pin))); #elif defined(CPU_FAM_STM32L4) || defined(CPU_FAM_STM32WB) || \ defined(CPU_FAM_STM32G4) || defined(CPU_FAM_STM32L5) || \ defined (CPU_FAM_STM32WL) periph_clk_en(AHB2, (RCC_AHB2ENR_GPIOAEN << _port_num(pin))); #elif defined(CPU_FAM_STM32U5) periph_clk_en(AHB2, (RCC_AHB2ENR1_GPIOAEN << _port_num(pin))); #elif defined(CPU_FAM_STM32MP1) periph_clk_en(AHB4, (RCC_MC_AHB4ENSETR_GPIOAEN << _port_num(pin))); #else periph_clk_en(AHB1, (RCC_AHB1ENR_GPIOAEN << _port_num(pin))); #endif /* set to analog mode, PUPD has to be 0b00 */ _port(pin)->MODER |= (0x3 << (2 * _pin_num(pin))); _port(pin)->PUPDR &= ~(0x3 << (2 * _pin_num(pin))); } void gpio_irq_enable(gpio_t pin) { EXTI_REG_IMR |= (1 << _pin_num(pin)); } void gpio_irq_disable(gpio_t pin) { EXTI_REG_IMR &= ~(1 << _pin_num(pin)); } int gpio_read(gpio_t pin) { return (_port(pin)->IDR & (1 << _pin_num(pin))); } void gpio_set(gpio_t pin) { _port(pin)->BSRR = (1 << _pin_num(pin)); } void gpio_clear(gpio_t pin) { _port(pin)->BSRR = (1 << (_pin_num(pin) + 16)); } void gpio_toggle(gpio_t pin) { if (gpio_read(pin)) { gpio_clear(pin); } else { gpio_set(pin); } } void gpio_write(gpio_t pin, int value) { if (value) { gpio_set(pin); } else { gpio_clear(pin); } } #ifdef MODULE_PERIPH_GPIO_IRQ int gpio_init_int(gpio_t pin, gpio_mode_t mode, gpio_flank_t flank, gpio_cb_t cb, void *arg) { int pin_num = _pin_num(pin); int port_num = _port_num(pin); /* set callback */ isr_ctx[pin_num].cb = cb; isr_ctx[pin_num].arg = arg; /* enable clock of the SYSCFG module for EXTI configuration */ #if !defined(CPU_FAM_STM32WB) && !defined(CPU_FAM_STM32MP1) && \ !defined(CPU_FAM_STM32WL) #ifdef CPU_FAM_STM32F0 periph_clk_en(APB2, RCC_APB2ENR_SYSCFGCOMPEN); #elif defined(CPU_FAM_STM32G0) || defined(CPU_FAM_STM32C0) periph_clk_en(APB12, RCC_APBENR2_SYSCFGEN); #elif defined(CPU_FAM_STM32U5) periph_clk_en(APB3, RCC_APB3ENR_SYSCFGEN); #else periph_clk_en(APB2, RCC_APB2ENR_SYSCFGEN); #endif #endif /* initialize pin as input */ gpio_init(pin, mode); /* enable global pin interrupt */ #if defined(CPU_FAM_STM32L5) || defined(CPU_FAM_STM32U5) NVIC_EnableIRQ(EXTI0_IRQn + pin_num); #elif defined(CPU_FAM_STM32F0) || defined(CPU_FAM_STM32L0) || \ defined(CPU_FAM_STM32G0) || defined(CPU_FAM_STM32C0) if (pin_num < 2) { NVIC_EnableIRQ(EXTI0_1_IRQn); } else if (pin_num < 4) { NVIC_EnableIRQ(EXTI2_3_IRQn); } else { NVIC_EnableIRQ(EXTI4_15_IRQn); } #elif defined(CPU_FAM_STM32MP1) if (pin_num < 5) { NVIC_EnableIRQ(EXTI0_IRQn + pin_num); } else if (pin_num < 6) { NVIC_EnableIRQ(EXTI5_IRQn); } else if (pin_num < 10) { NVIC_EnableIRQ(EXTI6_IRQn + pin_num - 6); } else if (pin_num < 11) { NVIC_EnableIRQ(EXTI10_IRQn); } else if (pin_num < 12) { NVIC_EnableIRQ(EXTI11_IRQn); } else if (pin_num < 14) { NVIC_EnableIRQ(EXTI12_IRQn + pin_num - 12); } else if (pin_num < 15) { NVIC_EnableIRQ(EXTI14_IRQn); } else { NVIC_EnableIRQ(EXTI15_IRQn); } #else if (pin_num < 5) { NVIC_EnableIRQ(EXTI0_IRQn + pin_num); } else if (pin_num < 10) { NVIC_EnableIRQ(EXTI9_5_IRQn); } else { NVIC_EnableIRQ(EXTI15_10_IRQn); } #endif /* configure the active flank */ EXTI_REG_RTSR &= ~(1 << pin_num); EXTI_REG_RTSR |= ((flank & 0x1) << pin_num); EXTI_REG_FTSR &= ~(1 << pin_num); EXTI_REG_FTSR |= ((flank >> 1) << pin_num); #if defined(CPU_FAM_STM32G0) || defined(CPU_FAM_STM32L5) || \ defined(CPU_FAM_STM32U5) || defined(CPU_FAM_STM32C0) /* enable specific pin as exti sources */ EXTI->EXTICR[pin_num >> 2] &= ~(0xf << ((pin_num & 0x03) * 8)); EXTI->EXTICR[pin_num >> 2] |= (port_num << ((pin_num & 0x03) * 8)); #elif defined(CPU_FAM_STM32MP1) /* enable specific pin as exti sources */ EXTI->EXTICR[pin_num >> 2] &= ~(0xf << ((pin_num & 0x03) * 4)); EXTI->EXTICR[pin_num >> 2] |= (port_num << ((pin_num & 0x03) * 4)); #else /* enable specific pin as exti sources */ SYSCFG->EXTICR[pin_num >> 2] &= ~(0xf << ((pin_num & 0x03) * 4)); SYSCFG->EXTICR[pin_num >> 2] |= (port_num << ((pin_num & 0x03) * 4)); #endif #if defined(CPU_FAM_STM32G0) || defined(CPU_FAM_STM32L5) || \ defined(CPU_FAM_STM32U5) || defined(CPU_FAM_STM32MP1) || \ defined(CPU_FAM_STM32C0) /* clear any pending requests */ EXTI->RPR1 = (1 << pin_num); EXTI->FPR1 = (1 << pin_num); #else /* clear any pending requests */ EXTI_REG_PR = (1 << pin_num); #endif /* unmask the pins interrupt channel */ EXTI_REG_IMR |= (1 << pin_num); return 0; } void isr_exti(void) { #if defined(CPU_FAM_STM32G0) || defined(CPU_FAM_STM32L5) || \ defined(CPU_FAM_STM32U5) || defined(CPU_FAM_STM32MP1) || \ defined(CPU_FAM_STM32C0) /* get all interrupts handled by this ISR */ uint32_t pending_rising_isr = (EXTI->RPR1 & EXTI_MASK); uint32_t pending_falling_isr = (EXTI->FPR1 & EXTI_MASK); /* clear by writing a 1 */ EXTI->RPR1 = pending_rising_isr; EXTI->FPR1 = pending_falling_isr; /* only generate interrupts against lines which have their IMR set */ uint32_t pending_isr = (pending_rising_isr | pending_falling_isr) & EXTI_REG_IMR; #else /* read all pending interrupts wired to isr_exti */ uint32_t pending_isr = (EXTI_REG_PR & EXTI_MASK); /* clear by writing a 1 */ EXTI_REG_PR = pending_isr; /* only generate soft interrupts against lines which have their IMR set */ pending_isr &= EXTI_REG_IMR; #endif /* iterate over all set bits */ uint8_t pin = 0; while (pending_isr) { pending_isr = bitarithm_test_and_clear(pending_isr, &pin); isr_ctx[pin].cb(isr_ctx[pin].arg); } cortexm_isr_end(); } #endif /* MODULE_PERIPH_GPIO_IRQ */