1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/drivers/include/periph/uart.h
benpicco e8cbf1ea90
Merge pull request #16681 from benpicco/drivers/dose-collision
drivers/dose: make use of UART collision detection feature
2021-12-08 20:48:28 +01:00

390 lines
12 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 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 */
#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);
#ifdef __cplusplus
}
#endif
#endif /* PERIPH_UART_H */
/** @} */