问题:
1、DMA buffsize与实际采集波形对应数据量不一致。输出数据少了,且在BuffSize低于64,GPIO没有输出;问题出在哪里呢?
2、DMA 在输出结束中断后关闭DMA_Cmd(DMA2_Stream5,DISABLE);然后再main函数中修改memory中数据,最后开启DMA;
发现 输出数据达不到预期效果;问:该怎样实现DMA开关?
代码:- //宏定义 DMA 传送数据量
- #define BUFF_LEN 64
- //初始化GPIO+DMA2
- void Test_RAM2GPIO_DMA(void)
- {
- GPIO_InitTypeDef GPIO_InitStruct;
- TIM_TimeBaseInitTypeDef TIM_BaseInitStructure;
- DMA_InitTypeDef DMA_InitStructure;
- int len = sizeof(g_buff_disp)/8;
- int i;
- for(i=0; i<len; i+=8)
- {
- g_buff_disp = 0xffff & (~(1 << 13));
- g_buff_disp[i+1] = 0xffff | (1 << 13);
- g_buff_disp[i+2] = 0x0000 ;
- g_buff_disp[i+3] = 0x0000 | (1 << 13);
- g_buff_disp[i+4] = 0xffff & (~(1 << 13));
- g_buff_disp[i+5] = 0xffff;
- g_buff_disp[i+6] = 0xffff & (~(1 << 13));
- g_buff_disp[i+7] = 0xffff;
- }
- my_memcpy32(g_buff_DMA0, g_buff_disp, BUFF_LEN/2);
- //my_memcpy32(g_buff_DMA1, g_buff_disp, BUFF_LEN/2);
- RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1, ENABLE);
- RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA2, ENABLE);
- RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);
- //GPIO
- GPIO_InitStruct.GPIO_Pin = GPIO_Pin_All;
- GPIO_InitStruct.GPIO_Mode = GPIO_Mode_OUT;
- GPIO_InitStruct.GPIO_OType = GPIO_OType_PP;
- GPIO_InitStruct.GPIO_Speed = GPIO_Speed_100MHz;
- GPIO_Init(GPIOB, &GPIO_InitStruct);
-
- //DMA2 Stream6 channel2
- DMA_InitStructure.DMA_Channel = DMA_Channel_6;
- DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&(GPIOB->ODR); //aSRC_Const_Buffer
- DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)&g_buff_DMA0;
- DMA_InitStructure.DMA_DIR = DMA_DIR_MemoryToPeripheral;
- DMA_InitStructure.DMA_BufferSize = BUFF_LEN;//以ODR为单位数据量
- 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_PeripheralDataSize_HalfWord;
- DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
- DMA_InitStructure.DMA_Priority = DMA_Priority_VeryHigh;
- DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;
- DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;
- DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;//单次
- DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
- DMA_Init(DMA2_Stream5, &DMA_InitStructure);
- //双缓存模式 ,默认循环。传送完全部 buff0之后再传送buff1
- //双模式下才能在传输完成时修改源/目标地址
- //DMA_DoubleBufferModeConfig(DMA2_Stream5,(uint32_t)&g_buff_DMA1,DMA_Memory_0);
- //DMA_DoubleBufferModeCmd(DMA2_Stream5,ENABLE);
- DMA_ClearITPendingBit(DMA2_Stream5, DMA_IT_TCIF5);
- DMA_ITConfig(DMA2_Stream5, DMA_IT_TC, ENABLE);
- DMA_Cmd(DMA2_Stream5,ENABLE);
- //TIM1
- TIM_BaseInitStructure.TIM_Period = 1260-1;
- TIM_BaseInitStructure.TIM_Prescaler = 1;
- TIM_BaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
- TIM_BaseInitStructure.TIM_ClockDivision = 0;
- TIM_BaseInitStructure.TIM_RepetitionCounter = 0;
- TIM_TimeBaseInit(TIM1, &TIM_BaseInitStructure);
-
- TIM_ARRPreloadConfig(TIM1, ENABLE);
- TIM_SelectOutputTrigger(TIM1,TIM_TRGOSource_Update);
- TIM_DMACmd(TIM1, TIM_DMA_Update, ENABLE);
- //TIM1开启
- TIM_Cmd(TIM1, ENABLE);
- }
- // main 函数中移动buff
- void moveLeft(u32 *pData, UI16 n)
- {
- u32 head = *pData;
- n=n/2;
- for(u16 i=0; i<n-1;i++)
- {
- *(pData+i) = *(pData+i+1);
- }
- *(pData+n-1) = head;
- }
- volatile u8 g_updata_flag = 0;
- void my_Animat_Left(void)
- {
- if(g_updata_flag)
- {
- moveLeft((u32 *)g_buff_disp, (BUFF_LEN-DUMMY_N)/2);
- my_memcpy32(g_buff_DMA0, g_buff_disp, (BUFF_LEN)/2);
-
- DMA_Cmd(DMA2_Stream5,ENABLE);
- //TIM_Cmd(TIM1, ENABLE);
- g_updata_flag = 0;
- }
- }
- //中断
- #ifndef RESERVED_MASK
- #define RESERVED_MASK (uint32_t)0x0F7D0F7D
- #endif
- void DMA2_Stream5_IRQHandler(void)
- {
- static int cnt = 0;
- if( DMA2->HISR & DMA_IT_TCIF5)
- {
- DMA2->HIFCR = (uint32_t)(DMA_IT_TCIF5 & RESERVED_MASK);
-
- ResetLed();
- SetLed();
- ResetLed();
- cnt=cnt &0x0F;
- if(!cnt)
- {
- g_updata_flag = 1;
- DMA_Cmd(DMA2_Stream5,DISABLE);
- }
- row ++;
- }
- }
复制代码
逻辑分析仪 采集波形:如下图可见 16个数据 小于64,而且 第12个数据之后出现一段较大空闲。不知原因。
|
楼主既然设置了DMA传输数据长度:
DMA_InitStructure.DMA_BufferSize = BUFF_LEN;
按理就不应该在中断中再计数cnt了,
因为设置的传输模式中:
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
DMA计数达到100时必定产生中断,当中断发生时,DMA的下一次传输仍然在进行,这时的cnt不会与DMA同步了,以cnt作为关闭DMA的时间,应该会漏掉不少DMA传输数据了。
建议:
如果要关闭DMA,应该在中断到来时就关闭,与cnt无关。
同时,要再次进入DMA传输时,恐怕要再初始化一次DMA了,
其实可以设一个缓冲数组,每次DMA中断将数据复制到缓冲区,DMA不停止工作,需要数据时从缓冲区读。
评分
查看全部评分
https://www.stmcu.org.cn/module/forum/thread-612887-1-1.html
Done !
你好, 怎么在论坛里邀请,ST技术答疑呢?
7楼,请问怎样尽可能以最直接的速度关闭DMA ?
评分
查看全部评分
/*-------中断函数部分代码--------*/
/*---------打印信息------------*/
00:00:00: HT IRQ n=15
00:00:00: Start sed n=32
00:00:00: GPIO_DMA_cfg init ok
发现:半中断后,数据量寄存器并没有等于 32/2=16.注: 配置没用 FIFO。
这个应该是在读取 NDTR寄存器时,继续传送了。导致读取那一刻的值小于 16.
评分
查看全部评分
大家会帮忙回答的,如果没有我会在签到帖里悬赏
试过了。 问题是发送100个数据,实际只有50个。
DMA控制的GPIO输出频率怎么调节。 我尝试调节TIM1,更新频率。逻辑分析仪得到 IO翻转变快了。道理上说不通。按理说,TIM更新变快只改变 DMA传送次数变快,难道还会改变每次发送的bit更快?horrible!
问题2,建议在DMA2的中断内,开启一个计时器中断,用计时器调用移位函数,这样速度会比在主程序调用要快。
评分
查看全部评分
评分
查看全部评分
这边一共只有12楼啊。。
Sorry!#11