/* * Copyright (C) 2015-2017 Freie Universität Berlin * * 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_efm32 * @ingroup drivers_periph_gpio * @{ * * @file * @brief Low-level GPIO driver implementation * * @author Hauke Petersen * @author Ryan Kurte * @author Bas Stottelaar * * @} */ #include "cpu.h" #include "board.h" #include "periph/gpio.h" #include "pm_layered.h" #include "em_gpio.h" #ifdef MODULE_PERIPH_GPIO_IRQ /** * @brief Number of external interrupt lines. */ #define NUMOF_IRQS (GPIO_PIN_MAX + 1) /** * @brief Hold one interrupt context per interrupt line */ static gpio_isr_ctx_t isr_ctx[NUMOF_IRQS]; #endif /* MODULE_PERIPH_GPIO_IRQ */ static inline GPIO_Port_TypeDef _port_num(gpio_t pin) { return ((pin & 0xf0) >> 4); } static inline uint32_t _pin_num(gpio_t pin) { return (pin & 0x0f); } int gpio_init(gpio_t pin, gpio_mode_t mode) { /* check for valid pin */ if (pin == GPIO_UNDEF) { return -1; } /* enable clocks */ #if defined(_SILICON_LABS_32B_SERIES_0) || defined(_SILICON_LABS_32B_SERIES_1) CMU_ClockEnable(cmuClock_HFPER, true); #endif CMU_ClockEnable(cmuClock_GPIO, true); /* configure pin */ GPIO_PinModeSet(_port_num(pin), _pin_num(pin), mode >> 1, mode & 0x1); #if defined(_SILICON_LABS_32B_SERIES_0) GPIO_DriveModeSet(_port_num(pin), gpioDriveModeStandard); #endif return 0; } int gpio_read(gpio_t pin) { return GPIO_PinInGet(_port_num(pin), _pin_num(pin)); } void gpio_set(gpio_t pin) { GPIO_PinOutSet(_port_num(pin), _pin_num(pin)); } void gpio_clear(gpio_t pin) { GPIO_PinOutClear(_port_num(pin), _pin_num(pin)); } void gpio_toggle(gpio_t pin) { GPIO_PinOutToggle(_port_num(pin), _pin_num(pin)); } void gpio_write(gpio_t pin, bool value) { if (value) { GPIO_PinOutSet(_port_num(pin), _pin_num(pin)); } else { GPIO_PinOutClear(_port_num(pin), _pin_num(pin)); } } #ifdef MODULE_PERIPH_GPIO_IRQ static inline uint32_t _pin_mask(gpio_t pin) { return (1 << _pin_num(pin)); } int gpio_init_int(gpio_t pin, gpio_mode_t mode, gpio_flank_t flank, gpio_cb_t cb, void *arg) { int result = gpio_init(pin, mode); if (result != 0) { return result; } /* just in case, disable the interrupt for this pin */ gpio_irq_disable(pin); /* store interrupt callback */ isr_ctx[_pin_num(pin)].cb = cb; isr_ctx[_pin_num(pin)].arg = arg; /* enable interrupts */ GPIO_ExtIntConfig(_port_num(pin), _pin_num(pin), _pin_num(pin), flank & GPIO_RISING, flank & GPIO_FALLING, false); NVIC_ClearPendingIRQ(GPIO_EVEN_IRQn); NVIC_ClearPendingIRQ(GPIO_ODD_IRQn); NVIC_EnableIRQ(GPIO_EVEN_IRQn); NVIC_EnableIRQ(GPIO_ODD_IRQn); /* enable IRQ */ gpio_irq_enable(pin); return 0; } void gpio_irq_enable(gpio_t pin) { unsigned pin_extirq = _pin_mask(pin); #if IS_ACTIVE(MODULE_PM_LAYERED) && defined(GPIO_INT_PM_BLOCKER) /* block pm mode if the irq is about to be enabled */ if (!(GPIO_EnabledIntGet() & pin_extirq)) { pm_block(GPIO_INT_PM_BLOCKER); } #endif GPIO_IntEnable(pin_extirq); } void gpio_irq_disable(gpio_t pin) { unsigned pin_extirq = _pin_mask(pin); #if IS_ACTIVE(MODULE_PM_LAYERED) && defined(GPIO_INT_PM_BLOCKER) /* unblock pm mode if the irq is about to be disabled */ if (GPIO_EnabledIntGet() & pin_extirq) { pm_unblock(GPIO_INT_PM_BLOCKER); } #endif GPIO_IntDisable(pin_extirq); } /** * @brief Actual interrupt handler for both even and odd pin index numbers. */ static void gpio_irq(void) { for (int i = 0; i < NUMOF_IRQS; i++) { if (GPIO_IntGet() & (1 << i)) { isr_ctx[i].cb(isr_ctx[i].arg); GPIO_IntClear(1 << i); } } cortexm_isr_end(); } /** * @brief External interrupt handler for even pin index numbers */ void isr_gpio_even(void) { gpio_irq(); } /** * @brief External interrupt handler for odd pin index numbers */ void isr_gpio_odd(void) { gpio_irq(); } #endif /* MODULE_PERIPH_GPIO_IRQ */