2017-11-13 16:08:08 +01:00
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2017-11-29 19:50:50 +01:00
|
|
|
* @ingroup cpu_efm32
|
2017-11-13 16:08:08 +01:00
|
|
|
* @ingroup drivers_periph_gpio
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Low-level GPIO driver implementation
|
|
|
|
*
|
2018-10-11 12:20:37 +02:00
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
2017-11-13 16:08:08 +01:00
|
|
|
* @author Ryan Kurte <ryankurte@gmail.com>
|
|
|
|
* @author Bas Stottelaar <basstottelaar@gmail.com>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "cpu.h"
|
2022-10-17 23:02:45 +02:00
|
|
|
#include "board.h"
|
2017-11-13 16:08:08 +01:00
|
|
|
|
|
|
|
#include "periph/gpio.h"
|
2022-10-17 23:02:45 +02:00
|
|
|
#include "pm_layered.h"
|
2017-11-13 16:08:08 +01:00
|
|
|
|
|
|
|
#include "em_gpio.h"
|
|
|
|
|
2018-09-21 08:17:38 +02:00
|
|
|
#ifdef MODULE_PERIPH_GPIO_IRQ
|
2017-11-13 16:08:08 +01:00
|
|
|
/**
|
|
|
|
* @brief Number of external interrupt lines.
|
|
|
|
*/
|
2019-06-19 01:43:18 +02:00
|
|
|
#define NUMOF_IRQS (GPIO_PIN_MAX + 1)
|
2017-11-13 16:08:08 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Hold one interrupt context per interrupt line
|
|
|
|
*/
|
|
|
|
static gpio_isr_ctx_t isr_ctx[NUMOF_IRQS];
|
2018-10-09 14:52:33 +02:00
|
|
|
#endif /* MODULE_PERIPH_GPIO_IRQ */
|
2017-11-13 16:08:08 +01:00
|
|
|
|
|
|
|
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 */
|
2022-10-17 23:02:05 +02:00
|
|
|
#if defined(_SILICON_LABS_32B_SERIES_0) || defined(_SILICON_LABS_32B_SERIES_1)
|
2017-11-13 16:08:08 +01:00
|
|
|
CMU_ClockEnable(cmuClock_HFPER, true);
|
2022-10-17 23:02:05 +02:00
|
|
|
#endif
|
2017-11-13 16:08:08 +01:00
|
|
|
CMU_ClockEnable(cmuClock_GPIO, true);
|
|
|
|
|
|
|
|
/* configure pin */
|
|
|
|
GPIO_PinModeSet(_port_num(pin), _pin_num(pin), mode >> 1, mode & 0x1);
|
2020-06-26 08:58:27 +02:00
|
|
|
#if defined(_SILICON_LABS_32B_SERIES_0)
|
2017-11-13 16:08:08 +01:00
|
|
|
GPIO_DriveModeSet(_port_num(pin), gpioDriveModeStandard);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-09-21 08:17:38 +02:00
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
2024-10-22 16:28:58 +02:00
|
|
|
void gpio_write(gpio_t pin, bool value)
|
2018-09-21 08:17:38 +02:00
|
|
|
{
|
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
2017-11-13 16:08:08 +01:00
|
|
|
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 */
|
2022-10-17 23:02:45 +02:00
|
|
|
gpio_irq_disable(pin);
|
2017-11-13 16:08:08 +01:00
|
|
|
|
|
|
|
/* store interrupt callback */
|
|
|
|
isr_ctx[_pin_num(pin)].cb = cb;
|
|
|
|
isr_ctx[_pin_num(pin)].arg = arg;
|
|
|
|
|
|
|
|
/* enable interrupts */
|
2022-10-15 15:39:21 +02:00
|
|
|
GPIO_ExtIntConfig(_port_num(pin), _pin_num(pin), _pin_num(pin),
|
2022-10-17 23:02:45 +02:00
|
|
|
flank & GPIO_RISING, flank & GPIO_FALLING, false);
|
2017-11-13 16:08:08 +01:00
|
|
|
|
|
|
|
NVIC_ClearPendingIRQ(GPIO_EVEN_IRQn);
|
|
|
|
NVIC_ClearPendingIRQ(GPIO_ODD_IRQn);
|
|
|
|
|
|
|
|
NVIC_EnableIRQ(GPIO_EVEN_IRQn);
|
|
|
|
NVIC_EnableIRQ(GPIO_ODD_IRQn);
|
|
|
|
|
2022-10-17 23:02:45 +02:00
|
|
|
/* enable IRQ */
|
|
|
|
gpio_irq_enable(pin);
|
|
|
|
|
2017-11-13 16:08:08 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_irq_enable(gpio_t pin)
|
|
|
|
{
|
2022-10-17 23:02:45 +02:00
|
|
|
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);
|
2017-11-13 16:08:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_irq_disable(gpio_t pin)
|
|
|
|
{
|
2022-10-17 23:02:45 +02:00
|
|
|
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);
|
2017-11-13 16:08:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @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();
|
|
|
|
}
|
2018-10-09 14:52:33 +02:00
|
|
|
#endif /* MODULE_PERIPH_GPIO_IRQ */
|