07 STM32F4定时器中断

1、综述

  (1)STM32F40x系列总共最多有14个定时器。分有:
    高级定时器:TIM1 和 TIM9;
    通用定时器:TIM2~TIM5 , TIM9~TIM14;
    基本定时器:TIM6 和 TIM7;


  (2)计数器的三种模式:
  <1>向上计数:计数器从零开始计数,一直计数到自动加载值(TIMx_ARR),然后重新从0开始计数,并产生一个计数器溢出事件;
  <2>向下计数:计数器从自动装入的值(TIMx_ARR)开始,向下计数到0,然后从自动装入的值重新开始,并产生一个计数器向下溢出事件;
  <3>中央对齐模式(向上/向下计数模式):从0开始,计数到自动装入的值,产生一个计数器上溢出事件,然后向下计数到0,又产生一个计数器下溢出事件,然后又从0开始重新计数。


2、相关寄存器

  (1)计数器当前值寄存器 CNT
  CNT是定时器的计数器,存储着当前定时器的计数值。


  (2)预分频寄存器 TIMx_PSC
 &emsp该寄存器对时钟进行分频,然后提供给计数器,作为计数频率。PSC是16位寄存器,存储着预分频器值。计数器计数频率为时钟频率除以(PSC+1)。
  注意:这里,定时器的时钟来源有4个:
  <1>内部时钟(CK_INT)
  <2>外部时钟模式1:外部输入脚(TIx)
  <3>外部时钟模式2:外部触发输入(ETR)
  <4>内部触发输入(ITRx):时钟级联,A为B提供时钟


  (3)自动重装载寄存器 TIMx_ARR
  ARR为要装载到实际自动重载寄存器的值。该寄存器在物理上对应着2个寄存器。一个是程序员可以直接操作的,另外一个是程序员看不到的,叫做“影子寄存器”。这里不做深入讨论。


  (4)控制寄存器1 TIMx_CR1
  16位寄存器,低10位有效。但是,我们仅关注其最低位(位0),称作CEN位,该位是计数器使能位,必须置1,才能让定时器开始计数。


  (5)DMA中断使能寄存器 TIMx_DIER
  16位寄存器,我们仅关心其最低位(位0),该位是更新中断允许位,要使用定时器中断的功能,那么该位要置1,来允许由于更新事件所产生的中断。


  (6)状态寄存器 TIMx_SR
  该寄存器用来标记当前与定时器相关的各种事件/中断是否发生。在这里,我们主要关注它的最低位(位0,UIF位),该位在发生更新事件时由硬件置1.但是需要通过软件清零。


3、通用定时器的配置步骤(TIM3为例)

  (1)使能TIM3时钟

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);

  TIM3挂载在总线APB1之下,所以,需要使能相应的时钟。


  (2)初始化定时器参数,设置自动装载值,分频系数,计数方式等

void TIM_TimeBaseInit(TIMx,&结构体)

  例如:

//初始化定时器参数,设置自动重装值,分频系数,计数方式等
TIM_TimeBaseStructure.TIM_Prescaler = psc;//定时器分频
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;//向上计数模式
TIM_TimeBaseStructure.TIM_Period = arr;//自动重装载值
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
//TIM_TimeBaseStructure.TIM_RepetitionCounter = //不需要,高级定时器才使用
TIM_TimeBaseInit(TIM3,&TIM_TimeBaseStructure);

  (3)设置TIM3_DIER允许更新中断

//允许定时器3更新中断
TIM_ITConfig(TIM3,TIM_IT_Update,ENABLE);

  (4)中断优先级设置

//中断优先级设置
NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;//定时器3中断  通道名称定义在顶层头文件中
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;//抢占优先级1
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;//响应优先级3
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//
NVIC_Init(&NVIC_InitStructure);//初始化NVIC

  (5)使能定时器

//使能定时器3
TIM_Cmd(TIM3,ENABLE);

  (5)编写中断服务函数
  在中断产生后,通过状态寄存器的值,判断此次产生的是哪个类型的中断,然后在执行相关操作。在处理完中断之后,应该对SR寄存器的相应标志位清除。

/*编写中断服务函数*/
void TIM3_IRQHandler(void)
{
    if(TIM_GetITStatus(TIM3,TIM_IT_Update) == SET)//更新中断
    {
        LED1 = !LED1;
    }
    TIM_ClearITPendingBit(TIM3,TIM_IT_Update);//清除中断标志位
}

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容