1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/cpu/stm32f1/periph/uart.c

140 lines
3.0 KiB
C
Raw Normal View History

2014-06-11 14:59:24 +02:00
/*
* Copyright (C) 2014-2016 Freie Universität Berlin
2014-06-11 14:59:24 +02:00
*
* 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.
2014-06-11 14:59:24 +02:00
*/
/**
* @ingroup cpu_stm32f1
* @{
*
* @file
2014-06-11 14:59:24 +02:00
* @brief Low-level UART driver implementation
*
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
* @author Thomas Eichinger <thomas.eichinger@fu-berlin.de>
*
* @}
*/
#include <math.h>
#include "cpu.h"
#include "board.h"
#include "periph_conf.h"
#include "periph/uart.h"
#include "periph/gpio.h"
2014-06-11 14:59:24 +02:00
2014-07-03 10:36:46 +02:00
#include "sched.h"
#include "thread.h"
2014-06-11 14:59:24 +02:00
/**
* @brief Allocate memory to store the callback functions.
*/
static uart_isr_ctx_t isr_ctx[UART_NUMOF];
2014-06-11 14:59:24 +02:00
static inline USART_TypeDef *dev(uart_t uart)
2014-06-11 14:59:24 +02:00
{
return uart_config[uart].dev;
}
2014-06-11 14:59:24 +02:00
static void clk_en(uart_t uart)
{
if (uart_config[uart].bus == APB1) {
RCC->APB1ENR |= uart_config[uart].rcc_pin;
2014-06-11 14:59:24 +02:00
}
else {
RCC->APB2ENR |= uart_config[uart].rcc_pin;
2014-06-11 14:59:24 +02:00
}
}
int uart_init(uart_t uart, uint32_t baudrate, uart_rx_cb_t rx_cb, void *arg)
2014-06-11 14:59:24 +02:00
{
uint32_t bus_clk;
2014-06-11 14:59:24 +02:00
uint16_t mantissa;
uint8_t fraction;
/* make sure the given device is valid */
if (uart >= UART_NUMOF) {
return -1;
2014-06-11 14:59:24 +02:00
}
/* save ISR context */
isr_ctx[uart].rx_cb = rx_cb;
isr_ctx[uart].arg = arg;
/* configure RX and TX pin */
gpio_init(uart_config[uart].rx_pin, GPIO_DIR_IN, GPIO_NOPULL);
gpio_init_af(uart_config[uart].tx_pin, GPIO_AF_OUT_PP);
/* enable the clock */
clk_en(uart);
/* reset UART configuration -> defaults to 8N1 mode */
dev(uart)->CR1 = 0;
dev(uart)->CR2 = 0;
dev(uart)->CR3 = 0;
/* calculate and apply baudrate */
bus_clk = (uart_config[uart].bus == APB1) ? CLOCK_APB1 : CLOCK_APB2;
bus_clk /= baudrate;
mantissa = (uint16_t)(bus_clk / 16);
fraction = (uint8_t)(bus_clk - (mantissa * 16));
dev(uart)->BRR = ((mantissa & 0x0fff) << 4) | (fraction & 0x0f);
/* enable the UART's global interrupt and activate it */
NVIC_EnableIRQ(uart_config[uart].irqn);
dev(uart)->CR1 = (USART_CR1_UE | USART_CR1_TE |
USART_CR1_RE | USART_CR1_RXNEIE);
2014-06-11 14:59:24 +02:00
return 0;
}
2015-10-20 16:27:05 +02:00
void uart_write(uart_t uart, const uint8_t *data, size_t len)
2014-06-11 14:59:24 +02:00
{
2015-10-20 16:27:05 +02:00
for (size_t i = 0; i < len; i++) {
while(!(dev(uart)->SR & USART_SR_TXE)) {}
dev(uart)->DR = data[i];
2014-06-11 14:59:24 +02:00
}
}
static inline void irq_handler(uart_t uart)
2014-06-11 14:59:24 +02:00
{
uint32_t status = dev(uart)->SR;
if (status & USART_SR_RXNE) {
char data = (char)dev(uart)->DR;
isr_ctx[uart].rx_cb(isr_ctx[uart].arg, data);
2014-06-11 14:59:24 +02:00
}
if (status & USART_SR_ORE) {
2015-10-20 16:27:05 +02:00
/* ORE is cleared by reading SR and DR sequentially */
dev(uart)->DR;
2015-10-20 16:27:05 +02:00
}
if (sched_context_switch_request) {
thread_yield();
2014-06-11 14:59:24 +02:00
}
}
#ifdef UART_0_ISR
void UART_0_ISR(void)
2014-06-11 14:59:24 +02:00
{
irq_handler(0);
2014-06-11 14:59:24 +02:00
}
#endif
#ifdef UART_1_ISR
void UART_1_ISR(void)
2014-06-11 14:59:24 +02:00
{
irq_handler(1);
}
#endif
#ifdef UART_2_ISR
void UART_2_ISR(void)
{
irq_handler(2);
2014-06-11 14:59:24 +02:00
}
#endif