STM32外设DMA使用总结

STM32外设DMA使用总结:

1、根据需要选择DAM模式:

(1)循环模式—DMA_Mode =  DMA_Mode_Circular

(2)正常模式—DMA_Mode =  DMA_Mode_Normal

2、对于DMA1的Chanel3,对应外设为USART3的RX

试想:如果串口接收中断和DAM中断同时打开,CPU如何相应?

(1)中断优先级不同:这好说,支持嵌套中断(NVIC)的Cortex-M3自然优先服务中断优先级高的

(2)中断优先级相同:处理原则,先来先处理;若同时到来,中断号低的优先处理

查询手册可知,DMA(IRQn number 13)会先于USART3(39)被CPU处理

3、设置DMA模式为循环模式,则:

(1)只开DMA中断,关闭外设中断,将只进一次DMA服务函数

(2)开启DMA中断,开启外设中断,才会循环进DMA服务函数

4、在开启多个DMA中断时,注意每个中断的优先级设置

5、在中断函数中,禁止DMA使能后要重新设置DMA传输数据量大小并开启DMA

void DMA1_Channel3_IRQHandler(void)
{
	if(DMA_GetITStatus(DMA1_FLAG_TC3))
	{
		DMA_ClearITPendingBit(DMA1_IT_GL3);
		/* 处理数据期间关闭DMA,防止数据篡改 */
		DMA_Cmd(DMA1_Channel3, DISABLE);	

		/* 处理数据 */

		/* 重新设置DMA传输数据量大小并开启DMA */
		DMA1_Channel3->CNDTR = DMA_CH3_RECV_BYTES;
		DMA1_Channel3->CCR |= DMA_CCR3_EN;
	}
}

6、DMA配置选项:

typedef struct
{
	/* 设置DMA传输的外设基地址,比如USART的DMA传输,基地址为(uint32_t)USART3-DR */
	uint32_t DMA_PeripheralBaseAddr;
	/* 设置DMA内存基地址,存放DMA传输数据的内存地址 */
	uint32_t DMA_MemoryBaseAddr;
	/* 操作的外设是源地(从外设取数据)、目的地(发送数据到外设) */
	uint32_t DMA_DIR;
	/* 一次DMA传输数据量的大小 */
	uint32_t DMA_BufferSize;
	/* 外设地址是否自动递增,比如对于USART的DMA传输,外设地址是固定的 */
	uint32_t DMA_PeripheralInc;
	/* 同上,只不过针对的是内存地址 */
	uint32_t DMA_MemoryInc;
	/* 设置外设数据宽度,8bits/16bits/31bits */
	uint32_t DMA_PeripheralDataSize;
	/* 同上 */
	uint32_t DMA_MemoryDataSize;
	/* 选择DAM模式,循环模式还是正常模式 */
	uint32_t DMA_Mode;
	/* DMA通道的优先级,DMA_Priority_VeryHigh,DMA_Priority_High,DMA_Priority_Medium,DMA_Priority_Low  */
	/* 如果开启多个DMA通道的话,记得设置合适的值 */
	uint32_t DMA_Priority;
	/* 设置DMA是否为Mem到Mem模式 */
	uint32_t DMA_M2M;
}DMA_InitTypeDef;
时间: 2024-12-28 14:01:17

STM32外设DMA使用总结的相关文章

STM32之DMA+ADC

借用小甲鱼的经典:各位互联网的广大网友们.大家早上中午晚上好..(打下小广告,因为小甲鱼的视频真的很不错).每次看小甲鱼的视频自学都是比较轻松愉快的..我在想,如果小甲鱼出STM32的视频,我会一集不漏的听的.哈.好了..学习到了STM32的DMA模块..琢磨了一下中文参考手册,官方是这样描述的: 直接存储器存取(DMA)用来提供在外设和存储器之间或者存储器和存储器之间的高速数据传输.无须CPU干预,数据可以通过DMA快速地移动,这就节省了CPU的资源来做其他操作. 是的,无需CPU干预,可以想

STM32之DMA

一.DMA简介 1.DMA简介 DMA(Direct Memory Access:直接内存存取)是一种可以大大减轻CPU工作量的数据转移方式. CPU有转移数据.计算.控制程序转移等很多功能,但其实转移数据(尤其是转移大量数据)是可以不需要CPU参与.比如希望外设A的数据拷贝到外设B,只要给两种外设提供一条数据通路,再加上一些控制转移的部件就可以完成数据的拷贝. DMA就是基于以上设想设计的,它的作用就是解决大量数据转移过度消耗CPU资源的问题.有了DMA使CPU更专注于更加实用的操作--计算.

STM32 外设地址映射管理硬件基地址和偏移地址

片上外设区分为四条总线,根据外设速度的不同,不同总线挂载着不同的外设, APB1 挂载低速外设,APB2 和 AHB 挂载高速外设.相应总线的最低地址我们称为该总线的基地址,总线基地址也是挂载在该总线上的首个外设的地址.APB1 总线的地址最低,因此片上外设就从这这个地址开始,也称外设基地址. (1)总线基地址 从存储器映射那张图的 Block2 可以看到,分为 4 大块,每块都有一个起始地址,这个起始地址就是基地址,然后到下一块起始地址的时候就会和前一块地址出现偏差,这个差值就是偏移量,即相对

STM32 基DMA的DAC波形发生器

DAC是STM32系列的一个基本外设,可以将数字信号转化成模拟信号,这次我将使用DAC来输出一个特定波形. 首先确定工作方法,由于我目前在做的简易示波器在输出波形的同时还需要显示输入信号,所以不能占用太多CPU时间,于是就选用了基于DMA的ADC. 使用DMA只需告诉DMA外设它要怎么搬移数据就可以处理其他事. 首先定义一下 #define DAC_DHR12R1    (u32)&(DAC->DHR12R1)   //DAC DATA buff 作为DMA的外设数据地址 首先是初始化输出管

STM32 UART DMA实现未知数据长度接收

串口通信是经常使用到的功能,在STM32中UART具有DMA功能,并且收发都可以使用DMA,使用DMA发送基本上大家不会遇到什么问题,因为发送的时候会告知DMA发送的数据长度,DMA按照发送的长度直接发送就OK了,但是使用DMA接收时候就不同了,因为有时候数据接收并不是每一次都是定长的,但是DMA只在接收数据长度和设定数据长度相同的时候才可以触发中断,告诉MCU数据接收完毕,针对这个问题,解决方法如下,有一点复杂,但是很管用. UART在传输一个字节的时候,首先拉低,传输起始位,然后在是LSB

STM32 ADC DMA 中断模式多通道读取ADC转换值

因为做一个电机控制项目,其中需要用到用PWM触发ADC,读取ADC的转换值. 这里有两个关键问题: 第一,如何使PWM触发ADC转换? 第二,如何在ADC多通道扫描模式下,读取各个channel的adc转换值? 第三,如何使用DMA读取多通道ADC转换值? 第一个问题,首先需要将ADC设置成外部触发转换,第二需要选择触发是发生在Timer的上升沿还是下降沿.程序如下: adc.ADC_Resolution = ADC_Resolution_12b; adc.ADC_ScanConvMode =E

STM32 使用DMA+DAC+TIMER 输出正弦波

之前已经简单论述过,根据我个人菜鸟的了解与认识,对之前的知识进行整理回顾: DMA:我的理解就是一个通道,或者是一座桥梁.在静态内存到静态内存,或者外设到静态内存间的一个通讯的通道.建立这个通道的好处是:可以抛开CPU,不占用CPU的资源,直接使用这块内存的内容,速度也会加快. DAC:STM32F103中有两个DAC,可以同时使用.DAC的作用就是将数字量转化为模拟量(电压),在这就不作太多的讲解. TIMER:定时器.不作讲解. 那么对于使用DMA+DAC+TIMER产生正弦波的原理或过程,

STM32串口DMA超时接收方法,可大大节约CPU时间

//超时时间定义#define        UART1_TimeoutComp 2  //20ms#define        UART2_TimeoutComp 10  //100ms#define        UART3_TimeoutComp 10  //100ms u8 UART1_Timeout,UART2_Timeout,UART3_Timeout;u16 UART1_FlagTemp,UART2_FlagTemp,UART3_FlagTemp;u8 uart1_data_tem

stm32 spi dma tft lcd

http://habrahabr.ru/post/139384 http://electronics.stackexchange.com/questions/100685/full-duplex-slave-spi-dma-and-interrupts-on-stm32f103 https://my.st.com/public/STe2ecommunities/mcu/Lists/cortex_mx_stm32/DispForm.aspx?ID=51426&Source=/public/STe2