sinadz的笔记 https://bbs.21ic.com/?589180 [收藏] [复制] [RSS]

日志

浅谈STM32的DMA模块的使用

已有 1778 次阅读2009-10-18 11:29 |系统分类:单片机| STM32

浅谈STM32的DMA模块的使用

              接触MCU的编程已经有几年的历史,刚开始是从PLCC语言的)学习与应用,后来有学习了8bitsAVR单片机,对MCU有一定的了解;最近接触了STM32Chip,发现其优点不只是单单宣传的32bitsMCU的速度,而且在编程的风格上也有了较大的变化(与我原有的编程习惯);特别是有一个与CPU并行运行的模块:DMA,对它印象深刻。现在就和大家一起讨论学习这个Module,谈谈我在学习中一些感悟和大家一起分享;如有对DMA理解和应用上的错误和偏差,欢迎“拍砖”和提出更正,虾米在此感谢大家了。哈哈!


什么是STM32DMA?其全称是:Direct Memory Access;根据ST公司提供的相关信息,DMASTM32中一个独立与Cortex-M3内核的模块,有点类似与ADCPWMTIMER等模块;主要功能是通信“桥梁”的作用,可以将所有外设映射的寄存器“连接”起来,这样就可以高速问各寄存器,其传输不受CPU的支配,传输还是双向的;例如,从“表面”上看,它可以将flash中的数据与储存器中变量建立通讯,还可以将一外设的积存器或缓冲器与另外设的寄存器或缓冲器建立双向通讯,有点像把外设硬件之间用“导线”连接在一起了。其间的通讯不占CPU资源,访问速度高,对于实时性强的应用将是一个很好的选择;当然,对于实时性非常强的,建议还是采用专用的DSP芯片。


怎样启用DMA?首先,众所周知的是初始化,任何设备启用前都要对其进行初始化,要对模块初始化,还要先了解该模块相应的结构及其函数,以便正确的设置;由于DMA较为复杂,我就只谈谈DMA的基本结构和和常用函数,这些都是ST公司提供在库函数中的。


1、  下面代码是一个标准DMA设置,当然实际应用中可根据实际情况进行裁减:


 


  DMA_DeInit(DMA_Channel1);


  上面这句是给DMA配置通道,根据ST提供的资料,STM3210FxDMA包含7个通道(CH1~CH7),也就是说可以为外设或memory提供7座“桥梁”(请允许我使用桥梁一词,我觉得更容易理解,哈哈,别“拍砖”呀!);


 


  DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;


  上面语句中的DMA_InitStructure是一个DMA结构体,在库中有声明了,当然使用时就要先定义了;DMA_PeripheralBaseAddr是该结构体中一个数据成员,给DMA一个起始地址,好比是一个buffer起始地址,数据流程是:外设寄存器à DMA_PeripheralBaseAddàmemory中变量空间(或flash中数据空间等),ADC1_DR_Address是我定义的一个地址变量;


 


  DMA_InitStructure.DMA_MemoryBaseAddr = (u32)ADC_ConvertedValue;


  上面这句很显然是DMA要连接在Memory中变量的地址,ADC_ConvertedValue是我自己在memory中定义的一个变量;


 


  DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;


  上面的这句是设置DMA的传输方向,就如前面我所说的,DMA可以双向传输,也可以单向传输,这里设置的是单向传输,如果需要双向传输:把DMA_DIR_PeripheralSRC改成DMA_DIR_PeripheralDST即可。


 


 


 DMA_InitStructure.DMA_BufferSize = 2;


  上面的这句是设置DMA在传输时缓冲区的长度,前面有定义过了buffer的起始地址:ADC1_DR_Address ,为了安全性和可靠性,一般需要给buffer定义一个储存片区,这个参数的单位有三种类型:ByteHalfWordword,我设置的2half-word(见下面的设置)32位的MCU1half-word16 bits


 


 DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;


  上面的这句是设置DMA的外设递增模式,如果DMA选用的通道(CHx)有多个外设连接,需要使用外设递增模式:DMA_PeripheralInc_Enable;我的例子里DMA只与ADC1建立了联系,所以选用DMA_PeripheralInc_Disable


 


DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;


上面的这句是设置DMA的内存递增模式,DMA访问多个内存参数时,需要使用DMA_MemoryInc_Enable,当DMA只访问一个内存参数时,可设置成:DMA_MemoryInc_Disable


 


 DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;


上面的这句是设置DMA在访问时每次操作的数据长度。有三种数据长度类型,前面已经讲过了,这里不在叙述。


 


DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;


与上面雷同。在此不再说明。


 


  DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;


  上面的这句是设置DMA的传输模式:连续不断的循环模式,若只想访问一次后就不要访问了(或按指令操作来反问,也就是想要它访问的时候就访问,不要它访问的时候就停止),可以设置成通用模式:DMA_Mode_Normal


 


  DMA_InitStructure.DMA_Priority = DMA_Priority_High;


  上面的这句是设置DMA的优先级别:可以分为4级:VeryHighHigh,Medium,Low.


 


  DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;


  上面的这句是设置DMA2memory中的变量互相访问的


 


  DMA_Init(DMA_Channel1,&DMA_InitStructure);


  前面那些都是对DMA结构体成员的设置,在次再统一对DMA整个模块做一次初始化,使得DMA各成员与上面的参数一致。


 


  /*DMA Enable*/


  DMA_Cmd(DMA_Channel1,ENABLE);


  哈哈哈!这一句我想我就不罗嗦了,大家一看就明白。


 


至此,整个DMA总算设置好了,但是,DMA通道又是怎样与外设联系在一起的呢?哈哈,这也是我当初最想知道的一个事情,别急!容我想喝口茶~~~~~~哈哈哈!


要使DMA与外设建立有效连接,这不是DMA自身的事情,是各个外设的事情,每个外设都有 一个xxx_DMACmd(XXXx,Enable )函数,如果使DMAADC建立有效联系,就使用ADC_DMACmd(ADC1,Enable); (这里我启用了ADC中的ADC1模块)。


 


路过

鸡蛋

鲜花

握手

雷人

发表评论 评论 (2 个评论)

回复 eydj2008 2010-8-16 13:13
写得好 不过 这里 你理解错了 “ DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address; 上面语句中的DMA_InitStructure是一个DMA结构体,在库中有声明了,当然使用时就要先定义了;DMA_PeripheralBaseAddr是该结构体中一个数据成员,给DMA一个起始地址,好比是一个buffer起始地址,数据流程是:外设寄存器à DMA_PeripheralBaseAddàmemory中变量空间(或flash中数据空间等),ADC1_DR_Address是我定义的一个地址变量;” ADC1_DR_Address 应该是外设地址才对 也就是源数据地址 说通俗一点就是从这个地址取数据 比如: #define ADC1_DR_Address ((u32)0x4001244C) //ADC1_BASE 0x4001 2400 详见技术手册P139 #define ADC3_DR_Address ((u32)0x40013C4C)
回复 eydj2008 2010-8-16 13:15
而这个地址是死的 一个外设对应一个地址 不是你自己定义的一个地址变量