mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
441 lines
14 KiB
C
441 lines
14 KiB
C
/*
|
|
* Copyright (C) 2014-2015 Freie Universität Berlin
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
/**
|
|
* @defgroup drivers_periph_uart UART
|
|
* @ingroup drivers_periph
|
|
* @brief Low-level UART peripheral driver
|
|
*
|
|
* This is a basic UART (Universal Asynchronous Receiver Transmitter) interface
|
|
* to allow platform independent access to the MCU's serial communication
|
|
* abilities. This interface is intentionally designed to be as simple as
|
|
* possible, to allow for easy implementation and maximum portability.
|
|
*
|
|
* 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
|
|
* symbol rate for initialization which is typically {9600, 19200, 38400, 57600,
|
|
* 115200} baud. Additionally, you should register a callback function that is
|
|
* 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
|
|
* 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.
|
|
*
|
|
* 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()`.
|
|
*
|
|
* # (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.
|
|
*
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Low-level UART peripheral driver interface definition
|
|
*
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
*/
|
|
|
|
#ifndef PERIPH_UART_H
|
|
#define PERIPH_UART_H
|
|
|
|
#include <errno.h>
|
|
#include <limits.h>
|
|
#include <stddef.h>
|
|
#include <stdint.h>
|
|
|
|
#include "periph_cpu.h"
|
|
#include "periph_conf.h"
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
/**
|
|
* @brief Threshold under which polling transfers are used instead of DMA
|
|
* TODO: determine at run-time based on baudrate
|
|
*/
|
|
#ifndef CONFIG_UART_DMA_THRESHOLD_BYTES
|
|
#define CONFIG_UART_DMA_THRESHOLD_BYTES 8
|
|
#endif
|
|
|
|
/**
|
|
* @brief Define default UART type identifier
|
|
*/
|
|
#ifndef HAVE_UART_T
|
|
typedef uint_fast8_t uart_t;
|
|
#endif
|
|
|
|
/**
|
|
* @brief Default UART undefined value
|
|
*/
|
|
#ifndef UART_UNDEF
|
|
#define UART_UNDEF (UINT_FAST8_MAX)
|
|
#endif
|
|
|
|
/**
|
|
* @brief Default UART device access macro
|
|
*/
|
|
#ifndef UART_DEV
|
|
#define UART_DEV(x) (x)
|
|
#endif
|
|
|
|
/**
|
|
* @brief Signature for receive interrupt callback
|
|
*
|
|
* @param[in] arg context to the callback (optional)
|
|
* @param[in] data the byte that was received
|
|
*/
|
|
typedef void(*uart_rx_cb_t)(void *arg, uint8_t data);
|
|
|
|
/**
|
|
* @brief Signature for receive start condition interrupt callback
|
|
*
|
|
* @param[in] arg context to the callback (optional)
|
|
*/
|
|
typedef void(*uart_rxstart_cb_t)(void *arg);
|
|
|
|
/**
|
|
* @brief Interrupt context for a UART device
|
|
*/
|
|
#ifndef HAVE_UART_ISR_CTX_T
|
|
typedef struct {
|
|
uart_rx_cb_t rx_cb; /**< data received interrupt callback */
|
|
void *arg; /**< argument to data received callback */
|
|
#ifdef MODULE_PERIPH_UART_RXSTART_IRQ
|
|
uart_rxstart_cb_t rxs_cb; /**< start condition received interrupt callback */
|
|
void *rxs_arg; /**< argument to start condition received callback */
|
|
#endif
|
|
} uart_isr_ctx_t;
|
|
#endif
|
|
|
|
/**
|
|
* @brief Possible UART return values
|
|
*
|
|
* @deprecated Use the errno constants directly instead.
|
|
*/
|
|
enum {
|
|
UART_OK = 0, /**< everything in order */
|
|
UART_NODEV = -ENODEV, /**< invalid UART device given */
|
|
UART_NOBAUD = -ENOTSUP, /**< given symbol rate is not applicable */
|
|
UART_NOMODE = -ENOTSUP, /**< given mode is not applicable */
|
|
UART_INTERR = -EIO, /**< all other internal errors */
|
|
};
|
|
|
|
/**
|
|
* @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
|
|
|
|
/**
|
|
* @brief Initialize a given UART device
|
|
*
|
|
* The UART device will be initialized with the following configuration:
|
|
* - 8 data bits
|
|
* - no parity
|
|
* - 1 stop bit
|
|
* - symbol rate as given
|
|
*
|
|
* If no callback parameter is given (rx_cb := NULL), the UART will be
|
|
* initialized in TX only mode.
|
|
*
|
|
* @param[in] uart UART device to initialize
|
|
* @param[in] baud desired symbol rate in baud
|
|
* @param[in] rx_cb receive callback, executed in interrupt context once
|
|
* for every byte that is received (RX buffer filled),
|
|
* set to NULL for TX only mode
|
|
* @param[in] arg optional context passed to the callback functions
|
|
*
|
|
* @retval 0 Success
|
|
* @retval -ENODEV Invalid UART device
|
|
* @retval -ENOTSUP Unsupported symbol rate
|
|
* @retval <0 On other errors
|
|
*/
|
|
int uart_init(uart_t uart, uint32_t baud, uart_rx_cb_t rx_cb, void *arg);
|
|
|
|
#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 */
|
|
|
|
/**
|
|
* @brief Get the CTS 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 CTS line.
|
|
*/
|
|
gpio_t uart_pin_cts(uart_t uart);
|
|
|
|
/**
|
|
* @brief Get the RTS 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 RTS line.
|
|
*/
|
|
gpio_t uart_pin_rts(uart_t uart);
|
|
|
|
#endif /* MODULE_PERIPH_UART_RECONFIGURE */
|
|
|
|
#if defined(MODULE_PERIPH_UART_RXSTART_IRQ) || DOXYGEN
|
|
|
|
/**
|
|
* @brief Configure the function that will be called when a start condition
|
|
* is detected.
|
|
*
|
|
* This will not enable / disable the generation of the RX start
|
|
* interrupt.
|
|
*
|
|
* @note You have to add the module `periph_uart_rxstart_irq` to your project
|
|
* to enable this function
|
|
*
|
|
* @param[in] uart The device to configure
|
|
* @param[in] cb The function called when a start condition is detected
|
|
* @param[in] arg Optional function argument
|
|
*/
|
|
void uart_rxstart_irq_configure(uart_t uart, uart_rxstart_cb_t cb, void *arg);
|
|
|
|
/**
|
|
* @brief Enable the RX start interrupt.
|
|
*
|
|
* @note You have to add the module `periph_uart_rxstart_irq` to your project
|
|
* to enable this function
|
|
*
|
|
* @param[in] uart The device to configure
|
|
*/
|
|
void uart_rxstart_irq_enable(uart_t uart);
|
|
|
|
/**
|
|
* @brief Disable the RX start interrupt.
|
|
*
|
|
* @note You have to add the module `periph_uart_rxstart_irq` to your project
|
|
* to enable this function
|
|
*
|
|
* @param[in] uart The device to configure
|
|
*/
|
|
void uart_rxstart_irq_disable(uart_t uart);
|
|
#endif /* MODULE_PERIPH_UART_RXSTART_IRQ */
|
|
|
|
#if defined(MODULE_PERIPH_UART_COLLISION) || DOXYGEN
|
|
/**
|
|
* @brief Enables collision detection check of the UART.
|
|
* This assumes the UART is connected to a bus where RX and TX are
|
|
* connected. After each sent byte it is checked whether the same
|
|
* byte could be received.
|
|
*
|
|
* This disables the RX interrupt.
|
|
*
|
|
* @note You have to add the module `periph_uart_rxstart_irq` to your project
|
|
* to enable this function
|
|
*
|
|
* @param[in] uart The device to configure
|
|
*/
|
|
void uart_collision_detect_enable(uart_t uart);
|
|
/**
|
|
* @brief Disables collision detection check of the UART.
|
|
*
|
|
* If an RX interrupt was configured before, it is enabled again.
|
|
*
|
|
* @note You have to add the module `periph_uart_rxstart_irq` to your project
|
|
* to enable this function
|
|
*
|
|
* @param[in] uart The device to configure
|
|
*/
|
|
void uart_collision_detect_disable(uart_t uart);
|
|
|
|
/**
|
|
* @brief Disables collision detection check of the UART.
|
|
*
|
|
* If an RX interrupt was configured before, it is enabled again.
|
|
*
|
|
* @note You have to add the module `periph_uart_rxstart_irq` to your project
|
|
* to enable this function
|
|
*
|
|
* @param[in] uart The device to probe
|
|
*
|
|
* @return true if a collision occurred during the last transder
|
|
*/
|
|
bool uart_collision_detected(uart_t uart);
|
|
#endif /* MODULE_PERIPH_UART_COLLISION */
|
|
|
|
/**
|
|
* @brief Setup parity, data and stop bits for a given UART device
|
|
*
|
|
* @param[in] uart UART device to configure
|
|
* @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
|
|
*
|
|
* @retval 0 Success
|
|
* @retval -ENOTSUP Given configuration not supported
|
|
* @retval <0 Other error
|
|
*/
|
|
int uart_mode(uart_t uart, uart_data_bits_t data_bits, uart_parity_t parity,
|
|
uart_stop_bits_t stop_bits);
|
|
|
|
/**
|
|
* @brief Write data from the given buffer to the specified UART device
|
|
*
|
|
* 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.
|
|
*
|
|
* @param[in] uart UART device to use for transmission
|
|
* @param[in] data data buffer to send
|
|
* @param[in] len number of bytes to send
|
|
*
|
|
*/
|
|
void uart_write(uart_t uart, const uint8_t *data, size_t len);
|
|
|
|
/**
|
|
* @brief Power on the given UART device
|
|
*
|
|
* @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);
|
|
|
|
/**
|
|
* @brief Enable the TX line one the given UART
|
|
*
|
|
* @note requires the `periph_uart_tx_ondemand` feature
|
|
*
|
|
* @param[in] uart the UART device start TX on
|
|
*/
|
|
void uart_enable_tx(uart_t uart);
|
|
|
|
/**
|
|
* @brief Disable the TX line one the given UART
|
|
*
|
|
* @note requires the `periph_uart_tx_ondemand` feature
|
|
*
|
|
* @param[in] uart the UART device to stop TX on
|
|
*/
|
|
void uart_disable_tx(uart_t uart);
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* PERIPH_UART_H */
|
|
/** @} */
|