[esp32教程] 5、UART使用

描述

0、前期准备

1、参考首篇文章搭建好esp32环境

2、准备好一块esp32开发开发板(本作者使用了esp32c3作为开发平台)

1、知识储备

1.1 概述

​ UART称为通用异步收发器,可以进行全双工/半双工数据通讯数据通讯,通讯距离取决于上拉驱动能力、波特率,一般只在电路板上使用,如果需要长距离通讯可以外接RS232(最长通讯距离15M)、RS485电平转换芯片,但是要注意如果外接的是RS485电平芯片,只能进行半双工通讯,在理想情况下使用9600波特率,其通讯距离可达1200M。

1.2 功能架构

​ 乐鑫EPS32C3带有2个UART控制器,不仅可以用来进行数据通讯,还能来用驱动红外发射管。(目前本文只用来作uart通讯)

1.3 uart配置流程介绍

LEDC的配置流程可分为以下3步

1、uart配置:设置波特率、数据位、停止位、奇偶校验位和流控

2、引脚绑定:设置UART使用哪几个引脚

3、 注册uart中断驱动(可选--建议进行该配置,本文给出例子就使用该方式)

1、uart配置

uart的配置主要是给 uart_config_t 结构体赋值,然后通过uart_param_config函数进行uart设置

以下对uart_config_t 结构体进行说明

typedef struct {
    int baud_rate;                      //波特率
    uart_word_length_t data_bits;       //数据位
    uart_parity_t parity;               //奇偶校验位
    uart_stop_bits_t stop_bits;         //停止位
    uart_hw_flowcontrol_t flow_ctrl;    //硬件流控
    uint8_t rx_flow_ctrl_thresh;        //硬件流控的RTS阈值
    union {
        uart_sclk_t source_clk;         //时钟源
        bool use_ref_tick  __attribute__((deprecated)); 
    };
} uart_config_t;

例子:

// uart1配置
// 115200 8 1 无奇偶校验位 无流控
uart_config_t uart_conf = {
    .baud_rate = 115200,
    .data_bits = UART_DATA_8_BITS,
    .stop_bits = UART_STOP_BITS_1,
    .parity = UART_PARITY_DISABLE,
    .flow_ctrl = UART_HW_FLOWCTRL_DISABLE,
    .source_clk = UART_SCLK_APB,
};

uart_param_config(UART_NUM_1, &uart_conf);
2、引脚绑定

设置UART使用哪几个引脚

esp_err_t uart_set_pin(uart_port_t uart_num, int tx_io_num, int rx_io_num, int rts_io_num, int cts_io_num)// 参数 :uart_num:串口编号 可查看uart.h,进行选择
//     tx_io_num:发送引脚编号,不需要可填:UART_PIN_NO_CHANGE
//     rx_io_num: 接收引脚编号,不需要可填:UART_PIN_NO_CHANGE
//     rts_io_num:rts引脚编号,不需要可填:UART_PIN_NO_CHANGE
//     cts_io_num:cts引脚编号,不需要可填:UART_PIN_NO_CHANGE

例子:

// 引脚绑定
uart_set_pin(UART_NUM_1, TX_PIN, RX_PIN, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE);
3、 注册uart中断驱动(配置可选)

通过注册uart中断,可以不用开一个任务实时的去读取串口,判断串口有没有数据发送过来,提高系统实时性。

中断驱动uart_driver_install函数参数说明如下:

esp_err_t uart_driver_install(uart_port_t uart_num, int rx_buffer_size, int tx_buffer_size, int queue_size, QueueHandle_t *uart_queue, int intr_alloc_flags);
// 参数 :uart_num:串口编号 可查看uart.h,进行选择
//     rx_buffer_size:uart接收数组大小
//     tx_buffer_size: uart发送数组大小
//     queue_size: uart接收消息队列大小,和新建的消息队列大小一致
//     uart_queue:uart接收消息队列句柄
//     intr_alloc_flags:中断标志位

例子:

QueueHandle_t uart_queue = NULL;    
uart_queue = xQueueCreate(8, 512);
uart_driver_install(UART_PORT, RX_BUFF_SIZE, RX_BUFF_SIZE, 8, &uart_queue, 0);
4、 其他函数

本文只介绍发送和接收两个函数,更多函数可参考乐鑫官网

int uart_read_bytes(uart_port_t uart_num, void *buf, uint32_t length, TickType_t ticks_to_wait);
//参数 : uart_num:串口编号 可查看uart.h,进行选择
//       buf:接收数组名
//       length:接收到的数据长度
//       ticks_to_wait:读取等待的RTOS 滴答个数
//返回 : -1:表示错误
//      其他:表示从uart的接收fifo读取到的数据

int uart_write_bytes(uart_port_t uart_num, const void *src, size_t size);
//参数 : uart_num:串口编号 可查看uart.h,进行选择
//       src:发送数组名
//       size:发送的数据长度
//返回 : -1:表示错误
//      其他:表示uart发送给发送fifo的数据

2、新建工程

idf.py create-project project_uart # 新建工程
cd project_uart
idf.py set-target esp32c3 # 设置工程使用的芯片

2、查看原理图确定uart引脚

ESP32

3、编写程序

#include < freertos/FreeRTOS.h >
#include < freertos/queue.h >
#include < freertos/task.h >

#include < driver/gpio.h >
#include < driver/uart.h >
#include < esp_log.h >

#include < string.h >

#define LOG_EVENT_TAG "uart"

QueueHandle_t uart_queue = NULL;

#define RX_BUFF_SIZE 1024

#define UART_PORT UART_NUM_2
#define RX_PIN GPIO_NUM_1
#define TX_PIN GPIO_NUM_0

uint8_t uart_buff[RX_BUFF_SIZE];

void UartEventHandle(void* param);

void init_uart(void) {
    uart_config_t uart_conf = {
        .baud_rate = 115200,
        .data_bits = UART_DATA_8_BITS,
        .stop_bits = UART_STOP_BITS_1,
        .parity = UART_PARITY_DISABLE,
        .flow_ctrl = UART_HW_FLOWCTRL_DISABLE,
        .source_clk = UART_SCLK_APB,
    };

    uart_param_config(UART_PORT, &uart_conf);
    uart_set_pin(UART_PORT, TX_PIN, RX_PIN, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE);

    uart_queue = xQueueCreate(8, 512);
    uart_driver_install(UART_PORT, RX_BUFF_SIZE, RX_BUFF_SIZE, 8, &uart_queue, 0);

    xTaskCreate(UartEventHandle, "uart_event_hanle", 1024, NULL, 5, NULL);
}

void UartEventHandle(void* param) {
    uart_event_t event;
    memset(&uart_data, 0, sizeof(UartData));
    uart_data.rx_status = pdFALSE;
    while (pdTRUE) {
        if (xQueueReceive(uart_queue, (void*)&event, portMAX_DELAY)) {
            switch (event.type) {
                case UART_DATA:
                    memset(&uart_data, 0, sizeof(UartData));
                    uart_data.rx_status = pdFALSE;
                    if (event.size <= sizeof(uart_data.buff)) {
                        uart_data.rx_status = pdTRUE;
                        uart_data.rx_uart_len = event.size;
                        uart_read_bytes(UART_PORT, uart_buff, event.size, portMAX_DELAY);
                        ESP_LOGI(LOG_EVENT_TAG, "uart_buff: %s", uart_buff);
                    }

                    break;
                case UART_FIFO_OVF:
                case UART_BUFFER_FULL:
                    uart_flush_input(UART_PORT);
                    xQueueReset(uart_queue);
                    break;
                default:
                    ESP_LOGI(LOG_EVENT_TAG, "uart event type: %d", event.type);
                    break;
            }
        }
    }
}

void app_main(void) {
    init_uart();

    while (1) {
        vTaskDelay(1000 / portTICK_RATE_MS);
    }
}

  审核编辑:汤梓红

打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分