一、概述
通用异步收发送器(UART)是一种硬件特性,它使用广泛适应的异步串行通信接口(如RS 232、RS 422、RS 485)来处理通信(即时序要求和数据帧)。UART提供了一种广泛采用和廉价的方法来实现不同设备之间的全双工或半双工数据交换。
ESP32芯片有三个UART控制器(UART 0、UART 1和UART 2),它们具有一组相同的寄存器,以便于编程和灵活性。
每个UART控制器都是独立配置的,参数包括波特率、数据比特长度、位序、停止位数、奇偶校验位等。所有控制器都与不同厂商的UART支持设备兼容,还可以支持红外数据关联协议(IRDA)。
ESP-IDF 编程指南——UART
二、API说明
以下 UART 接口位于 driver/include/driver/uart.h。
2.1 uart_param_config

2.2 uart_driver_install

2.3 uart_read_bytes

2.4 uart_write_bytes

2.5 uart_set_pin
ESP32的串口是支持引脚映射的,比如我的开发板串口一默认的是GPIO9和GPIO10,现在将TX、RX映射到GPIO4和GPIO5上。

三、编程流程
3.1 设置通信参数
如设置波特率、数据位、停止位等
在结构体中进行配置:
typedef struct {
int baud_rate; /*!< UART baud rate*/
uart_word_length_t data_bits; /*!< UART byte size*/
uart_parity_t parity; /*!< UART parity mode*/
uart_stop_bits_t stop_bits; /*!< UART stop bits*/
uart_hw_flowcontrol_t flow_ctrl; /*!< UART HW flow control mode (cts/rts)*/
uint8_t rx_flow_ctrl_thresh; /*!< UART HW RTS threshold*/
union {
uart_sclk_t source_clk; /*!< UART source clock selection */
bool use_ref_tick __attribute__((deprecated)); /*!< Deprecated method to select ref tick clock source, set source_clk field instead */
};} uart_config_t;
3.2 设置通信引脚
ESP32的串口是支持引脚映射的,比如我的开发板串口一默认的是GPIO9和GPIO10,现在将TX、RX映射到GPIO4和GPIO5上。
请调用函数uart_set_pin()并指定驱动程序应将Tx,Rx,RTS和CTS信号路由至的GPIO引脚号。
如果要为特定信号保留当前分配的管脚号,请传递宏UART_PIN_NO_CHANGE。
应该为不使用的引脚指定相同的宏。
// Set UART pins(TX: IO17 (UART2 default), RX: IO16 (UART2 default), RTS: IO18, CTS: IO19)ESP_ERROR_CHECK(uart_set_pin(UART_NUM_2, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE, 18, 19));
3.3 驱动程序安装
设置好通信引脚后,通过调用安装驱动程序uart_driver_install()并指定以下参数:
Tx环形缓冲区的大小
Rx环形缓冲区的大小
事件队列句柄和大小
分配中断的标志
该功能将为UART驱动程序分配所需的内部资源。
// Setup UART buffered IO with event queueconst int uart_buffer_size = (1024 * 2);QueueHandle_t uart_queue;// Install UART driver using an event queue hereESP_ERROR_CHECK(uart_driver_install(UART_NUM_2, uart_buffer_size,
uart_buffer_size, 10, &uart_queue, 0));
3.4 运行UART通信
发送数据的过程涉及以下步骤:
将数据写入Tx FIFO缓冲区
FSM序列化数据
FSM将数据发送出去
接收数据的过程类似,但是步骤相反:
FSM处理传入的串行流并将其并行化
FSM将数据写入Rx FIFO缓冲区
从Rx FIFO缓冲区读取数据
因此,应用程序将被限制为分别使用uart_write_bytes()和从相应的缓冲区写入和读取数据uart_read_bytes(),而FSM将完成其余的工作。
3.4.1 发送
准备好要传输的数据后,调用该函数uart_write_bytes()并将数据缓冲区的地址和数据长度传递给该函数。该函数将数据复制到Tx环形缓冲区(立即或在有足够空间可用之后),然后退出。当Tx FIFO缓冲区中有可用空间时,中断服务程序(ISR)将数据从Tx环形缓冲区移至后台的Tx FIFO缓冲区。下面的代码演示了此功能的用法。
// Write data to UART.char* test_str = 'This is a test string.n';uart_write_bytes(uart_num, (const char*)test_str, strlen(test_str));
该功能uart_write_bytes_with_break()类似于uart_write_bytes()但在传输结束时添加了一个串行中断信号。意味着它会在发送完数据之后,设置TX低电平一段时间(RTOS任务节拍为单位)。
// Write data to UART, end with a break signal.uart_write_bytes_with_break(uart_num, 'test breakn',strlen('test breakn'), 100);
将数据写入Tx FIFO缓冲区的另一个功能是uart_tx_chars()。不像uart_write_bytes(),此功能在可用空间之前不会阻塞。相反,它将写入可立即放入硬件Tx FIFO中的所有数据,然后返回已写入的字节数。
有一个“陪伴”功能uart_wait_tx_done(),可监视Tx FIFO缓冲区的状态并在其为空时返回。
// Wait for packet to be sentconst int uart_num = UART_NUM_2;ESP_ERROR_CHECK(uart_wait_tx_done(uart_num, 100)); // wait timeout is 100 RTOS ticks (TickType_t)
3.4.2 接收
UART接收到数据并将其保存在Rx FIFO缓冲区后,需要使用函数进行读出uart_read_bytes()。,这个函数会阻塞待在那里,直到读满需要的字节,或是超时。
在读取数据之前,您可以调用来检查Rx FIFO缓冲区中可用的字节数uart_get_buffered_data_len(),然后再读取相应的内容,这样就不会造成不必要的阻塞。下面给出了使用这些功能的示例。
// Read data from UART.const int uart_num = UART_NUM_2;uint8_t data[128];int length = 0;ESP_ERROR_CHECK(uart_get_buffered_data_len(uart_num, (size_t*)&length));length = uart_read_bytes(uart_num, data, length, 100);
如果不再需要Rx FIFO缓冲区中的数据,则可以通过调用清除缓冲区uart_flush()。
四、串口回环输出
这里我将GPIO4、GPIO5改成了GPIO23、GPIO18
#include * This is an example which echos any data it receives on UART1 back to the sender, * with hardware flow control turned off. It does not use UART driver event queue. * * - Port: UART1 * - Receive (Rx) buffer: on * - Transmit (Tx) buffer: off * - Flow control: off * - Event queue: off * - Pin assignment: see defines below */#define ECHO_TEST_TXD (GPIO_NUM_23)#define ECHO_TEST_RXD (GPIO_NUM_18)#define ECHO_TEST_RTS (UART_PIN_NO_CHANGE)#define ECHO_TEST_CTS (UART_PIN_NO_CHANGE)#define BUF_SIZE (1024)static void echo_task(void *arg){ /* Configure parameters of an UART driver, * communication pins and install the driver */ uart_config_t uart_config = { .baud_rate = 115200, .data_bits = UART_DATA_8_BITS, .parity = UART_PARITY_DISABLE, .stop_bits = UART_STOP_BITS_1, .flow_ctrl = UART_HW_FLOWCTRL_DISABLE, .source_clk = UART_SCLK_APB, }; uart_driver_install(UART_NUM_1, BUF_SIZE * 2, 0, 0, NULL, 0); uart_param_config(UART_NUM_1, &uart_config); uart_set_pin(UART_NUM_1, ECHO_TEST_TXD, ECHO_TEST_RXD, ECHO_TEST_RTS, ECHO_TEST_CTS); // Configure a temporary buffer for the incoming data uint8_t *data = (uint8_t *) malloc(BUF_SIZE); while (1) { // Read data from the UART int len = uart_read_bytes(UART_NUM_1, data, BUF_SIZE, 20 / portTICK_RATE_MS); // Write data back to the UART uart_write_bytes(UART_NUM_1, (const char *) data, len); }}void app_main(void){ xTaskCreate(echo_task, 'uart_echo_task', 1024, NULL, 10, NULL);} 五、串口队列接收 /********************************************************************* * INCLUDES */#include * LOCAL VARIABLES */static QueueHandle_t s_uart0Queue;static const char *TAG = 'board_uart';/********************************************************************* * PUBLIC FUNCTIONS *//** @brief 串口驱动初始化 @param 无 @return 无 */void UART_Init(void){ // Configure parameters of an UART driver, // communication pins and install the driver uart_config_t uart_config = { .baud_rate = 115200, .data_bits = UART_DATA_8_BITS, .parity = UART_PARITY_DISABLE, .stop_bits = UART_STOP_BITS_1, .flow_ctrl = UART_HW_FLOWCTRL_DISABLE }; uart_param_config(UART_NUM_0, &uart_config); // 配置串口0参数 uart_set_pin(UART_NUM_0, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE);// 配置串口0引脚 // Install UART driver, and get the queue. uart_driver_install(UART_NUM_0, BUF_SIZE * 2, BUF_SIZE * 2, 100, &s_uart0Queue, 0); // 安装UART驱动程序 // Create a task to handler UART event from ISR xTaskCreate(uartEventTask, 'uartEventTask', 2048, NULL, 12, NULL); }/********************************************************************* * LOCAL FUNCTIONS */static void uartEventTask(void *pvParameters){ uart_event_t event; uint8_t *pTempBuf = (uint8_t *)malloc(UART_MAX_NUM_RX_BYTES); for(;;) { // Waiting for UART event. if(xQueueReceive(s_uart0Queue, (void *)&event, (portTickType)portMAX_DELAY)) { bzero(pTempBuf, UART_MAX_NUM_RX_BYTES); switch(event.type) { // Event of UART receving data // We'd better handler data event fast, there would be much more data events than // other types of events. If we take too much time on data event, the queue might be full. case UART_DATA: // ESP_LOGI(TAG, '[UART DATA]: %d', event.size); uart_read_bytes(UART_NUM_0, pTempBuf, event.size, portMAX_DELAY); uart_write_bytes(UART_NUM_0, (const char *)pTempBuf, event.size); break; // Event of HW FIFO overflow detected case UART_FIFO_OVF: ESP_LOGI(TAG, 'hw fifo overflow'); // If fifo overflow happened, you should consider adding flow control for your application.
上一篇:ESP32学习笔记(5)——WiFi接口使用(STA和AP模式)
下一篇:ESP32学习笔记(3)——高分辨率定时器接口使用
推荐阅读最新更新时间:2026-03-25 13:28
- 用于 7VIN 至 16VIN、1.5V 和 1.2V 输出的 LTM4628EV DC/DC 模块稳压器的典型应用电路
- 使用 Analog Devices 的 LTC3728LIGN 的参考设计
- DER-406 - 适用于 A19 灯的 5.76 W 高 PF 非隔离降压-升压型 TRIAC 调光 LED 驱动器
- ADR5045B 5V 输出精密微功率并联模式电压基准的典型应用
- LT3970EDDB-3.42 2.5V 降压转换器的典型应用
- MC78M08BDTG 8V 电流调节器的典型应用
- LT1021DCN8-5 精密电压基准的典型应用
- DER-282 - 100W, 扁平(11 mm), LLC DC-DC转换器
- REF193 低压差开尔文连接电压基准的典型应用电路
- LT3088EM 线性稳压器用于添加软启动的典型应用



英飞凌PSoC 6 电位器控制LED闪烁状态并水墨屏显示——源码
Follow me第三季第4期任务
现代雷达系统的信号设计
CLC5612IMX
BFR340T






京公网安备 11010802033920号