RT-Thread驱动篇之串口驱动框架剖析及性能提升

原创 RTThread物联网操作系统 2022-01-19 18:39

前言:苦串口驱动久矣!


现状

串口驱动三种工作模式:轮询、中断、DMA。

轮询模式占用 CPU 最高,但是实现也是最简单的;DMA 占用 CPU 最少,实现也是最麻烦的;中断模式居中。

原串口驱动有以下几个问题:

1、中断模式,接收有缓存,发送没缓存

2、中断模式,读操作是非阻塞的,没有阻塞读;写操作因为没有缓存,只能阻塞写,没有非阻塞写。

3、中断接收过程,每往发送寄存器填充一个字符,就使用完成量等待发送完成中断,通过完成量进行进程调度次数和发送数据量同样多!

4、DMA 模式比较复杂,在实现上更复杂。

  1. 首先,接收有两种缓存方案,一种没有缓存,借用应用层的内存直接做 DMA 接收缓存;一种有缓存,用的和中断模式下相同的 fifo 数据结构。发送只有一种缓存方式,把应用层内存放到数据队列里做发送缓存。

  2. 无论哪种缓存方案,都没有考虑阻塞的问题。而是抛给串口驱动一个内存地址,就返回到应用层了。应用层要么动用 rt_device_set_rx_indicate rt_device_set_tx_complete 做同步——退化成 poll 模式,失去了 DMA 的优势;要么继续干其它工作——抛给串口驱动的内存可能引入隐患

  3. 为了防止 DMA 工作的时候又有新的读写需求,


对串口驱动的期望

轮询模式不在今天讨论计划内。下面所有的讨论都只涉及中断和 DMA 两种模式。

  • 无论哪种工作模式,都应该有至少一级缓存机制。

  • 无论哪种工作模式,都应该可以设置成阻塞或者非阻塞。

  • 默认是阻塞 io 模式;如果想用非阻塞工作模式,可以通过 open 或者 control 修改。

  • 读写阻塞特性是同步的,不存在阻塞写非阻塞读或者非阻塞写阻塞读两种模式。

  • 阻塞读的过程是,没有数据永久阻塞;有数据无论多少(小于等于期望数据量),返回读取的数据量。

  • 阻塞写的过程是,缓存空间为 0 阻塞等待缓存被释放;缓存空间不足先填满缓存,继续等待缓存被释放;缓存空间足够,把应用层数据拷贝到驱动缓存。最后返回搬到缓存的数据量。

  • 非阻塞读的过程是,没有数据返回 0;有数据,从 fifo 拷贝数据到应用层提供的内存,返回拷贝的数据量。

  • 非阻塞写的过程是,缓存为 0 ,返回 0;缓存不足返回写成功了多少数据;缓存足够,把数据搬移完,返回写成功的数据量。

  • 无论是轮询、中断、DMA 哪种模式,都应该可以实现 STREAM 特性。


中断模式下的理论实践

以下实现是在 NUC970 上完成的,有些特性可能不是通用的。例如,串口外设自带硬件 fifo ,uart1 是高速 uart 设备,fifo 有 64 字节。uart3 的 fifo 就只有 16 字节。

定义缓存数据结构

为实现上述需求,接收和发送都需要有如下一个 fifo

 1struct rt_serial_fifo
2{

3    rt_uint32_t buf_sz;
4    /* software fifo buffer */
5    rt_uint8_t *buffer;
6
7    rt_uint16_t put_index, get_index;
8
9    rt_bool_t is_full;
10};


注:别问我为啥不用 ringbuffer

大部分还是借用 struct rt_serial_rx_fifo 的实现的。增加了个 buf_sz 由 fifo 自己维护自己的缓存容量

针对 fifo 特意定义了三个函数,
rt_forceinline rt_size_t _serial_fifo_calc_data_len(struct rt_serial_fifo *fifo) 计算 fifo 中写入的数据量
rt_forceinline void _serial_fifo_push_data(struct rt_serial_fifo *fifo, rt_uint8_t ch) 压入一个数据(不完整实现,具体见下文)
rt_forceinline rt_uint8_t _serial_fifo_pop_data(struct rt_serial_fifo *fifo) 弹出一个数据(不完整实现,具体见下文)

读设备过程

读设备对应中断接收。

 1rt_inline int _serial_int_rx(struct rt_serial_device *serial, rt_uint8_t *data, int length)
2{
3    rt_size_t len, size;
4    struct rt_serial_fiforx_fifo;
5    rt_base_t level;
6
7    RT_ASSERT(serial != RT_NULL);
8
9    rx_fifo = (struct rt_serial_fifo*) serial->serial_rx;
10    RT_ASSERT(rx_fifo != RT_NULL);
11
12    /* disable interrupt */
13    level = rt_hw_interrupt_disable();
14
15    len = _serial_fifo_calc_data_len(rx_fifo);
16
17    if ((len == 0) &&                // non-blocking io mode
18        (serial->parent.open_flag & RT_DEVICE_OFLAG_NONBLOCKING) == RT_DEVICE_OFLAG_NONBLOCKING) {
19        /* enable interrupt */
20        rt_hw_interrupt_enable(level);
21        return 0;
22    }
23    if ((len == 0) &&                // blocking io mode
24        (serial->parent.open_flag & RT_DEVICE_OFLAG_NONBLOCKING) != RT_DEVICE_OFLAG_NONBLOCKING) {
25        do {
26            /* enable interrupt */
27            rt_hw_interrupt_enable(level);
28
29            rt_completion_wait(&(serial->completion_rx), RT_WAITING_FOREVER);
30
31            /* disable interrupt */
32            level = rt_hw_interrupt_disable();
33
34            len = _serial_fifo_calc_data_len(rx_fifo);
35        } while(len == 0);
36    }
37
38    if (len > length) {
39        len = length;
40    }
41
42    /* read from software FIFO */
43    for (size = 0; size < len; size++)
44    {
45        /* otherwise there's the data: */
46        *data = _serial_fifo_pop_data(rx_fifo);
47        data++;
48    }
49
50    rx_fifo->is_full = RT_FALSE;
51
52    /* enable interrupt */
53    rt_hw_interrupt_enable(level);
54
55    return size;
56}


简单说明就是:关中断,计算缓存数据量,如果为空判断是否需要阻塞。拷贝完数据,开中断。

这里需要注意的是,拷贝完数据后 fifo 必然不会是 full 的,rx_fifo->is_full = RT_FALSE 这句没有加在 _serial_fifo_pop_data 函数,所以上面说它的实现是不完整的。

写设备过程

写设备对应中断发送

 1rt_inline int _serial_int_tx(struct rt_serial_device *serial, const rt_uint8_t *data, int length)
2{
3    rt_size_t len, length_t, size;
4    struct rt_serial_fifo *tx_fifo;
5    rt_base_t level;
6    rt_uint8_t last_char = 0;
7
8    RT_ASSERT(serial != RT_NULL);
9
10    tx_fifo = (struct rt_serial_fifo*) serial->serial_tx;
11    RT_ASSERT(tx_fifo != RT_NULL);
12
13    size = 0;
14    do {
15        length_t = length - size;
16        /* disable interrupt */
17        level = rt_hw_interrupt_disable();
18
19        len = tx_fifo->buf_sz - _serial_fifo_calc_data_len(tx_fifo);
20
21        if ((len == 0) &&                // non-blocking io mode
22            (serial->parent.open_flag & RT_DEVICE_OFLAG_NONBLOCKING) == RT_DEVICE_OFLAG_NONBLOCKING) {
23            /* enable interrupt */
24            rt_hw_interrupt_enable(level);
25            break;
26        }
27
28        if ((len == 0) &&                // blocking io mode
29            (serial->parent.open_flag & RT_DEVICE_OFLAG_NONBLOCKING) != RT_DEVICE_OFLAG_NONBLOCKING) {
30            /* enable interrupt */
31            rt_hw_interrupt_enable(level);
32
33            rt_completion_wait(&(serial->completion_tx), RT_WAITING_FOREVER);
34
35            continue;
36        }
37
38        if (len > length_t) {
39            len = length_t;
40        }
41        /* copy to software FIFO */
42        while (len > 0)
43        {
44            /*
45             * to be polite with serial console add a line feed
46             * to the carriage return character
47             */

48            if (*data == '\n' &&
49                (serial->parent.open_flag & RT_DEVICE_FLAG_STREAM) == RT_DEVICE_FLAG_STREAM &&
50                last_char != '\r')
51            {
52                _serial_fifo_push_data(tx_fifo, '\r');
53
54                len--;
55                if (len == 0break;
56                last_char = 0;
57            } else if (*data == '\r') {
58                last_char = '\r';
59            } else {
60                last_char = 0;
61            }
62
63            _serial_fifo_push_data(tx_fifo, *data);
64
65            data++; len--; size++;
66        }
67
68        /* if the next position is read index, discard this 'read char' */
69        if (tx_fifo->put_index == tx_fifo->get_index)
70        {
71            tx_fifo->is_full = RT_TRUE;
72        }
73
74        // TODO: start tx
75        serial->ops->start_tx(serial);
76
77        /* enable interrupt */
78        rt_hw_interrupt_enable(level);
79    } while(size < length);
80
81    return size;
82}


简单说明就是:关中断,计算 fifo 剩余容量,如果空间不足判断是否阻塞。拷贝数据,开中断。

如果数据没拷贝完,继续上述过程,直到所有数据拷贝完成。
上述函数也实现了 STREAM 打开模式,检查 “r”“n” 不完整的问题。

特别注意:上述函数并没有执行写“发送寄存器”的操作,开中断前,这里执行了一句 serial->ops->start_tx(serial) 用于开启发送过程(这个的实现可能在不同芯片上略有差异)。

中断接收

 1        while (1) {
2            ch = serial->ops->getc(serial);
3            if (ch == -1break;
4
5            /* if fifo is full, discard one byte first */
6            if (rx_fifo->is_full == RT_TRUE) {
7                rx_fifo->get_index += 1;
8                if (rx_fifo->get_index >= rx_fifo->buf_sz) rx_fifo->get_index = 0;
9            }
10            /* push a new data */
11            _serial_fifo_push_data(rx_fifo, ch);
12
13            /* if put index equal to read index, fifo is full */
14            if (rx_fifo->put_index == rx_fifo->get_index)
15            {
16                rx_fifo->is_full = RT_TRUE;
17            }
18        }
19
20        rt_completion_done(&(serial->completion_rx));


先计算是否还有数据要发送,如果没有,调用 serial->ops->stop_tx(serial) 对应上面的 serial->ops->start_tx(serial) 。
因为硬件自带 fifo ,这里最多可以连续写 64 个字节。
因为发送 fifo 是往外弹出数据的,最后肯定是非满的。

未说明的问题

对于串口设备来讲,接收是非预期的,所以串口接收中断必须一直开着。发送就不一样了,没有发送数据的时候是可以不开发送中断的。
上文中提到的两个 ops 
start_tx stop_tx 正是开发送中断使能,关发送中断使能。另外,它俩还有更重要的作用。

在 NUC970 的设计上,只要发送寄存器为空就会有发送完成中断,并不是发送完最后一个字节才产生。正因为这个特性,当开发送中断使能的时候会立马进入中断。在中断里判断是否有数据要发送,刚好可以作为“启动发送”。

对于其它芯片,如果发送中断的含义是“发送完最后一个字节”,仅仅使能发送中断还不够,还需要软件触发发送中断。这是发送不同于接收的最重要的地方。

DMA 模式下的实现探讨

为什么上一节叫实践,这一节变成探讨了?
第一,笔者还没时间在 NUC970 上完成 DMA 的部分。
第二,有了上面中断模式的铺垫,DMA 模式也是轻车熟路。不觉得 NUC970 的硬件 fifo 就是 DMA 的翻版吗?

DMA 模式需要二级缓存机制。第一级缓存和中断模式用的 fifo 一样。这样 read write 两个函数的实现可以是一样的。
在此基础上,增加一个数组。如下是完整串口设备定义:

 1struct rt_serial_device
2{

3    struct rt_device          parent;
4
5    const struct rt_uart_ops *ops;
6    struct serial_configure   config;
7
8    void *serial_rx;
9    void *serial_tx;
10
11    rt_uint8_t serial_dma_rx[64];
12    rt_uint8_t serial_dma_tx[64];
13
14    cb_serial_tx _cb_tx;
15    cb_serial_rx _cb_rx;
16
17    struct rt_completion completion_tx;
18    struct rt_completion completion_rx;
19};
20typedef struct rt_serial_device rt_serial_t;


这两个数组作为 DMA 收发过程的缓存。

发送数据时,从 serial_tx 的 fifo 拷贝数据到 serial_dma_tx ,启动 DMA。发送完成后判断 serial_tx 的 fifo 是否还有数据,有数据继续拷贝,直到 fifo 为空关闭 DMA 发送。

接收数据时,在 DMA 中断里拷贝 serial_dma_rx 所有数据到 serial_rx 的 fifo 。如果 DMA 中断分完成一半中断和全部传输完成两种中断。可以分成两次中断,每次只处理一半数据,这样每次往 fifo 倒腾数据的时候,还有一半缓冲区可用,也不至于会担心仓促。

我们需要做的工作只有“怎么安全有效启动 DMA 发送。

底层驱动

以上都是串口设备驱动框架部分,下面说说和芯片操作紧密相关的部分

init 函数,负责注册设备到设备树。
configure 函数,负责串口外设初始化,包括波特率、数据位、流控等等。还有个重要的工作就是调用引脚复用配置函数。
control 函数,使能禁用收发等中断。
putc 函数,负责写发送寄存器,写寄存器前一定先判断发送寄存器是否可写是否为空,阻塞等。
getc 函数,负责读接收寄存器,读寄存器前一定先判断是否有有效数据,如果没有返回 -1。
start_tx 函数,使能发送中断,如果发送寄存器为空,触发发送中断。(如果芯片没有这个特性,需要想办法触发发送完成中断)
stop_tx 函数,禁用发送中断。
中断回调函数,负责处理中断,根据中断状态调用 
rt_hw_serial_isr 函数。

实机验证

中断模式在 NUC970 芯片下经过千万级数据收发测试的考验。测试环境有如下两种:

1、非阻塞 io;波特率 9600;串口调试工具:USR-TCP232 ,USR 出的调试工具。
串口调试工具定时 50ms 发送 30 个字符。NUC970 接收到数据后返回接收到的数据。

2、阻塞 io;波特率 115200;串口调试工具:USR-TCP232 ,USR 出的调试工具。
串口调试工具定时 10ms 发送 30 个字符。NUC970 接收到数据后返回接收到的数据。(串口调试助手发送了 200w 字节数据,接收到了相同个数字符!)

结论

因为 NUC970 芯片的特殊性,上面虽说使用的是中断模式,其实和 DMA 有点儿类似了。假如是没收发一个字节数据各对应一次中断,中断次数会比较多。

但是,在应用层来看,无论是中断还是 DMA 都是一样的——要么阻塞,要么非阻塞。


👇 点击阅读进入官网

点个在看你最好看

RTThread物联网操作系统 帮助您了解RT-Thread相关的资讯.
评论
  • 振动样品磁强计是一种用于测量材料磁性的精密仪器,广泛应用于科研、工业检测等领域。然而,其测量准确度会受到多种因素的影响,下面我们将逐一分析这些因素。一、温度因素温度是影响振动样品磁强计测量准确度的重要因素之一。随着温度的变化,材料的磁性也会发生变化,从而影响测量结果的准确性。因此,在进行磁性测量时,应确保恒温环境,以减少温度波动对测量结果的影响。二、样品制备样品的制备过程同样会影响振动样品磁强计的测量准确度。样品的形状、尺寸和表面处理等因素都会对测量结果产生影响。为了确保测量准确度,应严格按照规
    锦正茂科技 2025-02-28 14:05 154浏览
  •           近日受某专业机构邀请,参加了官方举办的《广东省科技创新条例》宣讲会。在与会之前,作为一名技术工作者一直认为技术的法例都是保密和侵权方面的,而潜意识中感觉法律有束缚创新工作的进行可能。通过一个上午学习新法,对广东省的科技创新有了新的认识。广东是改革的前沿阵地,是科技创新的沃土,企业是创新的主要个体。《广东省科技创新条例》是广东省为促进科技创新、推动高质量发展而制定的地方性法规,主要内容包括: 总则:明确立法目
    广州铁金刚 2025-02-28 10:14 111浏览
  • 应用趋势与客户需求,AI PC的未来展望随着人工智能(AI)技术的日益成熟,AI PC(人工智能个人电脑)逐渐成为消费者和企业工作中的重要工具。这类产品集成了最新的AI处理器,如NPU、CPU和GPU,并具备许多智能化功能,为用户带来更高效且直观的操作体验。AI PC的目标是提升工作和日常生活的效率,通过深度学习与自然语言处理等技术,实现更流畅的多任务处理、实时翻译、语音助手、图像生成等功能,满足现代用户对生产力和娱乐的双重需求。随着各行各业对数字转型需求的增长,AI PC也开始在各个领域中显示
    百佳泰测试实验室 2025-02-27 14:08 267浏览
  • Matter 协议,原名 CHIP(Connected Home over IP),是由苹果、谷歌、亚马逊和三星等科技巨头联合ZigBee联盟(现连接标准联盟CSA)共同推出的一套基于IP协议的智能家居连接标准,旨在打破智能家居设备之间的 “语言障碍”,实现真正的互联互通。然而,目标与现实之间总有落差,前期阶段的Matter 协议由于设备支持类型有限、设备生态协同滞后以及设备通信协议割裂等原因,并未能彻底消除智能家居中的“设备孤岛”现象,但随着2025年的到来,这些现象都将得到完美的解决。近期,
    华普微HOPERF 2025-02-27 10:32 241浏览
  • 美国加州CEC能效跟DOE能效有什么区别?CEC/DOE是什么关系?美国加州CEC能效跟DOE能效有什么区别?CEC/DOE是什么关系?‌美国加州CEC能效认证与美国DOE能效认证在多个方面存在显著差异‌。认证范围和适用地区‌CEC能效认证‌:仅适用于在加利福尼亚州销售的电器产品。CEC认证的范围包括制冷设备、房间空调、中央空调、便携式空调、加热器、热水器、游泳池加热器、卫浴配件、光源、应急灯具、交通信号模块、灯具、洗碗机、洗衣机、干衣机、烹饪器具、电机和压缩机、变压器、外置电源、消费类电子设备
    张工nx808593 2025-02-27 18:04 128浏览
  • 1,微软下载免费Visual Studio Code2,安装C/C++插件,如果无法直接点击下载, 可以选择手动install from VSIX:ms-vscode.cpptools-1.23.6@win32-x64.vsix3,安装C/C++编译器MniGW (MinGW在 Windows 环境下提供类似于 Unix/Linux 环境下的开发工具,使开发者能够轻松地在 Windows 上编写和编译 C、C++ 等程序.)4,C/C++插件扩展设置中添加Include Path 5,
    黎查 2025-02-28 14:39 149浏览
  • 一、VSM的基本原理震动样品磁强计(Vibrating Sample Magnetometer,简称VSM)是一种灵敏且高效的磁性测量仪器。其基本工作原理是利用震动样品在探测线圈中引起的变化磁场来产生感应电压,这个感应电压与样品的磁矩成正比。因此,通过测量这个感应电压,我们就能够精确地确定样品的磁矩。在VSM中,被测量的样品通常被固定在一个震动头上,并以一定的频率和振幅震动。这种震动在探测线圈中引起了变化的磁通量,从而产生了一个交流电信号。这个信号的幅度和样品的磁矩有着直接的关系。因此,通过仔细
    锦正茂科技 2025-02-28 13:30 106浏览
  • 在物联网领域中,无线射频技术作为设备间通信的核心手段,已深度渗透工业自动化、智慧城市及智能家居等多元场景。然而,随着物联网设备接入规模的不断扩大,如何降低运维成本,提升通信数据的传输速度和响应时间,实现更广泛、更稳定的覆盖已成为当前亟待解决的系统性难题。SoC无线收发模块-RFM25A12在此背景下,华普微创新推出了一款高性能、远距离与高性价比的Sub-GHz无线SoC收发模块RFM25A12,旨在提升射频性能以满足行业中日益增长与复杂的设备互联需求。值得一提的是,RFM25A12还支持Wi-S
    华普微HOPERF 2025-02-28 09:06 166浏览
  •         近日,广电计量在聚焦离子束(FIB)领域编写的专业著作《聚焦离子束:失效分析》正式出版,填补了国内聚焦离子束领域实践性专业书籍的空白,为该领域的技术发展与知识传播提供了重要助力。         随着芯片技术不断发展,芯片的集成度越来越高,结构也日益复杂。这使得传统的失效分析方法面临巨大挑战。FIB技术的出现,为芯片失效分析带来了新的解决方案。它能够在纳米尺度上对芯片进行精确加工和分析。当芯
    广电计量 2025-02-28 09:15 141浏览
  • 在2024年的科技征程中,具身智能的发展已成为全球关注的焦点。从实验室到现实应用,这一领域正以前所未有的速度推进,改写着人类与机器的互动边界。这一年,我们见证了具身智能技术的突破与变革,它不仅落地各行各业,带来新的机遇,更在深刻影响着我们的生活方式和思维方式。随着相关技术的飞速发展,具身智能不再仅仅是一个技术概念,更像是一把神奇的钥匙。身后的众多行业,无论愿意与否,都像是被卷入一场伟大变革浪潮中的船只,注定要被这股汹涌的力量重塑航向。01为什么是具身智能?为什么在中国?最近,中国具身智能行业的进
    艾迈斯欧司朗 2025-02-28 15:45 243浏览
  • RGB灯光无法同步?细致的动态光效设定反而成为产品客诉来源!随着科技的进步和消费者需求变化,电脑接口设备单一功能性已无法满足市场需求,因此在产品上增加「动态光效」的形式便应运而生,藉此吸引消费者目光。这种RGB灯光效果,不仅能增强电脑周边产品的视觉吸引力,还能为用户提供个性化的体验,展现独特自我风格。如今,笔记本电脑、键盘、鼠标、鼠标垫、耳机、显示器等多种电脑接口设备多数已配备动态光效。这些设备的灯光效果会随着音乐节奏、游戏情节或使用者的设置而变化。想象一个画面,当一名游戏玩家,按下电源开关,整
    百佳泰测试实验室 2025-02-27 14:15 140浏览
  • 更多生命体征指标风靡的背后都只有一个原因:更多人将健康排在人生第一顺位!“AGEs,也就是晚期糖基化终末产物,英文名Advanced Glycation End-products,是存在于我们体内的一种代谢产物” 艾迈斯欧司朗亚太区健康监测高级市场经理王亚琴说道,“相信业内的朋友都会有关注,最近该指标的热度很高,它可以用来评估人的生活方式是否健康。”据悉,AGEs是可穿戴健康监测领域的一个“萌新”指标,近来备受关注。如果站在学术角度来理解它,那么AGEs是在非酶促条件下,蛋白质、氨基酸
    艾迈斯欧司朗 2025-02-27 14:50 424浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦