e6 c4 O$ i8 r; k7 F ) a; B' J3 v. ~6 M/ {3 ]7 M& o ( f( i( l9 D( K5 e/ e" k7 H% x4 d 试验目的: 完成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个数都是这一个通道的值。 , }" P% f& t: {4、 DMA配置中,如果选择Normal模式,那么只执行一次DMA就停止了,如果设置成Circular模式,就可以连续开启DMA。数据宽度设置为Word,外设地址不变,存储器地址加1。 5 _3 ^' p" X4 s# q& P& k; {6 O+ T* X7 b. ^5 a* L- F |
/* ADC1 DMA Init */
/* ADC1 Init */1 f9 n1 }0 f5 V& B
hdma_adc1.Instance = DMA1_Channel1;+ v% @9 C. L9 M7 \5 {% x+ Q
hdma_adc1.Init.Direction = DMA_PERIPH_TO_MEMORY;1 C3 U, ~, f$ j5 ^- b
hdma_adc1.Init.PeriphInc = DMA_PINC_DISABLE;" r! A4 b, C9 Y5 O- U
hdma_adc1.Init.MemInc = DMA_MINC_ENABLE;
hdma_adc1.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD;) M! u; M# c0 F9 V
hdma_adc1.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD;
hdma_adc1.Init.Mode = DMA_NORMAL;+ [4 g3 t. E& f, r9 s3 g
hdma_adc1.Init.Priority = DMA_PRIORITY_HIGH;
if (HAL_DMA_Init(&hdma_adc1) != HAL_OK)
{
_Error_Handler(__FILE__, __LINE__);
}
__HAL_LINKDMA(adcHandle,DMA_Handle,hdma_adc1);! ? D; B" ]) p8 W2 a
8 b5 Z! l6 z8 [- m" @- |6 n
2.存储的空间可以定义成uint32_t,word的数量可以要采集次数的一半:
uint32_t uhADCxConvertedValue[131];
3.启动DMA的时候,传入缓冲区的地址按(uint32_t *)传入。
HAL_ADC_Start_DMA(&hadc1, (uint32_t*)&uhADCxConvertedValue, 262);! |, P" P& y8 \2 y2 T7 j6 ~8 _, K
4.这样进行了262次采样后,存放在uhADCxConvertedValue区域内的数据就是按16bit连续放过去的。. ]0 W" |/ V7 u' y$ a6 G
测亲的结果分享的。
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
论坛有一个人发过,你可以找找