0 }4 \4 d- f6 h- y2 O $ Q" B5 }( W1 I 试验目的: 完成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个数都是这一个通道的值。 ) u/ u" Y0 f" |, P) Q* B( ?4 j" t4、 DMA配置中,如果选择Normal模式,那么只执行一次DMA就停止了,如果设置成Circular模式,就可以连续开启DMA。数据宽度设置为Word,外设地址不变,存储器地址加1。 ' F! o3 u ^( z' Y, X+ b$ Y+ K$ D. z' @" o5 R9 E: J& P |
/* ADC1 DMA Init */; ?! U# J7 g( Y2 F5 M3 N; x
/* ADC1 Init */
hdma_adc1.Instance = DMA1_Channel1;
hdma_adc1.Init.Direction = DMA_PERIPH_TO_MEMORY;0 P, H2 a, [2 Y( N# v
hdma_adc1.Init.PeriphInc = DMA_PINC_DISABLE;
hdma_adc1.Init.MemInc = DMA_MINC_ENABLE;9 r5 s6 j- l& V8 A& m
hdma_adc1.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD;& L5 E- F1 ]. L+ p
hdma_adc1.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD;6 l! L- {& `6 H1 G5 J* d; P
hdma_adc1.Init.Mode = DMA_NORMAL;& b2 q8 T: u5 l1 i+ A; [7 Z$ ~
hdma_adc1.Init.Priority = DMA_PRIORITY_HIGH;
if (HAL_DMA_Init(&hdma_adc1) != HAL_OK)
{
_Error_Handler(__FILE__, __LINE__);
}& x) H- f0 a+ j& F0 q
__HAL_LINKDMA(adcHandle,DMA_Handle,hdma_adc1);
2.存储的空间可以定义成uint32_t,word的数量可以要采集次数的一半:: s/ _6 ^1 d7 V1 D
uint32_t uhADCxConvertedValue[131];: n. G; P8 R0 V, P
3.启动DMA的时候,传入缓冲区的地址按(uint32_t *)传入。3 C! M. x! v/ k y) }! U2 p
HAL_ADC_Start_DMA(&hadc1, (uint32_t*)&uhADCxConvertedValue, 262);
4.这样进行了262次采样后,存放在uhADCxConvertedValue区域内的数据就是按16bit连续放过去的。. M, O* K0 R6 c5 r
测亲的结果分享的。
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
论坛有一个人发过,你可以找找