STM32的ADC配合DMA采样以及STM32利用DSP库实现快速傅里叶变换,而当二者真正的结合到一起实现一个信号采样以及频域分析才可以发挥出很强大的功能。. t. _, g0 V' K7 D
2 J6 ^# ]( c, }8 U
本期我们就来演示,利用DMA+ADC+FFT实现信号采集与频域分析。' ^& g" D- |6 d6 ]3 y3 g
6 F1 B, S& O5 F
CUBEMX配置
8 ?( _& d; z* w: S8 p9 t0 |$ V
% k% m. l( ]+ G; ^7 x9 ?/ L
# Y/ ], e5 y, H) g t
+ ]7 n+ t9 k; I9 \# LADC配置这边除了之前的配置之外有一点区别,就是我们把循环采样给关掉,因为我们打算利用定时器来触发采样,这样子的话可以精准的控制采样率。
0 w& r G9 o1 [' f3 L4 m/ k/ Q
E, d3 ?. h9 I" r5 a! S# V- j! O
7 o) }( a4 ]% E. r9 [7 k3 t
& D U% X" K4 b- c0 v
这里我们选用TIM1的输入捕获来触发,当TIM1输出PWM波的时候,会选取上升沿来触发ADC采样,将采样的数据存储到DMA中。
/ w: O& w* z6 h) Z8 E+ H2 z R6 g
% k# D2 T4 V; {! P3 A( H8 I3 F
$ y$ D6 [2 O8 k# T& c. o& l9 A
9 W8 {1 g R2 P% p& M) n
定时器中我们设置PWM的频率为500HZ,这代表着我们的采样率为500HZ,之后生成我们的代码。(这里非常奇怪,我明明觉得这么算好像是1000HZ但实际只有500HZ,我哪里算错了?)
, ]7 K4 z$ N" r% w! I
2 S3 |) o) N$ h- r: ^代码编写$ F1 {' _. ?. |2 Y2 t& k
- #define FFT_LENGTH 1024//采样长度
% D# g# t( w! B f - arm_cfft_radix4_instance_f32 scfft;//定义scfft结构体
+ ^& `( O! H. O% U; Z" |; S' U - float FFT_InputBuf[FFT_LENGTH*2]; //FFT输入数组, o% b4 U: E- V
- float FFT_OutputBuf[FFT_LENGTH]; //FFT输出数组
& R( }. F8 Z' M- P! W- E4 R: V - uint16_t AD_Value[1024] = {0};//存放ADC的值
复制代码
9 P. ~1 j3 T* i/ A$ H+ ^上述代码需要定义在全局,否则可能会因为栈溢出(因为临时变量的定义是存放在栈中的)导致代码崩溃。
. N" }6 q1 y) M" S- HAL_TIM_PWM_Start(&htim1, TIM_CHANNEL_1); //用来触发adc采样
% K' M8 {1 y! ? - HAL_ADC_Start_DMA(&hadc1, (uint32_t *)AD_Value, FFT_LENGTH);//开启ADC
! h9 ?- e9 f, F" t4 O" x9 E - arm_cfft_radix4_init_f32(&scfft, FFT_LENGTH,0,1);
2 i/ F4 n( b; a. {1 y - HAL_Delay(1000);//稍微等待一会ADC一轮转换结束
- ~# z/ {+ ~( I* N) C% a, r5 ~ - for(int i=0; i < FFT_LENGTH; i++)
2 j( u1 f J+ z - {
1 o* i \( K$ O5 t - FFT_InputBuf[2*i]=AD_Value[i]*3.3/4096; //实部
2 N. {6 |- N$ o* z/ G* _' e1 ~ - FFT_InputBuf[2*i+1]=0; //虚部
2 f: J# J0 k$ ~1 K2 b - }2 B, e6 n r0 r" C
- arm_cfft_radix4_f32(&scfft,FFT_InputBuf); , d: X( [- Z0 T. P& Y0 Q
- //arm_cmplx_mag_f32(FFT_InputBuf,FFT_OutputBuf,FFT_LENGTH); //取模得幅值, |2 u e. H0 K' W8 i$ K$ q
- for(int i = 0;i<FFT_LENGTH;i++)
' G3 e3 v3 s& |$ f7 @0 m4 m' F - {) e* @3 O; B4 e* @2 i" [& L
- float32_t real = FFT_InputBuf[2 * i];
) B! ^* Z& m3 V - float32_t imag = FFT_InputBuf[2 * i + 1];& L( y; n' I' l& Z
- float32_t magnitude = sqrtf(real * real + imag * imag);: ?4 p) I! `. C& \& B
- 6 g# X/ W6 ? ?9 y8 L% o
- // 打印每个频率分量的模值, {4 `+ d* \' k# U) S0 X
- printf("Fre: %f \r\n", magnitude);9 K* y& n+ v I7 j
- }
复制代码 3 ]' l! {) H$ b2 C
生成代码,我们由定时器触发采样,原始信号来自于一个250HZ的方波。. V$ v1 K$ G0 C6 w" m8 G
% S3 g# V. H; }; n7 v( B. h) L* V$ B
$ k$ x% k7 V( F+ d2 P/ [
3 Q( C6 W7 K2 i
之后我们执行我们的FFT代码,首先我们先看一下采集的波形。9 e& W: j$ i$ x0 d a
6 [/ P+ h ~) w- K# b: ~* `$ G
" i: o: Q0 t4 j4 \' S7 Y* ~) x; J' g- g
这里可以看到外面采集的方波。
4 R/ k( ]) n# h- [
- F4 d* P5 t8 A1 n( g) c+ I+ k0 c$ @注意事项, v1 w3 [3 n3 {. X
(这里我的问题,根据奈奎斯特采样定理,采样频率应该至少是信号频率的两倍,否则会出现采样失真,因此为了能够顺利的采样,我将PWM的频率修改为100HZ)
9 G' C1 `7 J4 b- P! X0 v- n' e. K, A3 a9 z
% v" i; Y o) p) A: |' S, y
& ~- Q5 T; O; J8 ^
这里也非常奇怪,我觉得PWM的频率应该是100HZ的,但是不知道为什么测出来是50HZ(难不成我又算错了?)
" ]+ S1 P, F4 h* Y% n! A8 Y
3 J# o) F# A! p) G后来发现是我的分时系数算错啦~
: @+ i0 N/ s& H0 B( A, P& a# U7 I- n5 ~* U" H
变换结果* h6 N5 } r! r- |$ N0 t1 J
然后来看看傅里叶变换之后的结果。
: z% S: t, [# e9 O
* _3 s0 _: P- H9 h
& I/ D. m- B4 p' o( _
6 `% R! L9 V1 M" O b1 \, ~2 g(这个软件是我自己写的串口示波器)) z; y/ a3 ?9 Y
" n: j. k# m9 C9 y1 J2 I
每个FFT点对应的频率(f)与采样频率(Fs)和FFT的点数(N)之间的关系可以用以下的数学公式表示:4 [+ n7 _" |! f4 q2 G9 ]4 N
f = k × (Fs / N)( y( P9 N1 g0 X4 Y( ]6 f+ _, K" P
M! D3 u8 f; |9 ]6 [7 F7 U4 I2 W j! b
其中,k 是FFT结果的索引(从0到N-1),Fs是采样频率,N是FFT的点数(与采样点数相同)。( f& \5 @5 O& n# x& X. a
- A" }9 v8 H* F6 |9 X6 q
这里外设定的采样长度是1024,采样频率是500HZ,因此每个点之间的频率差是500/1024(这里按理来说1000更加方便计算但是不知道为什么1000就会卡死,1024就不会,这个也等我研究一下)。
# c* E: U$ |. d7 G& }& O2 y6 T$ k2 e
然后用Vofa+来算一下频率(自己的串口示波器不能显示下标,真是失败,我这两天就加上去这个功能)
4 p. E3 a$ C( F; z2 }' d
( Y6 D6 Y. m) J0 ], D- \
: S. O n" l' I( G I( C' S; G
5 V# A# b) ?6 k# n1 U$ }
首先是开始的这个大信号,毫无疑问就是是直流信号(低频)所带来的幅度。8 \2 f4 H9 V' B8 I! s* V
% C; S: f. D$ d) o这里我给数据加一个直流滤波,就可以把这个很大的直流量给滤掉了。- _6 F% H6 e, @. R- u4 G# S
- for(int i = 0;i<FFT_LENGTH;i++)
1 L: S3 }% K% B! Z - {
$ a# e c, a+ W& q4 N2 d/ F8 x/ H2 n - sum+=AD_Value[i];
i' V) i' `0 u6 v- B: c - }
% W/ c+ N. j+ r; V' d* l - sum/=1024;! m3 A+ S: Q! @3 a! P0 g2 ~3 ?/ p
- for(int i=0; i < FFT_LENGTH; i++)
" m- i& H. s; n - {
; h# c# a! h7 q/ ]; f1 M8 ~ - FFT_InputBuf[2*i]=(AD_Value[i]-sum)*3.3/4096; //实部
% m% p) H$ E8 u) Q+ i9 T* X - FFT_InputBuf[2*i+1]=0; //虚部2 M# k9 b1 h$ l/ X* x. M4 n
- }
复制代码 - l6 r _+ i/ x" n9 h* F" b& n
从图中也可以看到,FFT的结果不是一个单边频谱,而是一个双边频谱。5 d; ^- H/ H+ M) ^( A& D: a1 \
1 f: \) j9 g4 U
FFT(快速傅里叶变换)的结果默认是双边频谱,这主要是因为FFT是一种处理复数信号的方法,而复数信号在频域中既有正频率分量也有负频率分量。即使你输入的是实数信号,FFT算法也会将其视为复数信号来处理,并返回包括正频率和负频率在内的双边频谱。0 a% M! d! v$ g
* S" O6 I% ~& D. ^4 Q
1 Q5 D& R" V& D' l- U
# }& R8 ?/ f/ c/ B5 a两个峰值之间相差100个点,也就是存在50HZ的倍数波。
) P* E4 u' I; B& i7 g2 t, c8 ]- T B: y2 M7 d
" R+ c8 p- B, C {, n
; D% i1 K) J2 j" |9 k! }这是因为方波是由它的基波频率以及高次谐波所组成。8 _+ \' x- B0 m$ d9 ~
% Q& c! z( N3 Z G; d% c
( D9 R2 L6 L% Y4 d4 X
1 _6 `5 g3 d# K. T
: n1 Q" Y) N5 q F0 N
7 Q* B5 a5 W. w- ?# A4 _9 u' x0 o% O+ T/ h: g9 L
转载自:电路小白
4 Z! L. A2 J% J, P如有侵权请联系删除6 M, o. d# l9 m
5 S: m" [1 L( I+ v3 {% d( |6 \2 _0 f. H8 O2 Z# B9 M0 Y2 @& n, d
* t! \5 h0 L8 X, ^: \8 `+ U% v |
学习学习,正好需要FFT频谱分析