一、简介
曾经遇到客户有一个需求,需要用串口 DMA 的方式接收不定长度的数据,DMA 有个缺点就是在每次传输前需要设定好传输的字节长度,这种方式显然对于接收不定长度的数据来说没有那么灵活。但 DMA 也有着显著的优点,如可直接访问内存,减少 CPU 负担等。那么能否有比较好的方式能使用 DMA 去传输不定长的数据呢?当然是有的,以下便介绍其中一种通过定时器使串口 DMA 接收不定长数据的方式,以 NXP LPC5516 的 串口 DMA 例程为例。
二、方法思路
该方法使用到定时器中断和串口中断,具体思路如下。
- 使能串口的 RX Start 中断,当串口开始接收数据时会产生一个中断,
-
使能定时器中断、使能 DMA。
-
调用 USART_TransferReceiveDMA()函数,串口 DMA 开始接收数据,设置接收字节长度为最大字节长度。
-
当串口开始接收数据时,产生中断,进入中断函数,这时开启定时器,定时器时长为接收最大字节长度所需的时间,该时间可通过设定的串口波特率计算得出。
-
定时结束,触发定时器中断,进入定时中断函数,首先关闭定时器计时,然后读取 DMA 的 XFERCFG 寄存器获得剩余未读取的字节数,最后用设定的 DMA 接收最大字节数减去剩余未读取字节数便得出本次传输已接收的字节数(设定数 - 剩余未接收数 = 已接收数)。
- 待得到本次已接收的字节数后,便知道 Buffer 前面几个字节是本次接收到的数据,将其进行处理,最后将 DMA 重新设为空闲状态,同时准备第二次串口接收。
三、代码实现
- 使能串口及串口 Rx Start 中断,
- 使能定时器中断、DMA 中断并开启串口 DMA 接收。
- 进入串口 Rx Start 中断,开启定时器。
- 定时结束、进入定时器中断,在定时器中断中获取剩余未读取的字节数,计算已接收字节数。并对读取到的数据 buffer 进行处理。
- 将 DMA 重新进入空闲状态,并开启下一次串口 DMA 接收,以此循环。
四、测试
使用 LPC5516 EVK 烧录该工程,闭合跳帽JP12,打开串口工具,发送不同字节长度的数据,可以看到 RX 能正常读取。
五、总结
本文章介绍了一种串口 DMA 接收不定长度数据的方法,通过添加定时器定时最大字节长度的时间,通过剩余未读取的字节数,计算已读取的字节数,再对 Buffer 进行处理。当然该方式由于无论每次读取多少字节的数据,都会花费最大字节数的时间,故只能在对实时性没有特殊要求的情景下使用。以上便是串口 DMA 接收不定长数据的一种方式。