原创 [参加STM32博客大赛] 浅谈STM32的DMA模块的使用

2008-6-5 18:33 11946 4 14 分类: MCU/ 嵌入式

              


                浅谈STM32DMA模块的使用<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />


                              By <?xml:namespace prefix = st1 ns = "urn:schemas-microsoft-com:office:smarttags" />Jack Chang


                  Thu Jun 5 2008  18:30:49


                 E-mail:zhang-weihong@126.com QQ:179748613


前言:接触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模块)。


 


下面就以我的一个实例来和大家一起学习。这个实例仿真是成功的,我使用的是IAR Embedded Workbench IDE(v4.42) Manley Mini-kit评估板+ST-Link II来调试,是一个USART+ADC+DMA的例子。


首先,按照相关的资料提示,建立好项目路径、拷贝库文件和相关必要的文件,并建立一个新工程;还有就是设置好IAR IDE的相关设置。一切准备就绪后就开始修改相关的文档:


 conf.h文档的修改是根据应用中有启用的相关功能Module。如下图:


ADC模块的启用设置:


<?xml:namespace prefix = v ns = "urn:schemas-microsoft-com:vml" />


点击看大图 


DMA模块的启用设置:



点击看大图 


USART模块启用的设置:


点击看大图


 



其他的如flashGPIORCCNVICHSE等就不一一举例,因为这些几乎每个工程都要用到。


   


再回头看看我的main.c文档,


下图是相关的私有定义


点击看大图


 



USART的配制如下(96008 N1):


点击看大图


 



DMA的配置如下:



点击看大图 


ADC的配置如下图(采用了2A/D多通道的ADC模式):



点击看大图 


给各个模块配置的时钟使能如下图;在此想罗嗦一句就是在这个位置的设置我没有给DMA配置时钟,造成DMA功能无法正常使用,也就是前面所有说的与我原先的编程风格有较大的变化的原因,想说一句:在项目中所使用的所有功能模块都要在此配置时钟才能正常工作。



 点击看大图


 

 


GPIO的配置



点击看大图 


如果有开中断子程序,请在在NVIC中配置,还可以定义每个中断子程序的优先级别,由于我的工程没有用到,在此就不罗嗦了。


 


我项目应用到的文件目录如下图:



45510a7a-2732-4ec3-a8fe-36c12b448571.JPG 


我的项目仿真结果如下;PA0GND间接了一个1.5V的干电池,其结果是如中ADC_ConvertedValue[0]的值;PA1悬空,未接模拟信号。



点击看大图 


我的硬件仿真平台如下2图:红色的是我AVR mega16L Mini开发板,主要是利用了其中的POWERRS232



  点击看大图 


    点击看大图


 


PCUSART数据采集结果:(图中的显示是我将ADC转换后的值给字符串化了)


点击看大图



ST-link II仿真窗口的测试结果与USART采集到的数据结果差异说明:由于Manley stme32 Mini-kit评估板套件无法给ADCUSART同时组合仿真,我是分别分开仿真和USART数据传输的,所以出现差异。


 


心得  1、这次学习STM32MCU,最大的心得是改变我原有的编程思维,原来学习AVR的时候,文件库需要自己写,属于底层操作,如要自己写直接操作各外设的寄存器,复杂的还需要自己写设置函数,对底层要有非常的清楚的认识,编写的代码的工作量也比较大;稍有不慎,容易把寄存器弄错;调试也会花费较大的时间。而STM32的编程风格给我一个全新的视角(哈哈!别拍砖,虽然属菜鸟级别,但比较容易接受新的东西),感觉只要按其提供的资料建立起工程的基本构架,很多都是ST公司提供封装好的接口(类似windowsAPI),直接调用接口和做少许修改就可以完成你想要的工程结果。


2、  清晰的工程构架:ST提供一些工程案例的基本构架,只要熟悉这些构架,对构架做一些适当的修改,你将很轻松的完成你的任务,开发周期也可以缩短很多。请记住:牢记工程构架;


3、  哈哈!不想罗嗦的是STM32的速度和丰富的外设。(再罗嗦可能会被成批的砖头拍死,哈哈哈哈!)


4、  更有信心用STM32做其他更复杂一些的项目。祝大家好运!(别拍砖头呀!哈哈!)

PARTNER CONTENT

文章评论10条评论)

登录后参与讨论

用户441812 2014-8-30 09:16

DMA_DeInit(DMA_Channel1);不是设置,是复位操作 参数的单位有三种类型:Byte、HalfWord、word;恐怕你也不知道为什么要这样设置吧

用户377235 2013-7-28 10:47

ADC1_DR_Address是我定义的一个地址变量;这句的确说的有点不对(当然我看你后面设置的是内存到内存的传输,那么可能你可以自己设定,但前提是你要确保你设置的地址是没被占用这个是非常重要的),举个ADC采样的例子来说,那他的外设地址就是对应ADC转化后的地址,所以就是说这个地址是固定的,并非你随意设置的,第二就是那个“DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; 上面的这句是设置DMA的传输方向,就如前面我所说的,DMA可以双向传输,也可以单向传输,这里设置的是单向传输,如果需要双向传输:把DMA_DIR_PeripheralSRC改成DMA_DIR_PeripheralDST即可。”他是设置的是外设是作为数据的来源还是目的地的,这个固件库函数有说明,第三是“DMA_InitStructure.DMA_Mode = DMA_Mode_Circular”当你设置为循环模式传输时你的数据传输就不能配置为内存到内存的这一点固件库也有提醒。不过LZ花费那么多的精力去逐条解释这么多东西,实在令小菜鸟我佩服,虽有不妥之处,但必须赞赞赞

用户377235 2013-7-28 09:48

赞赞赞

用户349388 2012-3-8 11:06

1.“ADC1_DR_Address是我定义的一个地址变量”,这句话有误,这个地址在确定了外设之后是固定的,ADC1_DR_Address是ADC1转换结果寄存器地址,不是自己定义的,很容易误导人。 2.“如果需要双向传输:把DMA_DIR_PeripheralSRC改成DMA_DIR_PeripheralDST即可。”这句话有误,DMA_DIR_PeripheralDST是从内存传输到外设,没有什么双向一说。 3.“如果DMA选用的通道(CHx)有多个外设连接,需要使用外设递增模式”这句话有误,地址只是地址的地址,在ADC1_DR_Address基础上递增。 4.“DMA访问多个内存参数时,需要使用DMA_MemoryInc_Enable”,有误,不是什么多个内存参数,通常是传输到一个内存中的数组,传完一个数据后,数组的地址指针递增

用户468654 2011-7-10 19:44

“上面的这句是设置DMA的传输方向,就如前面我所说的,DMA可以双向传输,也可以单向传输,这里设置的是单向传输,如果需要双向传输:把DMA_DIR_PeripheralSRC改成DMA_DIR_PeripheralDST即可。”这句是不是有些问题,SRC应该指的是源,DST应该指的是目的吧?

用户24219 2009-6-23 09:27

写的一般,没什么深度!

用户95687 2009-5-22 14:12

分析的透彻!!! 俺理解了!!多谢博主分享!!

用户1359682 2008-11-10 13:53

男性青年随着性器官的发育,睾丸产生精子,前列腺和精囊等分泌精浆,两者合成精液,精液内主要成分为是水分和和少量有机成分。在100毫升精液内氨基酸1.25g,果糖0.1-0.5g, 糖类0.1g以及微量脂肪和无机盐,男子每次射精约2-6毫升,正常情况下,当精液达到一定量后,体内已无处可容纳,即所谓“满则溢”。其排出体外有4条途径,即遗精、手淫、性交和自流,所以正常男性,每周2-4次的排精发生,并不会影响身体,反之,较长时间的没有排精,会产生性情烦躁,记忆力不集中等不适感。

用户1327154 2008-8-1 17:40

学习中

用户155862 2008-7-10 13:56

我采用STM32F103系列, 串口DMA接收与DMA发送, 每隔多次通讯后, DMA发送总会少一个字符,而且少掉的字符会出现在一帧的任意位置, 导致PC机无法接收完整的一帧而报错. 后来把DMA发送改成串口中断发送,就解决了这一个问题. 我是采用BUSHOUND进行实时监控, 同时PC机在报错时进行查看后发现该问题,有没有高手指导一下。
相关推荐阅读
用户135981 2008-06-06 12:52
STM32方案:虚拟示波器
STM32项目/学习计划表项目名称 虚拟示波器请点击以下链接更新您的个人资料(包括“职位,部门,单位名称,地址,邮编,电子邮箱,电话”),以便我们能及时联系您并快速发送开发套件。http://spac...
EE直播间
更多
我要评论
10
4
关闭 站长推荐上一条 /3 下一条