2013-08-19 19:19:57 +02:00
|
|
|
/*
|
|
|
|
* uart.c - Implementation for the TelosB UART
|
|
|
|
* Copyright (C) 2013 Oliver Hahm <oliver.hahm@inria.fr>
|
|
|
|
*
|
2014-07-31 19:53:53 +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.
|
2013-08-19 19:19:57 +02:00
|
|
|
*/
|
2013-08-19 18:58:45 +02:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "board.h"
|
|
|
|
#include "kernel.h"
|
2013-08-19 19:19:57 +02:00
|
|
|
#include "board_uart0.h"
|
2013-08-19 18:58:45 +02:00
|
|
|
|
|
|
|
#define UART1_TX U1TXBUF
|
|
|
|
#define UART1_WAIT_TXDONE() while ( (U1TCTL & TXEPT) == 0 ) { _NOP(); }
|
|
|
|
|
|
|
|
#define BAUDRATE (115200ul)
|
|
|
|
|
2013-08-19 19:19:57 +02:00
|
|
|
static uint8_t calc_umctl(uint16_t br)
|
|
|
|
{
|
|
|
|
/* from TI slaa049 */
|
|
|
|
register uint8_t CMOD = 256 * br - 256 * (br + 1) / 2;
|
|
|
|
register uint8_t c = 0;
|
|
|
|
register int i = 0;
|
|
|
|
register uint8_t a = CMOD;
|
|
|
|
a <<= 1;
|
|
|
|
|
|
|
|
do {
|
|
|
|
if (a & 0x80) { /* Overflow to integer? */
|
|
|
|
a = a - 128 + CMOD; /* Yes, subtract 1.000000 */
|
|
|
|
c |= 0x80;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
a += CMOD; /* No, add fraction */
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i == 7) {
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
c >>= 1;
|
|
|
|
}
|
|
|
|
while (1);
|
|
|
|
}
|
|
|
|
|
2013-08-19 18:58:45 +02:00
|
|
|
void uart_init(void)
|
|
|
|
{
|
|
|
|
UCTL1 = SWRST; /* hold UART1 module in reset */
|
|
|
|
UCTL1 |= CHAR; /* 8-bit character */
|
|
|
|
|
|
|
|
/* 115200 baud, clocked from 4.8MHz SMCLK */
|
|
|
|
UTCTL1 |= SSEL1; /* UCLK = SCLK */
|
|
|
|
UBR01 = F_CPU / BAUDRATE;
|
|
|
|
UBR11 = (F_CPU / BAUDRATE) >> 8;
|
2013-08-19 19:19:57 +02:00
|
|
|
UMCTL1 = calc_umctl(F_CPU / BAUDRATE); /* set modulation */
|
2013-08-19 18:58:45 +02:00
|
|
|
|
|
|
|
ME2 |= UTXE1 + URXE1; /* enable UART1 TX/RX */
|
|
|
|
UCTL1 &= ~SWRST; /* clear UART1 reset bit */
|
2013-08-19 19:19:57 +02:00
|
|
|
|
|
|
|
IE2 |= URXIE1; /* enable rx interrupt */
|
|
|
|
IFG1 &= ~UTXIFG1;
|
2013-08-19 18:58:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int putchar(int c)
|
|
|
|
{
|
|
|
|
UART1_TX = c;
|
|
|
|
UART1_WAIT_TXDONE();
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t uart_readByte(void)
|
|
|
|
{
|
|
|
|
return U1RXBUF;
|
|
|
|
}
|
2013-08-19 19:19:57 +02:00
|
|
|
|
|
|
|
void usart1irq(void);
|
|
|
|
/**
|
|
|
|
* \brief the interrupt function
|
|
|
|
*/
|
|
|
|
interrupt(USART1RX_VECTOR) usart1irq(void)
|
|
|
|
{
|
|
|
|
/* Check status register for receive errors. */
|
|
|
|
if (U1RCTL & RXERR) {
|
|
|
|
if (U1RCTL & FE) {
|
|
|
|
puts("rx framing error");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (U1RCTL & OE) {
|
|
|
|
puts("rx overrun error");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (U1RCTL & PE) {
|
|
|
|
puts("rx parity error");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (U1RCTL & BRK) {
|
|
|
|
puts("rx break error");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Clear error flags by forcing a dummy read. */
|
2014-08-01 10:12:33 +02:00
|
|
|
volatile int c = U1RXBUF;
|
|
|
|
(void) c;
|
2013-08-19 19:19:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef MODULE_UART0
|
|
|
|
else if (uart0_handler_pid) {
|
2014-08-01 10:12:33 +02:00
|
|
|
volatile int c = U1RXBUF;
|
2013-08-19 19:19:57 +02:00
|
|
|
uart0_handle_incoming(c);
|
|
|
|
uart0_notify_thread();
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
}
|