第十三章 定时器中断实验 1.硬件平台:正点原子探索者STM32F407开发板 2.软件平台:MDK5.1 3.固件库版本:V1.4.0
实验8 定时器中断实验.zip
(474.94 KB)
这一章,我们将向大家介绍如何使用STM32F4的通用定时器,STM32F4的定时器功能十分强大,有TIME1和TIME8等高级定时器,也有TIME2~TIME5,TIM9~TIM14等通用定时器,还有TIME6和TIME7等基本定时器,总共达14个定时器之多。在本章中,我们将使用TIM3的定时器中断来控制DS1的翻转,在主函数用DS0的翻转来提示程序正在运行。本章,我们选择难度适中的通用定时器来介绍,本章将分为如下几个部分: 13.1 STM32F4通用定时器简介 13.2 硬件设计 13.3 软件设计 13.4下载验证 13.1 STM32F4通用定时器简介STM32F4的通用定时器包含一个16位或32位自动重载计数器(CNT),该计数器由可编程预分频器(PSC)驱动。STM32F4的通用定时器可以被用于:测量输入信号的脉冲长度(输入捕获)或者产生输出波形(输出比较和PWM)等。 使用定时器预分频器和RCC时钟控制器预分频器,脉冲长度和波形周期可以在几个微秒到几个毫秒间调整。STM32F4的每个通用定时器都是完全独立的,没有互相共享的任何资源。 STM3的通用TIMx (TIM2~TIM5和TIM9~TIM14)定时器功能包括: 1)16位/32位(仅TIM2和TIM5)向上、向下、向上/向下自动装载计数器(TIMx_CNT),注 意:TIM9~TIM14只支持向上(递增)计数方式。 2)16位可编程(可以实时修改)预分频器(TIMx_PSC),计数器时钟频率的分频系数为1~65535之间的任意数值。 3)4个独立通道(TIMx_CH1~4,TIM9~TIM14最多2个通道),这些通道可以用来作为: A.输入捕获 B.输出比较 C.PWM生成(边缘或中间对齐模式) ,注意:TIM9~TIM14不支持中间对齐模式 D.单脉冲模式输出 4)可使用外部信号(TIMx_ETR)控制定时器和定时器互连(可以用1个定时器控制另外一个定时器)的同步电路。 5)如下事件发生时产生中断/DMA(TIM9~TIM14不支持DMA): A.更新:计数器向上溢出/向下溢出,计数器初始化(通过软件或者内部/外部触发) B.触发事件(计数器启动、停止、初始化或者由内部/外部触发计数) C.输入捕获 D.输出比较 E.支持针对定位的增量(正交)编码器和霍尔传感器电路(TIM9~TIM14不支持) F.触发输入作为外部时钟或者按周期的电流管理(TIM9~TIM14不支持) 由于STM32F4通用定时器比较复杂,这里我们不再多介绍,请大家直接参考《STM32F4xx中文参考手册》第392页,通用定时器一章。下面我们介绍一下与我们这章的实验密切相关的几个通用定时器的寄存器(以下均以TIM2~TIM5的寄存器介绍,TIM9~TIM14的略有区别,具体请看《STM32F4xx中文参考手册》对应章节)。 首先是控制寄存器1(TIMx_CR1),该寄存器的各位描述如图13.1.1所示:
图13.1.1 TIMx_CR1寄存器各位描述 在本实验中,我们只用到了TIMx_CR1的最低位,也就是计数器使能位,该位必须置1,才能让定时器开始计数。接下来介绍第二个与我们这章密切相关的寄存器:DMA/中断使能寄存器(TIMx_DIER)。该寄存器是一个16位的寄存器,其各位描述如图13.1.2所示: 图13.1.2 TIMx_ DIER寄存器各位描述 这里我们同样仅关心它的第0位,该位是更新中断允许位,本章用到的是定时器的更新中断,所以该位要设置为1,来允许由于更新事件所产生的中断。 接下来我们看第三个与我们这章有关的寄存器:预分频寄存器(TIMx_PSC)。该寄存器用设置对时钟进行分频,然后提供给计数器,作为计数器的时钟。该寄存器的各位描述如图13.1.3所示:
图13.1.3 TIMx_ PSC寄存器各位描述这里,定时器的时钟来源有4个: 1)内部时钟(CK_INT) 2)外部时钟模式1:外部输入脚(TIx) 3)外部时钟模式2:外部触发输入(ETR),仅适用于TIM2、TIM3、TIM4 4)内部触发输入(ITRx):使用A定时器作为B定时器的预分频器(A为B提供时钟)。 这些时钟,具体选择哪个可以通过TIMx_SMCR寄存器的相关位来设置。这里的CK_INT时钟是从APB1倍频的来的,除非APB1的时钟分频数设置为1(一般都不会是1),否则通用定时器TIMx的时钟是APB1时钟的2倍,当APB1的时钟不分频的时候,通用定时器TIMx的时钟就等于APB1的时钟。这里还要注意的就是高级定时器以及TIM9~TIM11的时钟不是来自APB1,而是来自APB2的。 这里顺带介绍一下TIMx_CNT寄存器,该寄存器是定时器的计数器,该寄存器存储了当前定时器的计数值。 接着我们介绍自动重装载寄存器(TIMx_ARR),该寄存器在物理上实际对应着2个寄存器。一个是程序员可以直接操作的,另外一个是程序员看不到的,这个看不到的寄存器在《STM32F4xx中文参考手册》里面被叫做影子寄存器。事实上真正起作用的是影子寄存器。根据TIMx_CR1寄存器中APRE位的设置:APRE=0时,预装载寄存器的内容可以随时传送到影子寄存器,此时2者是连通的;而APRE=1时,在每一次更新事件(UEV)时,才把预装载寄存器(ARR)的内容传送到影子寄存器。 自动重装载寄存器的各位描述如图13.1.4所示: 图13.1.4 TIMx_ ARR寄存器各位描述 最后,我们要介绍的寄存器是:状态寄存器(TIMx_SR)。该寄存器用来标记当前与定时器相关的各种事件/中断是否发生。该寄存器的各位描述如图13.1.5所示: 图13.1.5 TIMx_ SR寄存器各位描述 关于这些位的详细描述,请参考《STM32F4xx中文参考手册》第429页。 只要对以上几个寄存器进行简单的设置,我们就可以使用通用定时器了,并且可以产生中断。 这一章,我们将使用定时器产生中断,然后在中断服务函数里面翻转DS1上的电平,来指示定时器中断的产生。接下来我们以通用定时器TIM3为实例,来说明要经过哪些步骤,才能达到这个要求,并产生中断。这里我们就对每个步骤通过库函数的实现方式来描述。首先要提到的是,定时器相关的库函数主要集中在固件库文件stm32f4xx_tim.h和stm32f4xx_tim.c文件中。定时器配置步骤如下: 1)TIM3时钟使能。 TIM3是挂载在APB1之下,所以我们通过APB1总线下的使能使能函数来使能TIM3。调用的函数是: RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE); ///使能TIM3时钟 2)初始化定时器参数,设置自动重装值,分频系数,计数方式等。 在库函数中,定时器的初始化参数是通过初始化函数TIM_TimeBaseInit实现的: voidTIM_TimeBaseInit(TIM_TypeDef*TIMx, TIM_TimeBaseInitTypeDef* TIM_TimeBaseInitStruct); 第一个参数是确定是哪个定时器,这个比较容易理解。第二个参数是定时器初始化参数结构体指针,结构体类型为TIM_TimeBaseInitTypeDef,下面我们看看这个结构体的定义: typedef struct { uint16_t TIM_Prescaler; uint16_t TIM_CounterMode; uint16_t TIM_Period; uint16_t TIM_ClockDivision; uint8_t TIM_RepetitionCounter; } TIM_TimeBaseInitTypeDef; 这个结构体一共有5个成员变量,要说明的是,对于通用定时器只有前面四个参数有用,最后一个参数TIM_RepetitionCounter是高级定时器才有用的,这里不多解释。 第一个参数TIM_Prescaler是用来设置分频系数的,刚才上面有讲解。 第二个参数TIM_CounterMode是用来设置计数方式,上面讲解过,可以设置为向上计数,向下计数方式还有中央对齐计数方式,比较常用的是向上计数模式TIM_CounterMode_Up和向下计数模式TIM_CounterMode_Down。 第三个参数是设置自动重载计数周期值,这在前面也已经讲解过。 第四个参数是用来设置时钟分频因子。 针对TIM3初始化范例代码格式: TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_TimeBaseStructure.TIM_Period = 5000; TIM_TimeBaseStructure.TIM_Prescaler =7199; TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); 3)设置TIM3_DIER允许更新中断。 因为我们要使用TIM3的更新中断,寄存器的相应位便可使能更新中断。在库函数里面定时器中断使能是通过TIM_ITConfig函数来实现的: void TIM_ITConfig(TIM_TypeDef* TIMx, uint16_t TIM_IT, FunctionalState NewState); 第一个参数是选择定时器号,这个容易理解,取值为TIM1~TIM17。 第二个参数非常关键,是用来指明我们使能的定时器中断的类型,定时器中断的类型有很多种,包括更新中断TIM_IT_Update,触发中断TIM_IT_Trigger,以及输入捕获中断等等。 第三个参数就很简单了,就是失能还是使能。 例如我们要使能TIM3的更新中断,格式为: TIM_ITConfig(TIM3,TIM_IT_Update,ENABLE ); 4)TIM3中断优先级设置。 在定时器中断使能之后,因为要产生中断,必不可少的要设置NVIC相关寄存器,设置中断优先级。之前多次讲解到用NVIC_Init函数实现中断优先级的设置,这里就不重复讲解。 5)允许TIM3工作,也就是使能TIM3。 光配置好定时器还不行,没有开启定时器,照样不能用。我们在配置完后要开启定时器,通过TIM3_CR1的CEN位来设置。在固件库里面使能定时器的函数是通过TIM_Cmd函数来实现的: void TIM_Cmd(TIM_TypeDef* TIMx, FunctionalState NewState) 这个函数非常简单,比如我们要使能定时器3,方法为: TIM_Cmd(TIM3, ENABLE); //使能TIMx外设 6)编写中断服务函数。 在最后,还是要编写定时器中断服务函数,通过该函数来处理定时器产生的相关中断。在中断产生后,通过状态寄存器的值来判断此次产生的中断属于什么类型。然后执行相关的操作,我们这里使用的是更新(溢出)中断,所以在状态寄存器SR的最低位。在处理完中断之后应该向TIM3_SR的最低位写0,来清除该中断标志。 在固件库函数里面,用来读取中断状态寄存器的值判断中断类型的函数是: ITStatus TIM_GetITStatus(TIM_TypeDef* TIMx, uint16_t) 该函数的作用是,判断定时器TIMx的中断类型TIM_IT是否发生中断。比如,我们要判断定时器3是否发生更新(溢出)中断,方法为:
后续内容由于文字限制,请下载pdf和源码学习
|