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

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

[复制链接]
STMCU小助手 发布时间:2022-1-18 22:49
通用定时器可以用来:
1、测量输入信号的脉冲长度,也就是输入捕获(请对此四字短语养成良好的形象哈)
2、产生输出波形,也就是输出比较或者PWM(同上)
) e6 X4 E8 D, B! i
     那葵花兄,通用的原理呢?又见葵花兄坦然自若的说:通用定时器是一个通过可编程预分频器(PSC)(注:这三个字母要有印象哈)驱动的16位自动装载计数器(CNT)(注:同上)构成。
    好了,大家撇开葵花兄,把焦点重新聚焦在我身上、那通用定时器有什么功能呢?我想肯定有人马上反应:有定时的功能、、额、、我也不能说你错是吧、、那我来具体点吧,其实也不是我具体,多亏了“葵花宝典”第STM32篇中文参考手册  :
1) 16 位向上 、向下 、向上/向下自动装载计数器 (TIMx_CNT)。
! w0 S/ }! Q0 H& E# ?' I, y2) 16 位可编程( 可以实时修改 )预分频器(TIMx_PSC)(TIMx_PSC)(TIMx_PSC),计数器时钟频率的分系为 1~ 65535 之间的任意数值。: B$ J$ R. `0 g3 _
3)4个独立通道( TIMx_CH1~4 TIMx_CH1~4 TIMx_CH1~4 TIMx_CH1~4), ~3 k- y# O. b' x( V& o
A.输入捕获
0 A4 I2 _, W3 E* p% _% {4 DB.输出比较
6 I! H, A) U- ?. P- K3 J& K% NC.PWM 生成 (边缘或中间对齐模式 ); U3 A3 R2 F: Z0 A( T
D.单脉冲模式输出; \4 B$ d5 [. r" q+ U! j; a
4)可使用外部信号( TIMx_ETR )控制定时器和互连(可以用1个定时器控制另外一个定时器)的同步电路。8 V: ]/ ^, F3 K1 o- J$ K& i4 F; L
5)如下事件发生时产生中断/DMA:
2 b2 P/ q- Q$ Z3 d1 L! bA.更新:计数器向上溢出/向下溢出,计数器初始化 ( 通过软件或者内部 /外部触发 )
1 g% w! a' N% x+ k8 F; ~# sB.触发事件  (计数器启动、停止初始化或者由内部/外部触发计数  )
6 y0 m8 l9 q0 }# P- z7 P% eC.输入捕获 0 y( ?1 j3 r( e* E5 |6 O
D.输出比较* S% i) O; l7 [% }5 f+ k
E.支持针对定位的增量  (正交 )编码器和霍尔传感电路 3 k$ a# ~, \; R$ Q. a3 N. s6 j
F.触发输入作为外部时钟或者按周期的电流管理
   当然啦,在此博客中不考虑那么多功能先哈、、先淡定、、若是被吓到了、、先去喝杯茶压压惊哈、、那我们怎么来实现定时的功能呢?
只要定时、当达到一定容量时,就会发生溢出、、聪明的人又知道我要说什么了吧?世上太多聪明的人了、、没错了、、把中断这位兄弟叫上、让溢出兄和中断兄合身、、(省略合身等待的时间)、对了、诞生了溢出中断、我们给取个好听的名字:更新中断、在这里,我们是利用中断来定时的、、
   接下来,容小弟介绍下与此博客有关系的寄存器(具体的位是什么功能我就不讲了哈,因为我们是采用库而不是操作寄存器,想知道的可以参考中文参考手册哈),大家掌声响起:欢迎控制兄(TIMx_CR1)

. K6 {8 i# U/ \$ W: D. V5 }
261716482798777.png
8 k% L$ a3 j. C1 X5 o3 S3 s
欢迎中断/DMA使能兄(TIMx_DIER):
261721185133335.png
% H: [# B9 m6 v1 y( i. B- Y
欢迎预分频器兄(TIMx_PSC):
261723384359882.png

3 z: P3 v( L2 b4 B  M: l$ a3 J! M
欢迎自动重装载寄存器兄(TIMx_ARR):
261725582163986.png
; D# s( p, D, h5 e* O2 m5 {. |# E+ p+ m$ w
好了、、这四位可以在旁边休息了(我们就不介绍了)、、因为我们的重点:库兄要出现了,接下来,用最热烈的掌声欢迎:
首先:有请挂载时钟大神ABP1(为什么要请这位大神呢?因为我们的通用定时器是挂载在这位ABP1的,证据如下)

+ `- B5 p7 g" O( [; {# {
261731028577156.png

6 p: L# J7 G! k+ {- Y
再者,我们有请第二大神:初始化大神:TIM_TimeBaseInit(为什么是他呢?为什么?究竟是为什么?啊哈、、请看)

0 M2 G/ f! O2 H4 B6 M( y3 R( K
261735027167733.png
. P3 X+ [0 p: f* A, b
看到了吧、、所以请不要怀疑大神的地位哈、、至于参数具体是哪些,大家可以打开#include "stm32f10x_tim.h"哈,在这里就不多说了、、相信大家都是聪明人、
前两位大神出来了,我们还有大神、、哎、、大神无处不在、、来、有请类型中断使能大神
void TIM_ITConfig(TIM_TypeDef* TIMx, u16 TIM_IT, FunctionalState  NewState)
     看到中断、、大家又有想到什么猫腻没??条件发射就是这样炼成的、、没错了、、就是中断优先级设置大神啦、、这个大家应该有印象、我就不细说了哈,没印象的话这位大神可就要躲到某个角落伤心了,
     看到以上大神、、其实还有一位大神没出现、、此大神没出现、、其他大神啥都不是、、对了、就是定时器启动大神
      void TIM_Cmd(TIM_TypeDef* TIMx, FunctionalState NewState)
      好了、、最后,编写中断服务函数,在中断函数里检查中断标志位和清中断标志位,还有你想要做的事,相信大家也是比较熟悉的了、、
   在这里,我们来谈谈时钟的问题、、大家可以看到系统时钟树图里:
3 F  {& |; [& ~
261829075919160.png
; Y8 X* q4 K: b& m
    看到我美丽销魂的涂鸦没、、由于我的系统时钟初始化是36MHZ的、而且设置APB1的预分频系数为2,所以定时器的时钟就为72MHZ,大家可以根据自己的系统时钟设置选择哈、、
到此,我们来总结下使用中断的定时器:
1.开启挂载在ABP1上的TIM3的时钟
2.对定时器进行初始化
3.设置中断类型
4.设置中断优先级
5.开启定时器3
6.编写中断服务函数 看一下参考代码:
  1. void Timer_3_Init(u16 arr,u16 psc)) T( I* U( L. `1 ]' a7 |! V
  2. {
    # a. K4 x1 H6 Z- f% j0 i& _: j* _2 v  e/ M

  3. 0 S4 w" r! L: C, K1 Z# H2 ]
  4.       //Tout = (arr + 1) * (psc + 1) / 72;
    ' @" E6 K/ N& u$ u/ A3 y0 k3 u5 z

  5. 2 x: W( |: u# B
  6.       TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;+ s9 @2 s4 o0 M
  7.       NVIC_InitTypeDef NVIC_InitStructure;) p- l8 W# V) L# }" L) i! O
  8. ' g( y: W2 c3 U
  9.       RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);                           //1  此数字对应上面的步骤3 q* Q! R) v: g/ N2 A

  10. 5 B/ I- k0 b! D8 r) S, F4 n
  11.       TIM_TimeBaseStructure.TIM_Period = arr;    //这个值在0~65535之间 为16位计数值   //2
    ) b1 U. ]; L7 m! i( _, A" o. I+ E
  12.       TIM_TimeBaseStructure.TIM_Prescaler = psc;
      ?9 f: @7 t, B( H3 A
  13.       TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
    ; p- Y4 Y+ H( A& t
  14.       TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  [9 |  ~1 i7 n! z3 c0 S- y
  15.       TIM_TimeBaseInit(TIM3, & TIM_TimeBaseStructure);
    + i* e$ p" m' e( h

  16. 8 z$ \! E  o! M+ h8 |
  17.       TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE );                                    //3
    ( n; D" l0 t' O7 ^* U
  18. / n2 O) c! N) K
  19.       NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;                                //4( N" L/ }8 a0 H
  20.       NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;6 m" \3 b6 Y9 O' x
  21.       NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;' O4 Q% {$ D* z# H- S9 N2 k; G
  22.       NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    4 |- ], E9 R% P9 K6 |) z
  23.       NVIC_Init(&NVIC_InitStructure);; D8 l0 _2 c& y+ ^( }. t8 r5 n
  24. : {/ J: E  ~* V, |
  25.     TIM_Cmd(TIM3, ENABLE);                                                         //5; ?4 X  b! [  r
  26. }2 }/ s9 Y# q2 ^1 V$ A$ R7 P) Z
  27. 4 c% d0 l, C; w" k
  28. void TIM3_IRQHandler(void)                                                         //6
    ) r" q7 ^& C% l
  29. {  g: K( n- o5 ^$ L
  30.     if(TIM_GetITStatus(TIM3, TIM_IT_Update) == SET)
      \% L4 g. h' {0 A. g+ y! T) c
  31.         {. s4 A8 i: l: ^) F
  32.         TIM_ClearITPendingBit(TIM3, TIM_IT_Update);
    7 g8 N$ S( r* J5 S, R4 f& `
  33.         LED1 = !LED1;
    0 L# V2 k( \' B% u" v
  34.         }
    # }; e& R5 S( o, j( C, K) q# u9 F3 T
  35. }
复制代码
3 E9 h$ a$ ?0 _/ L
好了、、说到这里、、通用果然是通用的、、对于所有的通用定时器都是同样的原理、、在这里我们只是实现了简单的定时(值加到某个程度就更新产生中断,通过LED灯显示)、、希望大家多多指教、、本博客比较滑稽、、望大家莫怪哈、、有什么说错的地方请像抓腐败那样提出来哈、、我虚心向您请教、、谢谢大家、、
: k! b" W+ p5 y3 r! \  A
收藏 评论0 发布时间:2022-1-18 22:49

举报

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