mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-15 17:52:45 +01:00
e7fbaf3815
- removed the __attribute__((naked)) from ISRs - removed ISR_ENTER() and ISR_EXIT() macros Rationale: Cortex-Mx MCUs save registers R0-R4 automatically on calling ISRs. The naked attribute tells the compiler not to save any other registers. This is fine, as long as the code in the ISR is not nested. If nested, it will use also R4 and R5, which will then lead to currupted registers on exit of the ISR. Removing the naked will fix this.
369 lines
7.8 KiB
C
369 lines
7.8 KiB
C
/*
|
|
* Copyright (C) 2014 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_stm32f4
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Low-level UART driver implementation
|
|
*
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
* @author Fabian Nack <nack@inf.fu-berlin.de>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include "cpu.h"
|
|
#include "thread.h"
|
|
#include "sched.h"
|
|
#include "periph_conf.h"
|
|
#include "periph/uart.h"
|
|
|
|
/* guard file in case no UART device was specified */
|
|
#if UART_NUMOF
|
|
/**
|
|
* @brief Each UART device has to store two callbacks.
|
|
*/
|
|
typedef struct {
|
|
uart_rx_cb_t rx_cb;
|
|
uart_tx_cb_t tx_cb;
|
|
void *arg;
|
|
} uart_conf_t;
|
|
|
|
/**
|
|
* @brief Unified interrupt handler for all UART devices
|
|
*
|
|
* @param uartnum the number of the UART that triggered the ISR
|
|
* @param uart the UART device that triggered the ISR
|
|
*/
|
|
static inline void irq_handler(uart_t uartnum, USART_TypeDef *uart);
|
|
|
|
/**
|
|
* @brief Allocate memory to store the callback functions.
|
|
*/
|
|
static uart_conf_t uart_config[UART_NUMOF];
|
|
|
|
int uart_init(uart_t uart, uint32_t baudrate, uart_rx_cb_t rx_cb, uart_tx_cb_t tx_cb, void *arg)
|
|
{
|
|
/* do basic initialization */
|
|
int res = uart_init_blocking(uart, baudrate);
|
|
if (res < 0) {
|
|
return res;
|
|
}
|
|
|
|
/* remember callback addresses */
|
|
uart_config[uart].rx_cb = rx_cb;
|
|
uart_config[uart].tx_cb = tx_cb;
|
|
uart_config[uart].arg = arg;
|
|
|
|
/* enable receive interrupt */
|
|
switch (uart) {
|
|
#if UART_0_EN
|
|
case UART_0:
|
|
NVIC_SetPriority(UART_0_IRQ_CHAN, UART_IRQ_PRIO);
|
|
NVIC_EnableIRQ(UART_0_IRQ_CHAN);
|
|
UART_0_DEV->CR1 |= USART_CR1_RXNEIE;
|
|
break;
|
|
#endif
|
|
#if UART_1_EN
|
|
case UART_1:
|
|
NVIC_SetPriority(UART_1_IRQ_CHAN, UART_IRQ_PRIO);
|
|
NVIC_EnableIRQ(UART_1_IRQ_CHAN);
|
|
UART_1_DEV->CR1 |= USART_CR1_RXNEIE;
|
|
break;
|
|
#endif
|
|
#if UART_2_EN
|
|
case UART_2:
|
|
NVIC_SetPriority(UART_2_IRQ_CHAN, UART_IRQ_PRIO);
|
|
NVIC_EnableIRQ(UART_2_IRQ_CHAN);
|
|
UART_2_DEV->CR1 |= USART_CR1_RXNEIE;
|
|
break;
|
|
#endif
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int uart_init_blocking(uart_t uart, uint32_t baudrate)
|
|
{
|
|
USART_TypeDef *dev = 0;
|
|
GPIO_TypeDef *port = 0;
|
|
uint32_t tx_pin = 0;
|
|
uint32_t rx_pin = 0;
|
|
uint8_t af = 0;
|
|
float clk = 0.0;
|
|
float divider;
|
|
uint16_t mantissa;
|
|
uint8_t fraction;
|
|
|
|
switch (uart) {
|
|
#if UART_0_EN
|
|
case UART_0:
|
|
dev = UART_0_DEV;
|
|
port = UART_0_PORT;
|
|
clk = UART_0_CLK;
|
|
tx_pin = UART_0_TX_PIN;
|
|
rx_pin = UART_0_RX_PIN;
|
|
af = UART_0_AF;
|
|
UART_0_CLKEN();
|
|
UART_0_PORT_CLKEN();
|
|
break;
|
|
#endif
|
|
#if UART_1_EN
|
|
case UART_1:
|
|
dev = UART_1_DEV;
|
|
port = UART_1_PORT;
|
|
clk = UART_1_CLK;
|
|
tx_pin = UART_1_TX_PIN;
|
|
rx_pin = UART_1_RX_PIN;
|
|
af = UART_1_AF;
|
|
UART_1_CLKEN();
|
|
UART_1_PORT_CLKEN();
|
|
break;
|
|
#endif
|
|
#if UART_2_EN
|
|
case UART_2:
|
|
dev = UART_2_DEV;
|
|
port = UART_2_PORT;
|
|
clk = UART_2_CLK;
|
|
tx_pin = UART_2_TX_PIN;
|
|
rx_pin = UART_2_RX_PIN;
|
|
af = UART_2_AF;
|
|
UART_2_CLKEN();
|
|
UART_2_PORT_CLKEN();
|
|
break;
|
|
#endif
|
|
}
|
|
|
|
/* configure pp mode with no pull for RX and TX pins */
|
|
port->OTYPER &= ~(1 << rx_pin | 1 << tx_pin);
|
|
port->PUPDR &= ~(3 << (rx_pin * 2) | 3 << (tx_pin * 2));
|
|
|
|
/* configure RX and TX pins, set pin to use alternative function mode */
|
|
port->MODER &= ~(3 << (rx_pin * 2) | 3 << (tx_pin * 2));
|
|
port->MODER |= 2 << (rx_pin * 2) | 2 << (tx_pin * 2);
|
|
/* and assign alternative function */
|
|
if (rx_pin < 8) {
|
|
port->AFR[0] &= ~(0xf << (rx_pin * 4));
|
|
port->AFR[0] |= af << (rx_pin * 4);
|
|
}
|
|
else {
|
|
port->AFR[1] &= ~(0xf << ((rx_pin - 8) * 4));
|
|
port->AFR[1] |= af << ((rx_pin - 8) * 4);
|
|
}
|
|
if (tx_pin < 8) {
|
|
port->AFR[0] &= ~(0xf << (tx_pin * 4));
|
|
port->AFR[0] |= af << (tx_pin * 4);
|
|
}
|
|
else {
|
|
port->AFR[1] &= ~(0xf << ((tx_pin - 8) * 4));
|
|
port->AFR[1] |= af << ((tx_pin - 8) * 4);
|
|
}
|
|
|
|
/* configure UART to mode 8N1 with given baudrate */
|
|
divider = clk / (16 * baudrate);
|
|
mantissa = (uint16_t)divider;
|
|
fraction = (uint8_t)((divider - mantissa) * 16);
|
|
dev->BRR = ((mantissa & 0x0fff) << 4) | (0x0f & fraction);
|
|
|
|
/* enable receive and transmit mode */
|
|
dev->CR3 = 0;
|
|
dev->CR2 = 0;
|
|
dev->CR1 = USART_CR1_UE | USART_CR1_TE | USART_CR1_RE;
|
|
|
|
return 0;
|
|
}
|
|
|
|
void uart_tx_begin(uart_t uart)
|
|
{
|
|
switch (uart) {
|
|
#if UART_0_EN
|
|
case UART_0:
|
|
UART_0_DEV->CR1 |= USART_CR1_TXEIE;
|
|
break;
|
|
#endif
|
|
#if UART_1_EN
|
|
case UART_1:
|
|
UART_1_DEV->CR1 |= USART_CR1_TXEIE;
|
|
break;
|
|
#endif
|
|
#if UART_2_EN
|
|
case UART_2:
|
|
UART_2_DEV->CR1 |= USART_CR1_TXEIE;
|
|
break;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
int uart_write(uart_t uart, char data)
|
|
{
|
|
USART_TypeDef *dev = 0;
|
|
|
|
switch (uart) {
|
|
#if UART_0_EN
|
|
case UART_0:
|
|
dev = UART_0_DEV;
|
|
break;
|
|
#endif
|
|
#if UART_1_EN
|
|
case UART_1:
|
|
dev = UART_1_DEV;
|
|
break;
|
|
#endif
|
|
#if UART_2_EN
|
|
case UART_2:
|
|
dev = UART_2_DEV;
|
|
break;
|
|
#endif
|
|
}
|
|
|
|
if (dev->SR & USART_SR_TXE) {
|
|
dev->DR = (uint8_t)data;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int uart_read_blocking(uart_t uart, char *data)
|
|
{
|
|
USART_TypeDef *dev = 0;
|
|
|
|
switch (uart) {
|
|
#if UART_0_EN
|
|
case UART_0:
|
|
dev = UART_0_DEV;
|
|
break;
|
|
#endif
|
|
#if UART_1_EN
|
|
case UART_1:
|
|
dev = UART_1_DEV;
|
|
break;
|
|
#endif
|
|
#if UART_2_EN
|
|
case UART_2:
|
|
dev = UART_2_DEV;
|
|
break;
|
|
#endif
|
|
}
|
|
|
|
while (!(dev->SR & USART_SR_RXNE));
|
|
*data = (char)dev->DR;
|
|
|
|
return 1;
|
|
}
|
|
|
|
int uart_write_blocking(uart_t uart, char data)
|
|
{
|
|
USART_TypeDef *dev = 0;
|
|
|
|
switch (uart) {
|
|
#if UART_0_EN
|
|
case UART_0:
|
|
dev = UART_0_DEV;
|
|
break;
|
|
#endif
|
|
#if UART_1_EN
|
|
case UART_1:
|
|
dev = UART_1_DEV;
|
|
break;
|
|
#endif
|
|
#if UART_2_EN
|
|
case UART_2:
|
|
dev = UART_2_DEV;
|
|
break;
|
|
#endif
|
|
}
|
|
|
|
while (!(dev->SR & USART_SR_TXE));
|
|
dev->DR = (uint8_t)data;
|
|
|
|
return 1;
|
|
}
|
|
|
|
void uart_poweron(uart_t uart)
|
|
{
|
|
switch (uart) {
|
|
#if UART_0_EN
|
|
case UART_0:
|
|
UART_0_CLKEN();
|
|
break;
|
|
#endif
|
|
#if UART_1_EN
|
|
case UART_1:
|
|
UART_1_CLKEN();
|
|
break;
|
|
#endif
|
|
#if UART_2_EN
|
|
case UART_2:
|
|
UART_2_CLKEN();
|
|
break;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
void uart_poweroff(uart_t uart)
|
|
{
|
|
switch (uart) {
|
|
#if UART_0_EN
|
|
case UART_0:
|
|
UART_0_CLKDIS();
|
|
break;
|
|
#endif
|
|
#if UART_1_EN
|
|
case UART_1:
|
|
UART_1_CLKDIS();
|
|
break;
|
|
#endif
|
|
#if UART_2_EN
|
|
case UART_2:
|
|
UART_2_CLKDIS();
|
|
break;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
#if UART_0_EN
|
|
void UART_0_ISR(void)
|
|
{
|
|
irq_handler(UART_0, UART_0_DEV);
|
|
}
|
|
#endif
|
|
|
|
#if UART_1_EN
|
|
void UART_1_ISR(void)
|
|
{
|
|
irq_handler(UART_1, UART_1_DEV);
|
|
}
|
|
#endif
|
|
|
|
#if UART_2_EN
|
|
void UART_2_ISR(void)
|
|
{
|
|
irq_handler(UART_2, UART_2_DEV);
|
|
}
|
|
#endif
|
|
|
|
static inline void irq_handler(uint8_t uartnum, USART_TypeDef *dev)
|
|
{
|
|
if (dev->SR & USART_SR_RXNE) {
|
|
char data = (char)dev->DR;
|
|
uart_config[uartnum].rx_cb(uart_config[uartnum].arg, data);
|
|
}
|
|
else if (dev->SR & USART_SR_TXE) {
|
|
if (uart_config[uartnum].tx_cb(uart_config[uartnum].arg) == 0) {
|
|
dev->CR1 &= ~(USART_CR1_TXEIE);
|
|
}
|
|
}
|
|
if (sched_context_switch_request) {
|
|
thread_yield();
|
|
}
|
|
}
|
|
|
|
#endif /* UART_NUMOF */
|