一、为什么要用DMA?
DMA 全称:Direct MemoryAccess 就是可以直接内存存取;
正是它可以直接操作内存所以具备以下优点:
而无需经过CPU去操作内存的存取,这样可以解放CPU出来干其他的事情;
因为他可以进行存储器时间的数据传输,而不需经过cpu,所以大大加快了数据传输速度—是一种高速的数据传输;
二.DMA有几种传输数据方式:
(1)内存到 内存之间的;即:SRAMßà SRAM
(2)内存到 外设之间的; (例如:串口收到的数据 从数据寄存器 à 内存)
(3) 外设到内存之间的;
三.传输的数据宽度是怎样的,数据是什么样的形式传输?DMA 能传输多大的数据量?
A. 数据源地址到数据目的地址 传输宽度或者说传输数据的形式,有几种:1)字节;2)半字;3)全字 [1字节=8bit 1半字=2字节=16位 1全字=2半字=4字节=32位]
B.传输的最大数据量是65536
我们来看一下stm32f103ve的数据手册第九章DMA 大概的了解一下DMA的特性:
四、怎样配置软件来使用DMA?
(1)配置dma
/*****************************************************************
*函数名称: Dma_Init
*功能描述: 利用DMA 把内存的数据 传输到flash 达到高速传输的目的
*
*输入参数:无
*返回值 :无
*其他说明:无
*当前版本:v1.0
*作者 :尹宣
*完成时间:2013年12月1日
*修改日期 版本号 修改人 修改内容
*-----------------------------------------------------------------
*
******************************************************************/
void Dma_Init(void)
{
DMA_InitTypeDef DMA_InitStructure;
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
/* DMA channel6 configuration */
DMA_DeInit(DMA1_Channel1);
DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)SRC_Const_Buffer; //外设地址
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)DST_Buffer; //内存地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //外设作为DMA的源端 DMA_DIR_PeripheralDST; //外设作为目的地址
DMA_InitStructure.DMA_BufferSize = BufferSize; //传输大小
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Enable; //外设地址增加
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //内存地址自增使能
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word; //内存存储方式:字节 DMA_MemoryDataSize_Word;//字(32位)
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Word;
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal; //DMA_Mode_Normal 正常模式,只传送一次; DMA_Mode_Circular:循环模式,不停的传送;
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_InitStructure.DMA_M2M = DMA_M2M_Enable;
DMA_Init(DMA1_Channel1, &DMA_InitStructure);
/* Enable DMA Channel1 Transfer Complete interrupt */
DMA_ITConfig(DMA1_Channel1, DMA_IT_TC, ENABLE);
/* Get Current Data Counter value before transfer begins */
CurrDataCounter= DMA_GetCurrDataCounter(DMA1_Channel1);
/* Enable DMA Channel6 transfer */
DMA_Cmd(DMA1_Channel1, ENABLE);
}
2.设置DMA优先级
/*****************************************************************
*函数名称: NVIC_Config
*功能描述: 配置DMA的中断优先级
*
*输入参数:无
*返回值 :无
*其他说明:无
*当前版本:v1.0
*作者 :尹宣
*完成时间:2013年12月1日
*修改日期 版本号 修改人 修改内容
*-----------------------------------------------------------------
*
******************************************************************/
void NVIC_Config(void)
{
NVIC_InitTypeDef NVIC_InitStructure;
/* Configure one bit for preemption priority -------------------------------- */
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);
/* Enable DMA channel1 IRQ Channel */
NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel1_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
}
3.在stm3210x_it.c 文件中添加 DMA中断处理函数
/*******************************************************************************
* Function Name : DMAChannel1_IRQHandler
* Description : This function handles DMA Stream 1 interrupt request.
* Input : None
* Output : None
* Return : None
*******************************************************************************/
void DMA1_Channel1_IRQHandler(void)
{
/* Test on DMA Channel1 Transfer Complete interrupt */
if(DMA_GetITStatus(DMA1_IT_TC1)) //DMA1_IT_TC1:通道1传输完成中断
{
/* Get Current Data Counter value after complete transfer */
CurrDataCounter= DMA_GetCurrDataCounter(DMA1_Channel1); //返回当前DMA通道1 剩余的待传输的数据数目
/* Clear DMA Channel1 Half Transfer, Transfer Complete and Global interrupt pending bits */
DMA_ClearITPendingBit(DMA1_IT_GL1); //清中断1全局中断
}
}
4主函数部分:
int main(void)
{
// int count;
uint32 judge;
NVIC_SetVectorTable(NVIC_VectTab_FLASH,0x00); // NVIC_VectTab_FLASH=0x08000000
// RCC_Config();
SysTick_Init();
GPIO_Config();
USART1_Init(19200);
NVIC_Config();
FLASH_SetLatency(FLASH_Latency_1); //延时2个时钟周期
/* Enable Prefetch Buffer --使能预取指缓存*/
FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
Dma_Init();
/* Get Current Data Counter value before transfer begins */
CurrDataCounter = DMA_GetCurrDataCounter(DMA1_Channel1); //返回当前DMA通道x, 剩余待传输数据数目
while( CurrDataCounter!=0) ;
judge = memcmp(SRC_Const_Buffer,DST_Buffer,BufferSize);
if(0==judge)
{
USART1_SendData(" Same !!!\r\n",sizeof(" Same !!!\r\n"));
}
else
{
USART1_SendData(" different !!!\r\n",sizeof(" different !!!\r\n"));
}
Delay_ms(1);
}
五、测试验证
1.我们先看看看下图
看点0:
问1:
使用内存窗口观测SRC_Const_Buffer和DST_Buffer所在的位置,可以发现SRC_Const_Buffer
地址为0x08001B24,,即Flash中;DST_Buffer地址为0x2000002c,即RAM中;
答1:
不能改变其值得变量(包括全局和局部)都是存储在FLASH中的,能改变的都储存在SRAM中
SRC_Const_Buffer 的定义:uc32 SRC_Const_Buffert
搜索了一下uc32的出处----
typedef const uint32_tuc32; /*!< Read Only */
DST_Buffer 的定义:u32DST_Buffer[BufferSize];
搜索了一下u32的出处----
Typedef uint32_t u32;
看点1:断点设置开始DMA传输前,
CurrDataCounter = DMA_GetCurrDataCounter(DMA1_Channel1);可以读出待传输的数据长度为0x20=32 跟我我们定义的待传输的数据长度是一样的;
看点2:开始传输前,目的数组DST_Buffer里面的全部为空即为:0x00;
2.传输完成时,CurrDataCounter 值已经为0,即待传输数据为0;同时目的数组 DST_Buffer 已经有数据,我们用对比了一下源数组 和目的数组 相同则返回0; 证实了是相同的;