STM32F4 hal库 ADC + DMA + DSP + FFT 实现920K波形频率的采集

使用开发板:立创开发板·天空星STM32F407

要实现920K波形频率的采集,使用定时器触发adc采集才能使adc采集频率可调,废话不多说接下来先进行cubemx的配置教程。

1、选择开发板

2、系统配置

3、时钟配置(重要)

(1)一定要使用外部晶振通道才能确保时钟的准确。

(2) 总线时钟配置成144MHz,因为stm32f407的adc采样率最高可以达到2.4M的速率,此时APB2总线经过2分频达到72MHz,ADC在经过2分频后,就可以达到ADC规定的最高时钟频率36MHz,adc的采样率计算公式:采样率 = ADC时钟频率 / (采样周期 + 12) 。这里选择3个Cycles,此时adc采样率就是最高的2.4M。

4、TIM配置

(1)使用定时器3,定时器3的时钟总线是APB1 72MHz,Trigger Event Selection选update Event,分频系数3,装载值13,此时定时器的频率为1846153Hz,不要开启定时器中断。频率太快了,开了会导致程序会死在中断。计数器的时钟频率 f  = TimeClockFren /((Prescaler + 1) * (Period + 1) );

5、ADC配置

(1)选一个adc通道,开启DMA,正常模式

如果选择循环模式,频率太高会导致程序死在dma中断中。

选择3Cycles,使adc采样达到2.4M

6、开启串口

7、生成工程(新手)

到此我们的工程就已经建好了。接下来进行代码的编写。

1、串口重定向

#include "stdio.h"
 #ifdef __GNUC__
     #define PUTCHAR_PROTOTYPE int _io_putchar(int ch)
 #else
     #define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
 #endif /* __GNUC__*/
 
 /******************************************************************
     *@brief  Retargets the C library printf  function to the USART.
     *@param  None
     *@retval None
 ******************************************************************/
 PUTCHAR_PROTOTYPE
 {
     HAL_UART_Transmit(&huart1, (uint8_t *)&ch,1,0xFFFF);
     return ch;
 }

2、添加DSP库

参考链接:https://blog.csdn.net/qq_34022877/article/details/117855263

3、傅里叶变换

(1)定义相关数组及宏定义

#define adc_SIZE 2048*2// ADC 采样大小的定义
uint32_t adcConvertValue[adc_SIZE]={0};// 存放 ADC 采样数据的数组
uint8_t flag=0;// 标记位,用于标识 ADC 采样是否完成

float32_t frequency ;// 用于存放计算结果的频率变量
// FFT 相关参数的定义
#define FFT_SIZE 2048
#define FFT_LEN FFT_SIZE 
#define SAMPLING_FREQUENCY 1846153.84615f //adc采样频率
float32_t inputSignal[FFT_SIZE*2];// FFT 输入信号数组
float32_t fftOutput[FFT_SIZE/2];// FFT 输出数组
uint32_t index_;// 存放 FFT 输出中最大值的索引

(2)创建一个fft的计算函数

void fftCalculate(void)// FFT 计算函数
{
    arm_cfft_f32(&arm_cfft_sR_f32_len2048, inputSignal, 0, 1);// 执行 FFT 计算
    arm_cmplx_mag_f32(inputSignal, fftOutput, FFT_LEN/2);// 计算 FFT 输出的幅度
    index_ = 0;// 查找 FFT 输出中的最大值
    float32_t maxValue = fftOutput[1];
//    for (uint32_t i = 1; i < FFT_LEN/2; i++)
//    {
//        if (fftOutput[i] > maxValue)
//        {
//            maxValue = fftOutput[i];
//            index = i;
//        }
//    }
		arm_max_f32(&fftOutput[1], FFT_LEN/2, &maxValue, &index_); // 使用 arm_max_f32 函数快速找到 FFT 输出中的最大值及其索引
    frequency = (float32_t)index_ * (float32_t)SAMPLING_FREQUENCY / (float32_t)FFT_SIZE;// 根据最大值的索引计算信号的频率

}

(3)在while中添加代码

		printf("frequency is:%f\n",frequency);
		if(flag==1)
		{
			flag=0;
			
			for(int j=0;j<FFT_SIZE;j++)//采样数据转换
			{
				inputSignal[j*2]=(adcConvertValue[j])*3.3f/4096.0f;//12位采样数字值对应为0-3.3伏
				inputSignal[j*2+1]=0;//交替插零,添加数据的虚部
			}

			fftCalculate();
			HAL_TIM_Base_Start(&htim3);	//重新启动定时器3
			HAL_ADC_Start_DMA(&hadc1, (uint32_t *)adcConvertValue,adc_SIZE);//重新开始下一轮采集
			
		}		

(4)在DMA中断中停止定时器和DMA,中断函数在stm32f4xx_it.c文件中。

extern uint8_t flag;
extern TIM_HandleTypeDef htim3;
extern ADC_HandleTypeDef hadc1;
void DMA2_Stream0_IRQHandler(void)
{
  /* USER CODE BEGIN DMA2_Stream0_IRQn 0 */

  /* USER CODE END DMA2_Stream0_IRQn 0 */
  HAL_DMA_IRQHandler(&hdma_adc1);
  /* USER CODE BEGIN DMA2_Stream0_IRQn 1 */
	HAL_TIM_Base_Stop(&htim3);//定时器3停止
	HAL_ADC_Stop_DMA(&hadc1);//停止ADC的DMA传输
	flag=1;//标志位置1,表示本次采样结束
  /* USER CODE END DMA2_Stream0_IRQn 1 */
}

DMA中断一次说明adc转换完成一次,此时把标志flag值赋值1,那么在while中把adc采集值赋值给傅里叶变换数组后,进行fft的计算,从而得出所测波形的频率。根据抽样定理我们可以知道,采样率要是最高频率的两倍,所以理论上要测1MHz频率时,adc的采样率要设置成2MHz。参考文章:https://blog.csdn.net/qq_59953808/article/details/131405743?spm=1001.2014.3001.5506

傅里叶参考:https://blog.csdn.net/qq_34022877/article/details/123190943?spm=1001.2014.3001.5502

附上效果图

  • 12
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值