Files
paparazzi/sw/airborne/mcu_periph/uart.c
T
Felix Ruess b8c77189c0 [pprzlink] update to C API with file descriptor param
This was already once proposed and mergd with #1589 (and reverted again) since with these changes the simulators don't work anymore.
So this needs some more work and checking before it can be integrated again.
2016-04-02 11:40:23 +02:00

299 lines
7.2 KiB
C

/*
* Copyright (C) 2010 The Paparazzi Team
*
* This file is part of paparazzi.
*
* paparazzi is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2, or (at your option)
* any later version.
*
* paparazzi is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with paparazzi; see the file COPYING. If not, write to
* the Free Software Foundation, 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*
*/
#include "mcu_periph/uart.h"
#if PERIODIC_TELEMETRY
#include "subsystems/datalink/telemetry.h"
#endif
#if USE_UART0
struct uart_periph uart0;
#if PERIODIC_TELEMETRY
static void send_uart0_err(struct transport_tx *trans, struct link_device *dev)
{
uint16_t ore = uart0.ore;
uint16_t ne_err = uart0.ne_err;
uint16_t fe_err = uart0.fe_err;
uint8_t _bus0 = 0;
pprz_msg_send_UART_ERRORS(trans, dev, AC_ID,
&ore, &ne_err, &fe_err, &_bus0);
}
#endif
#endif
#if USE_UART1
struct uart_periph uart1;
#if PERIODIC_TELEMETRY
static void send_uart1_err(struct transport_tx *trans, struct link_device *dev)
{
uint16_t ore = uart1.ore;
uint16_t ne_err = uart1.ne_err;
uint16_t fe_err = uart1.fe_err;
uint8_t _bus1 = 1;
pprz_msg_send_UART_ERRORS(trans, dev, AC_ID,
&ore, &ne_err, &fe_err, &_bus1);
}
#endif
#endif
#if USE_UART2
struct uart_periph uart2;
#if PERIODIC_TELEMETRY
static void send_uart2_err(struct transport_tx *trans, struct link_device *dev)
{
uint16_t ore = uart2.ore;
uint16_t ne_err = uart2.ne_err;
uint16_t fe_err = uart2.fe_err;
uint8_t _bus2 = 2;
pprz_msg_send_UART_ERRORS(trans, dev, AC_ID,
&ore, &ne_err, &fe_err, &_bus2);
}
#endif
#endif
#if USE_UART3
struct uart_periph uart3;
#if PERIODIC_TELEMETRY
static void send_uart3_err(struct transport_tx *trans, struct link_device *dev)
{
uint16_t ore = uart3.ore;
uint16_t ne_err = uart3.ne_err;
uint16_t fe_err = uart3.fe_err;
uint8_t _bus3 = 3;
pprz_msg_send_UART_ERRORS(trans, dev, AC_ID,
&ore, &ne_err, &fe_err, &_bus3);
}
#endif
#endif
#if USE_UART4
struct uart_periph uart4;
#if PERIODIC_TELEMETRY
static void send_uart4_err(struct transport_tx *trans, struct link_device *dev)
{
uint16_t ore = uart4.ore;
uint16_t ne_err = uart4.ne_err;
uint16_t fe_err = uart4.fe_err;
uint8_t _bus4 = 4;
pprz_msg_send_UART_ERRORS(trans, dev, AC_ID,
&ore, &ne_err, &fe_err, &_bus4);
}
#endif
#endif
#if USE_UART5
struct uart_periph uart5;
#if PERIODIC_TELEMETRY
static void send_uart5_err(struct transport_tx *trans, struct link_device *dev)
{
uint16_t ore = uart5.ore;
uint16_t ne_err = uart5.ne_err;
uint16_t fe_err = uart5.fe_err;
uint8_t _bus5 = 5;
pprz_msg_send_UART_ERRORS(trans, dev, AC_ID,
&ore, &ne_err, &fe_err, &_bus5);
}
#endif
#endif
#if USE_UART6
struct uart_periph uart6;
#if PERIODIC_TELEMETRY
static void send_uart6_err(struct transport_tx *trans, struct link_device *dev)
{
uint16_t ore = uart6.ore;
uint16_t ne_err = uart6.ne_err;
uint16_t fe_err = uart6.fe_err;
uint8_t _bus6 = 6;
pprz_msg_send_UART_ERRORS(trans, dev, AC_ID,
&ore, &ne_err, &fe_err, &_bus6);
}
#endif
#endif
#if USE_UART7
struct uart_periph uart7;
#if PERIODIC_TELEMETRY
static void send_uart7_err(struct transport_tx *trans, struct link_device *dev)
{
uint16_t ore = uart7.ore;
uint16_t ne_err = uart7.ne_err;
uint16_t fe_err = uart7.fe_err;
uint8_t _bus7 = 7;
pprz_msg_send_UART_ERRORS(trans, dev, AC_ID,
&ore, &ne_err, &fe_err, &_bus7);
}
#endif
#endif
#if USE_UART8
struct uart_periph uart8;
#if PERIODIC_TELEMETRY
static void send_uart8_err(struct transport_tx *trans, struct link_device *dev)
{
uint16_t ore = uart8.ore;
uint16_t ne_err = uart8.ne_err;
uint16_t fe_err = uart8.fe_err;
uint8_t _bus8 = 8;
pprz_msg_send_UART_ERRORS(trans, dev, AC_ID,
&ore, &ne_err, &fe_err, &_bus8);
}
#endif
#endif
#if PERIODIC_TELEMETRY
static void send_uart_err(struct transport_tx *trans __attribute__ ((unused)),
struct link_device *dev __attribute__ ((unused)))
{
static uint8_t uart_nb_cnt = 0;
switch (uart_nb_cnt) {
#if USE_UART0
case 0:
send_uart0_err(trans, dev); break;
#endif
#if USE_UART1
case 1:
send_uart1_err(trans, dev); break;
#endif
#if USE_UART2
case 2:
send_uart2_err(trans, dev); break;
#endif
#if USE_UART3
case 3:
send_uart3_err(trans, dev); break;
#endif
#if USE_UART4
case 4:
send_uart4_err(trans, dev); break;
#endif
#if USE_UART5
case 5:
send_uart5_err(trans, dev); break;
#endif
#if USE_UART6
case 6:
send_uart6_err(trans, dev); break;
#endif
#if USE_UART7
case 7:
send_uart7_err(trans, dev); break;
#endif
#if USE_UART8
case 8:
send_uart8_err(trans, dev); break;
#endif
default: break;
}
uart_nb_cnt++;
if (uart_nb_cnt == 9) {
uart_nb_cnt = 0;
}
}
#endif
void uart_periph_init(struct uart_periph *p)
{
p->rx_insert_idx = 0;
p->rx_extract_idx = 0;
p->tx_insert_idx = 0;
p->tx_extract_idx = 0;
p->tx_running = false;
p->ore = 0;
p->ne_err = 0;
p->fe_err = 0;
p->device.periph = (void *)p;
p->device.check_free_space = (check_free_space_t) uart_check_free_space;
p->device.put_byte = (put_byte_t) uart_put_byte;
p->device.put_buffer = (put_buffer_t) uart_put_buffer;
p->device.send_message = (send_message_t) uart_send_message;
p->device.char_available = (char_available_t) uart_char_available;
p->device.get_byte = (get_byte_t) uart_getch;
p->device.set_baudrate = (set_baudrate_t) uart_periph_set_baudrate;
#if PERIODIC_TELEMETRY
// the first to register do it for the others
register_periodic_telemetry(DefaultPeriodic, PPRZ_MSG_ID_UART_ERRORS, send_uart_err);
#endif
}
bool WEAK uart_check_free_space(struct uart_periph *p, long *fd __attribute__((unused)), uint16_t len)
{
int16_t space = p->tx_extract_idx - p->tx_insert_idx;
if (space <= 0) {
space += UART_TX_BUFFER_SIZE;
}
return (uint16_t)(space - 1) >= len;
}
// Weak implementation of put_buffer, byte by byte
void WEAK uart_put_buffer(struct uart_periph *p, long fd, const uint8_t *data, uint16_t len)
{
int i = 0;
for (i = 0; i < len; i++) {
uart_put_byte(p, fd, data[i]);
}
}
// Weak implementation of send_message, not needed for stream operation
void WEAK uart_send_message(struct uart_periph *p __attribute__((unused)), long fd __attribute__((unused)))
{
}
uint8_t WEAK uart_getch(struct uart_periph *p)
{
uint8_t ret = p->rx_buf[p->rx_extract_idx];
p->rx_extract_idx = (p->rx_extract_idx + 1) % UART_RX_BUFFER_SIZE;
return ret;
}
uint16_t WEAK uart_char_available(struct uart_periph *p)
{
int16_t available = p->rx_insert_idx - p->rx_extract_idx;
if (available < 0) {
available += UART_RX_BUFFER_SIZE;
}
return (uint16_t)available;
}
void WEAK uart_arch_init(void)
{
}