
前言 定时器作为微控制器不可缺少的外设,在STM32中也是如此。相信不少初学者学到定时器的时候对STM32的学习热情就大打折扣甚至想要放弃了,因为这一部分知识确实比较复杂。但是,如果你在之前对GPIO、串口通信、外部中断的学**把这些外设掌握了的话,学习这个新知识并不难。 例本章共计1万余字,从STM32定时器的原理、寄存器介绍、定时器配置以及定时器的几个常用的功能(如定时器中断、定时器输出比较PWM波形、定时器输入捕获测电平长度、定时器编码器模式应用等)的使用方法来教大家掌握定时器这一外设。 一、定时器基本介绍 1. STM32定时器 1、上来说就是用来定时的机器,是存在于STM32单片机中的一个外设。STM32总共有8个定时器,分别是2个高级定时器(TIM1、TIM8),4个通用定时器(TIM2、TIM3、TIM4、TIM5)和2个基本定时器(TIM5、TIM6),如下图所示: ![]() 这三种定时器的区别如下: ![]() 即:高级定时器具有捕获/比较通道和互补输出,通用定时器只有捕获/比较通道,基本定时器没有以上两者。 2. 通用定时器功能和特点 STM32的众多定时器中我们使用最多的是高级定时器和通用定时器,而高级定时器一般也是用作通用定时器的功能,下面我们就以通用定时器为例进行讲解,其功能和特点包括: 位于低速的APB1总线上(APB1) 16 位向上、向下、向上/向下(中心对齐)计数模式,自动装载计数器(TIMx_CNT)。 16 位可编程(可以实时修改)预分频器(TIMx_PSC),计数器时钟频率的分频系数 为 1~65535 之间的任意数值。 4 个独立通道(TIMx_CH1~4),这些通道可以用来作为: ① 输入捕获 ② 输出比较 ③ PWM 生成(边缘或中间对齐模式) ④ 单脉冲模式输出 可使用外部信号(TIMx_ETR)控制定时器和定时器互连(可以用 1 个定时器控制另外一个定时器)的同步电路。 如下事件发生时产生中断/DMA(6个独立的IRQ/DMA请求生成器): ①更新:计数器向上溢出/向下溢出,计数器初始化(通过软件或者内部/外部触发) ②触发事件(计数器启动、停止、初始化或者由内部/外部触发计数) ③输入捕获 ④输出比较 ⑤支持针对定位的增量(正交)编码器和霍尔传感器电路 ⑥触发输入作为外部时钟或者按周期的电流管理 STM32 的通用定时器可以被用于:测量输入信号的脉冲长度(输入捕获)或者产生输出波形(输出比较和 PWM)等。 使用定时器预分频器和 RCC 时钟控制器预分频器,脉冲长度和波形周期可以在几个微秒到几个毫秒间调整。STM32 的每个通用定时器都是完全独立的,没有互相共享的任何资源。 3. 计数器模式 通用定时器可以向上计数、向下计数、向上向下双向计数模式。 ①向上计数模式:计数器从0计数到自动加载值(TIMx_ARR),然后重新从0开始计数并且产生一个计数器溢出事件。 ②向下计数模式:计数器从自动装入的值(TIMx_ARR)开始向下计数到0,然后从自动装入的值重新开始,并产生一个计数器向下溢出事件。 ③中央对齐模式(向上/向下计数):计数器从0开始计数到自动装入的值-1,产生一个计数器溢出事件,然后向下计数到1并且产生一个计数器溢出事件;然后再从0开始重新计数。 ![]() 4. 定时器工作原理 a.定时器框图 下面就是STM32定时器的工作款图了,是学习STM32定时器必须要掌握的。很多学习者学会了通过库函数来配置定时器,实现了简单的应用却忽略了基本原理,这就对导致在复杂应用的设计上出现低级的错误。所以建议读者认真掌握定时器的工作框图,明白内在的原理。 ![]() 框图可以分为四个大部分(用红色笔表示出),分别是:①时钟产生器部分,②时基单元部分,③输入捕获部分、④输出比较部分。 b.时钟产生器部分 在第一部分时钟选择上,STM32定时器有四种时钟源选择(图中蓝色笔标识),分别是: ①内部时钟(CK_INT) ②外部时钟模式:外部触发输入(ETR) ③内部触发输入(ITRx):使用一个定时器作为另一个定时器的预分频器,如可以配置一个定时器Timer1而作为另一个定时器Timer2的预分频器。 ④外部时钟模式:外部输入脚(TIx) ![]() 这四种情况可由右图表示: 其中,内部触发输入口1~4除了ITR1/ITR2/ITR3/ITR4之外还有一种情况:用一个定时器作为另一个定时器的分频器。 外部捕获比较引脚有两种,分别是: 引脚1:TI1FP1或TI1F_ED 引脚2:TI2FP2 c.时基单元 时基单元就是定时器框图的第二部分,它包括三个寄存器,分别是:计数器寄存器(TIMx_CNT)、预分频器寄存器(TIMx_PSC)和自动装载寄存器(TIMx_ARR)。对这三个寄存器的介绍如下: 计数器寄存器(TIMx_CNT) 向上计数、向下计数或者中心对齐计数; 计数器寄存器(TIMx_CNT) 可将时钟频率按1到65535之间的任意值进行分频,可在运行时改变其设置值; 自动装载寄存器(TIMx_ARR) 如果TIMx_CR1寄存器中的ARPE位为0,ARR寄存器的内容将直接写入影子寄存器;如果ARPE为1,ARR寄存器的那日同将在每次的更新时间UEV发生时,传送到影子寄存器; 如果TIM1_CR1中的UDIS位为0,当计数器产生溢出条件时,产生更新事件。 d.输入捕获通道 IC1、2和IC3、4可以分别通过软件设置将其映射到TI1、TI2和TI3、TI4; 4个16位捕捉比较寄存器可以编程用于存放检测到对应的每一次输入捕捉时计数器的值; 当产生一次捕捉,相应的CCxIF标志位被置1;同时如果中断或DMA请求使能,则产生中断或DMA请求。 如果当CCxIF标志位已经为1,当又产生一个捕捉,则捕捉溢出标志位CCxOF将被置1。 ![]() e.输出比较通道(PWM) PWM模式运行产生: 定时器2、3和4可以产生4独立的信号 频率和占空比可以进行如下设定: 一个自动重载寄存器用于设定PWM的周期; 每个PWM通道有一个捕捉比较寄存器用于设定占空时间。 例如:产生一个40KHz的PWM信号:在定时器2的时钟为72MHz下,占空比为50% : 预分频寄存器设置为0 (计数器的时钟为TIM1CLK/(O+1)),自动重载寄存器设为 1799,CCRx寄存器设为899。 两种可设置PWM模式: 边沿对齐模式 中心对齐模式 ![]() 二、定时器中断应用 1.内部时钟选择 ![]() ![]() 除非APB1的分频系数是1,否则通用定时器的时钟等于APB1时钟的2倍。 默认调用SystemInit函数情况下: SYSCLK=72M AHB时钟=72M APB1时钟=36M 所以APB1的分频系数=AHB/APB1时钟=2 所以,通用定时器时钟CK_INT=2*36M=72M 2.计数器模式 向下计数模式:(时钟分频因子=1) ![]() ![]() 向下计数模式:(时钟分频因子=1) ![]() ![]() 中央对齐计数模式:(时钟分频因子=1 ARR=6) ![]() ![]() 3.定时器中断实验相关寄存器 计数器当前值寄存器CNT ![]() 预分频寄存器TIMx_PSC ![]() 自动重装载寄存器(TIMx_ARR) ![]() 控制寄存器1(TIMx_CR1) ![]() DMA中断使能寄存器(TIMx_DIER) ![]() 4. 常用库函数 定时器参数初始化:
结构体内部成员:
声明方式(一般):
定时器使能函数:
定时器中断使能函数
状态标志位获取和清除
5. 定时器中断实现步骤 ① 能定时器时钟。 RCC_APB1PeriphClockCmd(); ② 初始化定时器,配置ARR,PSC。 TIM_TimeBaseInit(); ③开启定时器中断,配置NVIC。 void TIM_ITConfig(); NVIC_Init(); ④ 使能定时器。 TIM_Cmd(); ⑥ 编写中断服务函数。 TIMx_IRQHandler(); 6. 应用实例 下面是使用定时器中断的代码,我们设置为每500ms中断一次,中断服务函数控制LED实现LED状态取反。时间计算方法为: Tout(溢出时间)=(ARR+1)(PSC+1)/Tclk
三、定时器PWM输出实验 1. 通用定时器PWM概述 PWM,英文名Pulse Width Modulation,是脉冲宽度调制缩写,它是通过对一系列脉冲的宽度进行调制,等效出所需要的波形(包含形状以及幅值),对模拟信号电平进行数字编码,也就是说通过调节占空比的变化来调节信号、能量等的变化,占空比就是指在一个周期内,信号处于高电平的时间占据整个信号周期的百分比,例如方波的占空比就是50%。PWM的功能有很多种,比如控制呼吸灯、控制直流电机或者舵机等驱动原件等等,是单片机的一个十分重要的功能。 在STM32单片机中,可以使用定时器的输出比较功能来产生PWM波: 即PWM模式可以产生一个由TIMx_ARR寄存器确定频率、由TIMx_CCRx寄存器确定占空比的信号。其框图如下图所示: ![]() 可见,横坐标是时间变量,纵坐标是CNT计数值,CNT计数值随着时间的推进会不断经历从0到ARR,清零复位再到ARR的这一过程。这之中还有一个数值是CCRx即比较值,通过比较值和输出配置可以使之输出高低电平逻辑,这样就产生了PWM波形。通过调节ARR的值可以调节PWM的周期,调节CCRx的值大小可以调节PWM占空比。 我们以通道1为例,详细讲解PWM的工作过程,如下图所示: ![]() 从最左边进入的是时钟源,由内部时钟(CNT)或者外部触发时钟(ETRF)输入,进入输入模式控制器,通过OCMR1寄存器的OC1M[2:0]位来配置PWM模式,之后进入一个选择器,由CCER寄存器的CC1P位来设置输出极性,最后由CCER寄存器的CC1E位来使能输出,然后通过OC1来输出PWM波。 CCR1:捕获比较(值)寄存器(x=1,2,3,4):设置比较值。 CCMR1: OC1M[2:0]位: 对于PWM方式下,用于设置PWM模式1【110】或者PWM模式2【111】 CCER:CC1P位:输入/捕获1输出极性。0:高电平有效,1:低电平有效。 CCER:CC1E位:输入/捕获1输出使能。0:关闭,1:打开。 2. PWM模式 PWM有PWM模式1和模式2两种模式,它们之间的区别用寄存器TIMx_CCMR1的OC1M[2:0]位来分析: ![]() 表中红色框标识的地方就是PWM模式1和模式2的定义和区别,可以简单理解为:PWM模式1的情况下,当前值小于比较值为有效电平;PWM模式2的情况下,当前值大于比较值为有效电平。 理解这一点对之后的PWM配置十分重要。 下面是对PWM模式1以及向上计数配置情况的说明: ![]() 3.相关寄存器介绍 捕获/比较寄存器1(TIMx_CCR1) ![]() 这里以寄存器1举例,其它的三个寄存器(CCR2、CCR3、CCR4)都是一样的 捕获比较模式寄存器1(TIMx_CCMR1) ![]() ![]() 可以看到,每个捕获/比较模式寄存器可以控制两个通道,这样的话每个定时器就对应两个捕获/比较模式寄存器。其最常用的位就是0C1M(OC2M)位了,这两个位是用来设置PWM模式的,有模式1和模式2两种,这就和前面所讲的对应上了。 捕获/比较使能寄存器(TIMx_CCER) ![]() 可以看到,位0(CC1E)和位1(CC1P)是捕获比较使能寄存器最常用的两个位,分别控制输出使能和输出极性,这就也和刚刚讲的对应上了。 自动重装载寄存器(TIMx_ARR) ![]() 这个寄存器不太常用,下面的库函数配置会讲解其库函数用法。 4. 定时器输出通道引脚 定时器输出PWM和定时器中断不同,定时器中断只需要开启这一外设即可工作,定时器输出PWM需要在单片机的引脚上输出实实在在的脉冲信号。 下面是定时器3的通道引脚,可以使用部分映射或者完全映射。其它定时器的引脚可以查看芯片手册。 ![]() 5.定时器PWM库函数配置 输出库函数配置 和定时器中断实验不同,在初始化时基单元之后,还需要对输出通道进行初始化:
其结构体成员如下:
初始化实例:
设置比较值函数
使能输出比较预装载
使能自动重装载的预装载寄存器允许位
输出配置步骤 ① 使能定时器3和相关IO口时钟。 使能定时器3时钟:RCC_APB1PeriphClockCmd(); 使能GPIOB时钟:RCC_APB2PeriphClockCmd(); ② 初始化IO口为复用功能输出。函数:GPIO_Init(); GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; ③ 这里我们是要把PB5用作定时器的PWM输出引脚,所以要重映射配置, 所以需要开启AFIO时钟。同时设置重映射。 RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE); GPIO_PinRemapConfig(GPIO_PartialRemap_TIM3, ENABLE); ④ 初始化定时器:ARR,PSC等:TIM_TimeBaseInit(); ⑤ 初始化输出比较参数:TIM_OC2Init(); ⑥ 使能预装载寄存器: TIM_OC2PreloadConfig(TIM3, TIM_OCPreload_Enable); ⑦ 使能定时器。TIM_Cmd(); ⑧ 不断改变比较值CCRx,达到不同的占空比效果:TIM_SetCompare2(); 6.实例 我们使用STM32单片机的定时器TIM3的PWM功能,输出占空比可变的PWM波,用来驱动LED灯,从而达到LED亮度由暗变亮,又从亮变暗,如此循环。代码如下:
总结 本章从STM32定时器的原理、寄存器介绍、定时器配置以及定时器的几个常用的功能(如定时器中断、定时器输出比较PWM波形)的使用方法来教大家掌握定时器这一外设。希望读者能够仔细学习,掌握这一重要的外设。 |
基于定时器捕获测量脉宽的应用示例
狂欢三】STM32C031使用TIM定时器DMA方式实现WS2812彩灯输出(三)
【狂欢三】STM32C031使用TIM定时器DMA方式实现PWM输出(二)
【狂欢三】STM32C031使用TIM定时器PWM输出
stm32使用定时器触发dma传输,启动dma没反应的几种情况的解决方法
定时器剩余通道是否可以做PWM输出呢?
基于STM32双定时器+ADC+DMA实战经验分享
基于STM32的定时器触发ADC时可能遇到的情形
【NUCLEO-U545RE-Q评测】5. 基本计时器
基于STM32的定时器不按设定超时产生中断