1.什么是DMA?
DMA:Direct Memory Access,直接存储器存取。是一种计算机系统中用于高效地实现数据传输的技术。DMA允许外设设备(如硬盘、显卡、网络适配器等)直接访问主内存,而不需要CPU的干预。
2.DMA的作用
DMA传输是将数据从一个地址空间搬运到另一个地址空间,提供在外设和存储器之间或者存储器和存储器之间的高速数据传输。这种数据传输无需CPU的干预,因此CPU可以专注于处理其他复杂的事物,提高了CPU的利用率。DMA最常见的用法是配合ADC的扫描模式,因为ADC规则组扫描有数据覆盖的缺陷。
3.DMA通道
STM32系列最多有12个独立可配置的通道: DMA1(7个通道), DMA2(5个通道);每个通道都支持软件触发和特定的硬件触发(红框)。
DMA请求映象
在同一个DMA模块上,多个请求间的优先权可以通过软件编程设置(共有四级:很高、高、中等和低),优先权设置相等时由硬件决定(请求0优先于请求1,依此类推)。
4.DMA框图
下面对该框图进行详细解释:
线路1是外设发送DMA请求,请求DMA搬运外设相关的数据;线路2是DMA搬运数据的过程,例如:将外设寄存器的数据搬运到RAM中
DMA的工作流程:
1.首先外设向DMA发送请求;
2.DMA控制器收到请求,触发DMA工作;
3.DMA控制器从AHB外设获取ADC采集的数据,存储到DMA通道中;
4.DMA控制器的DMA总线与总线矩阵协调,使用AHB把外设ADC采集的数据经由DMA通道存放到SRAM中。这个数据的传输过程中,完全不需要内核的参与,也就是不需要CPU的参与,实现了硬件自动化。绿色箭头代表了整体的线路流程。
1.DMA请求
如果外设想使用DMA搬运数据,必须先向DMA发送DMA请求,DMA接收到请求信号后,会返回给外设一个应答信号,当外设应答且DMA接收到外设的应答信号后,就会启动DMA传输,直到传输完毕。
2.通道
DMA有12个独立的可编程通道,DMA1有7个通道,DMA2有5个通道,不同的DMA通道对应着不同的外设请求,虽然每个DMA通道可以接收外设的DMA请求,但是同一时间只能接收一个。各个通道可以分别设置他们转运的源地址与目的地址,所以各个通道可以独立的工作。
3.仲裁器
当有多个DMA请求时,就会有先后响应的问题,这就由仲裁器管理, 管理DMA的优先级。
注:总线矩阵也有个仲裁器,如果DMA与CPU都想访问同一个目标,那么DMA就会暂停CPU的访问,防止发生冲突,不过总线仲裁器,仍然会保证CPU得到一般的带宽,使CPU能正常工作。这就是仲裁器的作用(调度各个通道)。
Cortex-M3内核:包含CPU、内核外设等。
Dcode总线:专门访问Flash
系统总线:访问其他东西
AHB从设备:就是DMA的寄存器。配置DMA参数。CPU可以通过CPU→系统→总线矩阵→AHB从设备;配置DMA。
DMA请求:就是DMA硬件触发源。
Flash接口控制器:因为Flash是只读的,不能写,配置Flash接口,可以向Flash写入数据
SRAM:是运行内存,读写都没问题
5.DMA简化结构图
起始地址:决定了方向(从哪里来到哪里去)
数据宽度:指定一次转运要按照多大的数据宽度来进行,数据宽度如下:
字节(Byte)8位、半字(HalfWord)16位、字(Word)32位
地址是否自增:指定一次转运完成之后,下一次转运,是不是要把地址移动到下一个位置去,相当于是指针p++的意思,比如ADC扫描模式。
传输计数器:指定总共需要转运几次。是一个自减计数器,比如写个5,DMA只能进行5次数据转运,转运过程中,每转运一次,计数器就自动减1,当传输计数器减到0之后,DMA就不会再进行数据转运了,另外,减到0之后,之前自增的地址,也会恢复到起始地址的位置。以方便之后DMA开始新一轮的转运。如果转运完成,传输计数器清0,这时再想给传输计数器赋值的话,先DMA失能-→写传输计数器→DMA使能。
自动重装器:传输计数器减到0之后,是否恢复到最初的值,如果不用自动重装器,转运一次,DMA就结束了。如果使用自动重装器,计数器减到0后,就会立即重装到初始值5。自动重装器决定了转运的模式:单次模式or循环模式。
DMA的触发控制:触发,就是决定DMA需要在什么时机进行转运的。触发源有硬件触发or软件触发。具体选择哪个触发源,由M2M(Memory to Memory)参数决定。
软件触发与循环模式不能同时使用,因为软件触发就是想把计数器清零,循环模式是清零后自动重装。如果同时用的话,DMA就停不下来了。
2:硬件触发:通过硬件的触发源触发DMA转运。
注:DMA的每个通道都有一个数据选择器,可以选择软件触发or硬件触发。每个通道的硬件触发源都是不同的。如果是软件触发的话,选择哪个通道都无所谓了。
6.数据转运+DMA
该图展示了源地址与目的地址都自增的情况,通过软件触发DMA转运。外设与存储器每进行一次转运后,地址都对应++。
7.ADC扫描模式+DMA
左边有7个通道,触发一次后,七个通道依次进行AD转换,转换结果都放到ADC_DR寄存器中,我们要做的是,在每个单独的通道转换完成之后,进行一次DMA数据转运,并且目的地址进行自增,这样数据就不会被覆盖了。
ADC如果是单次扫描,那DMA的传输计数器可以不自动重装,转换一轮就停止;如果ADC是连续扫描,那DMA就可以使用自动重装,在ADC启动下一轮转换的时候,DMA也启动下一轮转运,ADC与DMA同步工作。
8.DMA初始化结构体
typedef struct
{
uint32_t DMA_PeripheralBaseAddr; // 外设地址
uint32_t DMA_MemoryBaseAddr; // 存储器地址
uint32_t DMA_DIR; // 传输方向
uint32_t DMA_BufferSize; // 传输数目
uint32_t DMA_PeripheralInc; // 外设地址增量模式
uint32_t DMA_MemoryInc; // 存储器地址增量模式
uint32_t DMA_PeripheralDataSize; // 外设数据宽度
uint32_t DMA_MemoryDataSize; // 存储器数据宽度
uint32_t DMA_Mode; // 模式选择
uint32_t DMA_Priority; // 通道优先级
uint32_t DMA_M2M; // 存储器到存储器模式
} DMA_InitTypeDef;
1) DMAPeripheralBaseAddr: 外设地址,设定DMACPAR寄存器的值;一般设置为外设的数据寄存器地址,如果是存储器到存储器模式则设置为其中一个存储器地址。
2) DMAMemory0BaseAddr: 存储器地址,设定DMACMAR寄存器值;一般设置为我们自定义存储区的首地址。
3) DMADIR: 传输方向选择,可选外设到存储器、存储器到外设。它设定DMACCR寄存器的DIR[1:0]位的值。这里并没有存储器到存储器的方向选择,当使用存储器到存储器时,只需要把其中一个存储器当作外设使用即可。
4) DMABufferSize: 设定待传输数据数目,初始化设定DMACNDTR寄存器的值。
5) DMAPeripheralInc: 如果配置为DMAPeripheralIncEnable,使能外设地址自动递增功能,它设定DMACCR寄存器的PINC位的值;一般外设都是只有一个数据寄存器,所以一般不会使能该位。
6) DMAMemoryInc: 如果配置为DMAMemoryIncEnable,使能存储器地址自动递增功能,它设定DMACCR寄存器的MINC位的值;我们自定义的存储区一般都是存放多个数据的,所以要使能存储器地址自动递增功能。
7) DMAPeripheralDataSize: 外设数据宽度,可选字节(8位)、半字(16位)和字(32位),它设定DMACCR寄存器的PSIZE[1:0]位的值。
8) DMAMemoryDataSize: 存储器数据宽度,可选字节(8位)、半字(16位)和字(32位),它设定DMACCR寄存器的MSIZE[1:0]位的值。当外设和存储器之间传数据时,两边的数据宽度应该设置为一致大小。
9) DMAMode: DMA传输模式选择,可选一次传输或者循环传输,它设定DMACCR寄存器的CIRC位的值。例程我们的ADC采集是持续循环进行的,所以使用循环传输模式。
10) DMAPriority: 软件设置通道的优先级,有4个可选优先级分别为非常高、高、中和低,它设定DMACCR寄存器的PL[1:0]位的值。DMA通道优先级只有在多个DMA通道同时使用时才有意义,如果是单个通道,优先级可以随便设置。
11) DMAM2M: 存储器到存储器模式,使用存储器到存储器时用到,设定DMACCR的位14 MEN2MEN即可启动存储器到存储器模式。
9.DMA代码
#include "stm32f10x.h" // Device header
uint16_t MyDMA_Size;
void MyDMA_Init(uint32_t AddrA,uint32_t AddrB,uint16_t Size)
{
MyDMA_Size = Size;
//第一步:RCC开启DMA时钟
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);
//第二步:调用DMA_Init,初始化DMA各个参数
DMA_InitTypeDef DMA_InitStructure;
//外设站点的起始地址、数据宽度、是否自增
DMA_InitStructure.DMA_PeripheralBaseAddr = AddrA;
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Enable;
//存储器的起始地址、数据宽度、是否自增
DMA_InitStructure.DMA_MemoryBaseAddr = AddrB;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
//缓冲区大小(传输计数器)
DMA_InitStructure.DMA_BufferSize = Size;
//传输方向
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
//选择是否是存储器到存储器(硬件出发还是软件触发)
DMA_InitStructure.DMA_M2M = DMA_M2M_Enable;
//传输模式(是否使用自动重装)
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
//优先级
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_Init(DMA1_Channel1,&DMA_InitStructure);
//给DMA通电
DMA_Cmd(DMA1_Channel1,DISABLE);
}
//给传输计数器重新赋值
void MyDMA_Transfer(void)
{
DMA_Cmd(DMA1_Channel1,DISABLE); //修改计数时,需要先把DMA_Cmd失能
DMA_SetCurrDataCounter(DMA1_Channel1,MyDMA_Size);//设置计数器的值
DMA_Cmd(DMA1_Channel1,ENABLE);//使能
while(DMA_GetFlagStatus(DMA1_FLAG_TC1) == RESET); //等待转运完成
DMA_ClearFlag(DMA1_FLAG_TC1);//清除标志位(需要软件清除)
}
以上代码是DMA初始化部分,可直在main函数中调用,实现自己想要的功能。