STM32 DMA 应用之(一)SRAM 与flash 间数据传输
一、为什么要用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 !!!rn",sizeof(" Same !!!rn")); } else { USART1_SendData(" different !!!rn",sizeof(" different !!!rn")); } 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; 证实了是相同的; (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |