赞
踩
本文代码可以实现用单片机对一个未知频率的正弦波进行采样,经过运算后得到该频率。
实现方法为:通过定时器设置采样频率,DMA对采样到的数据进行传输,使用FFT对采样到的波形数据进行傅里叶变换,得到傅里叶序列,对其进行计算得到幅频特性序列,当幅值最大时对应的序列就是该未知波形的频率(序列需要乘采样频率再除采样点数)。
这里说一下如何配置采样频率,采样频率是完全由定时器来掌控的,通过改变下面这两个值
TIM_TimeBaseInitStructure.TIM_Period = 279; // 修改为满足采样频率的周期值
TIM_TimeBaseInitStructure.TIM_Prescaler = 1 - 1; // 修改为满足采样频率的预分频器值
预分频比较好理解,就是将自己单片机的时钟分频几次(除几)。例如我的单片机时钟为72MHz,如果我设置预分频器值为6,那么现在的时钟为72M÷6=12M。
而周期是在基础上再进行分割,例如我这里设置的周期为280,在计算时需要+1处理。72M÷281=256.2277KHz。
还有一个转换时间的问题,可以通过下面两行代码来更改。
RCC_ADCCLKConfig(RCC_PCLK2_Div4);//分频
ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 1, ADC_SampleTime_7Cycles5); //最后一个参数为采样时间的长度,单位是ADC时钟周期
分频和刚才说的一样,就是将ADC的时钟分频几次(除几)。这里我ADC的时钟与单片机的一致,分频4,故时钟为72MHz÷4=18MHz。
最后一个参数是指采样的时间长度。
This parameter can be one of the following values:
* @arg ADC_SampleTime_1Cycles5: Sample time equal to 1.5 cycles
* @arg ADC_SampleTime_7Cycles5: Sample time equal to 7.5 cycles
* @arg ADC_SampleTime_13Cycles5: Sample time equal to 13.5 cycles
* @arg ADC_SampleTime_28Cycles5: Sample time equal to 28.5 cycles
* @arg ADC_SampleTime_41Cycles5: Sample time equal to 41.5 cycles
* @arg ADC_SampleTime_55Cycles5: Sample time equal to 55.5 cycles
* @arg ADC_SampleTime_71Cycles5: Sample time equal to 71.5 cycles
* @arg ADC_SampleTime_239Cycles5: Sample time equal to 239.5 cycles
这些是官方所给的参数,到时候可根据实际情况带进去算,选择最合适的。
如何带入合适的参数呢
第一个方面
首先要自己确定好采样频率,必须要适中(大了小了都不行)。先解释为什么不能太小,采样定理规定采样频率必须要大于所采波形频率的2倍,如果采样频率定的很小,那么所采的波形范围就会变小。
之后要确定一下采样点数,采样点数的话当然是越多越好(越精确嘛)但是采样到的数据是要存在单片机里的,所以太多了单片机也存不下。一般采样点数为64、256、1024(官方给的库里就是这三个参数)。
最后用采样频率÷采样点数就得到了分辨率。举个例子,我这个代码中采样频率为256KHz,采样点数为256,所以采样的分辨率就是1KHz。
这个公式也解释了为什么采样频率不能定的很大,这样的话会导致分辨率变得很大。
第二个方面
采样频率有了,求倒数可得采样周期,也就是每隔多长时间采一个点。但是单片机的ADC采样不可能一瞬间就能把采到的模拟量转换为数字量,需要一段转换时间,所以转换时间一定要小于采样周期,否则得到的采样数据就会有问题。
转换时间太短也不好,可能会导致采样的值不准确(最理想的情况就是转换时间与采样周期相等)。
转换时间=采样时间长度÷时钟周期
下面就是关于配置ADC的全部代码
void ADC_GPIO_Configuration(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //使能GPIOA时钟 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1; //管脚 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入模式 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); //GPIO组 } void ADC_TIM3_Configuration(void) { TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); // 时钟分频(预分频器值)和周期值设置为满足 256 kHz 采样频率的配置 TIM_TimeBaseInitStructure.TIM_Period = 280; // 修改为满足采样频率的周期值 TIM_TimeBaseInitStructure.TIM_Prescaler = 1 - 1; // 修改为满足采样频率的预分频器值 TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM3, &TIM_TimeBaseInitStructure); TIM_SelectOutputTrigger(TIM3, TIM_TRGOSource_Update); TIM_Cmd(TIM3, ENABLE); } void ADC_DMA_NVIC_Configuration(void) { NVIC_InitTypeDef NVIC_InitStructure; NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel1_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); DMA_ClearITPendingBit(DMA1_IT_TC1); DMA_ITConfig(DMA1_Channel1,DMA1_IT_TC1,ENABLE); } void ADC_DMA_Configuration(void) { DMA_InitTypeDef DMA_InitStructure; RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1,ENABLE); DMA_DeInit(DMA1_Channel1); DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&ADC1->DR; DMA_InitStructure.DMA_MemoryBaseAddr = (u32)ADC_SourceData; DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; DMA_InitStructure.DMA_BufferSize = SAMPLS_NUM; 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_InitStructure.DMA_Priority = DMA_Priority_High; DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; DMA_Init(DMA1_Channel1, &DMA_InitStructure); DMA_Cmd(DMA1_Channel1, ENABLE);//使能DMA ADC_DMA_NVIC_Configuration(); } void ADC_Init_Configuration(void)//ADC配置函数 { ADC_InitTypeDef ADC_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); RCC_ADCCLKConfig(RCC_PCLK2_Div4);//分频 ADC_DeInit(ADC1); ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; ADC_InitStructure.ADC_ScanConvMode = DISABLE; ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T3_TRGO; ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; ADC_InitStructure.ADC_NbrOfChannel = 1; ADC_Init(ADC1, &ADC_InitStructure); ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 1, ADC_SampleTime_7Cycles5); //最后一个参数为 ADC_DMACmd(ADC1, ENABLE); ADC_Cmd(ADC1, ENABLE); ADC_ResetCalibration(ADC1); while(ADC_GetResetCalibrationStatus(ADC1)); ADC_StartCalibration(ADC1); while(ADC_GetCalibrationStatus(ADC1)); ADC_ExternalTrigConvCmd(ADC1, ENABLE); //ADC_SoftwareStartConvCmd(ADC1, ENABLE); } void Adc_Init(void) { ADC_GPIO_Configuration(); ADC_TIM3_Configuration(); ADC_DMA_Configuration(); ADC_Init_Configuration(); } //ADC_DMA中断服务程序 void DMA1_Channel1_IRQHandler(void) { if(DMA_GetITStatus(DMA1_IT_TC1) != RESET) { global.adc_finish_fg = true; DMA_ClearITPendingBit(DMA1_IT_TC1); } }
stm32F1会有官方给的FFT函数,这里分享给大家。
链接:https://pan.baidu.com/s/18ydSdSU6_8MIIK1tqq5Tlw?pwd=c8t6
提取码:c8t6
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);
官方给了3种,区别在于采样点数不同,采样点数越高,进行FFT计算后的精度也就越高。
这里要注意输入的数据是有要求的:输入的数据长度要与函数名给的一致,每个输入样本都应该是一个复数值,通常以结构体的形式表示,包含两个成员:实部和虚部。
void Get_FFT_Source_Data(EN_FFT_CHANNEL channel_idx) { u16 i; for(i=0; i<SAMPLS_NUM; i++) { FFT_SourceData[i] = ((signed short)ADC_SourceData[i]) << 16; } } void FFT_test(void) { Get_FFT_Source_Data(FFT_CHANNEL_1); cr4_fft_256_stm32(FFT_OutData, FFT_SourceData, SAMPLS_NUM); GetPowerMag(); // for(int i = 0; i < SAMPLS_NUM; i++) { printf("FFT_SourceData[%d]: %lu\n", i, FFT_SourceData[i]); // } // printf("FFT_Mag values:\n"); // for(int i = 0; i < SAMPLS_NUM / 2; i++) { printf("FFT_Mag[%d]: %lu\n", i, FFT_Mag[i]); // } }
void GetPowerMag(void) { signed short lX, lY; float X, Y, Mag; unsigned short i; unsigned long maxMag = 0; unsigned long secondMaxMag = 0; unsigned short maxIndex = 0; unsigned short secondMaxIndex = 0; for(i = 0; i < SAMPLS_NUM / 2; i++) { lX = (FFT_OutData[i] << 16) >> 16; lY = (FFT_OutData[i] >> 16); X = SAMPLS_NUM * ((float)lX) / 32768; Y = SAMPLS_NUM * ((float)lY) / 32768; Mag = sqrt(X * X + Y * Y) / SAMPLS_NUM; FFT_Mag[i] = (unsigned long)(Mag * 65536); if (FFT_Mag[i] > maxMag) { secondMaxMag = maxMag; secondMaxIndex = maxIndex; maxMag = FFT_Mag[i]; maxIndex = i; } else if (FFT_Mag[i] > secondMaxMag) { secondMaxMag = FFT_Mag[i]; secondMaxIndex = i; } } // printf("Maximum FFT_Mag value: %lu at index %u\n", maxMag, maxIndex); printf("波形频率:%u\nKHz", secondMaxIndex); }
得到FFT_Mag[i]之后,就快要成功了,在FFT_Mag[i]中找到最大的幅值所对应的序号,用序号×分辨率就得到了该波形的频率了。(这是理想情况)
实际情况是单片机只能采集0-3.3v的电压,在把波形送入单片机时必须要进行偏置,也就为信号混入了直流,可能会导致最大的幅值所对应的序号为0,所以我找的是第二大的点。(或者还有什么解决办法,大家可以在评论区说说)
int main(void) { Adc_Init(); USARTx_Init(115200); // 初始化串口 delay_init(72); // 初始化延时函数 while(1) { if(global.adc_finish_fg) { global.adc_finish_fg = true; FFT_test(); // printf("找到的最大值索引: %u\n", maxMagIndex); // for (int i = 0; i < SAMPLS_NUM; i++) // { // printf("adc值[%d]:%d\n",i, ADC_SourceData[i]); // } // 将标志复位为false,等待下一次DMA传输完成触发FFT计算 global.adc_finish_fg = false; } } }
最后附上完整代码链接:https://pan.baidu.com/s/1l6y9Iwpz7smP1SFnZNjVGQ?pwd=c8t6
提取码:c8t6
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。