有人使用STM32G4系列的通用型TIMER基于捕获功能对外来信号进行周期及占空比的测量。他用TIM3产生频率、占空比可调的PWM输出做为被测信号。TIM4用来进行频率测量,工作在复位从模式,被测信号接到其通道2的输入脚。然后经内部边沿检测和内部滤波电路后,兵分两路分别连接到IC1和IC2。显然IC2使用直接输入模式,IC1使用间接输入模式,如下图功能框图示意。: \1 j9 ^/ `. @ e
L' |1 U2 u" W7 V f& R* }) u0 B
* C: p" C) r! Z- e5 J* {
$ q% j! l: D. j, T
但他发现被测信号频率较高时,误差就明显变大了。当然,他也知道,任何测量肯定是有极限的。他现在就是想知道,能否基于现有方案将测量极限拉高点。比方说,他现在测量20KHz信号时就明显误差过大,导致测算结果难以采用。是否可以将可靠的测量结果提升到25KHz或更高呢。
: M8 b" y+ r& C2 O9 t" v9 z8 d( M- c; t3 ^, @0 Q1 _, n9 \
6 f3 f, l( [0 Z; w- i: q
9 F! x7 O7 `+ d我们不妨一起看看这个问题。依然保持相同的测量方案,TIM4工作在复位从模式,TIM3输出的被测信号连接到TIM4的CH2,然后兵分两路连接都TIM4的IC1与IC2。/ H/ I& S+ l( [
! f8 y" d2 ^! ~
显然,我们先要确定测量的计时起点。可以采用IC2的上沿捕获事件作为测量起点,即进入测试状态。也可以基于上沿触发信号产生定时器复位导致的更新事件加触发事件作为测量起点,我在下面就是使用后者来进行测量并组织相应代码。1 c, g; B3 G9 e
4 X: i: ~ f6 ~8 Q2 h' lIC1的下降沿触发捕获,捕获到的计数器值存放于变量Value_1stCap【结合上图来看】。
2 ]) O( e+ a: m" b% d) U9 ]' B, L+ P d' @$ n
IC2的上升沿作为TIM4的触发复位信号,且基于上升沿事件进行捕获, 捕获值存于Value_2ndCap。. R& n! G/ Y0 L' W' i! y, ?$ N
, H+ W2 P9 [* X; a8 F9 F" P从发生复位事件到发生第2次捕获期间,对TIM4的更新事件次数进行统计,总的更新事件次数计为Total_Num_OvEvent。
+ C* r. ^# H4 J% o
: }( p) d* k4 m; }( A从发生复位事件到发生第1次捕获期间,对TIM4的更新事件次数进行统计后,计为Front_Num_OvEvent。
$ q& {6 P2 ^( Z/ H4 L, u4 s7 o$ `, ^! I) `3 t% b( ~8 n
现在使用STM32CubeMx进行配置。重点关注TIM4的配置。TIM4的时基和捕获配置参数如下:' K! @. g1 h5 W3 x
" O1 ^8 X: t% b, L# k2 P
2 r5 s& a' T. O O1 n; a; u; T+ w6 [; B; Z
结合上面配置我们不难看出,TIM4的溢出周期为20ms,TI2FP2作为TIM4的复位触发信号,上沿触发,同时IC2针对输入信号的上沿进行捕获。IC1针对输入信号的下沿进行捕获。' q" v Q! o& M' r- g+ ]
- r+ n2 _8 \% m0 G+ ?
至于TIM3的配置没啥特别的,就是产生PWM输出信号,在代码里将调整其PSC分频系数和ARR参数以及CCR参数,以改变其频率和占空比。选择其通道1做PWM输出。下面测试中,TIM3输出的PWM波形的占空比固定为40%。
2 G5 Y$ ]6 V( x* j# G2 n- I: S" E8 E2 O- }- g
1 A* _8 ?1 {& f: N
- _( t. m) e: S- y1 z7 t
另外,STM32G4的系统主频配置为170MHz,使用HSE 时钟源。开启TIM4的中断响应使能。完成配置后创建工程。添加必要的用户初始及启动代码。% j! F$ i5 _1 F( ?$ B/ F: f6 e
* L8 J* l, ?* y6 A6 p+ a
- __IO uint32_t Vaule_2ndCap = 0u;
' n! r) p+ i$ K0 H3 v - __IO uint32_t Vaule_1stCap = 0u;
/ Z# W) P( U/ e2 h, E5 [+ |" ^+ i - ( j; K- W; x4 t) f7 H( I" `4 B5 c6 z
- B9 V- L N. z8 |$ ?* F: q; ?- __IO float Signal_Cycle= 0.0f;
7 A$ p3 W9 p' X$ w0 {' z - __IO float Signal_Duty = 0.0f;' a, J* K6 X; F
- __IO float Signal_Freq = 0.0f;0 v! q* k8 V1 I+ }7 l$ U
& e3 d8 i* R0 L: Q. D \- / L* \* u2 I4 y! v9 [
- __IO uint32_t Total_Num_OvEvent=0u;. W1 F6 N* V9 F- r$ n
- __IO uint32_t Front_Num_OvEvent=0u;
2 f5 n2 j% c" v2 [" E4 j; ~# U
: S! Z3 l. E) {; ?+ b- 6 d" \. Q" p% c% ]/ j* Y1 C: D0 B
- __IO uint32_t Num_OvEvent =0u ;7 \6 `1 ^/ J& I
- __IO uint32_t Measure_State = 0u;' s. r# {- {- @4 A. i
! T! h! K. u T1 E' H6 ?
0 q- L6 ^4 w' y8 s; b* b3 L- __IO uint32_t Clk_Internal;
- W7 L' \6 P! I! Y4 {; j
' ~5 [) \! `6 o4 v
* M3 r; X+ J- W1 {- / P. S0 C' y0 t4 E# K- B8 e J) V
) @1 v+ H) s- F$ N& c! Q9 f- __HAL_TIM_CLEAR_IT(&htim4, TIM_IT_UPDATE);
0 G1 |3 I$ Z! W8 R/ R& B+ l - __HAL_TIM_ENABLE_IT(&htim4, TIM_IT_UPDATE);
: U: n! ~( Q; X0 _ - ) X. f% ^3 i! m! v# X$ l6 b
- : ~, L# W% [* c; V$ k0 X# H8 w
- HAL_TIM_IC_Start_IT(&htim4, TIM_CHANNEL_2); //enable IC1 interrupt of TIM4
: s: R! w7 ]4 E5 ]; O - HAL_TIM_IC_Start_IT(&htim4, TIM_CHANNEL_1); //enable IC2 interrupt of TIM4& [' j& t1 f" E+ ^: ~
- 4 x0 g9 f& a2 r+ H. e
! v$ K8 x4 d% g- e# D4 c0 ^- __HAL_TIM_CLEAR_IT(&htim4, TIM_IT_UPDATE);
8 J6 F9 D5 o* e - __HAL_TIM_CLEAR_IT(&htim4, TIM_IT_TRIGGER);//enable Update interrupt of TIM4
1 T- k3 E* f3 @: K& P% a& E - 5 H( R- V ` r1 b4 m- `
- Clk_Internal = HAL_RCC_GetHCLKFreq(); //170MHz for G4 Series1 b n. r$ e P* B
- 1 p& C* }) n% F
- 7 ^+ t3 _$ ^/ ?( m( `( ]4 |
- Measure_State = 0x00; //initial state of Measuring$ E2 `3 J O6 _' }3 E
- Num_OvEvent = 0x00; //initial value of update count of TIM4
8 ^9 N1 U3 M7 m) y - 0 b2 ~4 I* |% f% Q4 W
5 I3 n2 y5 a+ i& v: p
2 s, _3 |0 Z& F2 _. f
3 N# P9 F3 K, F# ?5 W L) e0 c+ _- HAL_TIM_PWM_Start(&htim3, TIM_CHANNEL_1);
复制代码 9 a6 X" n- a ^% Z: T/ [
TIM4事件的所有中断共用同一个中断矢量入口。为了尽可能地测得较高的信号频率,中断处理代码尽量要简洁些,避免过多、过深的函数调用。
# l. K2 ~5 K( j1 Z; \/ d: w! T: c- }6 n
3 B: L" k! b) t4 z% W
. Y& z# N4 y$ D* i! ?
有关TIM4的捕获中断及更新中断的处理代码如下,包括IC1、IC2的捕获中断处理和更新中断处理代码,以及相关计算处理代码。* k- u% l7 X' m* p# j3 o
6 |( n) R' S) b0 h. y$ C% Y% M+ }- * [) i% y; L) z- e' F& ~
- void HAL_TIM4_PWM_Measure_IRQ(TIM_HandleTypeDef *htim)
$ d4 G9 y9 U! B - {
, I, I/ H/ Q) r# b - ) ?' z' U4 T# {' b. p; |, E
- if(__HAL_TIM_GET_FLAG(htim, TIM_FLAG_UPDATE) != RESET)* ^( J/ \6 \0 |# o6 w( J
- {
$ H( t# K3 U4 q! k - __HAL_TIM_CLEAR_IT(htim, TIM_IT_UPDATE);) W5 d+ z5 [# r# N/ n
- F: Z0 c/ ^1 F; ]2 l' ?
- if(__HAL_TIM_GET_FLAG(htim, TIM_FLAG_TRIGGER) != RESET)! W9 @* `7 Z) a5 ]
- {
* |4 C* E* J2 B0 _0 f- g - __HAL_TIM_CLEAR_IT(htim, TIM_IT_TRIGGER);
) k- ?- t+ O J$ j: z - " _- K6 W7 W. o( h
- if (Measure_State == 0)
o+ D8 Y+ R9 d8 b1 B
7 b5 q( x2 G# L* I# Q- {
3 g& v1 E8 ]" p7 S% Q5 ~ - Num_OvEvent = 0; // prepare count overflow events
- H: u/ }0 J2 b2 e5 b' ] - Measure_State = 0x01; //start measurement5 n$ B* V: W5 d4 k4 {5 y0 [/ g) n
- }
# l# ^+ P, Q6 ]6 @ - }9 a" ]: p' n. W3 h: F$ r) [1 d
- . Z. C7 ?+ o& p" }% }6 l! T
- else/ L1 v1 ^3 L4 c9 e! O# H( U; R
- {
, z8 \$ |2 r$ N - if(Measure_State != 0) : k! K& P) e: L% x; M3 Q7 y1 @
- {2 D5 i) q% W8 i% s1 p
- Num_OvEvent++;
0 O! k( J3 `1 w& z4 I - } 6 x! q5 ^* h5 f3 W
- }
4 M8 N, V* y; J2 E - }
3 r+ a4 r! w, _3 H. Z. I" J
+ V9 T! K( P8 F- J" q" F- /* Capture compare 1 event */3 S* J3 z' \: i/ K! }
- if(__HAL_TIM_GET_FLAG(htim, TIM_FLAG_CC1) != RESET)7 Z" `5 f* n' N& H7 o2 @/ m# _# ~/ i
- {
! b# H9 t, r/ k! f9 O& m& y - ) F, K6 u0 P3 O* |0 Q w; ]3 P
- __HAL_TIM_CLEAR_IT(htim, TIM_IT_CC1); ( p# I2 u! ]/ F' F+ M/ H
& l% Y" q" Y) C2 ?. j! Q/ |- & Q( v& W; p8 M& \
- if (Measure_State == 0x01)9 j6 I6 }8 f# _
- {
: d: _* A; O% X) X" O - Front_Num_OvEvent = Num_OvEvent;
4 j% @* u# P+ O+ g6 y1 Z - Measure_State =0x02;5 {7 E/ c9 O1 \1 D/ P P( j5 K
- }
! q) z+ ?% d, a0 ^* o
4 E6 X2 N9 r, ?1 q2 N7 m# u- }
) p' a7 s2 R9 W3 S4 { - : I! L# i" V% t' o: f& e2 ]
- /* Capture compare 2 event */) ~& s9 \+ @/ w" f0 m
- if(__HAL_TIM_GET_FLAG(htim, TIM_FLAG_CC2) != RESET), B. D: X( p5 R0 I k5 Q
- {- {( _- q g' q; R' O) |8 J
- __HAL_TIM_CLEAR_IT(htim, TIM_IT_CC2);
3 ~6 F6 @( Z; Q6 o4 K: y6 ~ - : }. s$ c/ Z) u5 }3 H. e/ \
' B: e, E! ^8 i: D# ?( I- if(Measure_State == 0x02)
' e6 m& x/ O; A4 [& P - {2 }) z d( s* D ?4 w
- Measure_State =0x03;
- C. X* |( M" A0 R: f4 L - Total_Num_OvEvent = Num_OvEvent;0 `6 c9 h( ]+ A. o Z4 E% {
- HAL_TIM4_IC_CaptureCallback(htim); //go to calculate pulse width and duty
6 f$ X. F c* c5 n) l3 }) T - Z' h6 `3 ~ g. ~' E: C/ r
- } ) A( }. X( o! g# x* M
- ) d7 B* Z7 |6 h3 m/ F
- } & y9 ]8 F3 i0 [$ }
7 S# B# y1 |, s$ g4 G1 H- . z0 t& J! x3 x8 I- Z
- }
) D2 d7 ?: q* L2 | - P0 W' t, d p- e% Q" D
- void HAL_TIM4_IC_CaptureCallback(TIM_HandleTypeDef *htim): X# \3 {, G$ {! m
- {
/ F) [+ g d$ Z7 Y1 Z$ Z) Y, r1 O, }
- Z. l( w: ^+ v2 N) t6 r- /* Get the Input Capture value */- U7 ^8 i* o: S9 B$ {8 F* b
- Vaule_1stCap = TIM4->CCR1;//HAL_TIM_ReadCapturedValue(htim, TIM_CHANNEL_1);% h* L) P! h. R& { y+ j: q* T
- Vaule_2ndCap = TIM4->CCR2;//HAL_TIM_ReadCapturedValue(htim, TIM_CHANNEL_2);
]9 Z# x) S/ e4 Z9 H1 j% r4 K - # r$ u$ `8 Q3 c) ]8 U
: P# ~. T9 {% ~7 @. \7 Y" D- Signal_Duty =(float)((Vaule_1stCap+(Front_Num_OvEvent*(TIM4_PERIOD+1)))*100u)/ \( y9 l. _8 D4 H# q* W8 O
- (float)(Vaule_2ndCap+(Total_Num_OvEvent*(TIM4_PERIOD+1)));
! t! `0 Q8 i% i9 |( k. b - ! f1 {8 q+ ^6 e1 l% v0 o( r; z# m- Y
- // Clk_Internal = HAL_RCC_GetHCLKFreq();5 \2 H5 }1 t% s" c, f
- 0 @/ }2 q9 j8 p3 h' h; x6 Z) K
- Signal_Freq =(float)(Clk_Internal/(TIM4_PSC+1))/(float)((Vaule_2ndCap+(Total_Num_OvEvent*(TIM4_PERIOD+1))));+ Z" Q& _8 c, o' Y l: {
- Signal_Cycle = (float) (1.000f/Signal_Freq) ;) T. I+ C# p3 r2 f% ~/ z
& T3 x7 ^: b" M7 S- , N) q1 R7 L" }5 I, |' r
- Measure_State = 0x00;
- g& i2 n& G# w- K - Num_OvEvent = 0x00;. s6 z8 S2 c" }" Y$ k
- __HAL_TIM_CLEAR_IT(htim, TIM_IT_CC1);
0 U' N9 }6 g4 B* m9 K - __HAL_TIM_CLEAR_IT(htim, TIM_IT_CC2);
2 Y! O' Z$ [, Q* Y; H/ \( l - __HAL_TIM_CLEAR_IT(htim, TIM_IT_UPDATE);7 _ v) {/ b! p. b! w7 c) K
- __HAL_TIM_CLEAR_IT(htim, TIM_IT_TRIGGER);' i. \1 |0 p; z; O. U; h" k3 A
-
1 y% |; G- {. W' Q6 ~ - }
复制代码 * M- ~; V6 d$ s! v3 _/ n" ]
现在基于上面的配置及用户代码进行测量验证。我们发现当TIM3输出的待测信号频率达到25KHz时,测得的频率及占空比就发生了明显的偏差。见下图标注问号的第4栏信息。- r/ O$ [( X r2 ^
9 g" E _- o3 @
: B" C$ ~6 } W7 W" x即当被测频率为25KHz,占空比为40%时,测得结果是下面的样子,明显偏差过大。
" `& R& P. v" G0 p" a- f( b$ s$ q- l: C
4 F: N( l/ B: Z1 g# {, f" A
如果说被测信号频率进一步提升的话,误差会变得更大。那么,这种情形是否有改善机会呢?即在当前的测试方案下,可准确测量的被测信号频率是否可以提高。
7 |/ @) H8 E. l4 M3 G3 [& |" Y- l& C) w+ X
目前的中断处理代码应该说比较精简,没有什么可以优化的余地了。% s; z! {! X5 g1 I) U' H/ D
2 n. [' X8 u+ D. t) d/ ]: ~% o
聊到这里,有人可能发现了,我前面配置TIM4时,它是用来完成测量任务的,它的分频系数PSC为169。即TIM4的计数器计数时钟为1MHz,其计数分辨率为1us。不难理解,这个PSC系数应该会直接影响TIM4的计数精度,按理会影响到测量结果,尤其被测信号频率较高时。
) [! D" U% z% W' B4 d' m5 _) O$ ?$ E* z
既然这样,我们将TIM4的PSC系数改为0,并适当调整其ARR值再来实施测量,看看结果会怎么样。见下图,TIM3输出的信号频率仍然是占空比为40% 、频率为25KHz的PWM信号。测量结果显著地明显改善,应该说此时结果是可以接受的,毕竟浮点运算也会带来些偏差。
# x9 @2 o2 }9 T' ~. F) p% y _1 |1 i/ X2 @$ z
7 Z' {7 ~( c. N1 @. R& a4 ~5 a: e; G L1 ?1 Q; o. u2 ~- J r
我们不妨在保持TIM3的PSC为0的条件下,将输出频率提升到50KHz、100KHz。继续看看测量结果,见下图:4 i' i4 T V) {" s/ k7 \( f9 D% I. [
9 o( B3 q, t% ~1 [( h( u2 F
7 I7 u8 N' `& I% I! L' d+ Z/ t0 O& N* A
从测试结果来看,当被测信号频率提升到50KHz,测量结果仍然很好,完全可以采用。即使当被测信号频率提升到100KHz时,测量结果虽发生了一些偏差,但此时的偏差相比PSC=169、被测信号为25KHz时的测试结果还要好得多。我把二者单列出来一起比较,见下图:) o9 ~& y4 n! |8 u/ U5 E' L/ d2 {
& k9 s B' ]' c w$ d6 {7 z: ~" f8 v: R
0 X, w, [& z7 W; R% Y
经过上面的讨论和验证,我们知道,在使用TIMER做信号的频率及占空比的测量时,当确定好测试方案后,为了尽可能地提升可以准确测试信号的频率,一方面代码要尽可能精简、优化,另一方面,因测试TIMER的分频系数会影响测试结果的精度及准确性,此时测试TIMER的分频系数要尽可能设置小、或不做分频,这点结合具体应用场景来定。( }% _" O: i9 y7 |& Z0 P
# t& z% Q# P0 h8 b关于上面的的话题,其实还可以有很多继续延伸的空间,因时间和篇幅问题,就先聊到这里。有兴趣的话,可以基于上面配置和分享的参考源码做些快速测试验证,也欢迎进一步地讨论交流。
# [' D% K# j2 Q8 j# G, H8 I
( a' S6 D! a4 V5 \. U' b; g, N! I% U4 m- h- `" y, ^6 d
转载自; 茶话MCU
* n: F. ~$ F' }# a/ h5 `如有侵权请联系删除/ e; c9 s2 V1 o7 _& L4 Q
& k" e; l3 c9 H7 s8 r& w! k5 F
8 n% c( ~2 o, ^. [5 }2 p2 k |