基于stm32F407ZGT6的三重ADC交替采样+DMA传输数据+定时器TRGO触发

好久没写博客了,最近挺忙的。近来有些好玩的实现,网上的资料并不是非常详细,打算慢慢写下来,希望别人能少走一点弯路。 因为希望提高ADC的采样率,这次我试着实现了一下三重ADC交替采样+DMA搬运至内存+TIM的TRGO触发采样(环境是stm32cubemx 6.5.0和keil 5) 

首先打开cubemx进行基础设置(设置时钟树,RCC,SYS)

然后设置ADC1(下图为具体设置,仅供参考)

此处我们将ADC_Mode设置为Triple regular simultaneous mode only,并打开DMA连续请求(为了使DMA能够填满数组,按照我们的预期工作)

外部触发源设置为TIM2的TRGO

由于已经设置了ADC模式,ADC2和ADC3是跟随ADC1的,所以他们的模式是不能改变的,和ADC1一样设置就可以了

接下来我们设置DMA,记住要设置成circular模式,不然他就只会传输一次,选择一次传输Half Word,从外设搬运到内存

主要的设置就这些,串口的初始化已经省略了,后面我们将用串口重定向打印数据至串口(详见我之前的博客串口重定向发送和接收

现在我们只需要生成代码,打开工程

这里有个小细节:DMA和ADC的初始化顺序不能错了,要先初始化DMA再初始化ADC(其实用到DMA传输数据的外设都要在DMA初始化之后再进行初始化,因为不先初始化DMA的话,在初始化其他外设时,当配置到DMA相关的设置时会出现错乱,导致外设无法正常工作,比如ADC就会传输数据失败,但是DMA一直是处于busy状态)

下面我们进行垃圾核心代码的编写

我的初始化顺序(仅供参考)

1 MX_GPIO_Init();
2 MX_DMA_Init();
3 MX_ADC1_Init();
4 MX_ADC2_Init();
5 MX_ADC3_Init();
6 MX_DAC_Init();
7 MX_TIM2_Init();
8 MX_USART1_UART_Init();

主函数当中编写的代码如下:

1 HAL_ADC_Start(&hadc1);
2 HAL_ADC_Start(&hadc2);
3 HAL_ADC_Start(&hadc3);//启动ADC
4 HAL_ADCEx_MultiModeStart_DMA(&hadc1,(uint32_t *)ADC_value,3000);
5 HAL_TIM_Base_Start(&htim2);//启动TIM2,用于触发ADC采样

对于数据的处理,我更倾向于写在回调函数里面

1 void HAL_ADC_ConvCpltCallback(ADC_HandleTypeDef* hadc)
2 {
3   int i;
4   for(i = 0;i < 1000;i++)
5   {
6     printf("%f,%f,%f\n",ADC_value[3 * i] * 3 / 4096.0,ADC_value[3 * i + 1] * 3 / 4096.0,ADC_value[3 * i + 2] * 3 / 4096.0);
7   }
8   return;
9 }

我们再编译,烧录程序。 至此,我们就可以实现ADC三重交替采样了

 /*------------------------------------------------------------------------分割线------------------------------------------------------------------------*/

我思来想去感觉还是在回调函数里面改变一个标志位就可以了,把这么一大段处理放进去不太好,容易出现中断嵌套等我们不希望出现的现象

尽量把处理的步骤写到函数里面去执行,只需通过判断标志位即可达到同等效果

posted @ 2022-08-02 12:41  砕月之殇  阅读(1362)  评论(0编辑  收藏  举报
/*粒子线条,鼠标移动会以鼠标为中心吸附的特效*/