2015-09-01 15:57:23 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2023-06-13 15:56:24 +02:00
|
|
|
* @ingroup cpu_msp430
|
2017-06-22 15:43:17 +02:00
|
|
|
* @ingroup drivers_periph_gpio
|
2015-09-01 15:57:23 +02:00
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Low-level GPIO driver implementation
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "bitarithm.h"
|
2023-06-14 16:18:15 +02:00
|
|
|
#include "container.h"
|
|
|
|
#include "cpu.h"
|
2015-09-01 15:57:23 +02:00
|
|
|
#include "periph/gpio.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Number of possible interrupt lines: 2 ports * 8 pins
|
|
|
|
*/
|
|
|
|
#define ISR_NUMOF (16U)
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Number of pins on each port
|
|
|
|
*/
|
|
|
|
#define PINS_PER_PORT (8U)
|
|
|
|
|
|
|
|
static msp_port_t *_port(gpio_t pin)
|
|
|
|
{
|
|
|
|
switch (pin >> 8) {
|
2023-06-14 16:18:15 +02:00
|
|
|
case 1:
|
|
|
|
return &PORT_1.base;
|
|
|
|
case 2:
|
|
|
|
return &PORT_2.base;
|
|
|
|
case 3:
|
|
|
|
return &PORT_3.base;
|
|
|
|
case 4:
|
|
|
|
return &PORT_4.base;
|
|
|
|
case 5:
|
|
|
|
return &PORT_5.base;
|
|
|
|
case 6:
|
|
|
|
return &PORT_6.base;
|
|
|
|
default:
|
|
|
|
return NULL;
|
2015-09-01 15:57:23 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-26 09:27:54 +02:00
|
|
|
static inline uint8_t _pin(gpio_t pin)
|
|
|
|
{
|
|
|
|
return (uint8_t)(pin & 0xff);
|
|
|
|
}
|
|
|
|
|
2023-06-14 16:18:15 +02:00
|
|
|
static inline msp_port_p1_p2_t *_isr_port(gpio_t pin)
|
2015-09-01 15:57:23 +02:00
|
|
|
{
|
2023-06-14 16:18:15 +02:00
|
|
|
/* checking for (pin >> 8) <= 2 requires 6 byte of .text more than
|
|
|
|
* checking the resulting address */
|
|
|
|
msp_port_p1_p2_t *port = container_of(_port(pin), msp_port_p1_p2_t, base);
|
|
|
|
if ((port == &PORT_1) || (port == &PORT_2)) {
|
|
|
|
return port;
|
2015-09-01 15:57:23 +02:00
|
|
|
}
|
2023-06-14 16:18:15 +02:00
|
|
|
|
2015-09-01 15:57:23 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2016-02-20 14:43:26 +01:00
|
|
|
int gpio_init(gpio_t pin, gpio_mode_t mode)
|
2015-09-01 15:57:23 +02:00
|
|
|
{
|
|
|
|
msp_port_t *port = _port(pin);
|
2016-02-20 14:43:26 +01:00
|
|
|
|
|
|
|
/* check if port is valid and mode applicable */
|
|
|
|
if ((port == NULL) || ((mode != GPIO_IN) && (mode != GPIO_OUT))) {
|
2015-09-01 15:57:23 +02:00
|
|
|
return -1;
|
|
|
|
}
|
2016-02-20 14:43:26 +01:00
|
|
|
|
2017-06-12 17:24:30 +02:00
|
|
|
/* set pin direction */
|
2016-02-20 14:43:26 +01:00
|
|
|
if (mode == GPIO_OUT) {
|
|
|
|
port->DIR |= _pin(pin);
|
|
|
|
}
|
2016-06-28 10:23:57 +02:00
|
|
|
else {
|
|
|
|
port->DIR &= ~(_pin(pin));
|
|
|
|
}
|
2016-02-20 14:43:26 +01:00
|
|
|
|
2015-09-01 15:57:23 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_periph_mode(gpio_t pin, bool enable)
|
|
|
|
{
|
|
|
|
REG8 *sel;
|
2023-06-14 16:18:15 +02:00
|
|
|
msp_port_p1_p2_t *isrport = _isr_port(pin);
|
2015-09-01 15:57:23 +02:00
|
|
|
if (isrport) {
|
|
|
|
sel = &(isrport->SEL);
|
|
|
|
}
|
|
|
|
else {
|
2023-06-14 16:18:15 +02:00
|
|
|
msp_port_p3_p6_t *port = container_of(_port(pin), msp_port_p3_p6_t, base);
|
2015-09-01 15:57:23 +02:00
|
|
|
if (port) {
|
|
|
|
sel = &(port->SEL);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (enable) {
|
|
|
|
*sel |= _pin(pin);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*sel &= ~(_pin(pin));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int gpio_read(gpio_t pin)
|
|
|
|
{
|
|
|
|
msp_port_t *port = _port(pin);
|
|
|
|
if (port->DIR & _pin(pin)) {
|
|
|
|
return (int)(port->OD & _pin(pin));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return (int)(port->IN & _pin(pin));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_set(gpio_t pin)
|
|
|
|
{
|
|
|
|
_port(pin)->OD |= _pin(pin);
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_clear(gpio_t pin)
|
|
|
|
{
|
|
|
|
_port(pin)->OD &= ~(_pin(pin));
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_toggle(gpio_t pin)
|
|
|
|
{
|
|
|
|
_port(pin)->OD ^= _pin(pin);
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_write(gpio_t pin, int value)
|
|
|
|
{
|
|
|
|
if (value) {
|
|
|
|
_port(pin)->OD |= _pin(pin);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
_port(pin)->OD &= ~(_pin(pin));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-26 09:27:54 +02:00
|
|
|
#ifdef MODULE_PERIPH_GPIO_IRQ
|
|
|
|
/**
|
|
|
|
* @brief Interrupt context for each interrupt line
|
|
|
|
*/
|
|
|
|
static gpio_isr_ctx_t isr_ctx[ISR_NUMOF];
|
|
|
|
|
|
|
|
static int _ctx(gpio_t pin)
|
|
|
|
{
|
|
|
|
int i = bitarithm_lsb(_pin(pin));
|
2023-06-14 16:18:15 +02:00
|
|
|
return (_port(pin) == &PORT_1.base) ? i : (i + 8);
|
2018-08-26 09:27:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int gpio_init_int(gpio_t pin, gpio_mode_t mode, gpio_flank_t flank,
|
|
|
|
gpio_cb_t cb, void *arg)
|
|
|
|
{
|
2023-06-14 16:18:15 +02:00
|
|
|
msp_port_p1_p2_t *port = _isr_port(pin);
|
2018-08-26 09:27:54 +02:00
|
|
|
|
|
|
|
/* check if port, pull resistor and flank configuration are valid */
|
|
|
|
if ((port == NULL) || (flank == GPIO_BOTH)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* disable any activated interrupt */
|
|
|
|
port->IE &= ~(_pin(pin));
|
|
|
|
/* configure as input */
|
|
|
|
if (gpio_init(pin, mode) < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
/* save ISR context */
|
|
|
|
isr_ctx[_ctx(pin)].cb = cb;
|
|
|
|
isr_ctx[_ctx(pin)].arg = arg;
|
|
|
|
/* configure flank */
|
|
|
|
port->IES &= ~(_pin(pin));
|
|
|
|
port->IES |= (flank & _pin(pin));
|
|
|
|
/* clear pending interrupts and enable the IRQ */
|
|
|
|
port->IFG &= ~(_pin(pin));
|
|
|
|
gpio_irq_enable(pin);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_irq_enable(gpio_t pin)
|
|
|
|
{
|
2023-06-14 16:18:15 +02:00
|
|
|
msp_port_p1_p2_t *port = _isr_port(pin);
|
2018-08-26 09:27:54 +02:00
|
|
|
if (port) {
|
|
|
|
port->IE |= _pin(pin);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void gpio_irq_disable(gpio_t pin)
|
|
|
|
{
|
2023-06-14 16:18:15 +02:00
|
|
|
msp_port_p1_p2_t *port = _isr_port(pin);
|
2018-08-26 09:27:54 +02:00
|
|
|
if (port) {
|
|
|
|
port->IE &= ~(_pin(pin));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-14 16:18:15 +02:00
|
|
|
static inline void isr_handler(msp_port_p1_p2_t *port, int ctx)
|
2015-09-01 15:57:23 +02:00
|
|
|
{
|
make: fix sign-compare errors
cpu, nrf5x_common: fix sign-compare in periph/flashpage
drivers, periph_common: fix sign-compare in flashpage
cpu, sam0_common: fix sign-compare error in periph/gpio
cpu, cc2538: fix sign-compare in periph/timer
cpu, sam3: fix sign-compare in periph/gpio
cpu, stm32_common: fix sign-compare in periph/pwm
cpu, stm32_common: fix sign-compare in periph/timer
cpu, stm32_common: fix sign-compare in periph/flashpage
cpu, nrf5x_common: fix sign-compare in radio/nrfmin
cpu, samd21: fix sign-compare in periph/pwm
cpu, ezr32wg: fix sign-compare in periph/gpio
cpu, ezr32wg: fix sign-compare in periph/timer
drivers, ethos: fix sign-compare
sys, net: fix sign-compare
cpu, atmega_common: fix sign-compare error
cpu, msp430fxyz: fix sign-compare in periph/gpio
boards, msb-430-common: fix sign-compare in board_init
driver, cc2420: fix sign-compared
sys/net: fix sign-compare in gnrc_tftp
driver, pcd8544: fix sign-compare
driver, pn532: fix sign-compare
driver, sdcard_spi: fix sign-compare
tests: fix sign_compare
sys/net, lwmac: fix sign_compare
pkg, lwip: fix sign-compare
boards, waspmote: make CORECLOCK unsigned long to fix sign_compare error
tests, sock_ip: fix sign compare
tests, msg_avail: fix sign compare
tests, sock_udp: fix sign compare
boards: fix sign-compare for calliope and microbit matrix
2017-10-31 11:57:40 +01:00
|
|
|
for (unsigned i = 0; i < PINS_PER_PORT; i++) {
|
2015-09-01 15:57:23 +02:00
|
|
|
if ((port->IE & (1 << i)) && (port->IFG & (1 << i))) {
|
|
|
|
port->IFG &= ~(1 << i);
|
|
|
|
isr_ctx[i + ctx].cb(isr_ctx[i + ctx].arg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ISR(PORT1_VECTOR, isr_port1)
|
|
|
|
{
|
|
|
|
__enter_isr();
|
2023-06-14 16:18:15 +02:00
|
|
|
isr_handler(&PORT_1, 0);
|
2015-09-01 15:57:23 +02:00
|
|
|
__exit_isr();
|
|
|
|
}
|
|
|
|
|
|
|
|
ISR(PORT2_VECTOR, isr_port2)
|
|
|
|
{
|
|
|
|
__enter_isr();
|
2023-06-14 16:18:15 +02:00
|
|
|
isr_handler(&PORT_2, 8);
|
2015-09-01 15:57:23 +02:00
|
|
|
__exit_isr();
|
|
|
|
}
|
2018-08-26 09:27:54 +02:00
|
|
|
#endif /* MODULE_PERIPH_GPIO_IRQ */
|