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选择更新时间来引起。(这段新手看不懂没关系,不耽误使用)

工程建立

时钟配置

image-20211213205147618

image-20211213205045357

ADC配置

image-20211213205305370

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

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

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

image-20211213205611953

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

image-20211213211113320

采样率控制在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

配置串口

image-20211213205856436

代码生成

image-20211213210039737

image-20211213210101284

image-20211213210118878

代码编写

串口重定向

image-20211213210432148

#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;
}

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

image-20211213210527822

#include <stdio.h>

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

image-20211213220835401

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

ADC采集代码

image-20211213215904440

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优化掉。

image-20211213220302661

HAL_TIM_Base_Start(&htim3);                           //开启定时器3
HAL_ADCEx_Calibration_Start(&hadc1);                  //AD校准,F4不用校准没用这行函数。
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); //数据打印,查看结果
}

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

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

extern uint8_t AdcConvEnd;//引入外部变量

image-20211213220458038

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,供电线,信号发生器接过来的夹子线

在这里插入图片描述

运行结果

image-20211213214544474

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

image-20211213214752190

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

5

k

20

=

100

k

5k*20=100k

5k20=100k采样率为100k验证完毕。

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

后记

本文章收录于:

唐承乾的电赛小站

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

配套程序:

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

版权声明:本文为CSDN博主「四臂西瓜」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/qq_34022877/article/details/121941236

生成海报
点赞 0

四臂西瓜

我还没有学会写个人说明!

暂无评论

发表评论

相关推荐

STM32F103+CubeMX+ADC采集直流

STM32F103CubeMXADC采集直流 前言 本文主要讲解如何使用单片机的内部ADC去采集直流量。需要对ADC和cubemx有一定的使用经历。 所需工具: 开发板:STM32F103C8T6STM32CubeMXID

唐承乾的电赛小站

唐承乾的电赛小站 系列文章 带*的内容,不是本人所写。 扫盲 stm32cubemx输出pwm波,实现呼吸灯*STM32hal库定时器实现微秒延迟*STM32多串口实现printf——基于cubemxstm32简易

唐承乾的电赛小站

唐承乾的电赛小站 系列文章 带*的内容,不是本人所写。 扫盲 stm32cubemx输出pwm波,实现呼吸灯*STM32hal库定时器实现微秒延迟*STM32多串口实现printf——基于cubemxstm32简易