开源小项目-基于STM32F103的频率测量实现 V1.1(ADC+TIM+DMA+FFT)

news2025/1/11 7:13:22

目录

一、快速傅里叶变换(FFT)

1.1 工作原理

1.2 应用

1.3 FFT官方支持库

二、使用外设简介

2.1 ADC外设

2.2 TIM外设

3.3 DMA外设

三、代码设计过程

3.1 初始化

3.2 DMA中断数据处理

3.3 其他自定义函数

1. 计算信号频率

2. 计算信号幅度

3. 计算信号相位

4. 完整代码流程

5. 注意事项

3.4 采样频率选择及配置      

四、调用示例

五、频率测量现象演示

5.1 采样频率为20Khz

5.2 采样频率为1Khz

六、FFT误差原因分析及减小误差策略

6.1 误差原因分析

6.2 FFT的优缺点补充

1. 优点

2. 缺点

七、总谐波失真计算



        该项目是基于STM32F1的频率测量,采用FFT快速傅里叶变换,配合ADC、DMA、TIM等外设实现频率测量,频率测量偏差小于千分之8。主要运用知识点:

  • 1. ADC单通道配置与使用
  • 2. 定时器配置及使用
  • 3. DMA数据搬运配置与使用
  • 4. C语言结构体、数组使用

项目源码文件下载地址:fft_measure V1.1.zip - 蓝奏云


一、快速傅里叶变换(FFT)

        快速傅里叶变换(Fast Fourier Transform,简称FFT)是一种高效的算法,用于计算离散傅里叶变换(Discrete Fourier Transform,简称DFT)及其逆变换。傅里叶变换是一种将信号从时域(或空间域)转换到频域的数学方法,反之亦然。这意味着它可以将复杂的波形分解成一系列不同频率的正弦波和余弦波的组合,每个波都有其对应的振幅和相位。这一过程对于信号处理、图像处理、数据分析等领域至关重要。

1.1 工作原理

        FFT通过将DFT的计算分解为较小部分的递归算法实现加速。核心思想是利用了傅里叶变换的对称性和周期性,将原问题分解为两个或更多的相似但规模较小的问题,最终通过分治策略合并这些小问题的答案以获得最终结果。特别是,它利用了蝶形运算,减少了乘法和加法的总数,这是FFT效率提升的关键。

1.2 应用

FFT在众多领域有着广泛的应用:

  • 信号处理:分析信号的频率组成,滤波,压缩,降噪等。
  • 图像处理:图像压缩,边缘检测,纹理分析等。
  • 通信:频谱分析,调制解调,信号同步等。
  • 音频处理:音乐合成,音调调整,噪声消除等。
  • 医学:心电图、脑电图等生物信号的分析。
  • 天文学:分析星系光谱,探测宇宙背景辐射等。

1.3 FFT官方支持库

        本例中使用STM32提供的DSP库函数进行FFT运算,仅支持F1系列单片机的计算。下载链接地址:STM32_FFT官方库.zip - 蓝奏云。下载后解压文件会得到如下图1所示的源码。

图1   STM32_FFT官方库

        其中可以看到官方提供了两个FFT计算的.s文件,FFT计算点数不同会直接影响到采集的数据精度,因此本文后面采用的是1024点的.s文件用于FFT计算。

        然后打开stm32_dsp.h文件可以看到,官方提供了三个计算FFT的函数,如下:

/* Radix-4 complex FFT for STM32, in assembly  */
/* 64 points*/
void cr4_fft_64_stm32(void *pssOUT, void *pssIN, u16 Nbin);
/* 256 points */
void cr4_fft_256_stm32(void *pssOUT, void *pssIN, u16 Nbin);
/* 1024 points */
void cr4_fft_1024_stm32(void *pssOUT, void *pssIN, u16 Nbin);

        其中每个参数的详细说明如下:

1. *pssOUT: FFT运算后输出的频域数组

2. *pssIN: 输入的时域采样信号数组

3. Nbin: 需要计算的FFT点数。

二、使用外设简介

2.1 ADC外设

        模数转换,即Analog-to-Digital Converter,常称ADC,是指将连续变量的模拟信号转换为离散的数字信号的器件。其工作原理涉及对输入的模拟电压进行采样,并将其量化为一定位数的数字值。关键参数包括分辨率(通常以位数表示,如8位、10位、12位等)、转换速度(样本每秒,Sps)、精度、输入通道数量以及是否支持差分输入等。

  1. 分辨率:决定了ADC能够区分的最小电压变化量,2^n,其中n是位数。例如,一个12位的ADC可以区分2^12=4096个不同的电压等级。

  2. 转换速率(Conversion Rate/Sampling Rate):单位时间内完成采样并转换的次数,通常以样本每秒(Samples Per Second, Sps)表示。

  3. 精度:除了分辨率外,还涉及到非线性误差、增益误差和偏置误差等因素,决定了转换结果与真实值的接近程度。

  4. 输入多路复用:许多ADC支持多个输入通道,通过多路复用器选择要转换的信号。

  5. 参考电压:决定了ADC能够测量的最大电压范围,通常是ADC的满量程电压。

  6. 电源和噪声抑制:良好的电源去耦和适当的布局对于减少噪声、提高转换精度至关重要。

        其中ADC部分的主要代码初始化如下:

    /* ----------------------- ADC结构体初始化 ----------------------- */
    ADC_InitTypeDef ADC_InitStructure;
    ADC_InitStructure.ADC_Mode               = ADC_Mode_Independent;
    ADC_InitStructure.ADC_ScanConvMode       = DISABLE;
    ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
    ADC_InitStructure.ADC_ExternalTrigConv   = ADC_ExternalTrigConv_T2_CC2;  /* 使用定时器2触发ADC采样 */
    ADC_InitStructure.ADC_DataAlign          = ADC_DataAlign_Right;
    ADC_InitStructure.ADC_NbrOfChannel       = 1;
    ADC_Init(ADC1, &ADC_InitStructure);
    
    ADC_RegularChannelConfig(ADC1, ADC_GPIO_CH, 1, ADC_SampleTime_28Cycles5);
    
    ADC_Cmd(ADC1,ENABLE);       /* 使能ADC */
    ADC_DMACmd(ADC1, ENABLE);
    ADC_ExternalTrigConvCmd(ADC1, ENABLE);  /* 外部触发使能 */

2.2 TIM外设

        TIM外设,全称为Timer(定时器),是嵌入式系统中常见的外设之一,它主要用于实现定时和计数功能。TIM外设可以实现如下几种功能:

  1. 定时功能:TIM外设可以基于内部时钟或者外部输入信号进行计数,当计数值达到预设值时,可以产生中断或者DMA请求,从而触发处理器执行特定任务。这使得定时器可以用来实现精确的时间延迟、周期性任务调度等。

  2. 计数功能:连接到外部输入引脚时,TIM可以作为计数器,记录外部脉冲的数量,适用于测量频率、脉冲宽度等。

  3. 输出比较:通过比较计数器的值与预设的比较值,可以控制输出引脚的状态,实现PWM(脉宽调制)信号生成,这对于电机控制、LED亮度调节等应用非常有用。

  4. 输入捕获:捕捉外部输入信号的上升沿或下降沿,可用于测量输入信号的频率、脉冲宽度,实现编码器计数等功能。

  5. 触发和同步:高级定时器还可以与其他外设(如DAC、ADC)进行同步,用于复杂的控制应用,如在数据采集时精确控制ADC的启动时间。

        其中TIM外设部分的初始化代码如下:

    /* ----------------------- TIM结构体初始化 ----------------------- */
    TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
    TIM_TimeBaseStructure.TIM_Period        = TIM_ARR - 1;          /* 自动重装载寄存器周期 */
    TIM_TimeBaseStructure.TIM_Prescaler     = TIM_PSC - 1;          /* 预分频值 */
    TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;         /* 设置时钟分割 */
    TIM_TimeBaseStructure.TIM_CounterMode   = TIM_CounterMode_Up;   /* TIM向上计数模式 */
    TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);
    
    TIM_OCInitTypeDef TIM_OCInitStructure;
    TIM_OCInitStructure.TIM_OCMode      = TIM_OCMode_PWM1;          /* 选择定时器模式:TIM脉冲宽度调制模式1 */
    TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;   /* 比较输出使能 */
    TIM_OCInitStructure.TIM_Pulse       = TIM_ARR/2;
    TIM_OCInitStructure.TIM_OCPolarity  = TIM_OCPolarity_Low;       /* 输出极性:TIM输出比较极性低 */
    TIM_OC2Init(TIM2, & TIM_OCInitStructure);                       /* 初始化外设TIM2_CH2 */
    
    TIM_Cmd(TIM2, ENABLE);

3.3 DMA外设

        DMA(Direct Memory Access,直接存储器访问)外设是一种专为高效数据传输设计的硬件控制器,它允许外部设备(如传感器、ADC、DAC、网络接口等)和系统内存(RAM)之间直接进行高速数据交换,而无需CPU的持续介入。这一机制显著减轻了CPU的负担,使得CPU可以在数据传输的同时执行其他任务,从而提高了系统的整体性能和响应速度。DMA的主要特点包括:

  1. 高效传输:DMA能够在不占用CPU资源的情况下,快速传输大量数据,特别适合于大数据量的连续传输,如音频、视频流处理,大容量存储读写等。

  2. 自动操作:DMA传输由硬件自动完成,一旦配置好DMA控制器(包括源地址、目标地址、传输量和控制参数),数据传输就会在指定的触发条件下自动开始并完成。

  3. 减少CPU干预:通过DMA,CPU不必参与每次数据的读写操作,仅在传输开始和结束时(或发生错误时)接收中断通知,这样就释放了CPU来执行其他计算密集型任务。

  4. 多通道支持:现代DMA控制器通常支持多个独立通道,每个通道可以被分配给不同的外设,同时进行数据传输,增加了系统的并行处理能力。

  5. 优先级管理:DMA控制器能够管理不同通道的优先级,确保关键数据的及时传输,特别是在多个外设同时请求DMA服务时。

  6. 灵活配置:DMA传输可以配置为单次传输或循环传输,对于需要周期性数据更新的应用(如采样数据的连续采集)非常有用。

  7. 对齐要求:为了优化传输效率,源和目标地址通常需要按照数据传输宽度(如字节、半字、全字)对齐。

        其中DMA部分的主要代码初始化如下:

    /* ----------------------- DMA结构体初始化 ----------------------- */
    DMA_InitTypeDef DMA_InitStructure;
    DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&(ADC1->DR);   /* ADC1对应地址 */
    DMA_InitStructure.DMA_MemoryBaseAddr     = (uint32_t)&adcValue[0];  /* 存储数据数组地址 */
    DMA_InitStructure.DMA_DIR                = DMA_DIR_PeripheralSRC;   /* 方向(从外设到内存) */
    DMA_InitStructure.DMA_BufferSize         = ADC_LEN;                 /* 数据长度 */
    DMA_InitStructure.DMA_PeripheralInc      = DMA_PeripheralInc_Disable;/* 外设地址固定 */
    DMA_InitStructure.DMA_MemoryInc          = DMA_MemoryInc_Enable;    /* 内存地址固定 */
    DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; /* 外设数据单位 */
    DMA_InitStructure.DMA_MemoryDataSize     = DMA_MemoryDataSize_HalfWord; /* 内存数据单位 */
    DMA_InitStructure.DMA_Mode               = DMA_Mode_Circular;       /* DMA模式 循环传输 */
    DMA_InitStructure.DMA_Priority           = DMA_Priority_High;       /* 优先级 高 */
    DMA_InitStructure.DMA_M2M                = DMA_M2M_Disable;         /* 禁止内存到内存的传输 */
    DMA_Init(ADC_DMA_CH, &DMA_InitStructure);
    
    DMA_ITConfig(ADC_DMA_CH, DMA_IT_TC, ENABLE); /* 使能DMA传输完成中断 */

三、代码设计过程

3.1 初始化

       代码的初始化分为两部分,一部分是初始化硬件引脚信息;另一部分是初始化内置结构体信息(用于保存待测频率的基本信息)。

        ①硬件引脚初始化:采用STM32F1的ADC引脚(ADC1 PA1)、DMA搬运(DMA1 通道1)、TIM定时器(TIM2 通道2),整体初始化代码设计如下:(注意打开C99 Mode

/* -------------------------- 宏定义区 -------------------------- */
/* 使用的ADC引脚 PA2 */
#define     ADC_GPIO_PORT   GPIOA
#define     ADC_GPIO_PIN    GPIO_Pin_2
#define     ADC_GPIO_CH     ADC_Channel_2

/* 这里不需改动 */
#define     ADC_TIM_CH      TIM_OC2Init
#define     ADC_DMA_CH      DMA1_Channel1

/* 定时器采集频率 */
#define     TIMx            TIM2
#define     TIM_ARR         (72u)
#define     TIM_PSC         (5u)

/* FFT采集点数 */
#define     FFT_POINTS      (1024u)
#define     ADC_LEN         (FFT_POINTS)
#define     M_PI            (3.1415926f)

/**
 * @brief  adc外设引脚初始化
 * @param  [in]
 *   @arg  None
 * @retval None
 */
static void fft_adc_pin_init(void)
{
    /* ----------------------- RCC时钟初始化 ----------------------- */
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);
    RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
    RCC_ADCCLKConfig(RCC_PCLK2_Div6);   /* 设置ADC分频因子 最大时钟不能超过14MHz */
    
    /* ----------------------- GPIO结构体初始化 ----------------------- */
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Pin   = ADC_GPIO_PIN;
    GPIO_InitStructure.GPIO_Mode  = GPIO_Mode_AIN;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(ADC_GPIO_PORT, &GPIO_InitStructure);
    
    /* ----------------------- ADC结构体初始化 ----------------------- */
    ADC_InitTypeDef ADC_InitStructure;
    ADC_InitStructure.ADC_Mode               = ADC_Mode_Independent;
    ADC_InitStructure.ADC_ScanConvMode       = DISABLE;
    ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
    ADC_InitStructure.ADC_ExternalTrigConv   = ADC_ExternalTrigConv_T2_CC2;  /* 使用定时器2触发ADC采样 */
    ADC_InitStructure.ADC_DataAlign          = ADC_DataAlign_Right;
    ADC_InitStructure.ADC_NbrOfChannel       = 1;
    ADC_Init(ADC1, &ADC_InitStructure);
    
    ADC_RegularChannelConfig(ADC1, ADC_GPIO_CH, 1, ADC_SampleTime_28Cycles5);
    
    ADC_Cmd(ADC1,ENABLE);       /* 使能ADC */
    ADC_DMACmd(ADC1, ENABLE);
    ADC_ExternalTrigConvCmd(ADC1, ENABLE);  /* 外部触发使能 */
    
    /* ----------------------- TIM结构体初始化 ----------------------- */
    TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
    TIM_TimeBaseStructure.TIM_Period        = TIM_ARR - 1;          /* 自动重装载寄存器周期 */
    TIM_TimeBaseStructure.TIM_Prescaler     = TIM_PSC - 1;          /* 预分频值 */
    TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;         /* 设置时钟分割 */
    TIM_TimeBaseStructure.TIM_CounterMode   = TIM_CounterMode_Up;   /* TIM向上计数模式 */
    TIM_TimeBaseInit(TIMx, &TIM_TimeBaseStructure);
    
    TIM_OCInitTypeDef TIM_OCInitStructure;
    TIM_OCInitStructure.TIM_OCMode      = TIM_OCMode_PWM1;          /* 选择定时器模式:TIM脉冲宽度调制模式1 */
    TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;   /* 比较输出使能 */
    TIM_OCInitStructure.TIM_Pulse       = TIM_ARR/2;
    TIM_OCInitStructure.TIM_OCPolarity  = TIM_OCPolarity_Low;       /* 输出极性:TIM输出比较极性低 */
    ADC_TIM_CH(TIMx, & TIM_OCInitStructure);                        /* 初始化外设TIMx_CHx */
    
    TIM_Cmd(TIMx, ENABLE);
    
    /* ----------------------- DMA结构体初始化 ----------------------- */
    DMA_InitTypeDef DMA_InitStructure;
    DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&(ADC1->DR);   /* ADC1对应地址 */
    DMA_InitStructure.DMA_MemoryBaseAddr     = (uint32_t)&adcValue[0];  /* 存储数据数组地址 */
    DMA_InitStructure.DMA_DIR                = DMA_DIR_PeripheralSRC;   /* 方向(从外设到内存) */
    DMA_InitStructure.DMA_BufferSize         = ADC_LEN;                 /* 数据长度 */
    DMA_InitStructure.DMA_PeripheralInc      = DMA_PeripheralInc_Disable;/* 外设地址固定 */
    DMA_InitStructure.DMA_MemoryInc          = DMA_MemoryInc_Enable;    /* 内存地址固定 */
    DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; /* 外设数据单位 */
    DMA_InitStructure.DMA_MemoryDataSize     = DMA_MemoryDataSize_HalfWord; /* 内存数据单位 */
    DMA_InitStructure.DMA_Mode               = DMA_Mode_Circular;       /* DMA模式 循环传输 */
    DMA_InitStructure.DMA_Priority           = DMA_Priority_High;       /* 优先级 高 */
    DMA_InitStructure.DMA_M2M                = DMA_M2M_Disable;         /* 禁止内存到内存的传输 */
    DMA_Init(ADC_DMA_CH, &DMA_InitStructure);
    
    DMA_ITConfig(ADC_DMA_CH, DMA_IT_TC, ENABLE); /* 使能DMA传输完成中断 */
    
    /* ----------------------- NVIC结构体初始化 ----------------------- */
    NVIC_InitTypeDef NVIC_InitStructure;
    NVIC_InitStructure.NVIC_IRQChannel                   = DMA1_Channel1_IRQn;  /* DMA中断服务函数 */
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority        = 3;
    NVIC_InitStructure.NVIC_IRQChannelCmd                = ENABLE;
    NVIC_Init(&NVIC_InitStructure);
    
    DMA_Cmd(ADC_DMA_CH, ENABLE);
    
    ADC_ResetCalibration(ADC1);                 /* 使能复位校准 */
    while(ADC_GetResetCalibrationStatus(ADC1)); /* 等待复位校准结束 */
    ADC_StartCalibration(ADC1);                 /* 开启AD校准 */
    while(ADC_GetCalibrationStatus(ADC1));      /* 等待校准结束 */
}

        ②内置结构体信息初始化:该结构体的成员变量如下:

typedef struct {
    unsigned char flag;         /* DMA数据搬运完成标志位 */
    unsigned short index;       /* 频率所在的fft数组索引 */
    unsigned short amplitude;   /* 幅值 mV */
    float sample_freq;          /* 采样频率 根据奈奎斯特采样定理 待测频率需小于等于采样频率的1/2 */
    float freq;                 /* 待测频率 hz */
    float phase;                /* 相位 ° */
} STRUCT_WAVE_TYPEDEF;

        结构体实例化代码如下:

static STRUCT_WAVE_TYPEDEF wave;                /* 定义内置结构体 保存波形信息 */

/**
 * @brief  wave结构体初始化
 * @param  [in]
 *   @arg  None
 * @retval None
 */
static void fft_struct_init(void)
{
    wave.flag = 0;
    wave.index = FFT_POINTS / 2;
    wave.sample_freq = SystemCoreClock/(TIM_ARR*TIM_PSC);
    
    wave.freq = 500;
    wave.amplitude = 3300;
    wave.phase = 0;
}

3.2 DMA中断数据处理

        我们在通过前面的一系列配置后,完成了DMA的数据搬运配置工作,然后我们需要在DMA的中断服务函数中去拿数据,代码配置如下:

/**
 * @brief  DMA1通道1中断服务函数
 * @param  [in]
 *   @arg  None
 * @retval None
 */
void DMA1_Channel1_IRQHandler(void)
{
    if(DMA_GetITStatus(DMA1_IT_TC1) == SET) {
        DMA_ClearITPendingBit(DMA1_IT_TC1); /* 清除中断标志位 */
        wave.flag = 1;
        DMA_Cmd(ADC_DMA_CH, DISABLE); /* 失能DMA1通道1 以防在中断过程中发生DMA传输 */
    }
}

        这里主要是在中断服务函数触发中断标志后,代表ADC数据获取成功,将标志位置为1,等待后续运算。

3.3 其他自定义函数

        我们通过DMA中断服务函数触发是否成功来判定ADC数据是否获取成功,当获取成功后,我们需要使用ADC的数据进行FFT的运算,该部分代码如下:

        /* 1. 将得到的ADC数据处理后放入fft_input_buff数组中 */
        for(i=0; i<FFT_POINTS; i++) {
            fft_input_buff[i] = ((signed short)(adcValue[i])) << 16;  /* 左移16位 高16位存放实部 */
        }
        /* 2. 调用FFT库计算 */
        cr4_fft_1024_stm32(fft_output_buff, fft_input_buff, FFT_POINTS);

        而完成FFT运算之后,如果我们需要得到信号的频率、幅值、相位等信息还要计算各次谐波的幅度值,拿到幅度值之后,就可以通过幅度值数组计算波形的信息了。该函数定义如下:

/**
 * @brief  取模获取信号谐波分量的幅值
 * @param  [in]
 *   @arg  None
 * @retval None
 */
static void fft_get_harmonic_data(void)
{
    float X,        /* 实部分量 */
          Y,        /* 虚部分量 */
          magnitude;/* 幅值 */
    unsigned short i;
    signed short iX, iY;
    
    const float scale_factor = (FFT_POINTS / 32768.0f);     /* 预先计算缩放因子 避免循环内重复计算 */
    
    for (i=0; i <FFT_POINTS/2; i++) {
        /* fft_output_buff是FFT的频谱数组 为复数形式 Z = X + bi */
        iX = (signed short)(fft_output_buff[i] >> 16);      /* 提取复数的实部数据 高16位并确保是有符号的 */
        iY = (signed short)(fft_output_buff[i] & 0xFFFF);   /* 提取复数的虚部数据 低16位并确保是有符号的 */
        
        /* 使用预先计算的缩放因子进行计算 */
        X = iX * scale_factor;
        Y = iY * scale_factor;
        magnitude = sqrtf(X * X + Y * Y); /* 使用sqrtf 针对浮点数优化的sqrt版本 */
        
        iX = (signed short)((int)magnitude >> 16);
        iY = (signed short)((int)magnitude & 0xFFFF);
        phase_buff[i] = atan2f(iY, iX) * 180 / M_PI;
        
        /* 根据索引决定最终缩放并存储 特殊处理直流分量 */
        amplitude_buff[i] = (magnitude * ((i == 0) ? 32768.0f : 65536.0f)); /* 第0个点是直流信号(低频)的幅值 */
    }
}

1. 计算信号频率

        我们可以通过上面函数(fft_get_harmonic_data)可以获取到波形信号的所有谐波分量的幅值,当找到该幅值数组harmonic_buff中的最大幅值即为对应的待测波形频率(Fs),可通过如下公式计算得出:

Fs = (F0 / N) * max

        其中F0是采样频率,N是FFT的点数,max是数组中最大幅值对应的索引。这个频率Fs即为信号的主要频率成分,即我们需要的待测频率。因此我们可以通过函数findMax_index找到数组中的最大值索引值,函数代码如下:

/**
 * @brief  查找数组中的最大值所在的数组索引
 * @param  [in]
 *   @arg  *arr - 待查找数组
 *   @arg  len - 查找数组长度
 * @retval None
 */
static int findMax_index(int *arr, int len)
{
    int i;
    int max_val = arr[0];
    int max_index = 0;
    
    if (len <= 0) {
        return -1; /* 错误值 表示数组为空或非法大小 */
    }
    for (i = 1; i < len; i++) {   /* 遍历数组 */
        if (arr[i] > max_val) { /* 如果找到更大的值 */
            max_val = arr[i];
            max_index = i;      /* 更新最大值及其索引 */
        }
    }
    return max_index;
}

2. 计算信号幅度

        当我们得到信号的频率后,信号的峰值幅度可以通过harmonic_buff[max]计算得到,但通常需要根据FFT的具体实现进行适当的缩放。最终的峰峰值Vpp(即信号的最大幅度变化范围)可以通过公式获得,公式如下:

Vpp = harmonic_buff[max] / (N / 2)

        这里的缩放因子N/2是因为FFT的结果包含了正负频率部分,且对于实数信号,能量分布在正负频率上是相同的,所以原始信号的能量需要平均分配到这两个部分。

3. 计算信号相位

        然后我们为了获取信号的相位信息,需要利用FFT结果的复数形式。假设out_buff[max]可以表示为一个复数,拆分为实部a和虚部b,即:

out_buff[max] = a + bi

        其中实部(a)和虚部(b)的计算公式如下:

实部:a = out_buff[max] >> 16

虚部:b = out_buff[max] & 0xFFFF

        注意:这里假设out_buff[max]是经过适当缩放以适应整数表示的复数。因此信号的相位可以通过atan2(a, b)函数计算得到,该函数返回的是弧度值。atan2相比普通的atan函数,能正确处理所有象限的角度,避免了正负符号和0的问题。

4. 完整代码流程

/**
 * @brief  fft采集并计算波形信息
 * @param  [in]
 *   @arg  None
 * @retval None
 * @note   该函数需定时执行 注意定时执行周期不宜太短
 */
void fft_measure_handle(void *priv)
{
    unsigned short i = 0;
    signed short real_part, imaginary_part; /* 定义实部和虚部分量 */
    double sqrt_harmonic_sum = 0;  /* 所有谐波分量的平方和 */
    
    if(wave.flag == 1) {
        /* 1. 将得到的ADC数据处理后放入fft_input_buff数组中 */
        for(i=0; i<FFT_POINTS; i++) {
            fft_input_buff[i] = ((signed short)(adcValue[i])) << 16;  /* 左移16位 高16位存放实部 */
        }
        /* 2. 调用FFT库计算 */
        cr4_fft_1024_stm32(fft_output_buff, fft_input_buff, FFT_POINTS);
        
        /* 3. 获取信号谐波分量的幅值 */
        fft_get_harmonic_data();
        
        /* 4. 获取fft频率数组最大值索引 */
        wave.index = findMax_index(&amplitude_buff[1], FFT_POINTS / 2 - 1) + 1; /* 第0个点是FFT直流分量 所以从第一个数据开始取 索引值index需要加1 */
        
        /* 5. 计算波形的频率 幅值 相位等信息 */
        wave.freq = wave.index * wave.sample_freq / (float)FFT_POINTS;  /* 频率计算公式: Fs = index * (F0 / points) */
        wave.amplitude = 2 * amplitude_buff[wave.index] / FFT_POINTS;   /* 幅值计算公式: Vpp = 2 * buff[index] / points */
        
        wave.phase = phase_buff[wave.index];
        
        /* 6. 计算总谐波失真 */
        for(i=1; i<FFT_POINTS/2; i++) {
            sqrt_harmonic_sum += (float)amplitude_buff[i] * (float)amplitude_buff[i];
        }
        wave.THD = sqrt(sqrt_harmonic_sum) / (float)amplitude_buff[0];
        
        /* 7. 将flag置0 开启下一次DMA搬运 */
        wave.flag = 0;
        
        /* 8. 重新使能DMA1通道1 开始工作 */
        DMA_Cmd(ADC_DMA_CH, ENABLE);
    }
}

5. 注意事项

  • 确保在进行相位计算前,正确地将out_buff[max]转换为实部和虚部,且考虑是否需要进一步的缩放以匹配实际的物理量。
  • 计算得到的相位可能需要根据具体应用场景调整,比如从弧度转换为度,或根据信号起始相位进行校正。
  • 在实际应用中,还需考虑FFT窗口效果、频谱泄露等对结果的影响,可能需要采取相应的预处理或后处理措施来提高分析准确性。

3.4 采样频率选择及配置      

        由于在初始化中是使用的定时器+DMA采集ADC引脚输入的波形频率,因此指定定时器的频率大小就是指定采样频率的大小,而根据奈奎斯特采样定理可知,采样频率(定时器的频率)需大于等于待测频率的2倍才行,通常选择3~5倍左右。因此我们可以通过指定定时器的频率从而指定待测ADC输入波形的可被采集的频率范围。定时器频率设置如下:

/**
 * @brief  设置定时器采样频率(标准频率)
 * @param  [in]
 *   @arg  定时器采样频率 单位Hz
 * @retval None
 */
void fft_set_sample_freq(float freq)
{
    float period;
    unsigned short psc;
    unsigned short arr;
    wave.sample_freq = freq;
    period = SystemCoreClock / freq; /* 获取周期值 */
    
    TIM_ARRPreloadConfig(TIMx, DISABLE);
    if(period < 720) { /* psc*arr小于720 采用重装载值的方式更新定时器频率 */
        psc = 1; // 预分频器设为1
        arr = (unsigned short)(period / (psc + 1));
        TIMx->ARR = arr - 1;
        TIMx->CCR2 = (TIMx->ARR + 1) / 2;/* 保证占空比为50% */
    } else { /* psc*arr大于等于720 采用预分频值的方式更新定时器频率 */
        psc = (unsigned short)(period / 720); // 设置预分频器 确保 period / (psc + 1) < 720
        arr = (unsigned short)(period / (psc + 1));
        TIMx->ARR = arr - 1;
        TIMx->CCR2 = (TIMx->ARR + 1) / 2; // 保证占空比为50%
    }
    TIMx->PSC = psc;
    TIMx->EGR = TIM_PSCReloadMode_Immediate;
    TIM_PrescalerConfig(TIMx, psc, TIM_PSCReloadMode_Immediate);
    TIM_ARRPreloadConfig(TIMx, ENABLE);
    TIM_OC2PreloadConfig(TIMx, TIM_OCPreload_Enable);
}

        这里需要注意:不能使用太高的定时器采样频率,因为太高的定时器采样频率会占用CPU的资源,可能会导致采样失败。

四、调用示例

        关于代码如何使用,详见下部分代码:

/* 硬件层 */
#include "delay.h"
#include "usart.h"
#include "timer.h"
#include "fft_measure.h"

/* ---------------- 变量定义区 ---------------- */
float freq, phase, thd;  /* ADC采集得到的频率 */
unsigned short amplitude;
static unsigned short fft_timer_id = 0;

int main(void)
{
    delay_init();                // 延时函数初始化
    usart1_init(115200);         // 串口1初始化
    basic_tim_init();            // 定时器任务矩阵初始化
    
    fft_measure_init();          // ADC+TIM+DMA硬件初始化
    fft_set_sample_freq(280000); // 通过该函数可自定义采样频率
    
    /* 定时任务创建 */
    if(fft_timer_id == 0) {
        fft_timer_id = timer_add_task(fft_measure_handle, NULL, 100);
    }
    
    while (1) {
        delay_ms(100);
        if(fft_get_convert_flag()) {
            freq = fft_get_freq();              /* 获取频率 */
            amplitude = fft_get_amplitude();    /* 获取振幅 */
            phase = fft_get_phase();            /* 获取相位 */
            thd = fft_get_THD();                /* 获取总谐波失真*/
            Debug("freq:%.3fHz  amp:%dmV  phase:%.3f°  thd:%.3f%%  %.3fdB\r\n", freq, amplitude, phase, thd, fft_get_THD_dB());
        }
    }
}

        关于上部分中,定时器任务的创建主要使用的是自写的任务矩阵定时器设计,需要的可私信(事先声明有偿的,项目开源,知识付费,给个打赏就行)。

1. 通过函数 fft_set_sample_freq 可自定义采样频率,但测试下来这是极限频率了,如果需要更高的采样频率,可以根据需要改动源码中的代码结构,需要的可自行实现。

2. 通过函数 timer_add_task 可实现定时器任务矩阵的定时器任务创建,函数的第一个参数是定时器需实现的函数;第二个参数是该函数的参数;第三个参数是该函数调用的周期,单位为ms。

        串口数据打印部分示例见下图2:

图2   串口数据打印部分示例

五、频率测量现象演示

5.1 采样频率为20Khz

        源码中定时器初始频率设置为20KHz,因此可测得的频率范围为10Khz以下,当在信号发生器一端设置波形为正弦波、频率为9500Hz、占空比为50%、幅度值为3.3V时,设置的图像见下图3:

图3   信号发生器设置1

        此时使用Keil 自带的debug窗口进行观测,其测得的波形信息如下图4所示:

图4   wave波形信息1

        由上图可以得知,波形的频率与信号发生器设置的频率大体一致,由些许误差,误差率为:(9500-9492.1875) / 9500 = 0.000822,可以看到测量精度还是挺高的。

5.2 采样频率为1Khz

        我们可以通过调用函数fft_set_sample_freq即可任意设置定时器的频率,即采样频率,当然也可以在初始化部分去配置定时器的重装载值和预分频值。见下图5。

图5   重新定义定时器采样频率

        接着需要在信号发生器一端设置波形:更改为方波、频率为300Hz、占空比为50%、幅度值为3.3V时,设置的图像见下图6:

图6   信号发生器设置2

        接着重新进入Keil的debug进行调试,可以看到如下图7所示波形信息:

图7   wave波形信息2

        通过上图可以看到,方波的幅值和相位都比较接近信号发生器的波形数据,待测频率与信号发生器的差别也不算很大,偏差率为:(335-337.890625)/335 = -0.00863

六、FFT误差原因分析及减小误差策略

6.1 误差原因分析

  1. FFT分辨率与采样频率:FFT的分辨率确实由采样频率F0与FFT点数N决定,即分辨率=F0/N。提高分辨率意味着能更精细地分辨相邻频率成分,在保持N不变时,增加F0会提高分辨率。而为了减小低频信号测量的误差,应当降低采样频率,但必须遵循奈奎斯特采样定理,即采样频率至少为信号最高频率的两倍。

  2. ADC读取延迟:ADC转换存在采样时间和转换时间,这导致实际采样间隔与理论设定值不符,从而引入误差。可以通过优化ADC配置,如使用更快的采样率、减少采样保持时间、采用更高的转换精度等方法来减少这种误差。

  3. 量化误差:ADC将连续的模拟信号量化为有限位数的数字信号,此过程必然会产生量化误差,尤其是在信号幅度接近ADC分辨率极限时。选择更高位数的ADC可以减小量化误差。

  4. 频谱泄漏:非整周期信号在进行FFT时会产生频谱泄漏,导致能量分散到邻近的频率上,影响频率成分的准确识别。采用窗函数可以一定程度上减少频谱泄漏。

  5. DC偏移和噪声:信号中的直流偏移和随机噪声也会干扰FFT分析,影响幅度和频率的准确度。预处理如高通滤波去除DC偏移、低通滤波减少噪声等可以改善分析质量。

6.2 FFT的优缺点补充

1. 优点

  • 高精度与宽频率范围:FFT能够提供高精度的频谱分析,理论上可测量高达几百万赫兹的信号频率,远超过传统的时域测量方法,适用于复杂信号的分析。

  • 微弱信号检测:FFT具有很高的灵敏度,即使信号幅度很小,也能通过累计多次采样来检测到,适用于微弱信号的分析,如电磁辐射、声纳信号等。

  • 多信号同时分析:在单一FFT结果中,可以同时观察到信号中的多种频率成分,便于分析信号的组成结构。

2. 缺点

  • 计算复杂度:虽然FFT相对于DFT大大降低了计算复杂度,但对于大数据量的处理仍然需要较大的计算资源。

  • 低频信号处理时间长:对于低频信号,由于其周期长,需要较长的采样时间才能获得足够的周期数,导致分析过程耗时较长。

  • 系统资源占用:在高采样率下,特别是配合DMA使用时,FFT处理和数据搬运会占用较多的系统资源,影响实时性,可能导致主循环响应变慢。

  • 频谱泄漏与混叠:如前所述,FFT分析易受频谱泄漏和频率混叠的影响,需要合理选择窗函数和采样率来减小这些影响。

---------------------------------------------------------------------------------------------------------------------------------

分割线(以下是续写内容)

---------------------------------------------------------------------------------------------------------------------------------

七、总谐波失真计算

        前面实现了基于FFT的频率采集,后续学习了一些FFT的相关理论后,补充如下概念:

        总谐波失真(Total Harmonic Distortion,简称THD)是衡量信号失真程度的重要指标,特别是在音频、电力电子设备和通信系统等领域。THD表示一个信号中谐波分量(除了基波之外的频率分量)相对于基波的比例。具体来说,THD是所有谐波分量的平方和与基波分量平方的比值的平方根,通常用百分比表示。其公式表示为:

图8   总谐波失真计算公式

        其中:V1是基波幅度,V2,V3……Vn是谐波幅度。

/**
 * @brief  fft采集并计算波形信息
 * @param  [in]
 *   @arg  None
 * @retval None
 * @note   该函数需定时执行 注意定时执行周期不宜太短
 */
void fft_measure_handle(void *priv)
{
    unsigned short i = 0;
    double sqrt_harmonic_sum = 0;  /* 所有谐波分量的平方和 */
    
    if(wave.flag == 1) {
        /* 1. 将得到的ADC数据处理后放入fft_input_buff数组中 */
        for(i=0; i<FFT_POINTS; i++) {
            fft_input_buff[i] = ((signed short)(adcValue[i])) << 16;  /* 左移16位 高16位存放实部 */
        }
        /* 2. 调用FFT库计算 */
        cr4_fft_1024_stm32(fft_output_buff, fft_input_buff, FFT_POINTS);
        
        /* 3. 获取信号谐波分量的幅值 */
        fft_get_harmonic_data();
        
        /* 4. 获取fft频率数组最大值索引 */
        wave.index = findMax_index(&amplitude_buff[1], FFT_POINTS / 2 - 1) + 1; /* 第0个点是FFT直流分量 所以从第一个数据开始取 索引值index需要加1 */
        
        /* 5. 计算波形的频率 幅值 相位等信息 */
        wave.freq = wave.index * wave.sample_freq / (float)FFT_POINTS;  /* 频率计算公式: Fs = index * (F0 / points) */
        wave.amplitude = 2 * amplitude_buff[wave.index] / FFT_POINTS;   /* 幅值计算公式: Vpp = 2 * buff[index] / points */
        wave.phase = phase_buff[wave.index];
        
        /* 6. 计算总谐波失真 */
        for(i=1; i<FFT_POINTS/2; i++) {
            sqrt_harmonic_sum += (float)amplitude_buff[i] * (float)amplitude_buff[i];
        }
        wave.THD = sqrt(sqrt_harmonic_sum) / (float)amplitude_buff[0];
        
        /* 7. 将flag置0 开启下一次DMA搬运 */
        wave.flag = 0;
        
        /* 8. 重新使能DMA1通道1 开始工作 */
        DMA_Cmd(ADC_DMA_CH, ENABLE);
    }
}

        故周期运算中增加该部分算法,实现如第7步步骤所示。设计驱动函数中的接口函数如下:

/**
 * @brief  获取FFT的总谐波失真 单位%
 * @param  [in]
 *   @arg  None
 * @retval FFT总谐波失真
 */
float fft_get_THD(void)
{
    if(wave.flag == 0) {
        return (wave.THD);
    }
    return (0);
}

/**
 * @brief  获取FFT的总谐波失真 单位dB
 * @param  [in]
 *   @arg  None
 * @retval FFT总谐波失真
 */
float fft_get_THD_dB(void)
{
    if(wave.flag == 0) {
        return (20 * log10(wave.THD));
    }
    return (0);
}

        后续将逐渐补充其他相关知识点......

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1971621.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

G-EVAL: NLG Evaluation using GPT-4 with Better Human Alignment

文章目录 题目摘要方法实验分析相关工作结论 题目 G-EVAL:使用GPT-4进行NLG评估&#xff0c;具有更好的人类一致 论文地址&#xff1a;https://arxiv.org/abs/2303.16634 项目地址&#xff1a;https://github.com/nlpyang/geval 摘要 自然语言生成(NLG)系统生成的文本质量难以…

网站开发涉及到的技术内容介绍——前端

一、网站开发涉及前后端交互模式 一般来说网站项目分为两种【前后端不分离、前后端分离】,其中前后端不分离的项目是(后端语言可以直接与前端的html等内容糅合在一起,后端语言可以操作数据进行动态渲染前端的html文件进行展示)。后端分离的项目是(后端语言不会直接对前端的…

c++| c++11左右值引用,完美转发,可变参数模板

c| c11的新特性 左&#xff0c;右值引用什么是左值&#xff0c;右值左值引用和右值引用右值引用解决什么问题呢&#xff1f;移动构造万能引用形式 完美转发格式 lambada表达式格式 可变参数模板可变参数模板实现打印不同类型emplace_push以list的emplace_back的实现举例 左&…

掌握 HTTP 请求的艺术:理解 cURL GET 语法

在 Web 开发和服务器通信领域&#xff0c;掌握 HTTP 请求对于寻求有效与 Web 服务和 API 交互的开发人员至关重要。在众多可用于发出 HTTP 请求的工具中&#xff0c;cURL 脱颖而出&#xff0c;成为全球开发人员信赖的强大且多功能的命令行工具。在这篇博文中&#xff0c;我们将…

MySQL分组查询有关知识总结

目录 4. 分组查询&#xff08;group by&#xff09; 4.1 概述 4.2 分组函数 4.2.1 单个使用 4.2.2 组合使用 4.2.3 注意&#xff01; 4.3 group by 4.3.1 单个字段 4.3.2 多个字段 4.3.3 提醒&#xff01; 4.4 having 4.5 分组查询演示 4. 分组查询&#xff08;…

海外仓怎么入局?货代还有发展的空间吗?

很多货代小伙伴可能都有一个疑惑&#xff1a;海外仓虽然火爆&#xff0c;但是怎么入局呢&#xff1f;显然&#xff0c;自建海外仓的试错代价太高了&#xff0c;成本高、选址难、管理起来更是一地鸡毛。既然无法自建&#xff0c;那转换赛道&#xff0c;管理别人的仓库&#xff0…

iOS面试之属性关键字(二):常见面试题

Q:ARC下&#xff0c;不显式指定任何属性关键字时&#xff0c;默认的关键字都有哪些&#xff1f; 对应基本数据类型默认关键字是:atomic,readwrite,assign 对于普通的 Objective-C 对象:atomic,readwrite,strong Q&#xff1a;atomic 修饰的属性是怎么样保存线程安全的&#x…

如何将本地下载的切片电影进行合成一个视频

合成后的时长1&#xff1a;41 没有合成前是50个电影切片 注明:电影切片不是真实的切片名称而是自定义从0-50的数字 import requests # with open("电影.m3u8","r") as f: # n0 # for line in f: # if line.startswith("#"): …

SpringBoot项目如何安装Selenium自动化(详解)

目录 一、打开intellij idea&#xff0c;创建Maven项目 二、添加依赖 三、在Test路径下创建自动化文件 3.1 项目结构 3.2 代码 四、运行自动化 前言&#xff1a; java版本最低要求为8。电脑至少已安装一种浏览器&#xff0c;如&#xff1a;Chrome&#xff08;推荐&#xff09;、…

十日Python项目——第五日(商品数据)

#前言&#xff1a; 在最近十天我会用Python做一个购物类项目&#xff0c;会用到DjangoMysqlRedisVue等。 今天是第五天&#xff0c;主要负责撰写响应具体的商品数据。若是有不懂大家可以先阅读我的前四篇博客以能够顺承。 若是大家基础有不懂的&#xff0c;小编前面已经关于…

RJ45空包弹网口描述与应用

RJ45空包弹网口&#xff0c;通常指的是RJ45接口的空芯线缆&#xff08;通常称为“空包”&#xff09;和相应的连接器。这种线缆和连接器组合常用于网络布线中&#xff0c;特别是在需要将网络信号从一端传输到另一端&#xff0c;同时保持信号完整性和隔离性的场合。 描述&#…

C++初阶:list的使用和模拟实现

关于list可以先看一下这个文档&#xff1a;list文档 一.list的介绍和使用 1.1 list的介绍 list实际上就是链表&#xff0c;是带头双向循环链表。 1.2 list的使用 list的使用跟我们以前用C语言实现时的一样。push&#xff0c;pop&#xff0c;insert等等。 1.2.1list的构造 …

【从零开始一步步学习VSOA开发】开发环境搭建

开发环境搭建 开发 VSOA 首先需要搭建开发环境&#xff0c;这里讲解 Windows 下 C/C 开发环境搭建方法。 下载 IDE 并申请授权码 SylixOS 的开发和部署需要 RealEvo-IDE 的支持&#xff0c;因此您需要先获取 RealEvo-IDE 的安装包和注册码。 RealEvo-IDE 分为体验版和商业版…

简单的 微服务netflix 学习

简单的 微服务netflix 学习 一.Eureka 学习 1. 服务简单搭建 1.1 首先确定pom文件 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-…

FPGA入门系列:计数器

目录 微信公众号获取更多FPGA相关源码&#xff0c;且微信公众号为首发&#xff0c;第一时间获取最新文章&#xff1a; 计数器是一种典型的时序器件&#xff0c;常用于对时钟脉冲的个数进行计数&#xff0c;还用于定时、分频、产生同步脉冲等 按触发方式分&#xff1a;同步计数…

国内SSL证书颁发机构哪家服务更优质?

SSL证书作为保障网站数据传输安全的关键工具&#xff0c;其重要性不言而喻。选择一个可靠的SSL证书代理商&#xff0c;不仅能够提供多样化的证书类型&#xff0c;而且能在众多品牌中进行比较&#xff0c;选择最适合自己的、性价比更高的产品。此外&#xff0c;优质的代理商还能…

JAVA项目基于Spring Boot的多媒体素材库

目录 一、前言 二、技术介绍 三、项目实现流程 四、论文流程参考 五、核心代码截图 专注于大学生实战开发、讲解和毕业答疑等辅导&#xff0c;获取源码后台 一、前言 信息化管理时代已至&#xff0c;计算机与互联网技术深度融合于生活点滴。本系统源于对用户需求的深刻洞…

Spark学习之SaprkCore

FlinkCore 1、JavaAPI 1、创建一个Topic并写入数据 向Kafka写数据 如果topic不存在则会自动创建一个副本和分区数都是1的topic package com.shujia.kafka;import org.apache.kafka.clients.producer.KafkaProducer; import org.apache.kafka.clients.producer.ProducerRecord…

JAVASpring学习Day2

面向切面编程 (AOP) 面向切面编程是一种编程范式&#xff0c;用于在程序中分离关注点&#xff0c;例如日志记录、事务管理和安全性。它主要由以下几个关键组成部分构成&#xff1a; 连接点 (Join Point)&#xff1a;在程序执行过程中可以插入切面的点&#xff0c;通常是方法的…

I2C 设备驱动编写流程

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、修改设备树1、 IO 修改或添加2、在 i2c1 节点追加 ap3216c 子节点3、查看设备树节点创建是否成功 二、AP3216C 驱动编写 前言 提示&#xff1a;这里可以添…