STM32串口DMA容易忽视的问题
扫描二维码
随时随地手机看文章
昨天晚上在STM32串口DMA的问题上纠结了好长时间,所以今天上午写篇博客来谈谈我对串口DMA发送的理解
讨论三个问题:1、什么叫串口DMA 请求;2、串口简要复习;3、串口DMA发送流程。
1、什么叫串口DMA 请求(战舰STM32开发板)
说这个问题之前先简单回顾DMA的基本特性。先导出原子哥的PPT内容:
DMA全称Direct Memory Access,即直接存储器访问。
DMA传输将数据从一个地址空间复制到另一个地址空间。当CPU初始化这个传输动作,传输动作本身是由DMA控制器来实现和完成的。
STM32有两个DMA控制器(DMA2只存在于大容量产品中),DMA1有7个通道,DMA2有5个通道,每个通道专门用来管理来自于一个或者多个外设对存储器的访问请求。还有一个仲裁器来协调各个DMA请求的优先权。
作用:为CPU减负!
下面上图
DMA各通道
从外设(TIMx、ADCx、SPIx、I2Cx、和USARTx)产生的DMA请求,通过逻辑或输入到DMA控制器,这就意味着同时只能有一个请求有效(在同一时间,就只能使用其中的一个,其他通道也类似)。外设的DMA请求,可以通过设置相应的外设寄存器中的控制位,被独立的开启或关闭。
看到这里串口DMA请求的定义就出来了:先举个例子:譬如我想用串口发送数据到PC机,一般做法是CPU通过APB1/APB2总线往串口写数据,然后串口发送;或者是CPU通过总线从串口取(读)数据;如果此时CPU处理的任务非常多,这就会耗费CPU的处理时间。
而我们如果用DMA来传送数据那就很快了,DMA会通过自己的特定通道将数据从一个地址空间复制到另一个地址空间,而且不需要CPU的干预。所以串口DMA请求的定义是:串口要发送/接收数据会请求DMA来完成数据的写和读。(注意这句话只针对本文!)
2、串口简要复习(库函数)
这里给大家分享个帖子关于printf重定向和半主机模式的理解:
http://www.openedv.com/forum.php?mod=viewthread&tid=44609&highlight=printf
这里简要复习一下战舰STM32串口的知识点。举个例子:我想用PC机往STM32串口发送个数据,然后STM32接收到后将数据原样返回给PC机(显示在串口调试助手中)。思路:
首先我们会在串口初始化函数中配置串口接收中断,当你PC机往串口发送数据时,串口会产生接收中断;并在中断服务函数中,判断接没接收完,如果接收完毕就会把接收状态寄存器的接收完成标志位置1。到这儿是接收中断的作用。。
然后在主函数中,我们不断检测接收状态寄存器的接收完成标志,看接没接收完。如果接收完了,就通过for循环将接收到的数据,全部发给串口的USART_DR寄存器(当向该寄存器写数据时,串口就会自动发送)。于是我们通过PC机发给串口的数据就会被STM32通过串口发回来了,显示在串口调试助手中。
3、串口DMA发送流程(库函数)
战舰实验:通过按键KEY0来控制DMA发送,每按一次KEY0,DMA就传送一次数据到USART1,然后在TFTLCD模块上显示进度信息。这个现在理解就不难了吧。
从主函数入手,首先我们在STM32的SRAM中开辟一段5200(这个数字越大DMA传输越慢,反之越快)字节的空间SendBuff[],作为我们的存储器(内存),存储要往外设(USART1)发送的数据。
然后我们在SendBuff[]中写满TEXT_TO_SEND[]里的内容。这里战舰往SendBuff[]里写的算法非常巧妙,应该学习;
现在SendBuff[]中已经存满了要发送给串口的数据。这时我们还要检测KEY0有没有按下,如果按下了,就通过两个函数:1、使能串口发送(在STM32库函数stm32f10x_usart.c中,配置USART_CR3中的DMA使能位DMAT);2、使能DMA1通道4,启动传输(在dam.c中,最后也是调用了stm32f10x_dma.c中的DMA_Cmd();函数来使能通道)。加上之前(本文没有)对DMA参数的初始化(初始化存储器和外设基地址等),就可以启动一次传输了。
传输时间问题:传输时间跟波特率密切相关。比如波特率是9600,意思就是每秒钟发送或接受的速率为9600bit/s,8bit=1Byte(字节),故当波特率为9600时,串口的传输速率为1200字节每秒。上文我们申请的存储器(内存)空间是5200字节,故串口要传输完这个大小的空间内容内容,预计需要5200/1200约4秒多。。。