2022-04-27 10:57:58 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2022 Christian Amsüss
|
|
|
|
*
|
|
|
|
* 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_ll
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Peripheral GPIO Low-Level API implementation for the EFM32 family
|
|
|
|
*
|
|
|
|
* @author Christian Amsüss <chrysn@fsfe.org>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "periph/gpio_ll.h"
|
|
|
|
#include "periph_cpu.h"
|
|
|
|
#include "periph_conf.h"
|
|
|
|
|
2024-01-19 14:44:45 +01:00
|
|
|
int gpio_ll_init(gpio_port_t port, uint8_t pin, gpio_conf_t conf)
|
2022-04-27 10:57:58 +02:00
|
|
|
{
|
|
|
|
GPIO_Mode_TypeDef mode;
|
2022-04-28 11:12:52 +02:00
|
|
|
|
2024-01-19 14:44:45 +01:00
|
|
|
bool initial = conf.initial_value;
|
2022-04-28 11:12:52 +02:00
|
|
|
|
2024-01-19 14:44:45 +01:00
|
|
|
switch (conf.state) {
|
2022-04-27 10:57:58 +02:00
|
|
|
case GPIO_DISCONNECT:
|
|
|
|
mode = gpioModeDisabled;
|
2024-01-23 13:59:15 +01:00
|
|
|
initial = (conf.pull == GPIO_PULL_UP);
|
2022-04-27 10:57:58 +02:00
|
|
|
break;
|
|
|
|
case GPIO_INPUT:
|
2024-01-19 14:44:45 +01:00
|
|
|
switch (conf.pull) {
|
2022-04-28 11:12:52 +02:00
|
|
|
case GPIO_FLOATING:
|
|
|
|
mode = gpioModeInput;
|
|
|
|
break;
|
|
|
|
case GPIO_PULL_UP:
|
|
|
|
initial = true;
|
|
|
|
mode = gpioModeInputPull;
|
|
|
|
break;
|
|
|
|
case GPIO_PULL_DOWN:
|
|
|
|
initial = false;
|
|
|
|
mode = gpioModeInputPull;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
2022-04-27 10:57:58 +02:00
|
|
|
break;
|
|
|
|
case GPIO_OUTPUT_PUSH_PULL:
|
2022-04-28 11:12:52 +02:00
|
|
|
/* ignoring pull */
|
2022-04-27 10:57:58 +02:00
|
|
|
mode = gpioModePushPull;
|
|
|
|
break;
|
2022-04-28 09:39:47 +02:00
|
|
|
case GPIO_OUTPUT_OPEN_DRAIN:
|
2024-01-19 14:44:45 +01:00
|
|
|
switch (conf.pull) {
|
2022-04-28 11:12:52 +02:00
|
|
|
case GPIO_FLOATING:
|
|
|
|
mode = gpioModeWiredAnd;
|
|
|
|
break;
|
|
|
|
case GPIO_PULL_UP:
|
|
|
|
mode = gpioModeWiredAndPullUp;
|
|
|
|
break;
|
|
|
|
default: /* including PULL_DOWN, which makes no sense here*/
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
2022-04-28 09:39:47 +02:00
|
|
|
break;
|
|
|
|
case GPIO_OUTPUT_OPEN_SOURCE:
|
2024-01-19 14:44:45 +01:00
|
|
|
switch (conf.pull) {
|
2022-04-28 11:12:52 +02:00
|
|
|
case GPIO_FLOATING:
|
|
|
|
mode = gpioModeWiredOr;
|
|
|
|
break;
|
|
|
|
case GPIO_PULL_DOWN:
|
|
|
|
mode = gpioModeWiredOrPullDown;
|
|
|
|
break;
|
|
|
|
default: /* including PULL_UP, which makes no sense here*/
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
2022-04-28 09:39:47 +02:00
|
|
|
break;
|
|
|
|
case GPIO_USED_BY_PERIPHERAL:
|
|
|
|
/* Needs to be configured to what the peripheral actually needs
|
|
|
|
* instead (eg. DISABLED for analog input, some output for timers and
|
|
|
|
* UARTs etc); fall-through */
|
2022-04-27 10:57:58 +02:00
|
|
|
default:
|
|
|
|
/* Some probably are by the hardware, but not yet by this
|
|
|
|
* implementation */
|
|
|
|
return -ENOTSUP;
|
|
|
|
}
|
|
|
|
|
2022-04-28 11:12:52 +02:00
|
|
|
GPIO_PinModeSet(port, pin, mode, initial);
|
2022-04-27 10:57:58 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2024-01-19 14:44:45 +01:00
|
|
|
gpio_conf_t gpio_ll_query_conf(gpio_port_t port, uint8_t pin)
|
2022-04-27 10:57:58 +02:00
|
|
|
{
|
2024-01-19 14:44:45 +01:00
|
|
|
gpio_conf_t result = { 0 };
|
2022-04-27 10:57:58 +02:00
|
|
|
GPIO_Mode_TypeDef mode = GPIO_PinModeGet(port, pin);
|
|
|
|
|
2024-01-19 14:44:45 +01:00
|
|
|
result.pull = GPIO_FLOATING;
|
2022-04-28 11:12:52 +02:00
|
|
|
|
2022-04-27 10:57:58 +02:00
|
|
|
switch (mode) {
|
|
|
|
case gpioModePushPull:
|
2024-01-19 14:44:45 +01:00
|
|
|
result.state = GPIO_OUTPUT_PUSH_PULL;
|
2022-04-27 10:57:58 +02:00
|
|
|
break;
|
2022-04-28 09:39:47 +02:00
|
|
|
case gpioModeWiredOr:
|
2024-01-19 14:44:45 +01:00
|
|
|
result.state = GPIO_OUTPUT_OPEN_SOURCE;
|
2022-04-28 09:39:47 +02:00
|
|
|
break;
|
2022-04-28 11:12:52 +02:00
|
|
|
case gpioModeWiredOrPullDown:
|
2024-01-19 14:44:45 +01:00
|
|
|
result.state = GPIO_OUTPUT_OPEN_SOURCE;
|
|
|
|
result.pull = GPIO_PULL_DOWN;
|
2022-04-28 11:12:52 +02:00
|
|
|
break;
|
2022-04-28 09:39:47 +02:00
|
|
|
case gpioModeWiredAnd:
|
2024-01-19 14:44:45 +01:00
|
|
|
result.state = GPIO_OUTPUT_OPEN_DRAIN;
|
2022-04-28 09:39:47 +02:00
|
|
|
break;
|
2022-04-28 11:12:52 +02:00
|
|
|
case gpioModeWiredAndPullUp:
|
2024-01-19 14:44:45 +01:00
|
|
|
result.state = GPIO_OUTPUT_OPEN_DRAIN;
|
|
|
|
result.pull = GPIO_PULL_UP;
|
2022-04-28 11:12:52 +02:00
|
|
|
break;
|
2022-04-27 10:57:58 +02:00
|
|
|
case gpioModeInput:
|
2024-01-19 14:44:45 +01:00
|
|
|
result.state = GPIO_INPUT;
|
2022-04-27 10:57:58 +02:00
|
|
|
break;
|
2022-04-28 11:12:52 +02:00
|
|
|
case gpioModeInputPull:
|
2024-01-19 14:44:45 +01:00
|
|
|
result.state = GPIO_INPUT;
|
|
|
|
result.pull = GPIO_PinOutGet(port, pin) ?
|
2022-04-28 11:12:52 +02:00
|
|
|
GPIO_PULL_UP :
|
|
|
|
GPIO_PULL_DOWN;
|
|
|
|
break;
|
2022-04-27 10:57:58 +02:00
|
|
|
case gpioModeDisabled:
|
|
|
|
/* Fall-through: There is no error reporting here */
|
|
|
|
default:
|
2024-01-19 14:44:45 +01:00
|
|
|
result.state = GPIO_DISCONNECT;
|
2024-01-23 13:59:15 +01:00
|
|
|
if (GPIO_PinOutGet(port, pin)) {
|
|
|
|
result.pull = GPIO_PULL_UP;
|
|
|
|
}
|
2022-04-27 10:57:58 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2024-01-19 14:44:45 +01:00
|
|
|
result.initial_value = (gpio_ll_read_output(port) >> pin) & 1;
|
|
|
|
|
|
|
|
return result;
|
2022-04-27 10:57:58 +02:00
|
|
|
}
|