你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

【经验分享】STM32之通用定时器

[复制链接]
STMCU小助手 发布时间:2022-1-18 22:49
通用定时器可以用来:
1、测量输入信号的脉冲长度,也就是输入捕获(请对此四字短语养成良好的形象哈)
2、产生输出波形,也就是输出比较或者PWM(同上)
, S6 `' V+ i5 y% v; N( j
     那葵花兄,通用的原理呢?又见葵花兄坦然自若的说:通用定时器是一个通过可编程预分频器(PSC)(注:这三个字母要有印象哈)驱动的16位自动装载计数器(CNT)(注:同上)构成。
    好了,大家撇开葵花兄,把焦点重新聚焦在我身上、那通用定时器有什么功能呢?我想肯定有人马上反应:有定时的功能、、额、、我也不能说你错是吧、、那我来具体点吧,其实也不是我具体,多亏了“葵花宝典”第STM32篇中文参考手册  :
1) 16 位向上 、向下 、向上/向下自动装载计数器 (TIMx_CNT)。3 ~5 M; N1 D( [8 @. V( I, n9 ]
2) 16 位可编程( 可以实时修改 )预分频器(TIMx_PSC)(TIMx_PSC)(TIMx_PSC),计数器时钟频率的分系为 1~ 65535 之间的任意数值。) ^$ Y2 ^: A4 M$ O4 G% O
3)4个独立通道( TIMx_CH1~4 TIMx_CH1~4 TIMx_CH1~4 TIMx_CH1~4)* }- d. A/ J2 J/ r# O
A.输入捕获
* g$ b4 [( ?- Z& {3 h1 gB.输出比较 * Q! h6 K' m6 C9 N/ _  y
C.PWM 生成 (边缘或中间对齐模式 )
6 c$ Z5 D/ q5 W; JD.单脉冲模式输出0 R. v; }) i$ ]* {. C  ^/ p# j
4)可使用外部信号( TIMx_ETR )控制定时器和互连(可以用1个定时器控制另外一个定时器)的同步电路。5 e8 f8 I( ?0 u, k+ m
5)如下事件发生时产生中断/DMA:, Z& B' }7 L- r; v
A.更新:计数器向上溢出/向下溢出,计数器初始化 ( 通过软件或者内部 /外部触发 )* t9 @$ M8 O$ }3 i* E
B.触发事件  (计数器启动、停止初始化或者由内部/外部触发计数  )
1 R* S1 C3 Y! O& z7 \C.输入捕获 " a- W0 v+ x! D0 O# P$ N* k
D.输出比较  C4 ^+ ~6 v. E. B1 Q1 r/ F
E.支持针对定位的增量  (正交 )编码器和霍尔传感电路 & `( N- r$ m0 S- Z; L6 K
F.触发输入作为外部时钟或者按周期的电流管理
   当然啦,在此博客中不考虑那么多功能先哈、、先淡定、、若是被吓到了、、先去喝杯茶压压惊哈、、那我们怎么来实现定时的功能呢?
只要定时、当达到一定容量时,就会发生溢出、、聪明的人又知道我要说什么了吧?世上太多聪明的人了、、没错了、、把中断这位兄弟叫上、让溢出兄和中断兄合身、、(省略合身等待的时间)、对了、诞生了溢出中断、我们给取个好听的名字:更新中断、在这里,我们是利用中断来定时的、、
   接下来,容小弟介绍下与此博客有关系的寄存器(具体的位是什么功能我就不讲了哈,因为我们是采用库而不是操作寄存器,想知道的可以参考中文参考手册哈),大家掌声响起:欢迎控制兄(TIMx_CR1)

" s% Q$ D0 o" e7 G: @
261716482798777.png
, g: A) Q; ?% J% R  h8 W
欢迎中断/DMA使能兄(TIMx_DIER):
261721185133335.png
( G0 n9 G8 \; d8 |/ d
欢迎预分频器兄(TIMx_PSC):
261723384359882.png

8 x+ E7 L* o; E! i
欢迎自动重装载寄存器兄(TIMx_ARR):
261725582163986.png

* `9 \  ~) \; x$ ~
好了、、这四位可以在旁边休息了(我们就不介绍了)、、因为我们的重点:库兄要出现了,接下来,用最热烈的掌声欢迎:
首先:有请挂载时钟大神ABP1(为什么要请这位大神呢?因为我们的通用定时器是挂载在这位ABP1的,证据如下)
" [0 S5 m! Y" u: ~6 O" R
261731028577156.png
% g; g3 w, @9 v4 M- N/ `- }2 l
再者,我们有请第二大神:初始化大神:TIM_TimeBaseInit(为什么是他呢?为什么?究竟是为什么?啊哈、、请看)
/ R# F$ M" Y! J( |
261735027167733.png

- s# R, D6 O8 U3 q2 j
看到了吧、、所以请不要怀疑大神的地位哈、、至于参数具体是哪些,大家可以打开#include "stm32f10x_tim.h"哈,在这里就不多说了、、相信大家都是聪明人、
前两位大神出来了,我们还有大神、、哎、、大神无处不在、、来、有请类型中断使能大神
void TIM_ITConfig(TIM_TypeDef* TIMx, u16 TIM_IT, FunctionalState  NewState)
     看到中断、、大家又有想到什么猫腻没??条件发射就是这样炼成的、、没错了、、就是中断优先级设置大神啦、、这个大家应该有印象、我就不细说了哈,没印象的话这位大神可就要躲到某个角落伤心了,
     看到以上大神、、其实还有一位大神没出现、、此大神没出现、、其他大神啥都不是、、对了、就是定时器启动大神
      void TIM_Cmd(TIM_TypeDef* TIMx, FunctionalState NewState)
      好了、、最后,编写中断服务函数,在中断函数里检查中断标志位和清中断标志位,还有你想要做的事,相信大家也是比较熟悉的了、、
   在这里,我们来谈谈时钟的问题、、大家可以看到系统时钟树图里:

8 }: T- R9 n1 L4 p( v8 U, H
261829075919160.png
8 K- o, A# b: _- K9 v4 y# v* ~: h
    看到我美丽销魂的涂鸦没、、由于我的系统时钟初始化是36MHZ的、而且设置APB1的预分频系数为2,所以定时器的时钟就为72MHZ,大家可以根据自己的系统时钟设置选择哈、、
到此,我们来总结下使用中断的定时器:
1.开启挂载在ABP1上的TIM3的时钟
2.对定时器进行初始化
3.设置中断类型
4.设置中断优先级
5.开启定时器3
6.编写中断服务函数 看一下参考代码:
  1. void Timer_3_Init(u16 arr,u16 psc)
    7 y% i: U) {& s+ ^
  2. {
    ; ~$ _1 \$ w3 j! H& }
  3. 1 l" `: A" i( V2 _- E
  4.       //Tout = (arr + 1) * (psc + 1) / 72;# n( Q# L* D* [( X0 A

  5. 1 d) D  w& U/ a1 N4 i
  6.       TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
    0 {8 j4 Q& e5 ^
  7.       NVIC_InitTypeDef NVIC_InitStructure;
    6 k9 \. U& o$ I: ?* V

  8.   ^8 |! r* a8 a* B, O% x
  9.       RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);                           //1  此数字对应上面的步骤
    4 B4 u* v, D) B( |& x  a/ |

  10. # |' j- V8 {/ Z3 j/ {( B
  11.       TIM_TimeBaseStructure.TIM_Period = arr;    //这个值在0~65535之间 为16位计数值   //28 w3 M8 `* w& f& k
  12.       TIM_TimeBaseStructure.TIM_Prescaler = psc;: F$ o; K- ]3 m3 k
  13.       TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;; k6 E- E4 r( [; c4 L( k" N
  14.       TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;- c  b8 l& P$ Y; O. L( B
  15.       TIM_TimeBaseInit(TIM3, & TIM_TimeBaseStructure);. Y6 r, @& k5 K; i$ P6 p
  16. . D# a9 l' p7 U9 b  K- a
  17.       TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE );                                    //36 u# h- Y/ p2 o% H" ^$ \" _
  18. % M$ R& [" M: F. \# n5 i: @& b
  19.       NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;                                //4
    0 L1 p7 |/ ~1 s- g/ z% k( {
  20.       NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;( @" j) Y: ^0 X! }0 }# Z% V
  21.       NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;; Q$ o( B: K; W6 W: U7 h* K
  22.       NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;3 y- W# D' |7 W$ B
  23.       NVIC_Init(&NVIC_InitStructure);. v+ i$ L9 e5 P3 k7 r
  24. # s+ J. \* T* |% W7 C
  25.     TIM_Cmd(TIM3, ENABLE);                                                         //58 s/ N) L; s/ c+ C3 Y$ p/ \
  26. }) u% F( E4 j* k$ P* y; t, l; t2 v& W% F' @

  27. 4 s7 F  u+ H: c! g0 y1 x/ S) s
  28. void TIM3_IRQHandler(void)                                                         //6- W* j% J2 T( w) A3 I4 X, e
  29. {; a7 x! [+ ~& N8 A$ p
  30.     if(TIM_GetITStatus(TIM3, TIM_IT_Update) == SET)
    4 Y9 C. _$ q. o
  31.         {+ ?) r- {. b& G* I: B7 g) n
  32.         TIM_ClearITPendingBit(TIM3, TIM_IT_Update);9 d' h* p8 E$ Z; W
  33.         LED1 = !LED1;/ P1 v: X5 j. i3 [6 o
  34.         }
    5 P  i. B7 }5 E2 b0 R# t1 w
  35. }
复制代码
1 x# y9 b: u8 P/ D! Y1 w0 `1 c
好了、、说到这里、、通用果然是通用的、、对于所有的通用定时器都是同样的原理、、在这里我们只是实现了简单的定时(值加到某个程度就更新产生中断,通过LED灯显示)、、希望大家多多指教、、本博客比较滑稽、、望大家莫怪哈、、有什么说错的地方请像抓腐败那样提出来哈、、我虚心向您请教、、谢谢大家、、

* J+ F# e) ^% r8 ]
收藏 评论0 发布时间:2022-1-18 22:49

举报

0个回答
关于意法半导体
我们是谁
投资者关系
意法半导体可持续发展举措
创新和工艺
招聘信息
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
关注我们
st-img 微信公众号
st-img 手机版