mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
52107b2416
The esp8266 CPU has actually two hardware UART peripherals. UART0 is used by the boot ROM for flashing and serial output during boot, typically at a baudrate of 74880 bps until the bootloader or application sets the more standard 115200 baudrate. This UART0 device has two possible pins for TXD, GPIO1 and GPIO2, which are both set to TXD by the boot ROM. esp8266 modules will typically have GPIO1 labeled as the TX pin, but it is possible to use GPIO2 for that purpose even while flashing the device with esptool.py. The second device, UART1, also has two options for TXD, GPIO2 and GPIO7, and only one option for RXD, GPIO8. However, GPIO7 and GPIO8 are used by the flash internally so those options are not very useful unless maybe while running from IRAM with the flash disabled, for example for a debugger over UART1. This patch allows boards to override UART{0,1}_{R,T}XD in their periph_conf.h to configure the uart selection. Defining UART1_TX will make the UART_DEV(1) device available. Tested with: ```CFLAGS='-DUART1_TXD=GPIO2' make -C tests/periph_uart BOARD=esp8266-esp-12x flash term``` * Connected one USB-UART to the standard GPIO1 and GPIO3 for flashing and console. After flashing we see the manual test output at 115200 bps * Connected a second USB-UART with RX to GPIO2 running at 74880. Then run on the first console: ``` > init 1 74880 > send 1 hello ``` The word "hello" appears on the second UART connection. Note that GPIO2 is used during boot for UART0's TX until the application or bootloader set it to a regular GPIO, so some boot ROM messages at 74880 bps are visible. After running `init 1 74880` it is set to UART1's TX.
492 lines
15 KiB
C
492 lines
15 KiB
C
/*
|
|
* Copyright (C) 2018 Gunar Schorcht
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup cpu_esp_common
|
|
* @ingroup drivers_periph_uart
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Low-level UART driver implementation
|
|
*
|
|
* @author Gunar Schorcht <gunar@schorcht.net>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <assert.h>
|
|
|
|
#include "esp_common.h"
|
|
#include "cpu.h"
|
|
#include "irq_arch.h"
|
|
#include "log.h"
|
|
#include "sched.h"
|
|
#include "thread.h"
|
|
|
|
#include "periph/gpio.h"
|
|
#include "periph/uart.h"
|
|
|
|
#include "stdio_uart.h"
|
|
|
|
#include "esp/common_macros.h"
|
|
#include "rom/ets_sys.h"
|
|
#include "xtensa/xtensa_api.h"
|
|
#include "gpio_arch.h"
|
|
|
|
#define ENABLE_DEBUG 0
|
|
#include "debug.h"
|
|
|
|
#ifdef MCU_ESP32
|
|
|
|
#include "driver/periph_ctrl.h"
|
|
#include "soc/gpio_reg.h"
|
|
#include "soc/gpio_sig_map.h"
|
|
#include "soc/gpio_struct.h"
|
|
#include "soc/rtc.h"
|
|
#include "soc/uart_reg.h"
|
|
#include "soc/uart_struct.h"
|
|
|
|
#undef UART_CLK_FREQ
|
|
#define UART_CLK_FREQ rtc_clk_apb_freq_get() /* APB_CLK is used */
|
|
|
|
#else /* MCU_ESP32 */
|
|
|
|
#include "esp/iomux_regs.h"
|
|
#include "esp8266/uart_struct.h"
|
|
|
|
#ifdef MODULE_ESP_QEMU
|
|
#include "esp/uart_regs.h"
|
|
#endif /* MODULE_ESP_QEMU */
|
|
|
|
#define UART0 uart0
|
|
#define UART1 uart1
|
|
#define CPU_INUM_UART ETS_UART_INUM
|
|
|
|
#endif /* MCU_ESP32 */
|
|
|
|
struct uart_hw_t {
|
|
uart_dev_t* regs; /* pointer to register data struct of the UART device */
|
|
bool used; /* indicates whether UART is used */
|
|
uint32_t baudrate; /* used baudrate */
|
|
uart_data_bits_t data; /* used data bits */
|
|
uart_stop_bits_t stop; /* used stop bits */
|
|
uart_parity_t parity; /* used parity bits */
|
|
uart_isr_ctx_t isr_ctx; /* callback functions */
|
|
#ifdef MCU_ESP32
|
|
uint8_t mod; /* peripheral hardware module of the UART interface */
|
|
uint8_t signal_txd; /* TxD signal from the controller */
|
|
uint8_t signal_rxd; /* RxD signal to the controller */
|
|
uint8_t int_src; /* peripheral interrupt source used by the UART device */
|
|
#endif
|
|
};
|
|
|
|
/* hardware resources */
|
|
static struct uart_hw_t _uarts[] = {
|
|
{
|
|
.regs = &UART0,
|
|
.used = false,
|
|
.baudrate = STDIO_UART_BAUDRATE,
|
|
.data = UART_DATA_BITS_8,
|
|
.stop = UART_STOP_BITS_1,
|
|
.parity = UART_PARITY_NONE,
|
|
#ifdef MCU_ESP32
|
|
.mod = PERIPH_UART0_MODULE,
|
|
.signal_txd = U0TXD_OUT_IDX,
|
|
.signal_rxd = U0RXD_IN_IDX,
|
|
.int_src = ETS_UART0_INTR_SOURCE
|
|
#endif /* MCU_ESP32 */
|
|
},
|
|
#if defined(UART1_TXD) || defined(MCU_ESP32)
|
|
{
|
|
.regs = &UART1,
|
|
.used = false,
|
|
.baudrate = STDIO_UART_BAUDRATE,
|
|
.data = UART_DATA_BITS_8,
|
|
.stop = UART_STOP_BITS_1,
|
|
.parity = UART_PARITY_NONE,
|
|
#ifdef MCU_ESP32
|
|
.mod = PERIPH_UART1_MODULE,
|
|
.signal_txd = U1TXD_OUT_IDX,
|
|
.signal_rxd = U1RXD_IN_IDX,
|
|
.int_src = ETS_UART1_INTR_SOURCE
|
|
#endif /* MCU_ESP32 */
|
|
},
|
|
#endif /* defined(UART1_TXD) || defined(MCU_ESP32) */
|
|
#ifdef MCU_ESP32
|
|
{
|
|
.regs = &UART2,
|
|
.used = false,
|
|
.baudrate = STDIO_UART_BAUDRATE,
|
|
.data = UART_DATA_BITS_8,
|
|
.stop = UART_STOP_BITS_1,
|
|
.parity = UART_PARITY_NONE,
|
|
.mod = PERIPH_UART2_MODULE,
|
|
.signal_txd = U2TXD_OUT_IDX,
|
|
.signal_rxd = U2RXD_IN_IDX,
|
|
.int_src = ETS_UART2_INTR_SOURCE
|
|
},
|
|
#endif /* MCU_ESP32 */
|
|
};
|
|
|
|
/* declaration of external functions */
|
|
extern void uart_div_modify(uint8_t uart_no, uint32_t div);
|
|
|
|
/* forward declaration of internal functions */
|
|
static int _uart_set_baudrate(uart_t uart, uint32_t baudrate);
|
|
static int _uart_set_mode(uart_t uart, uart_data_bits_t data_bits,
|
|
uart_parity_t parity, uart_stop_bits_t stop_bits);
|
|
static uint8_t IRAM _uart_rx_one_char(uart_t uart);
|
|
static void _uart_tx_one_char(uart_t uart, uint8_t data);
|
|
static void _uart_intr_enable(uart_t uart);
|
|
static void _uart_config(uart_t uart);
|
|
static void IRAM _uart_intr_handler(void *para);
|
|
|
|
int uart_init(uart_t uart, uint32_t baudrate, uart_rx_cb_t rx_cb, void *arg)
|
|
{
|
|
DEBUG("%s uart=%d, rate=%d, rx_cb=%p, arg=%p\n", __func__, uart, baudrate, rx_cb, arg);
|
|
|
|
assert(uart < UART_NUMOF_MAX);
|
|
assert(uart < UART_NUMOF);
|
|
|
|
/* UART1 and UART2 have configurable pins */
|
|
if ((UART_NUMOF > 0 && uart == UART_DEV(1)) ||
|
|
(UART_NUMOF > 1 && uart == UART_DEV(2))) {
|
|
|
|
/* reset the pins when they were already used as UART pins */
|
|
if (gpio_get_pin_usage(uart_config[uart].txd) == _UART) {
|
|
gpio_set_pin_usage(uart_config[uart].txd, _GPIO);
|
|
}
|
|
if (gpio_get_pin_usage(uart_config[uart].rxd) == _UART) {
|
|
gpio_set_pin_usage(uart_config[uart].rxd, _GPIO);
|
|
}
|
|
|
|
/* try to initialize the pins as GPIOs first */
|
|
if ((uart_config[uart].txd != GPIO_UNDEF &&
|
|
gpio_init(uart_config[uart].txd, GPIO_OUT)) ||
|
|
(uart_config[uart].rxd != GPIO_UNDEF &&
|
|
gpio_init(uart_config[uart].rxd, GPIO_IN))) {
|
|
return -1;
|
|
}
|
|
|
|
/* store the usage type in GPIO table */
|
|
gpio_set_pin_usage(uart_config[uart].txd, _UART);
|
|
gpio_set_pin_usage(uart_config[uart].rxd, _UART);
|
|
|
|
#ifdef MCU_ESP32
|
|
/* connect TxD pin to the TxD output signal through the GPIO matrix */
|
|
GPIO.func_out_sel_cfg[uart_config[uart].txd].func_sel = _uarts[uart].signal_txd;
|
|
|
|
/* connect RxD input signal to the RxD pin through the GPIO matrix */
|
|
GPIO.func_in_sel_cfg[_uarts[uart].signal_rxd].sig_in_sel = 1;
|
|
GPIO.func_in_sel_cfg[_uarts[uart].signal_rxd].sig_in_inv = 0;
|
|
GPIO.func_in_sel_cfg[_uarts[uart].signal_rxd].func_sel = uart_config[uart].rxd;
|
|
#else
|
|
if (uart_config[uart].txd != GPIO_UNDEF) {
|
|
uint8_t mux = _gpio_to_iomux[uart_config[uart].txd];
|
|
IOMUX.PIN[mux] = (IOMUX.PIN[mux] & ~IOMUX_PIN_FUNC_MASK) |
|
|
IOMUX_FUNC(uart_config[uart].txd == GPIO2 ? 2 : 4);
|
|
}
|
|
if (uart_config[uart].rxd != GPIO_UNDEF) {
|
|
/* There's really only GPIO8 / FUNC(4) for this, but it is normally
|
|
* unusable because it is used by the internal flash. */
|
|
uint8_t mux = _gpio_to_iomux[uart_config[uart].rxd];
|
|
IOMUX.PIN[mux] = (IOMUX.PIN[mux] & ~IOMUX_PIN_FUNC_MASK) |
|
|
IOMUX_FUNC(4);
|
|
}
|
|
#endif
|
|
}
|
|
_uarts[uart].baudrate = baudrate;
|
|
|
|
/* register interrupt context */
|
|
_uarts[uart].isr_ctx.rx_cb = rx_cb;
|
|
_uarts[uart].isr_ctx.arg = arg;
|
|
|
|
/* enable and configure the according UART module */
|
|
uart_poweron(uart);
|
|
|
|
return UART_OK;
|
|
}
|
|
|
|
#if MODULE_PERIPH_UART_MODECFG
|
|
int uart_mode(uart_t uart, uart_data_bits_t data_bits, uart_parity_t parity,
|
|
uart_stop_bits_t stop_bits)
|
|
{
|
|
return _uart_set_mode(uart, data_bits, parity, stop_bits);
|
|
}
|
|
#endif
|
|
|
|
void uart_write(uart_t uart, const uint8_t *data, size_t len)
|
|
{
|
|
assert(uart < UART_NUMOF);
|
|
|
|
for (size_t i = 0; i < len; i++) {
|
|
_uart_tx_one_char(uart, data[i]);
|
|
}
|
|
}
|
|
|
|
void uart_poweron(uart_t uart)
|
|
{
|
|
assert(uart < UART_NUMOF);
|
|
|
|
#ifdef MCU_ESP32
|
|
periph_module_enable(_uarts[uart].mod);
|
|
#endif
|
|
_uart_config(uart);
|
|
}
|
|
|
|
void uart_poweroff(uart_t uart)
|
|
{
|
|
assert(uart < UART_NUMOF);
|
|
|
|
#ifdef MCU_ESP32
|
|
periph_module_disable(_uarts[uart].mod);
|
|
#endif
|
|
}
|
|
|
|
/* systemwide UART initializations */
|
|
void uart_system_init(void)
|
|
{
|
|
for (unsigned uart = 0; uart < UART_NUMOF; uart++) {
|
|
/* reset all UART interrupt status registers */
|
|
_uarts[uart].regs->int_clr.val = ~0;
|
|
}
|
|
}
|
|
|
|
void uart_print_config(void)
|
|
{
|
|
for (unsigned uart = 0; uart < UART_NUMOF; uart++) {
|
|
printf("\tUART_DEV(%u)\ttxd=%d rxd=%d\n", uart,
|
|
uart_config[uart].txd, uart_config[uart].rxd);
|
|
}
|
|
}
|
|
|
|
static void IRAM _uart_intr_handler(void *arg)
|
|
{
|
|
/* to satisfy the compiler */
|
|
(void)arg;
|
|
|
|
irq_isr_enter();
|
|
|
|
/* UART0, UART1, UART2 peripheral interrupt sources are routed to the same
|
|
interrupt, so we have to use the status to distinguish interruptees */
|
|
for (unsigned uart = 0; uart < UART_NUMOF; uart++) {
|
|
if (_uarts[uart].used) {
|
|
DEBUG("%s uart=%d int_st=%08x\n", __func__,
|
|
uart, _uarts[uart].regs->int_st.val);
|
|
|
|
if (_uarts[uart].used && _uarts[uart].regs->int_st.rxfifo_full) {
|
|
/* read one byte of data */
|
|
uint8_t data = _uart_rx_one_char(uart);
|
|
/* if registered, call the RX callback function */
|
|
if (_uarts[uart].isr_ctx.rx_cb) {
|
|
_uarts[uart].isr_ctx.rx_cb(_uarts[uart].isr_ctx.arg, data);
|
|
}
|
|
/* clear interrupt flag */
|
|
_uarts[uart].regs->int_clr.rxfifo_full = 1;
|
|
}
|
|
|
|
/* TODO handle other types of interrupts, for the moment just clear them */
|
|
_uarts[uart].regs->int_clr.val = ~0x0;
|
|
}
|
|
}
|
|
|
|
irq_isr_exit();
|
|
}
|
|
|
|
/* RX/TX FIFO capacity is 128 byte */
|
|
#define UART_FIFO_MAX 128
|
|
|
|
/* receive one data byte with wait */
|
|
static uint8_t IRAM _uart_rx_one_char(uart_t uart)
|
|
{
|
|
#if defined(MODULE_ESP_QEMU) && defined(MCU_ESP8266)
|
|
/* wait until at least von byte is in RX FIFO */
|
|
while (!FIELD2VAL(UART_STATUS_RXFIFO_COUNT, UART(uart).STATUS)) {}
|
|
|
|
/* read the lowest byte from RX FIFO register */
|
|
return UART(uart).FIFO & 0xff; /* only bit 0 ... 7 */
|
|
#else
|
|
/* wait until at least von byte is in RX FIFO */
|
|
while (!_uarts[uart].regs->status.rxfifo_cnt) {}
|
|
|
|
/* read the lowest byte from RX FIFO register */
|
|
return _uarts[uart].regs->fifo.rw_byte;
|
|
#endif
|
|
|
|
}
|
|
|
|
/* send one data byte with wait */
|
|
static void _uart_tx_one_char(uart_t uart, uint8_t data)
|
|
{
|
|
/* wait until at least one byte is available in the TX FIFO */
|
|
while (_uarts[uart].regs->status.txfifo_cnt >= UART_FIFO_MAX) {}
|
|
|
|
/* send the byte by placing it in the TX FIFO using MPU */
|
|
#ifdef MCU_ESP32
|
|
WRITE_PERI_REG(UART_FIFO_AHB_REG(uart), data);
|
|
#else /* MCU_ESP32 */
|
|
#ifdef MODULE_ESP_QEMU
|
|
UART(uart).FIFO = data;
|
|
#else /* MODULE_ESP_QEMU */
|
|
_uarts[uart].regs->fifo.rw_byte = data;
|
|
#endif /* MODULE_ESP_QEMU */
|
|
#endif /* MCU_ESP32 */
|
|
}
|
|
|
|
static void _uart_intr_enable(uart_t uart)
|
|
{
|
|
_uarts[uart].regs->int_ena.rxfifo_full = 1;
|
|
_uarts[uart].regs->int_clr.rxfifo_full = 1;
|
|
_uarts[uart].used = true;
|
|
|
|
DEBUG("%s %08x\n", __func__, _uarts[uart].regs->int_ena.val);
|
|
}
|
|
|
|
static void _uart_config(uart_t uart)
|
|
{
|
|
assert(uart < UART_NUMOF);
|
|
|
|
/* setup the baudrate */
|
|
if (_uart_set_baudrate(uart, _uarts[uart].baudrate) != UART_OK) {
|
|
return;
|
|
}
|
|
|
|
/* set number of data bits, stop bits and parity mode */
|
|
if (_uart_set_mode(uart, _uarts[uart].data, _uarts[uart].stop,
|
|
_uarts[uart].parity) != UART_OK) {
|
|
return;
|
|
}
|
|
|
|
/* reset the FIFOs */
|
|
_uarts[uart].regs->conf0.rxfifo_rst = 1;
|
|
_uarts[uart].regs->conf0.rxfifo_rst = 0;
|
|
_uarts[uart].regs->conf0.txfifo_rst = 1;
|
|
_uarts[uart].regs->conf0.txfifo_rst = 0;
|
|
|
|
if (_uarts[uart].isr_ctx.rx_cb) {
|
|
/* since reading can only be done byte by byte, we set
|
|
UART_RXFIFO_FULL_THRHD interrupt level to 1 byte */
|
|
_uarts[uart].regs->conf1.rxfifo_full_thrhd = 1;
|
|
|
|
/* enable the RX FIFO FULL interrupt */
|
|
_uart_intr_enable(uart);
|
|
|
|
#ifdef MCU_ESP32
|
|
/* route all UART interrupt sources to same the CPU interrupt */
|
|
intr_matrix_set(PRO_CPU_NUM, _uarts[uart].int_src, CPU_INUM_UART);
|
|
#endif /* MCU_ESP32 */
|
|
|
|
/* we have to enable therefore the CPU interrupt here */
|
|
xt_set_interrupt_handler(CPU_INUM_UART, _uart_intr_handler, NULL);
|
|
xt_ints_on(BIT(CPU_INUM_UART));
|
|
}
|
|
}
|
|
|
|
static int _uart_set_baudrate(uart_t uart, uint32_t baudrate)
|
|
{
|
|
DEBUG("%s uart=%d, rate=%d\n", __func__, uart, baudrate);
|
|
|
|
assert(uart < UART_NUMOF);
|
|
|
|
/* wait until TX FIFO is empty */
|
|
while (_uarts[uart].regs->status.txfifo_cnt != 0) { }
|
|
|
|
critical_enter();
|
|
|
|
_uarts[uart].baudrate = baudrate;
|
|
|
|
#ifdef MCU_ESP32
|
|
/* use APB_CLK */
|
|
_uarts[uart].regs->conf0.tick_ref_always_on = 1;
|
|
/* compute and set the integral and the decimal part */
|
|
uint32_t clk_div = (UART_CLK_FREQ << 4) / _uarts[uart].baudrate;
|
|
_uarts[uart].regs->clk_div.div_int = clk_div >> 4;
|
|
_uarts[uart].regs->clk_div.div_frag = clk_div & 0xf;
|
|
#else
|
|
/* compute and set clock divider */
|
|
uint32_t clk_div = UART_CLK_FREQ / _uarts[uart].baudrate;
|
|
_uarts[uart].regs->clk_div.val = clk_div & 0xFFFFF;
|
|
#endif
|
|
|
|
critical_exit();
|
|
|
|
return UART_OK;
|
|
}
|
|
|
|
static int _uart_set_mode(uart_t uart, uart_data_bits_t data_bits,
|
|
uart_parity_t parity, uart_stop_bits_t stop_bits)
|
|
{
|
|
DEBUG("%s uart=%d, data_bits=%d parity=%d stop_bits=%d\n", __func__,
|
|
uart, data_bits, parity, stop_bits);
|
|
|
|
assert(uart < UART_NUMOF);
|
|
|
|
critical_enter();
|
|
|
|
/* set number of data bits */
|
|
switch (data_bits) {
|
|
case UART_DATA_BITS_5: _uarts[uart].regs->conf0.bit_num = 0; break;
|
|
case UART_DATA_BITS_6: _uarts[uart].regs->conf0.bit_num = 1; break;
|
|
case UART_DATA_BITS_7: _uarts[uart].regs->conf0.bit_num = 2; break;
|
|
case UART_DATA_BITS_8: _uarts[uart].regs->conf0.bit_num = 3; break;
|
|
default: LOG_TAG_ERROR("uart", "invalid number of data bits\n");
|
|
critical_exit();
|
|
return UART_NOMODE;
|
|
}
|
|
/* store changed number of data bits in configuration */
|
|
_uarts[uart].data = data_bits;
|
|
|
|
/* set number of stop bits */
|
|
#ifdef MCU_ESP32
|
|
/* workaround for hardware bug when stop bits are set to 2-bit mode. */
|
|
switch (stop_bits) {
|
|
case UART_STOP_BITS_1: _uarts[uart].regs->conf0.stop_bit_num = 1;
|
|
_uarts[uart].regs->rs485_conf.dl1_en = 0;
|
|
break;
|
|
case UART_STOP_BITS_2: _uarts[uart].regs->conf0.stop_bit_num = 1;
|
|
_uarts[uart].regs->rs485_conf.dl1_en = 1;
|
|
break;
|
|
default: LOG_TAG_ERROR("uart", "invalid number of stop bits\n");
|
|
critical_exit();
|
|
return UART_NOMODE;
|
|
}
|
|
#else
|
|
switch (stop_bits) {
|
|
case UART_STOP_BITS_1: _uarts[uart].regs->conf0.stop_bit_num = 1; break;
|
|
case UART_STOP_BITS_2: _uarts[uart].regs->conf0.stop_bit_num = 3; break;
|
|
default: LOG_TAG_ERROR("uart", "invalid number of stop bits\n");
|
|
critical_exit();
|
|
return UART_NOMODE;
|
|
}
|
|
#endif
|
|
|
|
/* store changed number of stop bits in configuration */
|
|
_uarts[uart].stop = stop_bits;
|
|
|
|
/* set parity mode */
|
|
switch (parity) {
|
|
case UART_PARITY_NONE: _uarts[uart].regs->conf0.parity_en = 0;
|
|
break;
|
|
case UART_PARITY_EVEN: _uarts[uart].regs->conf0.parity = 0;
|
|
_uarts[uart].regs->conf0.parity_en = 1;
|
|
break;
|
|
case UART_PARITY_ODD: _uarts[uart].regs->conf0.parity = 1;
|
|
_uarts[uart].regs->conf0.parity_en = 1;
|
|
break;
|
|
default: LOG_TAG_ERROR("uart", "invalid or unsupported parity mode\n");
|
|
critical_exit();
|
|
return UART_NOMODE;
|
|
}
|
|
/* store changed parity in configuration */
|
|
_uarts[uart].parity = parity;
|
|
|
|
critical_exit();
|
|
|
|
return UART_OK;
|
|
}
|