1 v8 K v/ i V" W" L$ I6 @! ` * N" Z: U. J( _* Y) L 试验目的: 完成2路ADC采样(PA1,PA2)通过DMA的方式,转换结果放入uint32_tADC_ConvertedValue[2]数组中。ADC_ConvertedValue[1]放PA1(ADC第一通道的值),ADC_ConvertedValue[2]放PA2(ADC第二通道的值)。 具体配置: 程序分析: 首先定义数组存放ADC转换的值 uint32_t ADC_ConvertedValue[2]; 在主程序中加入HAL_ADC_Start_DMA(&hadc, ADC_ConvertedValue,2);开启ADC的DMA功能。 疑问及需要注意的地方: 1、 ADC配置中,由于时钟是48MHz,所以ClockPrescaler为Synchronous clock mode divided by 4,这样ADC的时钟为12MHz(ADC时钟不能超过14MHz) 2、 End of Conversion Selection配置没有试出是什么功能。 3、 ADC_Regular_ConversionMode配置转换的通道数及采样周期,此处如果设成1,那么ADC_ConvertedValue[2]的2个数都是这一个通道的值。 4、 DMA配置中,如果选择Normal模式,那么只执行一次DMA就停止了,如果设置成Circular模式,就可以连续开启DMA。数据宽度设置为Word,外设地址不变,存储器地址加1。 ) T9 I. k/ b; _% W! |% c3 T; v7 j% [8 S/ [ |
/* ADC1 DMA Init */
/* ADC1 Init */
hdma_adc1.Instance = DMA1_Channel1;
hdma_adc1.Init.Direction = DMA_PERIPH_TO_MEMORY;. ?& D$ W" _# m; z9 E! O1 W
hdma_adc1.Init.PeriphInc = DMA_PINC_DISABLE;
hdma_adc1.Init.MemInc = DMA_MINC_ENABLE;
hdma_adc1.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD;
hdma_adc1.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD;
hdma_adc1.Init.Mode = DMA_NORMAL;/ Q, U& w+ U( Z$ Q- x( }# j
hdma_adc1.Init.Priority = DMA_PRIORITY_HIGH;
if (HAL_DMA_Init(&hdma_adc1) != HAL_OK)3 e& G+ j, J% [+ E3 [2 Y8 C
{
_Error_Handler(__FILE__, __LINE__);
}
1 B# O) R o/ k. j/ h. b: w
__HAL_LINKDMA(adcHandle,DMA_Handle,hdma_adc1);
8 r' d& I9 k! P
2.存储的空间可以定义成uint32_t,word的数量可以要采集次数的一半:
uint32_t uhADCxConvertedValue[131];
3.启动DMA的时候,传入缓冲区的地址按(uint32_t *)传入。
HAL_ADC_Start_DMA(&hadc1, (uint32_t*)&uhADCxConvertedValue, 262); |& N. N# ]0 @6 `3 Q- A
4.这样进行了262次采样后,存放在uhADCxConvertedValue区域内的数据就是按16bit连续放过去的。
测亲的结果分享的。
pData不能是uint16_t的,因为DMA理论上是要有能力把数据“搬运”4GB地址空间里的任何位置,所以必须要uint32_t, 虽然pData是uint32_t 但是DMA从ADC的数据寄存器里搬运数据并不一定是按照32bit来复制的,而是根据Periphral 和Memory 的data width设置的,比如如果Memory 的数据宽度是Half Word那么dma就按照16bit递增地址来写数据
恩,是这样的
我当时做ADC_DMA也有这个困惑,从函数原型来看是用32bit
论坛有一个人发过,你可以找找