历史上的今天
返回首页

历史上的今天

今天是:2024年10月21日(星期一)

正在发生

2018年10月21日 | STM32学习之:DMA详解(2)

2018-10-21 来源:eefocus

DMA部分我用到的相对简单,当然,可能这是新东西,我暂时还用不到它的复杂功能吧。下面用问答的形式表达我的思路。


DMA有什么用?


       直接存储器存取用来提供在外设和存储器之间或者存储器和存储器之间的高速数据传输。无须CPU的干预,通过DMA数据可以快速地移动。这就节省了CPU的资源来做其他操作。


有多少个DMA资源?


       有两个DMA控制器,DMA1有7个通道,DMA2有5个通道。


数据从什么地方送到什么地方?


       外设到SRAM(I2C/UART等获取数据并送入SRAM);


       SRAM的两个区域之间;


       外设到外设(ADC读取数据后送到TIM1控制其产生不同的PWM占空比);


       SRAM到外设(SRAM中预先保存的数据送入DAC产生各种波形);


       ……还有一些目前还搞不清楚的。


DMA可以传递多少数据?


       传统的DMA的概念是用于大批量数据的传输,但是我理解,在STM32中,它的概念被扩展了,也许更多的时候快速是其应用的重点。数据可以从1~65535个。


直接存储器存取(Direct Memory Access,DMA)是计算机科学中的一种内存访问技术。它允许某些电脑内部的硬体子系统(电脑外设),可以独立地直接读写系统存储器,而不需绕道 CPU。在同等程度的CPU负担下,DMA是一种快速的数据传送方式。它允许不同速度的硬件装置来沟通,而不需要依于 CPU的大量中断请求。【摘自Wikipedia】


现在越来越多的单片机采用DMA技术,提供外设和存储器之间或者存储器之间的高速数据传输。当 CPU 初始化这个传输动作,传输动作本身是由 DMA 控制器 来实行和完成。STM32就有一个DMA控制器,它有7个通道,每个通道专门用来管理一个或多个外设对存储器访问的请求,还有一个仲裁器来协调各个DMA请求的优先权。


DMA 控制器和Cortex-M3核共享系统数据总线执行直接存储器数据传输。当CPU和DMA同时访问相同的目标(RAM或外设)时,DMA请求可能会停止 CPU访问系统总线达若干个周期,总线仲裁器执行循环调度,以保证CPU至少可以得到一半的系统总线(存储器或外设)带宽。


在发生一个事件后,外设发送一个请求信号到DMA控制器。DMA控制器根据通道的优先权处理请求。当DMA控制器开始访问外设的时候,DMA控制器立即发送给外设一个应答信号。当从DMA控制器得到应答信号时,外设立即释放它的请求。一旦外设释放了这个请求,DMA控制器同时撤销应答信号。如果发生更多的请求时,外设可以启动下次处理。


总之,每个DMA传送由3个操作组成:


1. 从外设数据寄存器或者从DMA_CMARx寄存器指定地址的存储器单元执行加载操作。


2. 存数据到外设数据寄存器或者存数据到DMA_CMARx寄存器指定地址的存储器单元。


3. 执行一次DMA_CNDTRx寄存器的递减操作。该寄存器包含未完成的操作数目。




仲裁器根据通道请求的优先级来启动外设/存储器的访问。优先级分为两个等级:软件(4个等级:最高、高、中等、低)、硬件(有较低编号的通道比拥有较高编号的通道有较高的优先权)。


可以在DMA传输过半、传输完成和传输错误时产生中断。


STM32中DMA的不同中断(传输完成、半传输、传输完成)通过“线或”方式连接至NVIC,需要在中断例程中进行判断。


进行DMA配置前,不要忘了在RCC设置中使能DMA时钟。STM32的DMA控制器挂在AHB总线上。


DMA总共有7个通道,各个通道的DMA映射关系如下:




外设的事件连接至相应DMA通道,每个通道均可以通过软件触发实现存储器内部的DMA数据传输(M2M模式)


Tips:库2.0中函数RCC_AHBPeriphClockCmd的参数由“RCC_AHBPeriph_DMA”改成“RCC_AHBPeriph_DMA1”(如果是DMA1控制器的话)。


DMA的传输标志位(CHTIFx、CTCIFx、CGIFx)由硬件设置为“1”,但需要软件清零,在中断服务程序中清除。当CGIFx(全局中断标志位)清零后,CHTIFx 和 CTCIFx均清零。


 


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


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


DMA_DeInit(DMA_Channel1);


上面这句是给DMA配置通道,根据ST提供的资料,STM3210Fx中DMA包含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定义一个储存片区,这个参数的单位有三种类型:Byte、HalfWord、word,我设置的2个 half-word(见下面的设置);32位的MCU中1个half-word占16 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级:VeryHigh,High,Medium,Low.


DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;


上面的这句是设置DMA的2个memory中的变量互相访问的


DMA_Init(DMA_Channel1,&DMA_InitStructure);


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


DMA_Cmd(DMA_Channel1,ENABLE);


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


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


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


 


一个简单的例子 transfer  a word data buffer from FLASH memory to embedded SRAM memory.

在V3.1.2库的位置

STM32F10x_StdPeriph_Lib_V3.1.2\Project\STM32F10x_StdPeriph_Examples\DMA\FLASH_RAM



DMA_DeInit(DMA1_Channel6);

  //peripheral base address

DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)SRC_Const_Buffer;

  //memory base address   

DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)DST_Buffer;

  //数据传输方向    Peripheral is source               

DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;

//缓冲区大小 Number of data to be transferred (0 up to 65535).数据传输数目     

DMA_InitStructure.DMA_BufferSize = BufferSize;

   // the Peripheral address register is incremented       

DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Enable;

  //the memory address register is incremented

DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;

//the Peripheral data width       

DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word; 

DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Word;

DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;

DMA_InitStructure.DMA_Priority = DMA_Priority_High;

//the DMAy Channelx will be used in memory-to-memory transfer

//DMA通道的操作可以在没有外设请求的情况下进行,这种操作就是存储器到存储器模式。

DMA_InitStructure.DMA_M2M = DMA_M2M_Enable;   

DMA_Init(DMA1_Channel6, &DMA_InitStructure);



DMA_ITConfig(DMA1_Channel6, DMA_IT_TC, ENABLE);




DMA_Cmd(DMA1_Channel6, ENABLE);

=======================================================================


外设的DMA请求映像



要使DMA与外设建立有效连接,这不是DMA自身的事情,是各个外设的事情,每个外设都有 一个


xxx_DMACmd(XXXx,Enable )函数,如果使DMA与ADC建立有效联系,就使用 ADC_DMACmd


(ADC1,Enable); (这里我启用了ADC中的ADC1模块)。



DMA_DeInit(DMA1_Channel1);

DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;

DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)&AD_Value;   

//u16  AD_Value[2];   不加&应该也可以  数组名 代表地址

DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;

DMA_InitStructure.DMA_BufferSize = 2;      //############## 改了

DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;

DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;  //##############     改了

DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;

DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;

DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;

DMA_InitStructure.DMA_Priority = DMA_Priority_High;

DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;

DMA_Init(DMA1_Channel1, &DMA_InitStructure);



DMA_Cmd(DMA1_Channel1, ENABLE);



ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;

ADC_InitStructure.ADC_ScanConvMode = ENABLE;

ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;

ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;

ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;

ADC_InitStructure.ADC_NbrOfChannel = 2;      //##############     改了

ADC_Init(ADC1, &ADC_InitStructure);

//内部温度传感器  添加这一句 


ADC_TempSensorVrefintCmd(ENABLE);

//##############     改了


//################ Channel 10(电位器)

ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 1, ADC_SampleTime_13Cycles5);

//###### 内部温度传感器  Channel 16 ###################

ADC_RegularChannelConfig(ADC1, ADC_Channel_16, 2, ADC_SampleTime_55Cycles5);


  使能ADC1的DMA请求映像

  ADC_DMACmd(ADC1, ENABLE);



ADC_Cmd(ADC1, ENABLE);


   //使用之前一定要校准

ADC_ResetCalibration(ADC1);


while(ADC_GetResetCalibrationStatus(ADC1));



ADC_StartCalibration(ADC1);


while(ADC_GetCalibrationStatus(ADC1));


 

ADC_SoftwareStartConvCmd(ADC1, ENABLE);


推荐阅读

史海拾趣

Heimann Optoelectronics Gmbh公司的发展小趣事

2018年,Hei Inc Optoelectronic Division抓住行业整合的机遇,成功并购了一家在光电子材料领域具有领先地位的企业。这次并购不仅使公司获得了先进的光电子材料生产技术,还极大地丰富了公司的产品线。通过整合双方资源和技术优势,公司在光电子器件的研发和生产上实现了质的飞跃。并购后的Hei Inc Optoelectronic Division不仅在市场上占据了更大的份额,还进一步提升了自身的技术实力和品牌影响力。

GSR Technology Ltd公司的发展小趣事

随着技术的成熟和市场的认可,GSR Technology Ltd开始寻求国际合作机会,以进一步扩大其市场影响力。公司与欧洲一家知名雷达制造商达成战略合作,共同开发面向全球市场的先进GSR系统。这一合作不仅提升了公司的产品竞争力,还为公司打开了欧洲市场的大门。随后,公司逐步在北美、亚洲等地设立分支机构,实现了全球化布局。

BOCA公司的发展小趣事

随着技术的不断进步,BOCA公司意识到单一产品线的局限性。为了拓宽市场,公司决定进军消费电子领域。BOCA通过与知名品牌合作,推出了一系列搭载其先进技术的电子产品,如智能手机、平板电脑等。这些产品凭借其卓越的性能和独特的设计,迅速在市场上获得了良好的口碑,为BOCA带来了丰厚的利润。

Axiomtek公司的发展小趣事

为了进一步提升竞争力,BOCA公司开始尝试产业链整合。公司不仅加强了对上游原材料供应商的控制,还通过并购和合作的方式,将下游的组装、销售等环节也纳入自己的体系。这一举措使得BOCA能够更好地控制产品质量和交货期,同时也降低了成本,提高了整体运营效率。

Cotco公司的发展小趣事

在电子行业快速发展的浪潮中,Cotco公司于XXXX年正式成立,由几位在电子行业有着丰富经验的企业家和技术专家共同创立。初期,公司专注于电子元器件的分销和贸易,凭借创始团队成员在行业内的人脉和资源,Cotco很快与多家知名电子元器件制造商建立了合作关系,开始为客户提供优质的元器件采购服务。随着市场需求的不断增长,Cotco的业务也逐渐扩展,开始涉足电子产品的设计和生产领域。

Good Will Instrument Co., Ltd.公司的发展小趣事

随着产品线的不断丰富和市场需求的持续增长,Cotco公司开始加大市场拓展力度。公司积极参加国内外各类电子行业展览和交流活动,与潜在客户和合作伙伴建立了广泛的联系。同时,Cotco也注重品牌建设,通过赞助行业活动、开展公益事业等方式提升品牌知名度和美誉度。经过几年的努力,Cotco已经成为电子行业内一个响当当的名字。

问答坊 | AI 解惑

JZC-23F继电器

还有一个问题:这个继电器的用什么样的三极管来带动?上面有“10A/28VDC,5A/220VAC,DC5V“字样,怎样解释?它对吸合电流又没有要求?…

查看全部问答>

RS232 起始位 停止位 校验位 需要自己实现吗?

ModBus帧格式为:从机地址(1字节),功能码(1字节),起始地址(2字节),数据长度(2字节),CRC码(2字节) 然后数据传输方式规定为串行异步:1位起始位,8位数据位,1偶校验位,1位停止位。 我的疑问是: 1)CRC只对数据位进行校验,那就是 ...…

查看全部问答>

求一个万年历汇编程序

求一个万年历汇编程序,我用的IC:AT89S52,DS1302,及其液晶显示屏一块,因不是很懂单片机,哪位能否提供一下相应的程序及其电路制作给我参考,谢谢!…

查看全部问答>

EVC 的wince5.0模拟器的问题

   我2台机子EVC都装了EVC的WINCE5.0模拟器SDK ,一台机子上EVC呈现编译能在WINCE5模拟器跑起来,而将此程序复制到另外一台机子的WINCE5模拟器却跑不起来,提示Cannot open(or one of its components).Make sure the path and filen~~不 ...…

查看全部问答>

S3C44B0X板配什么LCD好?学习用的,价格一般

麻烦大家推荐一下,要是有卖的话直接卖给我吧,免得我东找西找,好辛苦~…

查看全部问答>

【求助】eVB读写文件时“语句未结束”?

源代码如下: Private Sub Form_Load()    Dim word, meaning As String        Open \"words.dat\" For Input Access Read As #1    Do While Not EOF(1)      Input #1, word ...…

查看全部问答>

帮忙看一句代码是什么意思...10分谢谢

typedef struct { unsigned char bit0 : 1; unsigned char bit1 : 1; unsigned char bit2 : 1; unsigned char bit3 : 1; unsigned char bit4 : 1; unsigned char bit5 : 1; unsigned char bit6 : 1; unsigned char bit7 : 1; } tpBIT; # ...…

查看全部问答>

C#移动项目PPC 2003,连接ACCESS在MSDN里有吗??

C#移动项目PPC 2003连接ACCESS在MSDN里有吗? 我找了半天没找到呀!谁能帮助帮助我!!!…

查看全部问答>

【求助】也是中断式按键

不知道为什么,我的程序也是在没有按键的时候就直接进入中断程序。下面是我的一个很简单的程序,大家帮忙找找原因,多谢! #include <msp430x44x.h> void main(void) { int i; WDTCTL=WDTPW+WDTHOLD; //停止看门狗 L ...…

查看全部问答>

【求助】有关无线通讯(实在没办法了)

我已经从21号弄到现在了,本来年前想弄一点效果出来,但这几天一直试却实在调不通,所以只好向大家求助了。 我计划做两块电路板之间的无线通讯,用的是149,通讯模块是prt2000。 电路版已做好,板内控制灯等元件都是灵的,但是两块板之间的控一直 ...…

查看全部问答>