串口设备应用笔记

摘要

本应用笔记描述了如何使用 RT-Thread 的串口设备,包括串口配置、设备操作接口的应用。并给出了在正点原子 STM32F4 探索者开发板上验证的代码示例。

本文的目的和结构

本文的目的和背景

串口(通用异步收发器,常写作 UART、uart)是最为广泛使用的通信接口之一。在裸机平台或者是没有设备管理框架的 RTOS 平台上,我们通常只需要根据官方手册编写串口硬件初始化代码即可。引入了带设备管理框架的实时操作系统 RT-Thread 后,串口的使用则与裸机或者其它 RTOS 有很大的不同之处。RT-Thread 中自带 I/O 设备管理层,将各种各样的硬件设备封装成具有统一接口的逻辑设备,方便管理及使用。本文说明了如何在 RT-Thread 中使用串口。

本文的结构

本文首先给出使用 RT-Thread 的设备操作接口开发串口收、发数据程序的示例代码,并在正点原子 STM32F4 探索者开发板上验证。接着分析了示例代码的实现,最后深入地描述了 RT-Thread 设备管理框架与串口的联系。

问题阐述

RT-Thread 提供了一套简单的 I/O 设备管理框架,它把 I/O 设备分成了三层进行处理:应用层、I/O 设备管理层、硬件驱动层。应用程序通过 RT-Thread 的设备操作接口获得正确的设备驱动,然后通过这个设备驱动与底层 I/O 硬件设备进行数据(或控制)交互。RT-Thread 提供给上层应用的是一个抽象的设备操作接口,给下层设备提供的是底层驱动框架。

RT-Thread 设备管理框架

那么用户如何使用设备操作接口开发出跨平台的串口应用代码呢?

问题的解决

本文基于正点原子 STM32F4 探索者开发板,给出了串口的配置流程和应用代码示例。由于 RT-Thread 设备操作接口的通用性,因此这些代码与硬件平台无关,读者可以直接将它用在自己使用的硬件平台上。正点原子 STM32F4 探索者开发板使用的是 STM32F407ZGT6,具有多路串口。我们使用串口 1 作为 shell 终端,串口 2 作为实验用串口,测试数据收发。终端软件使用 putty。板载串口 1 带有 USB 转串口芯片,因此使用 USB 线连接串口 1 和 PC 即可;串口 2 则需要使用 USB 转串口模块连接到 PC。

正点原子 STM32F4 探索者

准备和配置工程

  • 下载 RT-Thread 源码

  • 进入 rt-thread\bsp\stm32f4xx-HAL 目录,在 env 命令行中输入 menuconfig,进入配置界面,使用 menuconfig 工具(学习如何使用)配置工程。

(1) 配置 shell 使用串口 1:RT-Thread Kernel —-> Kernel Device Object —-> 修改 the device name for console 为 uart1。

(2) 勾选 Using UART1、Using UART2,选择芯片型号为 STM32F407ZG,时钟源为外部 8MHz,如图所示:

使用 menuconfig 配置串口

  • 输入命令 scons —target=mdk5 -s 生成 keil 工程,打开工程后先修改 MCU 型号为 STM32F407ZETx,如图所示:
    检查芯片型号

  • 打开 putty,选择正确的串口,软件参数配置为 115200-8-1-N、无流控。如图所示:
    putty 配置

  • 编译、下载程序,按下复位后就可以在串口 1 连接的终端上看到 RT-Thread 标志 log 了,输入 list_device 命令能查看到 uart1、uart2 Character Device 就表示串口配置好了。
    使用 list_device 命令查看 uart 设备

加入串口相关代码

下载串口示例代码

添加示例代码到工程

本应用笔记示例代码 app_uart.c、app_uart.h,app_uart.c 中是串口相关操作的代码,方便阅读。app_uart.c 中提供了 4 个函数 uart_open、uart_putchar、uart_putstring、uart_getchar 以方便使用串口。app_uart.c 中的代码与硬件平台无关,读者可以把它直接添加到自己的工程。利用这几个函数在 main.c 中编写测试代码。main.c 源码如下:

  1. #include "app_uart.h"
  2. #include "board.h"
  3. void test_thread_entry(void* parameter)
  4. {
  5. rt_uint8_t uart_rx_data;
  6. /* 打开串口 */
  7. if (uart_open("uart2") != RT_EOK)
  8. {
  9. rt_kprintf("uart open error.\n");
  10. while (1)
  11. {
  12. rt_thread_delay(10);
  13. }
  14. }
  15. /* 单个字符写 */
  16. uart_putchar('2');
  17. uart_putchar('0');
  18. uart_putchar('1');
  19. uart_putchar('8');
  20. uart_putchar('\n');
  21. /* 写字符串 */
  22. uart_putstring("Hello RT-Thread!\r\n");
  23. while (1)
  24. {
  25. /* 读数据 */
  26. uart_rx_data = uart_getchar();
  27. /* 错位 */
  28. uart_rx_data = uart_rx_data + 1;
  29. /* 输出 */
  30. uart_putchar(uart_rx_data);
  31. }
  32. }
  33. int main(void)
  34. {
  35. rt_thread_t tid;
  36. /* 创建 test 线程 */
  37. tid = rt_thread_create("test",
  38. test_thread_entry,
  39. RT_NULL,
  40. 1024,
  41. 2,
  42. 10);
  43. /* 创建成功则启动线程 */
  44. if (tid != RT_NULL)
  45. rt_thread_startup(tid);
  46. return 0;
  47. }

这段程序实现了如下功能:

  • main 函数里面创建并启动了测试线程 test_thread_entry。

  • 测试线程调用 uart_open 函数打开指定的串口后,首先使用 uart_putchar 函数发送字符和 uart_putstring 函数发送字符串。

  • 接着在 while 循环里面调用 uart_getchar 函数读取接收到的数据并保存到局部变量 uart_rx_data 中,最后将数据错位后输出。

运行结果

编译、将代码下载到板卡,复位,串口 2 连接的终端软件 putty(软件参数配置为 115200-8-1-N、无流控)输出了字符 2、0、1、8 和字符串 Hello RT-Thread!。输入字符 ‘A’,串口 2 接收到将其错位后输出。实验现象如图所示:

实验现象

图中 putty 连接开发板的串口 2 作为测试串口。

进阶阅读

串口通常被配置为接收中断和轮询发送模式。在中断模式下,CPU 不需要一直查询等待串口相关标志寄存器,串口接收到数据后触发中断,我们在中断服务程序进行数据处理,效率较高。RT-Thread 官方 bsp 默认便是这种模式。

使用哪个串口

uart_open 函数用于打开指定的串口,它完成了串口设备回调函数设置、串口设备的开启和事件的初始化。源码如下:

  1. rt_err_t uart_open(const char *name)
  2. {
  3. rt_err_t res;
  4. /* 查找系统中的串口设备 */
  5. uart_device = rt_device_find(name);
  6. /* 查找到设备后将其打开 */
  7. if (uart_device != RT_NULL)
  8. {
  9. res = rt_device_set_rx_indicate(uart_device, uart_intput);
  10. /* 检查返回值 */
  11. if (res != RT_EOK)
  12. {
  13. rt_kprintf("set %s rx indicate error.%d\n",name,res);
  14. return -RT_ERROR;
  15. }
  16. /* 打开设备,以可读写、中断方式 */
  17. res = rt_device_open(uart_device, RT_DEVICE_OFLAG_RDWR |
  18. RT_DEVICE_FLAG_INT_RX );
  19. /* 检查返回值 */
  20. if (res != RT_EOK)
  21. {
  22. rt_kprintf("open %s device error.%d\n",name,res);
  23. return -RT_ERROR;
  24. }
  25. }
  26. else
  27. {
  28. rt_kprintf("can't find %s device.\n",name);
  29. return -RT_ERROR;
  30. }
  31. /* 初始化事件对象 */
  32. rt_event_init(&event, "event", RT_IPC_FLAG_FIFO);
  33. return RT_EOK;
  34. }

简要流程如下:

uart_open 函数流程图

uart_open 函数使用到的设备操作接口有:rt_device_find、rt_device_set_rx_indicate、rt_device_open。uart_open 函数首先调用 rt_device_find 根据串口名字获得串口句柄,保存在静态全局变量 uart_device 中,后面关于串口的操作都是基于这个串口句柄。这里的名字是在 drv_usart.c 中调用注册函数 rt_hw_serial_register 决定的,该函数将串口硬件驱动和 RT-Thread 设备管理框架联系起来了。

  1. /* register UART2 device */
  2. rt_hw_serial_register(&serial2,
  3. "uart2",
  4. RT_DEVICE_FLAG_RDWR | RT_DEVICE_FLAG_INT_RX,
  5. uart);

接着调用 rt_device_set_rx_indicate 设置串口接收中断的回调函数。最后调用 rt_device_open 以可读写、中断接收方式打开串口。它的第二个参数为标志,与上面提到的注册函数 rt_hw_serial_register 保持一致即可。

  1. rt_device_open(uart_device, RT_DEVICE_OFLAG_RDWR | RT_DEVICE_FLAG_INT_RX);

最后调用 rt_event_init 初始化事件。RT-Thread 中默认开启了自动初始化机制,因此用户不需要在应用程序中手动调用串口的初始化函数(drv_usart.c 中的 INIT_BOARD_EXPORT 实现了自动初始化)。用户实现的由宏 RT_USING_UARTx 选定的串口硬件驱动将自动关联到 RT-Thread 中来(drv_usart.c 中的 rt_hw_serial_register 实现了串口硬件注册)。

串口发送

uart_putchar 函数用于发送 1 字节数据。uart_putchar 函数实际上调用的是 rt_device_write 来发送一个字节,并采取了防出错处理,即检查返回值,失败则重新发送,并限定了超时。源码如下:

  1. void uart_putchar(const rt_uint8_t c)
  2. {
  3. rt_size_t len = 0;
  4. rt_uint32_t timeout = 0;
  5. do
  6. {
  7. len = rt_device_write(uart_device, 0, &c, 1);
  8. timeout++;
  9. }
  10. while (len != 1 && timeout < 500);
  11. }

调用 uart_putchar 发生的数据流向示意图如下:

uart_putchar 数据流

应用程序调用 uart_putchar 时,实际调用关系为:rt_device_write ==> rt_serial_write ==> drv_putc,最终数据通过串口数据寄存器发送出去。

串口接收

uart_getchar 函数用于接收数据,uart_getchar 函数的实现采用了串口接收中断回调机制和事件用于异步通信,它具有阻塞特性。相关源码如下:

  1. /* 串口接收事件标志 */
  2. #define UART_RX_EVENT (1 << 0)
  3. /* 事件控制块 */
  4. static struct rt_event event;
  5. /* 设备句柄 */
  6. static rt_device_t uart_device = RT_NULL;
  7.  
  8. /* 回调函数 */
  9. static rt_err_t uart_intput(rt_device_t dev, rt_size_t size)
  10. {
  11. /* 发送事件 */
  12. rt_event_send(&event, UART_RX_EVENT);
  13. return RT_EOK;
  14. }
  15. rt_uint8_t uart_getchar(void)
  16. {
  17. rt_uint32_t e;
  18. rt_uint8_t ch;
  19. /* 读取 1 字节数据 */
  20. while (rt_device_read(uart_device, 0, &ch, 1) != 1)
  21. {
  22. /* 接收事件 */
  23. rt_event_recv(&event, UART_RX_EVENT,RT_EVENT_FLAG_AND |
  24. RT_EVENT_FLAG_CLEAR,RT_WAITING_FOREVER, &e);
  25. }
  26. return ch;
  27. }

uart_getchar 函数内部有一个 while() 循环,先调用 rt_device_read 去读取一字节数据,没有读到则调用 rt_event_recv 等待事件标志,挂起调用线程;串口接收到一字节数据后产生中断,调用回调函数 uart_intput,回调函数里面调用了 rt_event_send 发送事件标志以唤醒等待该 event 事件的线程。调用 uart_getchar 函数发生的数据流向示意图如下:

uart_getchar 数据流

应用程序调用 uart_getchar 时,实际调用关系为:rt_device_read ==> rt_serial_read ==> drv_getc,最终从串口数据寄存器读取到数据。

I/O 设备管理框架和串口的联系

RT-Thread 自动初始化功能依次调用 hw_usart_init ==> rt_hw_serial_register ==> rt_device_register 完成了串口硬件初始化,从而将设备操作接口和串口驱动联系起来,我们就可以使用设备操作接口来对串口进行操作。

串口驱动和设备管理框架联系

更多关于 I/O 设备管理框架的说明和串口驱动实现细节,请参考《RT-Thread 编程手册》第 6 章I/O 设备管理

在线查看地址:链接

参考

本文所有相关的 API

注意: app_uart.h 文件不属于 RT-Thread。

API 列表

API 头文件
uart_open app_uart.h
uart_getchar app_uart.h
uart_putchar app_uart.h
rt_event_send rt-thread\include\rtthread.h
rt_event_recv rt-thread\include\rtthread.h
rt_device_find rt-thread\include\rtthread.h
rt_device_set_rx_indicate rt-thread\include\rtthread.h
rt_device_open rt-thread\include\rtthread.h
rt_device_write rt-thread\include\rtthread.h
rt_device_read rt-thread\include\rtthread.h

核心 API 详解

rt_device_open()

函数原型

  1. rt_err_t rt_device_open (rt_device_t dev, rt_uint16_t oflag)

函数参数

参数 描述
dev 设备句柄
oflag 访问模式

函数返回

返回值 描述
RT_EOK 正常
-RT_EBUSY 如果设备注册时指定的参数中包括 RT_DEVICE_FLAG_STANDALONE,此设备将不允许重复打开

此函数可根据设备句柄来打开设备。

oflag 支持以下参数:

  1. RT_DEVICE_OFLAG_CLOSE /* 设备已经关闭(内部使用)*/
  2. RT_DEVICE_OFLAG_RDONLY /* 以只读方式打开设备 */
  3. RT_DEVICE_OFLAG_WRONLY /* 以只写方式打开设备 */
  4. RT_DEVICE_OFLAG_RDWR /* 以读写方式打开设备 */
  5. RT_DEVICE_OFLAG_OPEN /* 设备已经打开(内部使用)*/
  6. RT_DEVICE_FLAG_STREAM /* 设备以流模式打开 */
  7. RT_DEVICE_FLAG_INT_RX /* 设备以中断接收模式打开 */
  8. RT_DEVICE_FLAG_DMA_RX /* 设备以 DMA 接收模式打开 */
  9. RT_DEVICE_FLAG_INT_TX /* 设备以中断发送模式打开 */
  10. RT_DEVICE_FLAG_DMA_TX /* 设备以 DMA 发送模式打开 */

注意事项

如果上层应用程序需要设置设备的接收回调函数,则必须以 INT_RX 或者 DMA_RX的方式打开设备,否则不会回调函数。

rt_device_find()

函数原型

  1. rt_device_t rt_device_find(const char *name)

函数参数

参数 描述
name 设备名称

函数返回

查找到对应设备将返回相应的设备句柄;否则返回 RT_NULL

此函数根据指定的设备名称查找设备。

rt_device_set_rx_indicate()

函数原型

  1. rt_err_t rt_device_set_rx_indicate(rt_device_t dev,
  2. rt_err_t (*rx_ind)(rt_device_t dev, rt_size_t size))

函数参数

参数 描述
dev 设备句柄
rx_ind 接收中断回调函数

函数返回

返回值 描述
RT_EOK 成功

此函数可设置一个回调函数,当硬件设备收到数据时回调以通知应用程序有数据到达。

当硬件设备接收到数据时,会回调这个函数并把收到的数据长度放在 size 参数中传递给上层应用。上层应用线程应在收到指示后,立刻从设备中读取数据。

rt_device_read()

函数原型

  1. rt_size_t rt_device_read (rt_device_t dev,
  2. rt_off_t pos,
  3. void *buffer,
  4. rt_size_t size)

函数参数

参数 描述
dev 设备句柄
pos 读取数据偏移量
buffer 内存缓冲区指针,读取的数据将会被保存在缓冲区中
size 读取数据的大小

函数返回

返回读到数据的实际大小(如果是字符设备,返回大小以字节为单位;如果是块设备,返回的大小以块为单位);如果返回 0,则需要读取当前线程的 errno 来判断错误状态。

此函数可从设备中读取数据

调用这个函数,会从设备 dev 中获得数据,并存放在 buffer 缓冲区中。这个缓冲区的最大长度是 size。pos 根据不同的设备类别存在不同的意义。

rt_device_write()

函数原型

  1. rt_size_t rt_device_write(rt_device_t dev,
  2. rt_off_t pos,
  3. const void *buffer,
  4. rt_size_t size)

函数参数

参数 描述
dev 设备句柄
pos 写入数据偏移量
buffer 内存缓冲区指针,放置要写入的数据
size 写入数据的大小

函数返回

返回写入数据的实际大小 (如果是字符设备,返回大小以字节为单位;如果是块设备,返回的大小以块为单位);如果返回 0,则需要读取当前线程的 errno 来判断错误状态。注:调用这个函数,会把缓冲区 buffer 中的数据写入到设备 dev 中。写入数据的最大长度是size。pos 根据不同的设备类别存在不同的意义。

此函数可向设备中写入数据。

原文: https://www.rt-thread.org/document/site/rtthread-application-note/driver/uart/an0001-rtthread-driver-uart/