2014-06-11 14:59:24 +02:00
|
|
|
/*
|
2016-03-16 10:27:43 +01:00
|
|
|
* Copyright (C) 2014-2016 Freie Universität Berlin
|
2014-06-11 14:59:24 +02:00
|
|
|
*
|
2016-03-16 10:27:43 +01: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
|
|
|
|
* @{
|
|
|
|
*
|
2015-05-22 07:34:41 +02:00
|
|
|
* @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"
|
2015-08-05 15:11:21 +02:00
|
|
|
#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.
|
|
|
|
*/
|
2016-03-16 10:27:43 +01:00
|
|
|
static uart_isr_ctx_t isr_ctx[UART_NUMOF];
|
2014-06-11 14:59:24 +02:00
|
|
|
|
2016-03-16 10:27:43 +01:00
|
|
|
static inline USART_TypeDef *dev(uart_t uart)
|
2014-06-11 14:59:24 +02:00
|
|
|
{
|
2016-03-16 10:27:43 +01:00
|
|
|
return uart_config[uart].dev;
|
|
|
|
}
|
2014-06-11 14:59:24 +02:00
|
|
|
|
2016-03-16 10:27:43 +01: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
|
|
|
}
|
2016-03-16 10:27:43 +01:00
|
|
|
else {
|
|
|
|
RCC->APB2ENR |= uart_config[uart].rcc_pin;
|
2014-06-11 14:59:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-16 10:27:43 +01: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
|
|
|
{
|
2016-03-16 10:27:43 +01:00
|
|
|
uint32_t bus_clk;
|
2014-06-11 14:59:24 +02:00
|
|
|
uint16_t mantissa;
|
|
|
|
uint8_t fraction;
|
|
|
|
|
2016-03-16 10:27:43 +01:00
|
|
|
/* make sure the given device is valid */
|
|
|
|
if (uart >= UART_NUMOF) {
|
|
|
|
return -1;
|
2014-06-11 14:59:24 +02:00
|
|
|
}
|
2016-03-16 10:27:43 +01:00
|
|
|
|
|
|
|
/* save ISR context */
|
|
|
|
isr_ctx[uart].rx_cb = rx_cb;
|
|
|
|
isr_ctx[uart].arg = arg;
|
|
|
|
|
2015-08-05 15:11:21 +02:00
|
|
|
/* configure RX and TX pin */
|
2016-03-16 10:27:43 +01:00
|
|
|
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++) {
|
2016-03-16 10:27:43 +01:00
|
|
|
while(!(dev(uart)->SR & USART_SR_TXE)) {}
|
|
|
|
dev(uart)->DR = data[i];
|
2014-06-11 14:59:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-16 10:27:43 +01:00
|
|
|
static inline void irq_handler(uart_t uart)
|
2014-06-11 14:59:24 +02:00
|
|
|
{
|
2016-03-16 10:27:43 +01: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
|
|
|
}
|
2016-03-16 10:27:43 +01:00
|
|
|
if (status & USART_SR_ORE) {
|
2015-10-20 16:27:05 +02:00
|
|
|
/* ORE is cleared by reading SR and DR sequentially */
|
2016-03-16 10:27:43 +01:00
|
|
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-16 10:27:43 +01:00
|
|
|
#ifdef UART_0_ISR
|
2014-10-16 23:13:01 +02:00
|
|
|
void UART_0_ISR(void)
|
2014-06-11 14:59:24 +02:00
|
|
|
{
|
2016-03-16 10:27:43 +01:00
|
|
|
irq_handler(0);
|
2014-06-11 14:59:24 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2016-03-16 10:27:43 +01:00
|
|
|
#ifdef UART_1_ISR
|
2014-10-16 23:13:01 +02:00
|
|
|
void UART_1_ISR(void)
|
2014-06-11 14:59:24 +02:00
|
|
|
{
|
2016-03-16 10:27:43 +01: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
|