STM32HAL ADC+TIM+DMA采集交流信号 基于cubemx

ADC+TIM+DMA采集交流

前言

本文主要讲解定时器触发ADC去采集交流信号,DMA把数据搬移到内存。

所需工具:

  • 开发板:STM32F103C8T6
  • STM32CubeMX
  • IDE: Keil-MDK

模式简介

ADC+TIM+DMA采集交流信号是电赛中使用范围最为广泛的一个技术。这个模式下单个ADC可以实现0-1M的任意可调采样率,采集20khz一下的信号轻轻松松。

F1的ADC支持许多触发信号,这里选择TIM3的TRGO事件作为触发信号,其中TRGO选择更新时间来引起。(这段新手看不懂没关系,不耽误使用)

工程建立

时钟配置

在这里插入图片描述

在这里插入图片描述

ADC配置

在这里插入图片描述

相对于ADC采集直流,这里的触发源不是软件上的一行代码来触发,而是让选择外部触发,这里选择TIM3的TRGO信号。

对于新手来说这里可能有疑惑,换成硬件触发有什么好处吗?查看系列的上一篇文章,软件触发ADC采样一次,需要写几行代码,才能让他们采集一次,如果我们想实现100hz的采样率,可以设置一个100hz的定时器中断,在中断里用代码(软件)触发ADC采样,这样确实可以达到100hz采样的效果。可是如果100k采样呢?CPU代码执行的速度是有限的,100hz可以勉强达到,100k就来不及了。但是我让TIM这样的硬件去触发ADC采样,ADC采集完成后,DMA硬件搬运数据,整个采集过程不需要CPU参与。

直观上看就是你告诉ADC,TIM,DMA你们仨给我100k采样率采集1000个点.说完这句话后,他们三就去采集了,CPU只需要等他们采集完成就可以。采集过程CPU不管的,也就是不需要写任何代码。

在这里插入图片描述

DMA配置为normal模式。如果配置成circular的话,ADC采集完成指定个数后,不会停下来,不方便管理。读者可以修改成circular看看效果。

在这里插入图片描述

采样率控制在100kz,那么TIM就需要产生100khz的TRGO的信号,我们这里选择的更新时间产生TRGO信号,那么TIM3的计数器从0计算到ARR的频率为100khz。于是我们这里设置PSC=0,ARR=720-1。换算下: 72 M 720 = 100 k \frac{72M}{720}=100k 72072M=100k

配置串口

在这里插入图片描述

代码生成

在这里插入图片描述

在这里插入图片描述在这里插入图片描述

代码编写

串口重定向

在这里插入图片描述

#include <stdio.h>

int fputc(int ch, FILE *f)
{
    
    
  HAL_UART_Transmit(&huart1, (uint8_t *)&ch, 1, 0xffff);
  return ch;
}

int fgetc(FILE *f)
{
    
    
  uint8_t ch = 0;
  HAL_UART_Receive(&huart1, &ch, 1, 0xffff);
  return ch;
}

这里是串口重定向的主体部分

在这里插入图片描述

#include <stdio.h>

在mian.c里面包含stdio.h头文件,mian.c里面就可以printf了。别的.c文件同理。

在这里插入图片描述

勾选MicroLIB库,否则没法使用printf

ADC采集代码

在这里插入图片描述

uint16_t adc_buff[200];//存放ADC采集的数据
/* 
AdcConvEnd用来检测ADC是否采集完毕
0:没有采集完毕
1:采集完毕,在stm32f1xx_it里的DMA完成中断进行修改
 */
__IO uint8_t AdcConvEnd = 0;

在main.c里面定义两个变量,一个存放ADC采集到的数据,一个标志ADC是否采集完毕。

特别注意__IO修饰AdcConvEnd。他的含义是volatile。避免AdcConvEnd被MDK优化掉。

在这里插入图片描述

HAL_TIM_Base_Start(&htim3);                           //开启定时器3
HAL_ADCEx_Calibration_Start(&hadc1);                  //AD校准
HAL_ADC_Start_DMA(&hadc1, (uint32_t *)adc_buff, 200); //让ADC1去采集200个数,存放到adc_buff数组里
while (!AdcConvEnd)                                   //等待转换完毕
    ;
for (uint16_t i = 0; i < 200; i++)
{
    
    
    printf("%.3f\n", adc_buff[i] * 3.3 / 4095); //数据打印,查看结果
}

这里写的采集程序,如每一步的含义都在注释里写明了。

希望读者养成随手写注释的好习惯。

在这里插入图片描述

AdcConvEnd = 1;

ADC采集,DMA搬运,当DMA搬运结束后,整个采集过程也就完成了。DMA搬运结束,程序会接收到DMA中断,就会执行DMA1_Channel1_IRQHandler函数,告诉CPU,采集完毕了。程序上则根据AdcConvEnd的变化,得知采集完毕。

硬件连接

引脚 连接对象 释义
PA9 CH340的RX 单片机的TX连接CH340的RX
PA10 CH340的TX 单片机的RX连接CH340的TX
PA0 信号发生器信号端 图中红线
GND 信号发生器地 跟信号发生器共地

在这里插入图片描述

在这里插入图片描述

上面总共有STlinkV2,ch340,供电线,信号发生器接过来的夹子线

运行结果

请添加图片描述

ADC去采集信号发生器产生的1k正弦信号,数据打印到VOFA上,结果如图。

请添加图片描述

为了验证采样率是否是100k,ADC去采集信号发生器产生的5k信号,打印到VOFA上,可以看到一个周期20个点。 5 k ∗ 20 = 100 k 5k*20=100k 5k20=100k采样率为100k验证完毕。

VOFA的使用可以在电赛小站里查看到教程。

后记

本文章收录于:

唐承乾的电赛小站

本文为系列文章中的冰山一角,欢迎进入小站查看。

配套程序:

STM32的ADC+DMA+TIM采集交流信号.zip-嵌入式文档类资源-CSDN文库

猜你喜欢

转载自blog.csdn.net/qq_34022877/article/details/121941236