Stm32 在DMA循环模式下使用ADC时,DMA内存缓冲区中的采样数据是否丢失或不正确?

Stm32 在DMA循环模式下使用ADC时,DMA内存缓冲区中的采样数据是否丢失或不正确?,stm32,Stm32,我的目的是通过ADC通道采样信号,DMA数据在STM32Fx板中移动。生成一个方波到ADC通道。如果使用DMA模式,则某些数据出现故障或被称为混乱。同样的结果也发生在STM32F207和STM32F373板上 (1) 当我使用ADC EOC中断采集转换后的数据时,数据阵列看起来像一个方波。这没关系 (2) 我想尝试DMA circle而不是EOC IRQ,但数据数组似乎混乱,一些数据丢失或不正确。如果采样率增加,情况可能会更糟。下面是我的测试结果 DMA中的波形变短 DMA模式的数据混乱,但使

我的目的是通过ADC通道采样信号,DMA数据在STM32Fx板中移动。生成一个方波到ADC通道。如果使用DMA模式,则某些数据出现故障或被称为混乱。同样的结果也发生在STM32F207和STM32F373板上

(1) 当我使用ADC EOC中断采集转换后的数据时,数据阵列看起来像一个方波。这没关系

(2) 我想尝试DMA circle而不是EOC IRQ,但数据数组似乎混乱,一些数据丢失或不正确。如果采样率增加,情况可能会更糟。下面是我的测试结果

DMA中的波形变短

DMA模式的数据混乱,但使用ADC EOC IRQ是一致的

enter code here

<<<< ADC config >>>>

/* ADC Common Init */
ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div8;
ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled;
ADC_CommonInitStructure.ADC_TwoSamplingDelay = 
ADC_TwoSamplingDelay_5Cycles;
ADC_CommonInit(&ADC_CommonInitStructure);

/* ADC1 DeInit */
ADC_StructInit(&ADC_InitStructure);

/* Configure the ADC1 in continuous mode */
ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;
ADC_InitStructure.ADC_ScanConvMode = ENABLE;
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructure.ADC_NbrOfConversion = 1;
ADC_Init(ADC1, &ADC_InitStructure);

/* ADC1 regular channels 6 configuration */ 
ADC_RegularChannelConfig(ADC1, ADC_Channel_6, 1, 
ADC_SampleTime_480Cycles);
ADC_EOCOnEachRegularChannelCmd(ADC1, ENABLE);

#ifdef __DMA_ENABLE__

/* Enable DMA request after last transfer (Single-ADC mode) */
ADC_DMARequestAfterLastTransferCmd(ADC1, ENABLE);

/* Enable ADC1 DMA since ADC1 is the Master*/
ADC_DMACmd(ADC1, ENABLE);

#else

ADC_ITConfig(ADC1, ADC_IT_EOC, ENABLE);
ADC_ITConfig(ADC1, ADC_IT_OVR, ENABLE);

NVIC_InitStructure.NVIC_IRQChannel = ADC_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);

#endif

/* Enable ADC1 */
ADC_Cmd(ADC1, ENABLE);


<<<< DMA config >>>>

/* DMA1 clock enable */
RCC_AHB1PeriphClockCmd( RCC_AHB1Periph_DMA2, ENABLE );
/* DMA1 Channel1 Config */
DMA_DeInit(DMA2_Stream0);

DMA_DoubleBufferModeConfig(DMA2_Stream0, (uint32_t)ADCValB, DMA_Memory_1);
DMA_DoubleBufferModeCmd(DMA2_Stream0, ENABLE);

DMA_InitStructure.DMA_Channel = DMA_Channel_0; 
DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)ADCValA;
DMA_InitStructure.DMA_PeripheralBaseAddr = ((uint32_t) ADC1) + 0x4C;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;
DMA_InitStructure.DMA_BufferSize = NUM_OF_ADC; // 512 buffer size
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_MemoryDataSize_HalfWord;
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;
DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;         
DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull;
DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;
DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
DMA_Init(DMA2_Stream0, &DMA_InitStructure);

DMA_ITConfig(DMA2_Stream0, DMA_IT_TC, ENABLE);
NVIC_InitStructure.NVIC_IRQChannel = DMA2_Stream0_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);

/* DMA1 Channel1 enable */
DMA_Cmd(DMA2_Stream0, ENABLE);
在此处输入代码
>
/*ADC公共初始化*/
ADC_CommonInitStructure.ADC_Mode=ADC_Mode_独立;
ADC_CommonInitStructure.ADC_Prescaler=ADC_Prescaler_Div8;
ADC_CommonInitStructure.ADC_DMAAccessMode=ADC_DMAAccessMode_已禁用;
ADC_CommonInitStructure.ADC_TwoSamplingDelay=
ADC两个采样延迟周期;
ADC_CommonInit(&ADC_CommonInit结构);
/*ADC1脱硝剂*/
ADC_init(&ADC_init structure);
/*在连续模式下配置ADC1*/
ADC_InitStructure.ADC_Resolution=ADC_Resolution_12b;
ADC_InitStructure.ADC_ScanConvMode=启用;
ADC_InitStructure.ADC_ContinuousConvMode=启用;
ADC_InitStructure.ADC_DataAlign=ADC_DataAlign_Right;
ADC_InitStructure.ADC_nbrof转换=1;
ADC_Init(ADC1和ADC_Init结构);
/*ADC1常规通道6配置*/
ADC_常规通道配置(ADC1、ADC_通道6、1、,
ADC_采样时间_480;周期);
ADC_EOCOnEachRegularChannelCmd(ADC1,启用);
#ifdef\uuu DMA\u启用__
/*上次传输后启用DMA请求(单ADC模式)*/
ADC_DMARequestAfterLastTransferCmd(ADC1,启用);
/*启用ADC1 DMA,因为ADC1是主设备*/
ADC\U DMACmd(ADC1,启用);
#否则
ADC_ITConfig(ADC1、ADC_IT_EOC、启用);
ADC_ITConfig(ADC1、ADC_IT_OVR、启用);
NVIC_InitStructure.NVIC_IRQChannel=ADC_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority=1;
NVIC_InitStructure.NVIC_IRQChannelCmd=启用;
NVIC_Init(&NVIC_InitStructure);
#恩迪夫
/*启用ADC1*/
ADC_Cmd(ADC1,启用);
>
/*DMA1时钟使能*/
RCC_AHB1PeriphLockCmd(RCC_AHB1Periph_DMA2,启用);
/*DMA1信道1配置*/
DMA_脱硝(DMA2_-0);
DMA_DoubleBufferModeConfig(DMA2_Stream0,(uint32_t)ADCValB,DMA_内存_1);
DMA_DoubleBufferModeCmd(DMA2_Stream0,启用);
DMA_InitStructure.DMA_Channel=DMA_Channel_0;
DMA_InitStructure.DMA_Memory0BaseAddr=(uint32_t)ADCValA;
DMA_InitStructure.DMA_peripheraldbaseaddr=(uint32_t)ADC1)+0x4C;
DMA_InitStructure.DMA_DIR=DMA_DIR_外设存储器;
DMA_InitStructure.DMA_BufferSize=NUM_OF_ADC;//512缓冲区大小
DMA_InitStructure.DMA_PeripheralInc=DMA_PeripheralInc_Disable;
DMA_InitStructure.DMA_MemoryInc=DMA_MemoryInc_Enable;
DMA_InitStructure.DMA_PeripheralDataSize=DMA_PeripheralDataSize\u半字;
DMA_InitStructure.DMA_MemoryDataSize=DMA_MemoryDataSize_半字;
DMA_InitStructure.DMA_Mode=DMA_Mode_循环;
DMA_InitStructure.DMA_Priority=DMA_Priority_Medium;
DMA_InitStructure.DMA_FIFOMode=DMA_FIFOMode_Disable;
DMA_InitStructure.DMA_FIFOThreshold=DMA_FIFOThreshold_半满;
DMA_InitStructure.DMA_MemoryBurst=DMA_MemoryBurst_Single;
DMA_InitStructure.DMA_PeripheralBurst=DMA_PeripheralBurst_Single;
DMA_Init(DMA2_Stream0和DMA_InitStructure);
DMA_ITConfig(DMA2_Stream0,DMA_IT_TC,ENABLE);
NVIC_InitStructure.NVIC_IRQChannel=DMA2_Stream0_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority=0;
NVIC_InitStructure.NVIC_IRQChannelCmd=启用;
NVIC_Init(&NVIC_InitStructure);
/*DMA1信道1启用*/
DMA_Cmd(DMA2_Stream0,启用);


最后,我希望结果应该与使用ADC EOC IRQ的结果相同。

根据您提供的数据表格视图,DMA似乎配置正确,因为数据看起来与从ADC EOC IRQ获得的数据几乎相同

enter code here

<<<< ADC config >>>>

/* ADC Common Init */
ADC_CommonInitStructure.ADC_Mode = ADC_Mode_Independent;
ADC_CommonInitStructure.ADC_Prescaler = ADC_Prescaler_Div8;
ADC_CommonInitStructure.ADC_DMAAccessMode = ADC_DMAAccessMode_Disabled;
ADC_CommonInitStructure.ADC_TwoSamplingDelay = 
ADC_TwoSamplingDelay_5Cycles;
ADC_CommonInit(&ADC_CommonInitStructure);

/* ADC1 DeInit */
ADC_StructInit(&ADC_InitStructure);

/* Configure the ADC1 in continuous mode */
ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b;
ADC_InitStructure.ADC_ScanConvMode = ENABLE;
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
ADC_InitStructure.ADC_NbrOfConversion = 1;
ADC_Init(ADC1, &ADC_InitStructure);

/* ADC1 regular channels 6 configuration */ 
ADC_RegularChannelConfig(ADC1, ADC_Channel_6, 1, 
ADC_SampleTime_480Cycles);
ADC_EOCOnEachRegularChannelCmd(ADC1, ENABLE);

#ifdef __DMA_ENABLE__

/* Enable DMA request after last transfer (Single-ADC mode) */
ADC_DMARequestAfterLastTransferCmd(ADC1, ENABLE);

/* Enable ADC1 DMA since ADC1 is the Master*/
ADC_DMACmd(ADC1, ENABLE);

#else

ADC_ITConfig(ADC1, ADC_IT_EOC, ENABLE);
ADC_ITConfig(ADC1, ADC_IT_OVR, ENABLE);

NVIC_InitStructure.NVIC_IRQChannel = ADC_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);

#endif

/* Enable ADC1 */
ADC_Cmd(ADC1, ENABLE);


<<<< DMA config >>>>

/* DMA1 clock enable */
RCC_AHB1PeriphClockCmd( RCC_AHB1Periph_DMA2, ENABLE );
/* DMA1 Channel1 Config */
DMA_DeInit(DMA2_Stream0);

DMA_DoubleBufferModeConfig(DMA2_Stream0, (uint32_t)ADCValB, DMA_Memory_1);
DMA_DoubleBufferModeCmd(DMA2_Stream0, ENABLE);

DMA_InitStructure.DMA_Channel = DMA_Channel_0; 
DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)ADCValA;
DMA_InitStructure.DMA_PeripheralBaseAddr = ((uint32_t) ADC1) + 0x4C;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;
DMA_InitStructure.DMA_BufferSize = NUM_OF_ADC; // 512 buffer size
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_MemoryDataSize_HalfWord;
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;
DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Disable;         
DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_HalfFull;
DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;
DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
DMA_Init(DMA2_Stream0, &DMA_InitStructure);

DMA_ITConfig(DMA2_Stream0, DMA_IT_TC, ENABLE);
NVIC_InitStructure.NVIC_IRQChannel = DMA2_Stream0_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);

/* DMA1 Channel1 enable */
DMA_Cmd(DMA2_Stream0, ENABLE);
依赖DMA和IRQ之间的唯一变量是DMA控制器和CPU之间可能存在意外的总线“冲突”,因为与使用IRQ时不同,它们都同时运行,可能导致等待状态

从STM32参考手册第13.4节:

DMA控制器通过与Cortex-M4®F core共享系统总线来执行直接内存传输。当CPU和DMA针对同一目的地(内存或外围设备)时,DMA请求可能会在某些总线周期内停止CPU对系统总线的访问

您观察到的与采样率相关的性能下降肯定证实了这一假设,因为总线矩阵必须仲裁DMA控制器和CPU之间更频繁的访问


如果看不到设置和读取缓冲区的其余代码,很难说应用程序代码的哪个方面可能导致此问题。

根据您提供的数据表格视图,DMA似乎配置正确,因为数据看起来与从ADC EOC IRQ获得的数据几乎相同

依赖DMA和IRQ之间的唯一变量是DMA控制器和CPU之间可能存在意外的总线“冲突”,因为与使用IRQ时不同,它们都同时运行,可能导致等待状态

从STM32参考手册第13.4节:

DMA控制器通过与Cortex-M4®F core共享系统总线来执行直接内存传输。当CPU和DMA针对同一目的地(内存或外围设备)时,DMA请求可能会在某些总线周期内停止CPU对系统总线的访问

您观察到的采样率依赖性性能下降肯定证实了这一假设,因为总线矩阵必须仲裁更多的fr