STM32 DMA

DMA简介:

• DMA(Direct Memory Access)直接存储器存取
• DMA可以提供外设和存储器或者存储器和存储器1之间的高速数据传输,无须CPU干预,节省了CPU的资源
• 12个独立可配置的通道2: DMA1(7个通道), DMA2(5个通道)
• 每个通道都支持软件触发和特定的硬件触发

• STM32F103C8T6 DMA资源:DMA1(7个通道)

1:外设寄存器实际上也是存储器,上面的外设和存储器或者存储器和存储器,本质上都是存储器之间数据转运,说成是外设和存储器只不过是STM32特别指定了可以转运外设的存储器而已
2:通道是数据转运的路径,从一个地方移动到另一个地方,就需要占用一个通道。如果有多个通道进行转运,那它们可以各转各的,互不干扰。这就是DMA的通道。

STM32存储器映象:


ROM是只读存储器,是一种非易失性,掉电不丢失的存储器
RAM是随机存储器,是一种易失性,掉电丢失的存储器
在这里插入图片描述
• 用灰色填充的是保留区域,没有使用
• 零地址是没有存储器的,但因为程序是从零地址执行的,所以需要把我们想要执行的程序,映射到零地址来,如果映射到Flash区,就是从Flash执行。如果映射到系统存储器区,就是从系统存储器区运行BootLoader。如果映射到SRAM,就是从SRAM运行。由BOOT0和BOOT1两个引脚选择。
在这里插入图片描述

DMA框图:

请添加图片描述

• 寄存器:寄存器是一种SRAM存储器,是特殊的存储器,一方面,CPU可以对寄存器进行读写,就像读写运行内存一样。另一方面,寄存器每一位背后都连接了一根导线,可以控制外设电路的状态。
请添加图片描述
可以看出,寄存器是硬件之间软件的桥梁,软件读写寄存器,就相当于控制硬件的执行。

这样,用DMA进行数据转运可以归为一类问题,就是从某个地址取内容,再放到某个地址上。

为了高效访问存储器,设计了一个总线矩阵,矩阵左面是主动单元,右面是被动单元。主动单元拥有存储器得访问权,被动单元的存储器只能被主动单元读写。

• 主动单元的总线:

DC_code:专门访问Flash
系统总线:访问除Flash的其他存储器
DMA:供各DMA使用

DMA的通道:

各个通道可以分别设置它们转运数据的源地址和目的地址,这样它们就可以各自独立得工作了

DMA内部的仲裁器:

DMA的各通道虽然可以独立转运数据,但最终一个DMA外设只有一条DMA总线,所以各通道只能分时复用这一条DMA总线。如果发生了冲突,那么就会由仲裁器,根据通道的优先级,决定谁先用,谁后用。

总线矩阵的仲裁器:

当DMA和CPU要同时访问同一个目标时,那么DMA会暂停CPU的访问,以防止冲突。但总线仲裁器仍会保证CPU得到一半的总线带宽,使CPU可以正常工作。

Flash的读写:

Flash是ROM的一种,通过总线直接访问它,无论是CPU还是DMA,都是只读的(只能读取数据,而不能写入)。如果DMA的目的地址填了Flash的区域,转运时就会出错。但Flash并不是一定不可写入,我们可以通过配置Flash接口控制器,对Flash进行写入。

DMA基本结构:

在这里插入图片描述
传输计数器的写入必须在DMA关闭时(DMA_cmd失能)。

DMA转运数据的条件:
• 开关控制,DMA_cmd必须使能
• 传输计数器大于零
• 触发源,必须有触发信号。

DMA请求:

DMA请求
其它触发源详见参考手册“DMA请求映像”

软硬件触发源的选择:

软硬件触发源的选择由EN和MEM2MEM位决定。
EN位控制数据选择器得工作与否:EN=0时不工作,EN=1时工作。
MEM2MEM位控制使用软件触发源或硬件触发源:MEM2MEM=1时使用软件触发源,MEM2MEM=1时使用硬件触发源。

硬件触发源:

硬件触发源的对应通道:

因为各通道的硬件触发源都不同,所以要使用某个硬件触发源时,就必须使用它对应的通道。

硬件触发源的DMA输出:

因为一个通道有多个硬件触发源,使用哪个硬件触发源由对应的外设是否启用了DMA输出,如果一个通道的多个硬件触发源都启用了DMA输出,那么每一个硬件触发源都可以触发。

硬件触发源的优先级:
默认优先级,对应通道号越小,优先级越高。

数据对齐:

在这里插入图片描述

DMA配置基本步骤:

第一步:开启DMA的RCC时钟。
第二步:使用DMA_Init初始化。
如果使用得是硬件触发,要开启外设DMA触发信号输出
第三步:调用DMA_cmd开启DMA

DMA相关函数:

DMA通道配置函数:

DMAy_Channelx: y可以是12,用于选择DMA,x可以是1~7(对于DMA1)或1~5(对于DMA2)。
DMA_InitStruct: 一个DMA_InitTypeDef类型的指针
void DMA_Init(DMA_Channel_TypeDef* DMAy_Channelx, DMA_InitTypeDef* DMA_InitStruct)

DMA通道配置结构体:

	DMA_InitTypeDef DMA_InitStructure;//创建一个DMA_InitTypeDef 结构体
	DMA_InitStructure.DMA_PeripheralBaseAddr;//外设站点的起始地址
	DMA_InitStructure.DMA_PeripheralDataSize;//外设站点的数据宽度
	DMA_InitStructure.DMA_PeripheralInc;//外设站点的地址是否自增
	DMA_InitStructure.DMA_MemoryBaseAddr;//存储器站点的起始地址
	DMA_InitStructure.DMA_MemoryDataSize;//存储器站点的数据宽度
	DMA_InitStructure.DMA_MemoryInc;//存储器站点的地址是否自增
	DMA_InitStructure.DMA_DIR;//传输方向
	DMA_InitStructure.DMA_BufferSize;//传输次数(传输计数器)
	DMA_InitStructure.DMA_Mode;//是否使用自动重装(不能用于软件触发)
	DMA_InitStructure.DMA_M2M;//选择硬件触发还是软件触发
	DMA_InitStructure.DMA_Priority;//通道优先级
成员可选值
DMA_PeripheralBaseAddr和DMA_MemoryBaseAddr一个32位的地址
DMA_PeripheralDataSizeDMA_PeripheralDataSize_Byte//外设站点的数据宽度为8bits
DMA_PeripheralDataSize_HalfWord//外设站点的数据宽度为16bits
DMA_PeripheralDataSize_Word//外设站点的数据宽度为32bits
DMA_PeripheralIncDMA_PeripheralInc_Enable//外设站点的地址自增
DMA_PeripheralInc_Disable//外设站点的地址不自增
DMA_MemoryDataSizeDMA_MemoryDataSize_Byte//存储器站点数据宽度为8bits
DMA_MemoryDataSize_HalfWord//存储器站点数据宽度为16bits
DMA_MemoryDataSize_Word//存储器站点数据宽度为32bits
DMA_MemoryIncDMA_MemoryInc_Enable//存储器站点的地址自增
DMA_MemoryInc_Disable//存储器站点的地址不自增
DMA_DIRDMA_DIR_PeripheralDST//将外设站点设为传输目的地(传输方向是从存储器站点(DMA_MemoryBaseAddr )到外设站点(DMA_PeripheralBaseAddr))
DMA_DIR_PeripheralSRC//将外设站点设为传输源头(传输方向是从外设站点(DMA_PeripheralBaseAddr)到存储器站点(DMA_MemoryBaseAddr ))
DMA_BufferSize0~65535
DMA_ModeDMA_Mode_Circular//使用自动重装
DMA_Mode_Normal//不使用自动重装
DMA_M2MDMA_M2M_Enable//使用软件触发
DMA_M2M_Disable//使用硬件触发(不使用软件触发)
DMA_PriorityDMA_Priority_VeryHigh//非常高
DMA_Priority_High//高
DMA_Priority_Medium//中等
DMA_Priority_Low//低

DMA传输计数器赋值:

DMAy_Channelx: y可以是1或2,用于选择DMA,x可以是1~7(对于DMA1)或1~5(对于DMA2)。
DataNumber: 要对传输计数器写入的值
这个函数必须在DMA通道关闭时使用。                
void DMA_SetCurrDataCounter(DMA_Channel_TypeDef* DMAy_Channelx, uint16_t DataNumber)

DMA查询传输计数器值函数:

DMAy_Channelx: y可以是12,用于选择DMA,x可以是1~7(对于DMA1)或1~5(对于DMA2)。
返回指定通道传输计数器的剩余未传输值
uint16_t DMA_GetCurrDataCounter(DMA_Channel_TypeDef* DMAy_Channelx)

DMA检查标志位函数:

检查指定通道指定标志是否已设置。
DMAy_FLAG: 指定要检查的标志位
返回SET(已设置) 或RESET(未设置)
FlagStatus DMA_GetFlagStatus(uint32_t DMAy_FLAG)

清除指定标志位
DMAy_FLAG: 指定要清除的标志位
DMA_ClearFlag(uint32_t DMAy_FLAG);


例子

DMA转运数组

示意图:

在这里插入图片描述

代码:

//待传输的数据个数
uint16_t My_DMASize;
//把AddrA为启始地址里的size个的数据转运到以AddrB为起始地址的储存里
void MyDMA_Init(uint32_t AddrA,uint32_t AddrB,uint16_t size)
{
	My_DMASize=size;	//待传输的数据为size个
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);//开启DMA的RCC时钟(DMA1挂在AHB总线上)。
	DMA_InitTypeDef DMA_InitStructure;//创建一个DMA_InitTypeDef初始化结构体
	DMA_InitStructure.DMA_PeripheralBaseAddr=AddrA;//设置AddrA为外设基地址
	DMA_InitStructure.DMA_PeripheralDataSize=DMA_PeripheralDataSize_Byte;//外设站点的数据宽度为8bits
	DMA_InitStructure.DMA_PeripheralInc=DMA_PeripheralInc_Enable;//外设站点的地址自增
	DMA_InitStructure.DMA_MemoryBaseAddr=AddrB;//设置AddrB为存储器基地址
	DMA_InitStructure.DMA_MemoryDataSize=DMA_MemoryDataSize_Byte;//存储器站点的数据宽度为8bits
	DMA_InitStructure.DMA_MemoryInc=DMA_MemoryInc_Enable;//存储器站点的地址自增
	DMA_InitStructure.DMA_DIR=DMA_DIR_PeripheralSRC;//将外设站点设为传输源头
	DMA_InitStructure.DMA_BufferSize=size;//待传输的数据为size个
	DMA_InitStructure.DMA_Mode=DMA_Mode_Normal;//不使用自动重装
	DMA_InitStructure.DMA_M2M=DMA_M2M_Enable;//使用软件触发
	DMA_InitStructure.DMA_Priority=DMA_Priority_Medium;//优先级中等
	DMA_Init(DMA1_Channel1,&DMA_InitStructure);//使用DMA_Init初始化。

	DMA_Cmd(DMA1_Channel1,DISABLE);//先不使能,需要时再使能
}
//开始DMA传输
void MyDMA_Transfer()
{
	DMA_Cmd(DMA1_Channel1,DISABLE);//DMA失能,准备写入传输计数器
	DMA_SetCurrDataCounter(DMA1_Channel1,My_DMASize);//将DMA1的通道1的传输计数器赋值为My_DMASize
	DMA_Cmd(DMA1_Channel1,ENABLE);//DMA使能,开始传输
	while(DMA_GetFlagStatus(DMA1_FLAG_TC1)==RESET);//等待传输完成
	DMA_ClearFlag(DMA1_FLAG_TC1);//清除传输完成标志位
}

ADC扫描模式+DMA

示意图:

在这里插入图片描述

代码

uint16_t DA_Value[4];//转换结果存放数组
//DMA和ADC初始化
void AD_Init()
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);//开启ADC1的时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);//开启GPIOA的时钟
	RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE);//开启DMA1的时钟
	
	RCC_ADCCLKConfig(RCC_PCLK2_Div6); //选择时钟6分频,ADCCLK = 72MHz / 6 = 12MHz
	/*GPIO初始化*/
	GPIO_InitTypeDef GPIO_InitStruct;
	GPIO_InitStruct.GPIO_Mode=GPIO_Mode_AIN;
	GPIO_InitStruct.GPIO_Pin=GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3;
	GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStruct);//将PA0、PA1、PA2和PA3引脚初始化为模拟输入
	
	ADC_RegularChannelConfig(ADC1,ADC_Channel_0,1,ADC_SampleTime_55Cycles5);//规则组序列1的位置,配置为通道0
	ADC_RegularChannelConfig(ADC1,ADC_Channel_1,2,ADC_SampleTime_55Cycles5);//规则组序列2的位置,配置为通道1
	ADC_RegularChannelConfig(ADC1,ADC_Channel_2,3,ADC_SampleTime_55Cycles5);//规则组序列3的位置,配置为通道2
	ADC_RegularChannelConfig(ADC1,ADC_Channel_3,4,ADC_SampleTime_55Cycles5);//规则组序列4的位置,配置为通道3
	
	ADC_InitTypeDef ADC_InitStructure;//定义结构体变量
	ADC_InitStructure.ADC_Mode=ADC_Mode_Independent;//模式,选择独立模式,即单独使用ADC1
	ADC_InitStructure.ADC_DataAlign=ADC_DataAlign_Right;//数据对齐,选择右对齐
	ADC_InitStructure.ADC_ExternalTrigConv=ADC_ExternalTrigConv_None;//外部触发,使用软件触发,不需要外部触发
	ADC_InitStructure.ADC_ContinuousConvMode=ENABLE;//连续转换,使能,每转换一次规则组序列后立刻开始下一次转换
	ADC_InitStructure.ADC_ScanConvMode=ENABLE;//扫描模式,使能,扫描规则组的序列,扫描数量由ADC_NbrOfChanne
	ADC_InitStructure.ADC_NbrOfChannel=4;//通道数,为4,扫描规则组的前4个通道
	ADC_Init(ADC1,&ADC_InitStructure);//将结构体变量交给ADC_Init,配置ADC1
	
	DMA_InitTypeDef DMA_InitStructure;//定义结构体变量
	DMA_InitStructure.DMA_PeripheralBaseAddr=(uint32_t)&ADC1->DR;//设置&ADC1->DR为外设基地址
	DMA_InitStructure.DMA_PeripheralDataSize=DMA_PeripheralDataSize_HalfWord;//外设站点的数据宽度为16bits(ADC数据寄存器为16位)
	DMA_InitStructure.DMA_PeripheralInc=DMA_PeripheralInc_Disable;//外设站点的地址不自增
	DMA_InitStructure.DMA_MemoryBaseAddr=(uint32_t)DA_Value;//设置DA_Value为存储器基地址
	DMA_InitStructure.DMA_MemoryDataSize=DMA_MemoryDataSize_HalfWord;//存储器站点的数据宽度为16bits
	DMA_InitStructure.DMA_MemoryInc=DMA_MemoryInc_Enable;//存储器站点的地址不自增
	DMA_InitStructure.DMA_DIR=DMA_DIR_PeripheralSRC;//将外设站点设为传输源头
	DMA_InitStructure.DMA_BufferSize=4;//待传输的数据为4个
	DMA_InitStructure.DMA_Mode=DMA_Mode_Circular;//使用自动重装
	DMA_InitStructure.DMA_M2M=DMA_M2M_Disable;//使用硬件触发
	DMA_InitStructure.DMA_Priority=DMA_Priority_Medium;//优先级中等
	DMA_Init(DMA1_Channel1,&DMA_InitStructure);//使用DMA_Init对DMA1通道1初始化。
	/*DMA和ADC使能*/
	DMA_Cmd(DMA1_Channel1,ENABLE);//DMA1的通道1使能
	ADC_DMACmd(ADC1,ENABLE);//ADC1触发DMA1的信号使能
	ADC_Cmd(ADC1,ENABLE);//ADC1使能
	/*ADC校准*/
	ADC_ResetCalibration(ADC1);
	while(ADC_GetResetCalibrationStatus(ADC1) == SET){}//固定流程,内部有电路会自动执行校准
	ADC_StartCalibration(ADC1);
	while(ADC_GetCalibrationStatus(ADC1) == SET){}
	/*ADC触发*/	
	ADC_SoftwareStartConvCmd(ADC1,ENABLE);//软件触发ADC开始工作,由于ADC处于连续转换模式,故触发一次后ADC就可以一直连续不断地工作
}

  • 20
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值