OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 3269|回复: 7

串口DMA很难?让我来说不!!!

[复制链接]

1

主题

1

帖子

0

精华

新手上路

积分
33
金钱
33
注册时间
2020-3-14
在线时间
8 小时
发表于 2020-5-15 01:48:09 | 显示全部楼层 |阅读模式
本帖最后由 li1229574727 于 2020-5-15 02:02 编辑

串口DMA

原文已在CSDN发布,博文阅读起来更舒服,CSDN博文链接:

STM32 | 串口DMA很难?其实就是如此简单!(超详细、附代码)https://blog.csdn.net/weixin_44524484/article/details/106029682



DMA利用好无疑会让串口使用起来更加高效,同时CPU还能处理自己的事情,但是DMA的使用却让代码变得更加的复杂,也使串口配置变得更加麻烦!麻烦???也许只是学习的方式不对,代码是变长了,但是思路清晰的话,也就是在原来串口加了点东西而已。本文让你轻轻松松学会串口DMA!定长数据传输与不定长数据传输都教会你!通过此文,希望能让更多人了解和会使用串口DMA,希望大家多多支持!

一、串口DMA的配置

本文针对串口2(USART2)如何进行DMA传输进行讲解,如果采用其他串口,注意要修改相应端口配置以及DMA通道。

1、串口的DMA请求映像

通过我之前的博文《 STM32 | DMA配置和使用如此简单(超详细)》可以知道,串口2(USART2)的接收(USART2_RX)和发送(USART2_TX)分别在DMA1控制器的通道6和通道7。下图为DMA1控制器的请求映像图。




2、资源说明



STM32F1中,USART2使用的是PA2(USART2_TX)和PA3(USART2_RX),为了方便阅读,使用到的资源列在下表。

外设        GPIO口        DMA请求映像通道        备注

USART2_TX        PA2        DMA1通道7        RAM->USART2的数据传输

USART2_RX        PA3        DMA1通道6        USART2->RAM的数据传输

3、DMA初始化配置

USART2的DMA配置在《 STM32 | DMA配置和使用如此简单(超详细)》中分别讲了库函数版和寄存器版两种配置,我这里直接搬用,不理解的可以看看《 STM32 | DMA配置和使用如此简单(超详细)》这篇文章。要注意的是库函数最大优势就是便于阅读,所以在DMA初始化配置中我们使用库函数。

首先,我们要先定义三个缓冲区(作全局定义),一个发送缓冲区,两个接收缓冲区,两个接收缓冲区是为了做双缓冲区,目的是为了防止后一次传输的数据覆盖前一次传输的数据,并且留出足够的时间让CPU处理缓冲区数据。双缓冲在串口DMA中有着很重要的意义并起着很大的作用!

  1. //USART2_MAX_TX_LEN和USART2_MAX_RX_LEN在头文件进行了宏定义,分别指USART2最大发送长度和最大接收长度
  2. u8 USART2_TX_BUF[USART2_MAX_TX_LEN];         //发送缓冲,最大USART2_MAX_TX_LEN字节
  3. u8 u1rxbuf[USART2_MAX_RX_LEN];                      //发送数据缓冲区1
  4. u8 u2rxbuf[USART2_MAX_RX_LEN];                      //发送数据缓冲区2
  5. u8 witchbuf=0;                                                    //标记当前使用的是哪个缓冲区,0:使用u1rxbuf;1:使用u2rxbuf
  6. u8 USART2_TX_FLAG=0;                                      //USART2发送标志,启动发送时置1
  7. u8 USART2_RX_FLAG=0;                                      //USART2接收标志,启动接收时置1
复制代码

要说明的是,实际上发送缓冲区可能用不上,因为我们要发送的数据内容和大小都不一定每次都是固定的,可以是变化的,我们只需重新指派新的发送缓冲区地址即可,但是在初始化中我们还是要指定一个发送缓冲区地址。
下面是DMA1_USART2的初始化函数。

  1. void DMA1_USART2_Init(void)
  2. {
  3.         DMA_InitTypeDef DMA1_Init;
  4.         NVIC_InitTypeDef NVIC_InitStructure;
  5.         
  6.         RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);                                                //使能DMA1时钟

  7.         //DMA_USART2_RX  USART2->RAM的数据传输
  8.         DMA_DeInit(DMA1_Channel6);                                                                                                //将DMA的通道6寄存器重设为缺省值
  9.         DMA1_Init.DMA_PeripheralBaseAddr = (u32)(&USART2->DR);                                        //启动传输前装入实际RAM地址
  10.         DMA1_Init.DMA_MemoryBaseAddr = (u32)u1rxbuf;                                            //设置接收缓冲区首地址
  11.         DMA1_Init.DMA_DIR = DMA_DIR_PeripheralSRC;                                                                //数据传输方向,从外设读取到内存
  12.         DMA1_Init.DMA_BufferSize = USART2_MAX_RX_LEN;                                                        //DMA通道的DMA缓存的大小
  13.         DMA1_Init.DMA_PeripheralInc = DMA_PeripheralInc_Disable;                                //外设地址寄存器不变
  14.         DMA1_Init.DMA_MemoryInc = DMA_MemoryInc_Enable;                                                        //内存地址寄存器递增
  15.         DMA1_Init.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;                        //数据宽度为8位
  16.         DMA1_Init.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;                                        //数据宽度为8位
  17.         DMA1_Init.DMA_Mode = DMA_Mode_Normal;                                                                        //工作在正常模式
  18.         DMA1_Init.DMA_Priority = DMA_Priority_High;                                                         //DMA通道 x拥有高优先级
  19.         DMA1_Init.DMA_M2M = DMA_M2M_Disable;                                                                        //DMA通道x没有设置为内存到内存传输
  20.          
  21.         DMA_Init(DMA1_Channel6,&DMA1_Init);                                                                         //对DMA通道6进行初始化
  22.         
  23.         //DMA_USART2_TX  RAM->USART2的数据传输
  24.         DMA_DeInit(DMA1_Channel7);                                                                                                //将DMA的通道7寄存器重设为缺省值
  25.         DMA1_Init.DMA_PeripheralBaseAddr = (u32)(&USART2->DR);                                        //启动传输前装入实际RAM地址
  26.         DMA1_Init.DMA_MemoryBaseAddr = (u32)USART2_TX_BUF;                              //设置发送缓冲区首地址
  27.         DMA1_Init.DMA_DIR = DMA_DIR_PeripheralDST;                                                                 //数据传输方向,从内存发送到外设
  28.         DMA1_Init.DMA_BufferSize = USART2_MAX_TX_LEN;                                                        //DMA通道的DMA缓存的大小
  29.         DMA1_Init.DMA_PeripheralInc = DMA_PeripheralInc_Disable;                                //外设地址寄存器不变
  30.         DMA1_Init.DMA_MemoryInc = DMA_MemoryInc_Enable;                                                        //内存地址寄存器递增
  31.         DMA1_Init.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;                        //数据宽度为8位
  32.         DMA1_Init.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;                                        //数据宽度为8位
  33.         DMA1_Init.DMA_Mode = DMA_Mode_Normal;                                                                        //工作在正常模式
  34.         DMA1_Init.DMA_Priority = DMA_Priority_High;                                                         //DMA通道 x拥有高优先级
  35.         DMA1_Init.DMA_M2M = DMA_M2M_Disable;                                                                        //DMA通道x没有设置为内存到内存传输

  36.         DMA_Init(DMA1_Channel7,&DMA1_Init);                                                                         //对DMA通道7进行初始化
  37.         
  38.         //DMA1通道6 NVIC 配置
  39.         NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel6_IRQn;                                //NVIC通道设置
  40.         NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 3 ;                                //抢占优先级
  41.         NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;                                                //子优先级
  42.         NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                                                        //IRQ通道使能
  43.         NVIC_Init(&NVIC_InitStructure);                                                                                        //根据指定的参数初始化NVIC寄存器

  44.         //DMA1通道7 NVIC 配置
  45.         NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel7_IRQn;                                //NVIC通道设置
  46.         NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 3 ;                                //抢占优先级
  47.         NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;                                                //子优先级
  48.         NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                                                       //IRQ通道使能
  49.         NVIC_Init(&NVIC_InitStructure);                                                                                        //根据指定的参数初始化NVIC寄存器

  50.         DMA_ITConfig(DMA1_Channel6,DMA_IT_TC,ENABLE);                                                        //开USART2 Rx DMA中断
  51.         DMA_ITConfig(DMA1_Channel7,DMA_IT_TC,ENABLE);                                                        //开USART2 Tx DMA中断

  52.         DMA_Cmd(DMA1_Channel6,ENABLE);                                                                           //使DMA通道6停止工作
  53.         DMA_Cmd(DMA1_Channel7,DISABLE);                                                                           //使DMA通道7停止工作
  54.          
  55.         USART_DMACmd(USART2, USART_DMAReq_Tx, ENABLE);                                                //开启串口DMA发送
  56.         USART_DMACmd(USART2, USART_DMAReq_Rx, ENABLE);                                                //开启串口DMA接收
  57. }
复制代码

相应配置的讲解在《 STM32 | DMA配置和使用如此简单(超详细)》中讲解过,这里不再叙述,要注意的是,我们采用中断方式进行DMA传输。正常情况下,我们传输完数据要进行相应的处理,那我们怎么知道什么时候数据传输完成了呢?这时候就借助DMA传输完成中断。既然打开了中断,那一定要注意中断优先级,这里特别指出串口的中断优先级应低于串口DMA通道的中断优先级。

4、串口配置

前面已经完成了DMA的配置,而DMA是不能单独使用的,所以不要忘了配置要用到的串口USART2。

  1. void Initial_UART2(unsigned long baudrate)
  2. {
  3.         //GPIO端口设置
  4.         GPIO_InitTypeDef GPIO_InitStructure;
  5.         USART_InitTypeDef USART_InitStructure;
  6.         NVIC_InitTypeDef NVIC_InitStructure;
  7.         
  8.         RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2 | RCC_APB2Periph_GPIOA, ENABLE);        //使能USART2,GPIOA时钟
  9.         
  10.         //USART2_TX   GPIOA.2初始化
  11.         GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;                                                                                //PA.2
  12.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;                                                                        //复用推挽输出
  13.         GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;                                                                //GPIO速率50MHz
  14.         GPIO_Init(GPIOA, &GPIO_InitStructure);                                                                                        //初始化GPIOA.2
  15.         
  16.         //USART2_RX          GPIOA.3初始化
  17.         GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;                                                                                //PA.3
  18.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;                                                        //浮空输入
  19.         GPIO_Init(GPIOA, &GPIO_InitStructure);                                                                                        //初始化GPIOA.3
  20.          
  21.         //USART 初始化设置
  22.         USART_InitStructure.USART_BaudRate = baudrate;                                                                        //串口波特率
  23.         USART_InitStructure.USART_WordLength = USART_WordLength_8b;                                                //字长为8位数据格式
  24.         USART_InitStructure.USART_StopBits = USART_StopBits_1;                                                        //一个停止位
  25.         USART_InitStructure.USART_Parity = USART_Parity_No ;                                                        //无奇偶校验位
  26.         USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;        //无硬件数据流控制
  27.         USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;                                        //收发模式
  28.         USART_Init(USART2, &USART_InitStructure);                                                                                 //初始化串口2
  29.         
  30.         //中断开启设置
  31.         USART_ITConfig(USART2, USART_IT_IDLE, ENABLE);                                                                        //开启检测串口空闲状态中断
  32.         USART_ClearFlag(USART2,USART_FLAG_TC);                                                                                        //清除USART2标志位
  33.         
  34.         USART_Cmd(USART2, ENABLE);                                                                                                                //使能串口2
  35.         
  36.         NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;                                                                //NVIC通道设置
  37.         NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 8;                                                //抢占优先级
  38.         NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;                                                                //响应优先级
  39.         NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                                                                        //IRQ通道使能
  40.         NVIC_Init(&NVIC_InitStructure);                                                                                                        //根据指定的参数初始化NVIC寄存器
  41.         
  42.         DMA1_USART2_Init();                                                                                                                                //DMA1_USART2初始化
  43. }

复制代码

可以注意到,我这里定义的串口中断抢占优先级低于DMA中断的抢占优先级。细心点可以看到,我开启了串口空闲中断。为什么呢?因为通常情况下我们是不知道接收数据的长度的,这样我们是没有办法利用DMA传输完成标志位来判断是否完成接收,所以我们这里采用串口空闲中断来判断数据是否接收完成,接收完成了会进入串口空闲中断。串口配置的最后进行了串口2(USART2)DMA的初始化,这样直接初始化串口也直接包括了串口DMA的初始化,主函数初始化只需一步即可。

二、串口DMA的使用

。。。。。。论坛看起来太不舒服了,有的格式没办法设置,如果喜欢还是去CSDN博客看吧!

原文链接:https://blog.csdn.net/weixin_44524484/article/details/106029682


以下为原文目录,一步一步教会你串口DMA!


正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

0

主题

42

帖子

0

精华

新手上路

积分
43
金钱
43
注册时间
2019-6-25
在线时间
0 小时
发表于 2020-5-15 01:48:12 | 显示全部楼层
回复 支持 反对

使用道具 举报

1

主题

36

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
357
金钱
357
注册时间
2015-12-11
在线时间
79 小时
发表于 2020-5-15 11:50:40 | 显示全部楼层
楼主大神,膜拜学习ing
回复 支持 反对

使用道具 举报

6

主题

412

帖子

0

精华

资深版主

Rank: 8Rank: 8

积分
2722
金钱
2722
注册时间
2019-8-14
在线时间
415 小时
发表于 2020-5-15 15:28:36 | 显示全部楼层
谢谢分享
回复 支持 反对

使用道具 举报

0

主题

98

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
347
金钱
347
注册时间
2018-3-20
在线时间
75 小时
发表于 2020-5-16 14:25:14 | 显示全部楼层
膜拜大佬
回复 支持 反对

使用道具 举报

10

主题

205

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1322
金钱
1322
注册时间
2015-3-3
在线时间
127 小时
发表于 2020-5-16 18:47:40 | 显示全部楼层
你这F1的肯定没啥大的问题,你可以试试F4的这个DMA所有串口同时开启DMA试试比较费劲了。
海纳百川者,荣耀伴一生!
回复 支持 反对

使用道具 举报

32

主题

236

帖子

0

精华

高级会员

Rank: 4

积分
993
金钱
993
注册时间
2017-8-11
在线时间
137 小时
发表于 2020-5-18 11:58:29 | 显示全部楼层
楼主的讲解,非常适合新手理解跟阅读。CSDN的原工程需要积分才能下载到源工程,作为开源移植对于想学习的童鞋还是有点难度。其实底层这些DMA网上资料很多很多,然而怎么用好双缓冲这个才是亮点(标志位提醒判断处理),我觉得这个才是楼主程序值得借鉴的地方。
遗憾的是,没有开源到工程文件
回复 支持 反对

使用道具 举报

0

主题

13

帖子

0

精华

初级会员

Rank: 2

积分
183
金钱
183
注册时间
2016-1-19
在线时间
54 小时
发表于 2020-6-12 10:40:33 | 显示全部楼层
谢谢分享,学习了
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-10 04:00

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表