用 STM32 通用定时器做微秒延时函数(STM32CubeMX版本) ' k, E& Q5 m5 N& {4 g7 t, O. B
概述
9 o* Y4 a+ z+ J- y; l# _& w 在使用 DHT11 的时候,时序通信需要微秒来操作,STM32CubeMX 自带一个系统时钟,但是实现的是毫秒级别的。因此就自己用通用计时器实现一个。) B* B* m0 m' f' F, ^ z, Y
8 ^- A/ @8 I1 t4 `* ]
) @# N8 o6 s' U0 N5 K
; I" M6 C" p4 k2 c% G) W6 E
5 R& K- Z# m4 J& _# \; G 环境:, w C: M$ X2 |$ \
$ Y; W) H7 }/ ^" C4 O2 B' L' @% D4 g8 }* |
开发板:STM32F4探索者(正点原子)8 [' ]* p5 N" @' C$ W
) D' L* r U- }9 T& a! C' l* F5 d% J; v3 b/ `
文章目录) h q3 i) I6 X4 A# k# e' F6 X
1.配置定时器时钟- z7 F$ ^8 q$ L8 ]( N) r) t& V
2.计数器时钟频率及计数模式, M) Y t3 G! o- V
' x) w$ G7 L0 U
" ]* [' z. A8 K/ w& f& B$ v: [预分频系数$ t2 a& A8 d' Q
计数器模式) }! ?8 s7 v/ r8 @2 s' X( J
自动重装载值1 B1 p* S. k. N% w3 d3 P i+ M- @+ `
3.打开定时器中断7 n9 {9 y, B; I$ @) Y8 f# X
4.具体实现代码 _! w, B3 K- d2 Y, W/ J9 I
5.代码测试
& A) b. J" m* B$ |. S1 p6 s
; D% o; Q. p/ R. ]' g* i- }( {2 E: Q/ V* {- Z, A! s# d! r" X# U
1.配置定时器时钟: T5 w+ k* ?8 _: M+ Q8 m& j
·选择时钟源% s$ ?: r2 N5 b+ b# v9 F
这里选择的是内部时钟,来自 RCC 的TIMxCLK,在通用定时器框图中我们可以看到如下:4 \7 [; h" C' Q" n) o
而我们可以在 STM32F4xx中文参考手册中找到,TIM2 在外设总线1(APB1上),因此其时钟为 84MHz,如下图所示:# `: g! n8 s* r8 d" c
2.计数器时钟频率及计数模式
" ]8 O$ V3 j- k" Q$ g v除了配置定时器的时钟,还需要配置计数器时钟频率,我们要实现微秒延时,因此计数器时钟频率应该是1MHz,
1 m$ K: n9 z- c$ L3 T7 u$ D" H
, S$ ^- L! y5 z3 ~5 ~! B
( {+ S* `3 u+ H8 v. c- r而要实现还需要以下3个参数:
- x2 c" f, A2 m( L: M; t# W预分频系数7 p! o$ y! @1 ~5 W" L# H
根据STM32F4xx中文参考手册中的时钟频率计算,如下图所示:
' @( f# j) o: j0 W其中fCK_PSC就是通用定时器框图中的CK_PSC, 即值为84MHz,而我们所要的计数器时钟频率1MHz' C0 @7 b+ S6 v, F! L6 V
5 Q# ~# }' H* Y7 ^/ ?5 @2 J
8 m6 _( q) `( I9 q
因此:
) [. W% D, U+ O3 x- PSC[15:0] = (fCK_PSC/CK_CNT) - 1
, {+ s7 P, x2 ~3 H! ]# z - = (84/1) - 1( p$ b* @' c' G7 e+ a
- = 83
复制代码 因此预分频系数为 838 f5 j! G7 n5 x# s- D
! _/ N. R. _: ]$ y1 W
6 R. f1 ~+ L. h2 `& k) H计数器模式' O" U( i$ Q3 T' j5 w% e
计数器这里采用向下计数模式,也就是 如设置计数值为 1000,那么每隔一个微秒,就减一,一直减到 0, V3 [5 H6 R8 |$ s1 N% O* s& D
: r$ O7 g/ H" y# @/ c+ ^- b! E) r) i) F9 M9 n
自动重装载值
& N' [' d0 u7 T$ S虽然我们并不使用自动重装载功能,但是,我们还是要对自动重装载寄存器进行赋值且不赋值为0即可,但是我测试时发送,如果为1,延时会出现偏差,因此这里赋值为 2,依据如下:
% u( W$ m" T/ ]" g' F! I. ?
; ?1 M G6 d" ~9 K+ A- k3.打开定时器中断# H2 S. [ n1 d. E2 B9 V* e
4.具体实现代码
. e" Y$ }1 Z7 T- volatile bool elapsed = false; //用于判断设置的计数值是否耗尽(向下计数模式),耗尽时,在中断中奖会设置为true# ]7 {- D! m: ~; i/ w2 g% Z
1 Z" E7 `$ z; P0 f- void setState(bool state)
% R; ]8 O, z1 X - {
; w9 Y3 A8 k8 m - elapsed = state;5 V2 w: U2 T% j' N L
- }
+ { B" `6 X' o. M* Q+ K* j ] - ( A5 O! }( x0 g0 z
- bool getState()
* A% r2 e" p. Z" C: }% S - {3 B5 [% C1 w1 A2 X; J
- return elapsed;. w9 ]* A9 \1 N! i6 R" j- A; [
- }- B3 Y2 K- {3 C1 F
$ K8 r1 q* \: l4 M7 `) ?+ t) }- void usDelay(uint32_t time)
: q- F9 N0 i) A1 _- S. H' r( E9 _& g - {1 U; k+ W) L0 C5 l8 P1 Z6 f
6 e+ A5 ?5 q& S6 ~2 a v$ f- __HAL_TIM_SetCounter(&htim2,time); //设置计数值
% A2 C$ h! `8 q& O2 ~. l: d - setState(false);
% _% d, b; \! ]5 z9 L! V - HAL_TIM_Base_Start_IT(&htim2); //开启定时器
0 U7 {8 e9 N9 ?: g - while(!getState()); //判断计数值是否耗尽" |; P- v- J/ s4 O. x
- HAL_TIM_Base_Stop_IT(&htim2); //关闭定时器
1 d' ]( J1 Y8 |5 W -
v- o" h2 l9 p3 w: j - }
复制代码 计数值耗尽回调函数; m2 n& d9 O; ]$ n% }
- void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)8 L: s# V6 Y' ~& k) Y' p
- { |9 E& e }; s; G' e$ Q1 a
- if (htim->Instance == TIM2) {- W2 t& x" F; {
- setState(true);
! B0 l( y& Q- S6 B& { t0 w - }9 \" ?5 x6 T9 l$ q, U
- 0 p7 X( T( n# L
- }
复制代码 5.代码测试/ `$ X8 P, v7 c8 J/ f
主函数的主循环中:3 @$ p; Q1 n2 Z, l8 \
- while (1)
$ K" m2 G( J& y. C9 x2 ` - {, [/ O0 G( h( r9 p1 k9 o
- /* USER CODE END WHILE */
0 u" z" t7 s7 ?/ s - 2 }6 ~# ?! b3 w( L% P
- /* USER CODE BEGIN 3 */" D# N# q% i* Y3 Z; J9 D, W
- printf("hello usDelay");
& f; U: C6 g7 q/ ]# E - for(int i = 0; i < 1000; i++)
6 Z5 }6 ^- T+ E7 { - {
. O& i9 ]: {6 R4 c# Y - usDelay(1000);( ~- [. U0 k* E
- }& Z* ]3 @7 T% r. }. z) B
- * F% W) j) z+ s
- }
复制代码 在串口调试助手中,可以看到如下,一秒打印一条语句
- U) k) _5 f; A* i# u$ X在后台回复“微秒延时”,将获取项目代码!8 c8 H- Z" z+ @5 H. E: J
后续也将分享如何使用该函数读取 DHT11 温湿度传感器的值,欢迎关注我与我交流!! j* H* l/ @# c F: r+ E) V
/ y9 V* N3 C: B3 C4 V! Y |