前言
定时器作为微控制器不可缺少的外设,在STM32中也是如此。相信不少初学者学到定时器的时候对STM32的学习热情就大打折扣甚至想要放弃了,因为这一部分知识确实比较复杂。但是,如果你在之前对GPIO、串口通信、外部中断的学习中把这些外设掌握了的话,学习这个新知识并不难。
软件版本:
STM32CubeMX:6.30
KEIL:5.0
FIyMcu:V0.993
开发板:STM32F103C8T6
一、定时器基本介绍
1、STM32定时器
上来说就是用来定时的机器,是存在于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、定时器工作原理
定时器框图
下面就是STM32定时器的工作款图了,是学习STM32定时器必须要掌握的。很多学习者学会了通过库函数来配置定时器,实现了简单的应用却忽略了基本原理,这就对导致在复杂应用的设计上出现低级的错误。所以建议读者认真掌握定时器的工作框图,明白内在的原理。
框图可以分为四个大部分(用红色笔表示出),分别是:①时钟产生器部分,②时基单元部分,③输入捕获部分、④输出比较部分。
时钟产生部分
在第一部分时钟选择上,STM32定时器有四种时钟源选择(图中蓝色笔标识),分别是:
①内部时钟(CK_INT)
②外部时钟模式:外部触发输入(ETR)
③内部触发输入(ITRx):使用一个定时器作为另一个定时器的预分频器,如可以配置一个定时器Timer1而作为另一个定时器Timer2的预分频器。
④外部时钟模式:外部输入脚(TIx)
这四种情况可以由下图表示:
其中,内部触发输入口1~4除了ITR1/ITR2/ITR3/ITR4之外还有一种情况:用一个定时器作为另一个定时器的分频器。
外部捕获比较引脚有两种,分别是:
引脚1:TI1FP1或TI1F_ED
引脚2:TI2FP2
时基单元
时基单元就是定时器框图的第二部分,它包括三个寄存器,分别是:计数器寄存器(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,当计数器产生溢出条件时,产生更新事件。
输入捕获通道
- IC1、2和IC3、4可以分别通过软件设置将其映射到TI1、TI2和TI3、TI4;
- 4个16位捕捉比较寄存器可以编程用于存放检测到对应的每一次输入捕捉时计数器的值;
- 当产生一次捕捉,相应的CCxIF标志位被置1;同时如果中断或DMA请求使能,则产生中断或DMA请求。
- 如果当CCxIF标志位已经为1,当又产生一个捕捉,则捕捉溢出标志位CCxOF将被置1。
输出比较通道
-
PWM模式运行产生:
定时器2、3和4可以产生4独立的信号频率和占空比可以进行如下设定:
一个自动重载寄存器用于设定PWM的周期;
每个PWM通道有一个捕捉比较寄存器用于设定占空时间。
例如:产生一个40KHz的PWM信号:在定时器2的时钟为72MHz下,占空比为50% :
预分频寄存器设置为0 (计数器的时钟为TIM1CLK/(O+1)),自动重载寄存器设为 1799,CCRx寄存器设为899。
- 两种可设置PWM模式:
边沿对齐模式
中心对齐模式
4、常用库函数
定时器参数初始化:
void TIM_TimeBaseInit(TIM_TypeDef* TIMx,TIM_TimeBaseInitTypeDef*TIM_TimeBaseInitStruct);
结构体内部成员:
typedef struct { uint16_t TIM_Prescaler; uint16_t TIM_CounterMode; uint16_t TIM_Period; uint16_t TIM_ClockDivision; uint8_t TIM_RepetitionCounter; } TIM_TimeBaseInitTypeDef;
声明方式(一般):
TIM_TimeBaseStructure.TIM_Period = 4999;
TIM_TimeBaseStructure.TIM_Prescaler =7199;
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);
定时器使能函数:
void TIM_Cmd(TIM_TypeDef* TIMx, FunctionalState NewState)
定时器中断使能函数
void TIM_ITConfig(TIM_TypeDef* TIMx, uint16_t TIM_IT, FunctionalState NewState);
状态标志位获取和清除
FlagStatus TIM_GetFlagStatus(TIM_TypeDef* TIMx, uint16_t TIM_FLAG);
void TIM_ClearFlag(TIM_TypeDef* TIMx, uint16_t TIM_FLAG);
ITStatus TIM_GetITStatus(TIM_TypeDef* TIMx, uint16_t TIM_IT);
void TIM_ClearITPendingBit(TIM_TypeDef* TIMx, uint16_t TIM_IT);
定时器中断实现步骤
① 能定时器时钟。
RCC_APB1PeriphClockCmd();
② 初始化定时器,配置ARR,PSC。
TIM_TimeBaseInit();
③开启定时器中断,配置NVIC。
void TIM_ITConfig();
NVIC_Init();
④ 使能定时器。
TIM_Cmd();
⑥ 编写中断服务函数。
TIMx_IRQHandler();
实验过程
实验步骤
1、新建工程,进行基本配置。
参照实验1,芯片选择“STM32F103C8”。
先点击Timers选择TIM2,并如图进行配置TIM2:
如图进行RCC配置:
如图进行SYS配置:
配置串口
点击Project Manager对项目进行命名和选择IDE(选择MDK-ARM):
再点击Code Generator对如图所示的勾选:
最后点击右上角的GENERATE CODE即可完成项目的创建。
项目实现
点击main.c文件,添加以下代码:
HAL_TIM_Base_Start_IT(&htim2);
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
static uint32_t time_cnt =0;
static uint32_t time_cnt1=0;
if(htim->Instance == TIM2)
{
if(++time_cnt >= 200)
{
time_cnt =0;
HAL_GPIO_TogglePin(GPIOA,GPIO_PIN_5);
}
if(++time_cnt1>=500)
{
time_cnt1=0;
printf("hello windows!\r\n");
}
}
}
重写printf需要添加头文件stdio.h
int fputc(int ch,FILE *f){
uint8_t temp[1]={ch};
HAL_UART_Transmit(&huart1,temp,1,2);
return ch;
}
编译无误后打开mcuisp软件选择项目所生成的.hex文件进行烧录,具体方法参照结尾部分寄存器实现流水灯
结果:
定时器两秒亮灯,五秒传“hello world!”
总结
本次实验通过定时器实现LED亮灭以及串口发送数据,使我名明白了定时器的原理和使用方法。定时器的运用并不难。