1)实验平台:正点原子stm32f103战舰开发板V4
2)平台购买地址:https://detail.tmall.com/item.htm?id=609294757420
3)全套实验源码+手册+视频下载地址: http://www.openedv.com/thread-340252-1-1.html##
第五十七章 DSP FFT实验
本章将使用DSP进行快速傅里叶变换(FFT)的运算测试。通过本章的学习,读者将学习到DSP库中FFT运算的简单应用。
本章分为如下几个章节:
57.1 硬件设计
57.2 程序设计
57.3 下载验证
57.1 硬件设计
57.1.1 例程功能
- 按下KEY0按键进行FFT计算测试,LCD显示计算耗时,USART输出计算耗时和计算结果
- LED0闪烁,指示程序正在运行
57.1.2 硬件资源 - LED
LED0 - PF9 - 按键
KEY0 - PE4 - USART1(PA9、PA10连接至板载USB转串口芯片上)
- 正点原子 2.8/3.5/4.3/7/10寸TFTLCD模块(仅限MCU屏,16位8080并口驱动)
- FPU(可选,使用后相关的计算耗时更少)
- 基本定时器6
57.1.3 原理图
本章使用的DSP库为软件库,因此没有对应的连接原理图。
57.2 程序设计
57.2.1 实验应用代码
本章实验的应用代码,如下所示:
int main(void)
{/* 必要初始化,代码省略 */while (1){key = key_scan(0);switch (key){case KEY0_PRES:{/* 初始化FFT测试 */fft_test_init(fft_inputbuf, FFT_LENGTH);TMR_ConfigCounter(BTMR_TMRX_INT, 0);g_timeout = 0;/* FFT计算测试 */fft_test(fft_inputbuf);time = TMR_ReadCounter(BTMR_TMRX_INT) + g_timeout * 0x10000;sprintf(buf, "%0.1fms\r\n", (float)time / 10);lcd_show_string(30 + 12 * 8, 160, 100, 16, 16, buf, BLUE);/* 对FFT运算结果取模求得幅值 */fft_test_get_output(fft_inputbuf, fft_outputbuf, FFT_LENGTH);printf("\r\n%d point FFT runtime:%0.1fms\r\n",FFT_LENGTH, (float)time / 10);printf("FFT Result:\r\n");for (i=0; i<FFT_LENGTH; i++){printf("fft_outputbuf[%d]:%f\r\n", i, fft_outputbuf[i]);}break;}default:{break;}}/* 省略其他代码 */}
}
从上面的代码中可以看出,在检测到KEY0按键被按下后,便会调用函数fft_test_init()以及相关函数对FFT运算测试进行初始化准备,随后调用函数fft_test()进行FFT运算测试,同时使用定时器统计运算耗时,最后调用函数fft_test_get_output()对FFT的运算结果取模,然后在串口调试助手上显示最终的结果。
以上提到的函数fft_test_init()、函数fft_test()和函数fft_test_get_output(),如下所示:
static arm_cfft_radix4_instance_f32 scfft;/*** @brief 初始化FFT测试* @param fft_inputbuf: FFT输入数组* @retval fft_length: FFT长度*/
void fft_test_init(float *fft_inputbuf, uint16_t fft_length)
{uint16_t i;arm_cfft_radix4_init_f32(&scfft, fft_length, 0, 1);/* 生成信号序列 */for (i=0; i<fft_length; i++){fft_inputbuf[2 * i] = 100 + /* 实部 */10 * arm_sin_f32(2 * PI * i / fft_length) +30 * arm_sin_f32(2 * PI * i * 4 / fft_length) +50 * arm_cos_f32(2 * PI * i * 8 / fft_length);fft_inputbuf[2 * i + 1] = 0; /* 虚部 */}
}/*** @brief FFT计算测试* @param fft_inputbuf: FFT输入数组* @retval 无*/
void fft_test(float *fft_inputbuf)
{arm_cfft_radix4_f32(&scfft, fft_inputbuf);
}/*** @brie f对FFT计算测试结果取模求得幅值* @param fft_inputbuf: FFT输入数组* @param fft_outputbuf: FFT输出数组* @retval fft_length: FFT长度* @retval 无*/
void fft_test_get_output( float *fft_inputbuf,float *fft_outputbuf,uint16_t fft_length)
{arm_cmplx_mag_f32(fft_inputbuf, fft_outputbuf, fft_length);
}
可以看到,这三个函数实际上是对DSP库中函数的简单封装,本质还是调用了DSP库中的函数。但是可以看到,在使用了DSP库后,进行FFT运算变得十分简便,这非常有利于缩短应用开发周期。
57.3 下载验证
在完成编译和烧录操作后,可以看到LCD上显示了本实验的相关信息,此时若按下KEY0按键进行FFT运算测试,测可以看到,在FFT运算结束后,LCD上显示了本次FFT运算的耗时时长,同时串口调试助手上也显示了本次FFT运算结果取模后的结果值。