2014-02-03 19:36:12 +01:00
|
|
|
/*
|
2015-10-19 14:51:20 +02:00
|
|
|
* Copyright (C) 2014-2015 Freie Universität Berlin
|
2014-02-03 19:36:12 +01:00
|
|
|
*
|
2014-08-27 18:47:31 +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-02-03 19:36:12 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2015-11-19 15:54:52 +01:00
|
|
|
* @defgroup drivers_periph_uart UART
|
|
|
|
* @ingroup drivers_periph
|
2014-02-03 19:36:12 +01:00
|
|
|
* @brief Low-level UART peripheral driver
|
|
|
|
*
|
2015-12-18 15:38:57 +01:00
|
|
|
* This is a basic UART (Universal Asynchronous Receiver Transmitter) interface
|
2018-12-06 10:08:56 +01:00
|
|
|
* to allow platform independent access to the MCU's serial communication
|
|
|
|
* abilities. This interface is intentionally designed to be as simple as
|
2019-02-11 09:35:23 +01:00
|
|
|
* possible, to allow for easy implementation and maximum portability.
|
2015-12-18 15:38:57 +01:00
|
|
|
*
|
2019-02-11 09:35:23 +01:00
|
|
|
* The simple interface provides capabilities to initialize and configure
|
|
|
|
* the serial communication module, which automatically enables for receiving
|
|
|
|
* data, as well as writing data to the UART port, which means transmitting
|
|
|
|
* data. The UART device and the corresponding pins need to be mapped in
|
|
|
|
* `RIOT/boards/ * /include/periph_conf.h`. Furthermore, you need to select the
|
2018-12-06 10:08:56 +01:00
|
|
|
* baudrate for initialization which is typically {9600, 19200, 38400, 57600,
|
2019-02-11 09:35:23 +01:00
|
|
|
* 115200} baud. Additionally, you should register a callback function that is
|
2018-12-06 10:08:56 +01:00
|
|
|
* executed in interrupt context when data is being received. The driver will
|
|
|
|
* then read the received data byte, call the registered callback function and
|
|
|
|
* pass the received data to it via its argument. The interface enforces the
|
|
|
|
* receiving to be implemented in an interrupt driven mode. Thus, you never know
|
|
|
|
* how many bytes are going to be received and might want to handle that in your
|
|
|
|
* specific callback function. The transmit function can be implemented in any
|
2019-02-11 09:35:23 +01:00
|
|
|
* way. You can also configure parity, the number of data and stop bits, i.e.
|
|
|
|
* such combinations as 8-E-1, 7-N-2 etc. 8-N-1 mode is set by default.
|
2015-12-18 15:38:57 +01:00
|
|
|
*
|
2018-12-06 10:08:56 +01:00
|
|
|
* By default the @p UART_DEV(0) device of each board is initialized and mapped
|
|
|
|
* to STDIO in RIOT which is used for standard input/output functions like
|
|
|
|
* `printf()` or `puts()`.
|
2015-12-18 15:38:57 +01:00
|
|
|
*
|
2018-12-06 10:06:50 +01:00
|
|
|
* # (Low-) Power Implications
|
|
|
|
*
|
|
|
|
* After initialization, the UART peripheral **should** be powered on and
|
|
|
|
* active. The UART can later be explicitly put to sleep and woken up by calling
|
|
|
|
* the uart_poweron() and uart_poweroff() functions. Once woken up using
|
|
|
|
* uart_poweron(), the UART **should** transparently continue it's previously
|
|
|
|
* configured operation.
|
|
|
|
*
|
|
|
|
* While the UART is active, the implementation might need to block certain
|
|
|
|
* power states.
|
|
|
|
*
|
2017-06-06 13:41:59 +02:00
|
|
|
* @{
|
|
|
|
*
|
2014-10-25 15:37:04 +02:00
|
|
|
* @file
|
2015-10-19 14:51:20 +02:00
|
|
|
* @brief Low-level UART peripheral driver interface definition
|
2014-02-03 19:36:12 +01:00
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*/
|
|
|
|
|
2015-03-23 21:07:50 +01:00
|
|
|
#ifndef PERIPH_UART_H
|
|
|
|
#define PERIPH_UART_H
|
2014-02-03 19:36:12 +01:00
|
|
|
|
2016-01-19 14:11:35 +01:00
|
|
|
#include <stddef.h>
|
2014-02-03 19:36:12 +01:00
|
|
|
#include <stdint.h>
|
2016-04-01 10:54:55 +02:00
|
|
|
#include <limits.h>
|
2014-02-03 19:36:12 +01:00
|
|
|
|
2015-10-19 14:51:20 +02:00
|
|
|
#include "periph_cpu.h"
|
2014-02-03 19:36:12 +01:00
|
|
|
#include "periph_conf.h"
|
2015-10-19 14:51:20 +02:00
|
|
|
/* TODO: remove once all platforms are ported to this interface */
|
|
|
|
#include "periph/dev_enums.h"
|
2014-02-03 19:36:12 +01:00
|
|
|
|
2014-10-13 15:49:17 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2015-10-19 14:51:20 +02:00
|
|
|
/**
|
|
|
|
* @brief Define default UART type identifier
|
|
|
|
*/
|
|
|
|
#ifndef HAVE_UART_T
|
|
|
|
typedef unsigned int uart_t;
|
2014-02-03 19:36:12 +01:00
|
|
|
#endif
|
2015-10-19 14:51:20 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Default UART undefined value
|
|
|
|
*/
|
|
|
|
#ifndef UART_UNDEF
|
2016-04-01 10:54:55 +02:00
|
|
|
#define UART_UNDEF (UINT_MAX)
|
2014-02-03 19:36:12 +01:00
|
|
|
#endif
|
2015-10-19 14:51:20 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Default UART device access macro
|
|
|
|
*/
|
|
|
|
#ifndef UART_DEV
|
|
|
|
#define UART_DEV(x) (x)
|
2014-02-03 19:36:12 +01:00
|
|
|
#endif
|
|
|
|
|
2014-07-28 14:45:51 +02:00
|
|
|
/**
|
2015-10-19 14:51:20 +02:00
|
|
|
* @brief Signature for receive interrupt callback
|
2014-07-28 14:45:51 +02:00
|
|
|
*
|
2015-10-19 14:51:20 +02:00
|
|
|
* @param[in] arg context to the callback (optional)
|
2014-07-28 14:45:51 +02:00
|
|
|
* @param[in] data the byte that was received
|
|
|
|
*/
|
2016-03-15 10:57:44 +01:00
|
|
|
typedef void(*uart_rx_cb_t)(void *arg, uint8_t data);
|
2014-07-28 14:45:51 +02:00
|
|
|
|
|
|
|
/**
|
2015-10-19 14:51:20 +02:00
|
|
|
* @brief Interrupt context for a UART device
|
2014-07-28 14:45:51 +02:00
|
|
|
*/
|
2015-10-19 14:51:20 +02:00
|
|
|
#ifndef HAVE_UART_ISR_CTX_T
|
|
|
|
typedef struct {
|
|
|
|
uart_rx_cb_t rx_cb; /**< data received interrupt callback */
|
|
|
|
void *arg; /**< argument to both callback routines */
|
|
|
|
} uart_isr_ctx_t;
|
|
|
|
#endif
|
2014-02-03 19:36:12 +01:00
|
|
|
|
2016-10-27 14:17:43 +02:00
|
|
|
/**
|
|
|
|
* @brief Possible UART return values
|
|
|
|
*/
|
|
|
|
enum {
|
|
|
|
UART_OK = 0, /**< everything in order */
|
|
|
|
UART_NODEV = -1, /**< invalid UART device given */
|
|
|
|
UART_NOBAUD = -2, /**< given baudrate is not applicable */
|
|
|
|
UART_INTERR = -3, /**< all other internal errors */
|
|
|
|
UART_NOMODE = -4 /**< given mode is not applicable */
|
|
|
|
};
|
|
|
|
|
2019-01-08 10:54:26 +01:00
|
|
|
/**
|
|
|
|
* @brief Definition of possible parity modes
|
|
|
|
*/
|
|
|
|
#ifndef HAVE_UART_PARITY_T
|
|
|
|
typedef enum {
|
|
|
|
UART_PARITY_NONE, /**< no parity */
|
|
|
|
UART_PARITY_EVEN, /**< even parity */
|
|
|
|
UART_PARITY_ODD, /**< odd parity */
|
|
|
|
UART_PARITY_MARK, /**< mark parity */
|
|
|
|
UART_PARITY_SPACE /**< space parity */
|
|
|
|
} uart_parity_t;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Definition of possible data bits lengths in a UART frame
|
|
|
|
*/
|
|
|
|
#ifndef HAVE_UART_DATA_BITS_T
|
|
|
|
typedef enum {
|
|
|
|
UART_DATA_BITS_5, /**< 5 data bits */
|
|
|
|
UART_DATA_BITS_6, /**< 6 data bits */
|
|
|
|
UART_DATA_BITS_7, /**< 7 data bits */
|
|
|
|
UART_DATA_BITS_8, /**< 8 data bits */
|
|
|
|
} uart_data_bits_t;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Definition of possible stop bits lengths in a UART frame
|
|
|
|
*/
|
|
|
|
#ifndef HAVE_UART_STOP_BITS_T
|
|
|
|
typedef enum {
|
|
|
|
UART_STOP_BITS_1, /**< 1 stop bit */
|
|
|
|
UART_STOP_BITS_2, /**< 2 stop bits */
|
|
|
|
} uart_stop_bits_t;
|
|
|
|
#endif
|
|
|
|
|
2014-02-03 19:36:12 +01:00
|
|
|
/**
|
2015-10-19 14:51:20 +02:00
|
|
|
* @brief Initialize a given UART device
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2014-02-03 19:36:12 +01:00
|
|
|
* The UART device will be initialized with the following configuration:
|
|
|
|
* - 8 data bits
|
|
|
|
* - no parity
|
|
|
|
* - 1 stop bit
|
2015-10-19 14:51:20 +02:00
|
|
|
* - baudrate as given
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2017-02-07 16:16:05 +01:00
|
|
|
* If no callback parameter is given (rx_cb := NULL), the UART will be
|
|
|
|
* initialized in TX only mode.
|
|
|
|
*
|
2015-10-19 14:51:20 +02:00
|
|
|
* @param[in] uart UART device to initialize
|
|
|
|
* @param[in] baudrate desired baudrate in baud/s
|
|
|
|
* @param[in] rx_cb receive callback, executed in interrupt context once
|
2017-02-07 16:16:05 +01:00
|
|
|
* for every byte that is received (RX buffer filled),
|
|
|
|
* set to NULL for TX only mode
|
2015-10-19 14:51:20 +02:00
|
|
|
* @param[in] arg optional context passed to the callback functions
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2016-10-27 14:17:43 +02:00
|
|
|
* @return UART_OK on success
|
|
|
|
* @return UART_NODEV on invalid UART device
|
|
|
|
* @return UART_NOBAUD on inapplicable baudrate
|
|
|
|
* @return UART_INTERR on other errors
|
2014-02-03 19:36:12 +01:00
|
|
|
*/
|
2015-10-19 14:51:20 +02:00
|
|
|
int uart_init(uart_t uart, uint32_t baudrate, uart_rx_cb_t rx_cb, void *arg);
|
2014-02-03 19:36:12 +01:00
|
|
|
|
2020-03-13 18:13:12 +01:00
|
|
|
#if defined(MODULE_PERIPH_UART_RECONFIGURE) || DOXYGEN
|
|
|
|
/**
|
|
|
|
* @brief Change the pins of the given UART back to plain GPIO functionality
|
|
|
|
*
|
|
|
|
* The pin mux of the RX and TX pins of the bus will be changed back to
|
|
|
|
* default (GPIO) mode and the UART is powered off.
|
|
|
|
* This allows to use the UART pins for another function and return to UART
|
|
|
|
* functionality again by calling @ref uart_init_pins
|
|
|
|
*
|
|
|
|
* If you want the pin to be in a defined state, call @ref gpio_init on it.
|
|
|
|
*
|
|
|
|
* @note Until this is implemented on all platforms, this requires the
|
|
|
|
* periph_uart_reconfigure feature to be used.
|
|
|
|
*
|
|
|
|
* @param[in] uart the device to de-initialize
|
|
|
|
*/
|
|
|
|
void uart_deinit_pins(uart_t uart);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initialize the used UART pins, i.e. RX and TX
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* After calling uart_init, the pins must be initialized (i.e. uart_init is
|
|
|
|
* calling this function internally). In normal cases, this function will not
|
|
|
|
* be used. But there are some devices, that use UART bus lines also for other
|
|
|
|
* purposes and need the option to dynamically re-configure one or more of the
|
|
|
|
* used pins. So they can take control over certain pins and return control back
|
|
|
|
* to the UART driver using this function.
|
|
|
|
*
|
|
|
|
* The pins used are configured in the board's periph_conf.h.
|
|
|
|
*
|
|
|
|
* @param[in] uart UART device the pins are configure for
|
|
|
|
*/
|
|
|
|
void uart_init_pins(uart_t uart);
|
|
|
|
|
|
|
|
#if DOXYGEN
|
|
|
|
/**
|
|
|
|
* @brief Get the RX pin of the given UART.
|
|
|
|
*
|
|
|
|
* @param[in] uart The device to query
|
|
|
|
*
|
|
|
|
* @note Until this is implemented on all platforms, this requires the
|
|
|
|
* periph_uart_reconfigure feature to be used.
|
|
|
|
*
|
|
|
|
* @return The GPIO used for the UART RX line.
|
|
|
|
*/
|
|
|
|
gpio_t uart_pin_rx(uart_t uart);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Get the TX pin of the given UART.
|
|
|
|
*
|
|
|
|
* @param[in] uart The device to query
|
|
|
|
*
|
|
|
|
* @note Until this is implemented on all platforms, this requires the
|
|
|
|
* periph_uart_reconfigure feature to be used.
|
|
|
|
*
|
|
|
|
* @return The GPIO used for the UART TX line.
|
|
|
|
*/
|
|
|
|
gpio_t uart_pin_tx(uart_t uart);
|
|
|
|
|
|
|
|
#endif /* DOXYGEN */
|
|
|
|
#endif /* MODULE_PERIPH_UART_RECONFIGURE */
|
|
|
|
|
2019-01-08 10:54:26 +01:00
|
|
|
/**
|
|
|
|
* @brief Setup parity, data and stop bits for a given UART device
|
|
|
|
*
|
2019-02-11 09:35:23 +01:00
|
|
|
* @param[in] uart UART device to configure
|
2019-01-08 10:54:26 +01:00
|
|
|
* @param[in] data_bits number of data bits in a UART frame
|
|
|
|
* @param[in] parity parity mode
|
|
|
|
* @param[in] stop_bits number of stop bits in a UART frame
|
|
|
|
*
|
|
|
|
* @return UART_OK on success
|
|
|
|
* @return UART_NOMODE on other errors
|
|
|
|
*/
|
|
|
|
int uart_mode(uart_t uart, uart_data_bits_t data_bits, uart_parity_t parity,
|
|
|
|
uart_stop_bits_t stop_bits);
|
|
|
|
|
2014-02-03 19:36:12 +01:00
|
|
|
/**
|
2015-10-19 14:51:20 +02:00
|
|
|
* @brief Write data from the given buffer to the specified UART device
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2015-10-19 14:51:20 +02:00
|
|
|
* This function is blocking, as it will only return after @p len bytes from the
|
|
|
|
* given buffer have been send. The way this data is send is up to the
|
|
|
|
* implementation: active waiting, interrupt driven, DMA, etc.
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2015-10-19 14:51:20 +02:00
|
|
|
* @param[in] uart UART device to use for transmission
|
|
|
|
* @param[in] data data buffer to send
|
|
|
|
* @param[in] len number of bytes to send
|
2014-05-14 10:46:15 +02:00
|
|
|
*
|
2014-02-03 19:36:12 +01:00
|
|
|
*/
|
2015-10-19 14:51:20 +02:00
|
|
|
void uart_write(uart_t uart, const uint8_t *data, size_t len);
|
2014-02-03 19:36:12 +01:00
|
|
|
|
2014-07-28 14:45:51 +02:00
|
|
|
/**
|
2015-10-19 14:51:20 +02:00
|
|
|
* @brief Power on the given UART device
|
2014-07-28 14:45:51 +02:00
|
|
|
*
|
|
|
|
* @param[in] uart the UART device to power on
|
|
|
|
*/
|
|
|
|
void uart_poweron(uart_t uart);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Power off the given UART device
|
|
|
|
*
|
|
|
|
* @param[in] uart the UART device to power off
|
|
|
|
*/
|
|
|
|
void uart_poweroff(uart_t uart);
|
|
|
|
|
2014-10-13 15:49:17 +02:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2015-03-23 21:07:50 +01:00
|
|
|
#endif /* PERIPH_UART_H */
|
2014-02-03 19:36:12 +01:00
|
|
|
/** @} */
|