STM32定时器

article/2025/10/10 19:32:15

目录

一 定时器的基本介绍

二 定时器的原理框图

1、定时器时钟

2 时基单元

3 捕获输入

4 PWM输出。

三 定时器的应用

1 定时器的基础定时计数功能

2 PWM比较输出

3 外部脉冲计数                                

a、外部触发输入(ETR--外部时钟模式2)                                                           

b、外部引脚输入(TIx--外部时钟模式1)                                                                      

c、捕获输入脉冲计数

d、输入捕获脉宽测量

定时器作为STM32的基础外设,配置还是比较复杂的,本章主要从定时器的使用,结合寄存器的配置和常用实例来理解定时器。

一 定时器的基本介绍

STM32的定时器主要分为三种:高级定时器(TIM1/TIM8)、通用定时器(TIM2-TIM5)、基本定时器(TIM6/TIM7),基本功能不多做介绍,到处都有。

二 定时器的原理框图

看起来比较复杂,可以分为四个部分:1定时器时钟、2时基单元、3捕获输入、4PWM输出。

1、定时器时钟

定时器时钟来源有四种: 

①内部时钟(CK_INT)。一般不做其他配置,默认为此时钟。

②外部时钟模式:外部触发输入(ETR)

③内部触发输入(ITRx):使用一个定时器作为另一个定时器的预分频器,如可以配置一个定时器Timer1而作为另一个定时器Timer2的预分频器。

定时器的触发可以是以上两种触发模式,也可以是这种通过一个定时器触发另一个定时器的工作方式。发出触发信号的定时器工作于主模式,接受触发信号而启动的定时器工作于从模式。

定时器之间相互联系的主要通过配置TIMx_SMCR寄存器中的TS比特位【ITR0 ITR1 ITR2 ITR3】来设置。

例如,如果要用TIM4去触发TIM2,需要将TIM2的Internal Triger配置成ITR3

④外部时钟模式:外部输入脚(TIx)

可以参考:(121条消息) 定时器定时中断&定时器外部时钟_tim_etrclockmode2config_tz得像个小孩的博客-CSDN博客

关于时钟来源不做展开讲解,可以在应用实例中体会。

2 时基单元

时基单元就是定时器框图的第二部分,它包括三个寄存器:计数器寄存器(TIMx_CNT)、预分频器寄存器(TIMx_PSC)和自动装载寄存器(TIMx_ARR)。对这三个寄存器的介绍如下:

计数器寄存器(TIMx_CNT)
向上计数、向下计数或者中心对齐计数;

预分频器寄存器(TIMx_PSC)
可将时钟频率按1到65535之间的任意值进行分频,可在运行时改变其设置值;

自动装载寄存器(TIMx_ARR)
此处有关介绍信息比较多,也比较简单,不多做介绍。

3 捕获输入

IC1、2和IC3、4可以分别通过软件设置将其映射到TI1、TI2和TI3、TI4;
4个16位捕捉比较寄存器可以编程用于存放检测到对应的每一次输入捕捉时计数器的值;
当产生一次捕捉,相应的CCxIF标志位被置1;同时如果中断或DMA请求使能,则产生中断或DMA请求。
如果当CCxIF标志位已经为1,当又产生一个捕捉,则捕捉溢出标志位CCxOF将被置1

4 PWM输出。

PWM,英文名Pulse Width Modulation,是脉冲宽度调制缩写,它是通过对一系列脉冲的宽度进行调制,等效出所需要的波形。

在STM32单片机中,可以使用定时器的输出比较功能来产生PWM波:
       即PWM模式可以产生一个由TIMx_ARR寄存器确定频率、由TIMx_CCRx寄存器确定占空比的信号。

以上为定时器框架的基本介绍,下面主要通过实例分解来详细学习定时器。

三 定时器的应用

1 定时器的基础定时计数功能

程序实例:STM32F407的TIM1定时800ms,800ms产生一次溢出中断

 TIM_TimeBaseInitTypeDef    TIM_TimeBaseStructure;                                                                     NVIC_InitTypeDef NVIC_InitStructure;

 RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1, ENABLE);
    
    TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);
    TIM_TimeBaseStructure.TIM_Period = 7999;        // 自动重装(TIMx_ARR)
    TIM_TimeBaseStructure.TIM_Prescaler =16799;     //800ms    预分频(TIMx_PSC)
    TIM_TimeBaseStructure.TIM_ClockDivision = 0;    //
    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;   计数模式TIMx->CR1
    TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure);  TIMx->CR1
    TIM_ITConfig(TIM1, TIM_IT_Update, ENABLE);  //允许更新中断  TIMx->DIER
    
    TIM_SetCounter(TIM1,0);                     //设置计数初始值
    TIM_ClearFlag(TIM1,TIM_IT_Update);    //清除更新中断标志
    
    NVIC_InitStructure.NVIC_IRQChannel = TIM1_UP_TIM10_IRQn; //打开定时器中断的通道
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0;  
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;    
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; 
    NVIC_Init(&NVIC_InitStructure);    

    TIM_Cmd(TIM1, ENABLE); //使能计数器  TIMx_CR1  -> CEN

配置中断服务函数:                                                                                                                          void TIM1_UP_TIM10_IRQHandler(void)
{
    if(TIM_GetITStatus(TIM1, TIM_IT_Update) == SET)
    {
     /*****处理事件********/
        TIM_ClearITPendingBit(TIM1, TIM_IT_Update);
    }
}

从定时器的框图和配置步骤来学习:

(1)RCC开启时钟;                                                                                                                (2)选择时基单元的时钟源,对于定时中断选择内部时钟源TIM_InternalClockConfig(TIM_TypeDef* TIMx);默认可以省略。                                                   (3)配置时基单元。包括预分频(TIMx_PSC)、自动重装(TIMx_ARR)、计数模式(TIMx_CNT)等                                                                                                                            (4)配置输出中断控制允许更新中断输出到NVIC                                                                    (5)配置NVIC,在NVIC中打开定时器中断的通道,并分配一个优先级                                    (6)运行控制,配置完成后,需要对计数器进行使能,否则计数器无法运行

  计数器更新时,触发中断,最后再写一个定时器中断服务函数。

2 PWM比较输出

a、实例:TIM3比较通道3,通道4输出PWM

void PWM_Iint(void)
{                                                                                                                                        GPIO_InitTypeDef GPIO_InitStructure                                                                            TIM_OCInitTypeDef TIM_OCInitStructure;
TIM_TimeBaseInitTypeDef    TIM_TimeBaseStructure;
    
 RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);    
  RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
  GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
  GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_DOWN;
  GPIO_Init(GPIOB, &GPIO_InitStructure);                      //先配置IO复用推挽输出
    
  GPIO_PinAFConfig(GPIOB, GPIO_PinSource0, GPIO_AF_TIM3);        //PB0映射为TIM3_ch3
  GPIO_PinAFConfig(GPIOB, GPIO_PinSource1, GPIO_AF_TIM3);        //PB1映射为TIM3_ch4
    
  TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);             
  TIM_TimeBaseStructure.TIM_Period = 60000;      //10HZ
  TIM_TimeBaseStructure.TIM_Prescaler =139; 
  TIM_TimeBaseStructure.TIM_ClockDivision = 0; 
  TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  
  TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);
    
  TIM_OCStructInit(&TIM_OCInitStructure);
  TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;   //设置PWM输出模式
  TIM_OCInitStructure.TIM_Pulse =0;  //占空比初始值
  TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
  TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
  TIM_OC3Init(TIM3, &TIM_OCInitStructure);
  TIM_OC4Init(TIM3, &TIM_OCInitStructure);
    
  TIM_OC3PreloadConfig(TIM3, TIM_OCPreload_Enable);
  TIM_OC4PreloadConfig(TIM3, TIM_OCPreload_Enable);
  TIM_ARRPreloadConfig(TIM3, ENABLE);
    
  TIM_Cmd(TIM3, ENABLE);                                                                                               TIM_SetCompare3(TIM3, pwm_cnt);                                                                             TIM_SetCompare4(TIM3, pwm_cnt);                                                                                                 }

b、根据以上实例讲解原理和配置步骤:                                                                                               (1)、根据选择的IO口配置RCC时钟;                                                                                           (2)、 根据IO口选择映射的定时器通道,本实例中映射TIM3_ch3和TIM3_ch4  ;                         (3)、初始化定时器TIM_TimeBaseInit。配置时基及计数模式,它决定PWM的频率;                   (4)、初始化输出比较参数TIM_OC3Init。配置TIMx->CR2,TIMx->CCMR2,TIMx->CCR3,TIMx->CCER中输出比较的相关寄存器;                                                                                              (5)、使能预装载定时器TIM_OC3PreloadConfig;                                                                        (6)、使能自动重装的预装载寄存器允许位 TIM_ARRPreloadConfig;                                          (7)、使能定时器TIM_Cmd;                                                                                                          (8)、设置改变比较器的值TIM-CCRx,达到不同占空比的效果。

c、寄存器讲解    根据配置步骤讲解寄存器:(1)、1/2/3步骤的讲解省略,直接学习捕获/比较的输出配置;以通道1为例,从最左边进入的是时钟源,由内部时钟(CNT)或者外部触发时钟(ETRF)输入,进入输出模式控制器,通过OCMR1寄存器的OC1M[2:0]位来配置PWM模式,之后进入一个选择器,由CCER寄存器的CC1P位来设置输出极性,最后由CCER寄存器的CC1E位来使能输出,然后通过OC1来输出PWM波。
(2)、初始化输出比较参数    TIMx_CCMR1的OC1M[2:0]配置输出模式(通道1有详细讲解,所以用通道1寄存器)

 TIMx_CCR3(比较器预装载值,决定占空比)

 TIMx_CCER中的CC3P和CC3E分别为输出极性和输出使能

  TIMx_CCMR2中的预装载使能(TIM_OC3PreloadConfig)

  TIMx_CR1中的ARPE自动装载使能(TIM_ARRPreloadConfig)

TIMx-CR1中的CEN 使能定时器TIM_Cmd

3 外部脉冲计数                                

脉冲计数我常用的方法主要左右三种:捕获输入、外部引脚输入(TIx--外部时钟模式1)、外部触发输入(ETR--外部时钟模式2)。下面对上述三种方法分别实例学习。

a、外部触发输入(ETR--外部时钟模式2)                                                           

对管脚PA12 (TIM1_ETR)上的脉冲进行计数

GPIO_InitTypeDef GPIO_InitStructure;                                                                              NVIC_InitTypeDef NVIC_InitStructure;
TIM_TimeBaseInitTypeDef    TIM_TimeBaseStructure;
    
 RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);    
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1, ENABLE);

 GPIO_PinAFConfig(GPIOA, GPIO_PinSource12, GPIO_AF_TIM1);        
 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12;
 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
 GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
 GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
 GPIO_Init(GPIOA, &GPIO_InitStructure);
    
 TIM_DeInit(TIM1);
 TIM_TimeBaseStructInit(&TIM_TimeBaseStructure); 
 TIM_TimeBaseStructure.TIM_Period = 0xFFFF;
 TIM_TimeBaseStructure.TIM_Prescaler =0;     
 TIM_TimeBaseStructure.TIM_ClockDivision = 0; 
 TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  
 TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure);
 TIM_ITConfig(TIM1,TIM_IT_Update,ENABLE);//使能中断,Update更新中断
    
 TIM_SetCounter(TIM1,0);
 TIM_ClearFlag(TIM1,TIM_IT_Update);
 TIM_ETRClockMode2Config(TIM1,TIM_ExtTRGPSC_OFF,TIM_ExtTRGPolarity_NonInverted,0);//SMCR->ETPS外部触发预分频;SMCR->ETP外部触发极性;SMCR->ETF外部触发滤波器;SMCR->ECE使能外部时钟模式2;


 NVIC_InitStructure.NVIC_IRQChannel = TIM1_UP_TIM10_IRQn;   
 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1;
 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; 
 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; 
 NVIC_Init(&NVIC_InitStructure);    

讲解:

PA12是TIM1_ETR管脚,采用外部时钟源模式2配置,PA12管脚每来一个上升沿脉冲,计数器值加1。

1. 由于此例中不需滤波器,因此在 TIMx_SMCR 寄存器中写入 ETF[3:0]=0000
2. 不需要分频,通过在 TIMx_SMCR 寄存器中写入 ETPS[1:0]=00 来设置预分频器
3. 通过在 TIMx_SMCR 寄存器中写入 ETP=0 来选择 ETR 引脚的上升沿检测
4. 通过在 TIMx_SMCR 寄存器中写入 ECE=1 来使能外部时钟模式 2
5. 通过在 TIMx_CR1 寄存器中写入 CEN=1 来使能计数器。综上配置

 TIM_ETRClockMode2Config(TIM1,TIM_ExtTRGPSC_OFF,TIM_ExtTRGPolarity_NonInverted,0);

b、外部引脚输入(TIx--外部时钟模式1)                                                                      

同样的功能,改到PB7,对管脚PB7 (TIM4_CH2)上的脉冲进行计数。如果是PA5(TIM2_CH1_ERT)这样的管脚,既可以配置外部引脚输入(TIx--外部时钟模式1),也可以配置外部触发输入(ETR--外部时钟模式2),具体要根据选择的管脚进行配置。

GPIO_InitTypeDef GPIO_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;
TIM_TimeBaseInitTypeDef    TIM_TimeBaseStructure;
TIM_ICInitTypeDef TIM_ICInitStructure;
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);    
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE);

GPIO_PinAFConfig(GPIOB, GPIO_PinSource7, GPIO_AF_TIM4);        
    
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
GPIO_Init(GPIOB, &GPIO_InitStructure);
    
TIM_DeInit(TIM4);
TIM_TimeBaseStructInit(&TIM_TimeBaseStructure); 
TIM_TimeBaseStructure.TIM_Period = 0xFFFF;
TIM_TimeBaseStructure.TIM_Prescaler =0;     
TIM_TimeBaseStructure.TIM_ClockDivision = 0; 
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  
TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure);

TIM_ICInitStructure.TIM_Channel=TIM_Channel_2;
TIM_ICInitStructure.TIM_ICPolarity=TIM_ICPolarity_Falling;//CCER->CC2P
TIM_ICInitStructure.TIM_ICSelection=TIM_ICSelection_DirectTI;//CCMR1-CC2S方向和输入脚的选择
TIM_ICInitStructure.TIM_ICPrescaler=TIM_ICPSC_DIV1;//输入捕获预分频CCMR1-IC2PSC
TIM_ICInitStructure.TIM_ICFilter=0x00;//输入捕获滤波CCMR1-IC2F
TIM_ICInit(TIM4, &TIM_ICInitStructure); 

TIM_SetCounter(TIM4,0);
TIM_ClearFlag(TIM4,TIM_IT_Update);
TIM_ETRClockMode1Config(TIM4,TIM_ExtTRGPSC_OFF,TIM_ExtTRGPolarity_NonInverted,0); //SMCR->ETPS外部触发预分频;SMCR->ETP外部触发极性;SMCR->ETF外部触发滤波器;SMCR->SMS外部时钟模式1;

TIM_SelectInputTrigger(TIM4, TIM_TS_TI2FP2);//SMCR->TS
TIM_ITConfig(TIM4, TIM_IT_Update, ENABLE);

NVIC_InitStructure.NVIC_IRQChannel = TIM4_IRQn;   
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1;
NVIC_InitStructure.NVIC_IRQChannelSubPriority =0;                            NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; 
NVIC_Init(&NVIC_InitStructure);

例如,要使递增计数器在 TI2 输入出现上升沿时计数,请执行以下步骤:
1. 通过在 TIMx_CCMR1 寄存器中写入 CC2S= 01 ”来配置通道 2 ,使其能够检测 TI2
入的上升沿。
2. 通过在 TIMx_CCMR1 寄存器中写入 IC2F[3:0] 位来配置输入滤波时间(如果不需要任
何滤波,请保持 IC2F=0000 )。
注意: 由于捕获预分频器不用于触发操作,因此无需对其进行配置。
3. 通过在 TIMx_CCER 寄存器中写入 CC2P=0 CC2NP=0 来选择上升沿极性。
4. 通过在 TIMx_SMCR 寄存器中写入 SMS=111 ,使定时器在外部时钟模式 1 下工作。
5. 通过在 TIMx_SMCR 寄存器中写入 TS=110 来选择 TI2 作为输入源。
6. 通过在 TIMx_CR1 寄存器中写入 CEN=1 来使能计数器。

c、捕获输入脉冲计数

同样的功能,仍然是PB7管脚,采用捕获输入脉冲计数的方式

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
  GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
  GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
  GPIO_Init(GPIOB, &GPIO_InitStructure);
    
    TIM_DeInit(TIM4);
    TIM_TimeBaseStructInit(&TIM_TimeBaseStructure); 
    TIM_TimeBaseStructure.TIM_Period = 0xFFFF;
    TIM_TimeBaseStructure.TIM_Prescaler =0;     
    TIM_TimeBaseStructure.TIM_ClockDivision = 0; 
    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  
    TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure);

    TIM_ICInitStructure.TIM_Channel=TIM_Channel_2;
    TIM_ICInitStructure.TIM_ICPolarity=TIM_ICPolarity_Falling;
    TIM_ICInitStructure.TIM_ICSelection=TIM_ICSelection_DirectTI;
    TIM_ICInitStructure.TIM_ICPrescaler=TIM_ICPSC_DIV1;
    TIM_ICInitStructure.TIM_ICFilter=0x00;
    TIM_ICInit(TIM4, &TIM_ICInitStructure);

    TIM_SetCounter(TIM4,0);
    TIM_ITConfig(TIM4, TIM_IT_CC2, ENABLE);

    NVIC_InitStructure.NVIC_IRQChannel = TIM4_IRQn;   
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; 
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; 
    NVIC_Init(&NVIC_InitStructure);    

然后在捕获中断函数里,计数++即可。达到和上述两种方法一样的效果。这几种方法要根据不同的管脚进行配置。捕获输入脉冲计数的方法,中断开销太大,每来一个脉冲就要进入中断++,不适合高频信号的计数。

d、输入捕获脉宽测量

 设计思想:在上升沿计数器设为0,然后下降沿读出计数器,就能得到相应时间了,但是这样其实有个小问题,就是有可能有溢出。我们如何解决这个问题呢?答案就是记录中断溢出的次数,然后在后面加上这些时间就好了。

   GPIO_InTIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
    TIM_ICInitTypeDef  TIM5_ICInitStruct;
    NVIC_InitTypeDef NVIC_InitStructure;
    
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM5,ENABLE);//TIM5时钟使能
   GPIO_PinAFConfig(GPIOA,GPIO_PinSource0,GPIO_AF_TIM5);//使能PORTA时钟
    
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; //GPIOA0
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;//复用功能
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;    //速度100MHz
    GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; //推挽复用输出
    GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_DOWN; //下拉
    GPIO_Init(GPIOA,&GPIO_InitStructure); //初始化PA0                                           
    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_TimeBaseInit(TIM5,&TIM_TimeBaseStructure);
    
    TIM5_ICInitStruct.TIM_Channel = TIM_Channel_1;//选择输入端 IC1映射到TI1上
    TIM5_ICInitStruct.TIM_ICFilter = 0x00;//上升沿捕获
    TIM5_ICInitStruct.TIM_ICPolarity = TIM_ICPolarity_Rising;//映射到TI1上
    TIM5_ICInitStruct.TIM_ICPrescaler = TIM_ICPSC_DIV1;//配置输入分频,不分频
    TIM5_ICInitStruct.TIM_ICSelection = TIM_ICSelection_DirectTI;// 配置输入滤波器 不滤波
    TIM_ICInit(TIM5 ,&TIM5_ICInitStruct);
    TIM_ITConfig(TIM5,TIM_IT_Update|TIM_IT_CC1,ENABLE);
    TIM_Cmd(TIM5,ENABLE );     //使能定时器5
    
    NVIC_InitStructure.NVIC_IRQChannel = TIM5_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=2;//抢占优先级3
    NVIC_InitStructure.NVIC_IRQChannelSubPriority =2;        //子优先级3
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;            //IRQ通道使能
    NVIC_Init(&NVIC_InitStructure);    //根据指定的参数初始化VIC寄存器、   

u8  TIM5CH1_CAPTURE_STA=0;    //输入捕获状态                            
u32    TIM5CH1_CAPTURE_VAL;    //输入捕获值(TIM2/TIM5是32位)
//定时器5中断服务程序     
void TIM5_IRQHandler(void)
{             
     if((TIM5CH1_CAPTURE_STA&0X80)==0)//还未成功捕获    
    {
        //溢出
        if(TIM_GetITStatus(TIM5, TIM_IT_Update) != RESET)
        {         
            if(TIM5CH1_CAPTURE_STA&0X40)//已经捕获到高电平了
            {
                if((TIM5CH1_CAPTURE_STA&0X3F)==0X3F)//高电平太长了
                {
                    TIM5CH1_CAPTURE_STA|=0X80;        //标记成功捕获了一次
                    TIM5CH1_CAPTURE_VAL=0XFFFFFFFF;
                }else TIM5CH1_CAPTURE_STA++;
            }     
        }
        //未溢出
        if(TIM_GetITStatus(TIM5, TIM_IT_CC1) != RESET)//捕获1发生捕获事件
        {    
            if(TIM5CH1_CAPTURE_STA&0X40)        //捕获到一个下降沿         
            {                  
                TIM5CH1_CAPTURE_STA|=0X80;        //标记成功捕获到一次高电平脉宽
                TIM5CH1_CAPTURE_VAL=TIM_GetCapture1(TIM5);//获取当前的捕获值.
                 TIM_OC1PolarityConfig(TIM5,TIM_OCPolarity_High); //CC1P=0 设置为上升沿捕获
            }else                                  //还未开始,第一次捕获上升沿
            {
                TIM5CH1_CAPTURE_STA=0;            //清空
                TIM5CH1_CAPTURE_VAL=0;
                TIM5CH1_CAPTURE_STA|=0X40;        //标记捕获到了上升沿
                TIM_Cmd(TIM5,DISABLE );     //关闭定时器5
                 TIM_SetCounter(TIM5,0);
                 TIM_OC1PolarityConfig(TIM5,TIM_OCPolarity_Low);      //CC1P=1 设置为下降沿捕获
                TIM_Cmd(TIM5,ENABLE );     //使能定时器5
            }            
        }                                                
     }
    TIM_ClearITPendingBit(TIM5, TIM_IT_CC1|TIM_IT_Update); //清除中断标志位
}

1、 开启 TIM5 时钟,配置 PA0 为复用功能(AF2),并开启下拉电阻。
2、初始化 TIM5,设置 TIM5 的 ARR 和 PSC。                                                                             3、设置 TIM5 的输入捕获参数,开启输入捕获。
4、使能捕获和更新中断(设置 TIM5 的 DIER 寄存器)。
5、设置中断优先级,编写中断服务函数。
6、使能定时器(设置 TIM5 的 CR1 寄存器)。


http://chatgpt.dhexx.cn/article/jnBQJ046.shtml

相关文章

SpringBoot定时器

SpringBoot定时器 1 介绍2 注解3 代码4 Cron表达式1. 格式2. 取值3. 特殊字符4. 经典案例 5 Cron既然那么麻烦就生成吧 1 介绍 定时器是一种控制任务延时调用,或者周期调用的技术。 作用:定时邮件、短信发送、更新数据、同步数据、检查数据库和缓存数据…

555定时器

555定时器 美国signetics公司1972年研制,取代机械式定时器,因为输入端有3个5k欧的电阻得名 电路结构 输入引脚 输出引脚 1.因为要接电容,电流较大,需要用oc门进行输出 2.需要接上拉电阻接vcc功能引脚 5号引脚对参考电压的影…

SysTick 定时器

11.1关于 SysTick 定时器 SysTick定时器(又名系统滴答定时器)是存在于Cortex-M3的一个定时器,只要是ARM Cotex-M系列内核的MCU都包含这个定时器。使用内核的SysTick定时器来实现延时,可以不占用系统定时器,节约资源。由于SysTick是在CPU核内…

JavaScript之定时器

定时器 一、 setTimeout() 定时器二、停止 setTimeout() 定时器三、setInterval() 定时器四、清除setInterval() 定时器五、电子时钟案例 在很多页面中,我们都可以看到一些倒计时或者和时间相关的效果,今天小熊将就JavaScript里面的倒计时做一概述。 首先…

STM32-通用定时器-定时器中断

1 STM32的定时器 STM32F103ZET6一共有8个定时器,其中分别为:高级定时器(TIM1、TIM8);通用定时器(TIM2、TIM3、TIM4、TIM5);基本定时器(TIM6、TIM7)。 …

Python——定时器

1.定时器 Timer定时器源码实现,和自定义一个线程方式一样,都是继承Thread类,重写了run()方法,只是实现的功能是延时执行一个函数或方法。 (1)线程定时器(Timer)解释: Timer类是Thread的子类&a…

[JavaEE]定时器

专栏简介: JavaEE从入门到进阶 题目来源: leetcode,牛客,剑指offer. 创作目标: 记录学习JavaEE学习历程 希望在提升自己的同时,帮助他人,,与大家一起共同进步,互相成长. 学历代表过去,能力代表现在,学习能力代表未来! 目录: 1.定时器的概念 2.标准库中的定时器 3.实现定时…

定时器的作用

一、简介。 在很多时候,我们设计网页时,为了某种表现形式,会使用到定时器这一功能,如:为了保证用户有仔细阅读我们的用户条款,我们会给确认按钮设置只有条款被打开,并超过5秒才允许点击。 二、…

定时器简介

文章目录 一.定时器基本介绍A.CPU时序B.定时器的原理 二.定时/计数器的相关寄存器A.定时器工作方式寄存器(TMOD)B.控制寄存器(TCON) 三.定时器的四种工作方式图解 一.定时器基本介绍 A.CPU时序 振荡周期:CPU外部晶振…

定时器基本常识

1.概念解读 1.1定时器和计数器,电路一样 1.2定时或者计数的本质就是让单片机某个部件数数 1.3当定时器用的时候,靠内部震荡电路数数 1.4当计数器用的时候,书外面的信号,读取针脚的数据 2.定时器怎么定时 定时器的本质原理&a…

定时器详解

1. 什么是定时器(timer) 定时器实际上就是Soc当中的一个内部外设。 (1)定时器与计数器 定时器常与计数器扯到一起,计数器也是soc当中的一个内部外设,计数器顾名思义是用来计数的,就和我们的秒…

定时器(Timer)

一、定时器是什么? 定时器类似于我们生活中的闹钟,可以设定一个时间来提醒我们。 而定时器是指定一个时间去执行一个任务,让程序去代替人工准时操作。 标准库中的定时器: Timer 方法作用void schedule(TimerTask task, long delay)指定dela…

STM32-定时器详解

目录 前言 一、定时器基本介绍 1. STM32定时器 2. 通用定时器功能和特点 3. 计数器模式 4. 定时器工作原理 a.定时器框图 b.时钟产生器部分 c.时基单元 d.输入捕获通道 e.输出比较通道(PWM) 二、定时器中断应用 1.内部时钟选择 2.计数器模式 …

typedef和#define

typedef是c语言中一个重要的关键字其作用是为一种数据类型定义了一个新的名字这里的类型包括(int,char,double 等)和自定义数据类型,通俗一点来说就是为一种数据类型起一个别名 举个例子: 定义一个整型变量a并将其初始化为666&a…

typedef和define的区别、typedef的具体用法

typedef最核心的用法:给数据类型取别名,这个别名既可以是此数据类型的替换,也是指向此数据类型的指针。 具体用法(对普通数据类型取别名): 对结构体数据类型取别名: typedef与define的区别&…

#define与typedef的区别

目录 (1)原理不同 (2)功能不同 (3)作用域不同 (4)对指针的操作不同 (5)补充 a.指针常量 b.常量指针 typedef和define都是替一个对象取一个别名&#x…

C语言中的typedef

C语言中的"typedef" 一、什么是typedef typedef是用于定义新的类型名,在编程中可以用typedef来定义新的类型名来代替已有的类型名 格式: typedef 已有类型名 新的类型名 通俗点说,就是为已有的类型取别名,例如 老鼠&am…

define 与typedef的区别

define 与typedef大体功能都是使用时给一个对象取一个别名,增强程序的可读性,但它们在使用时有以下几点区别: 1.定义不一样 define定义后面不用加分号,并且它的别名在对象的前面 typedef需要加分号,并且它的别后面替…

C语言学习笔记---typedef 简介

在单片机和操作系统中 typedef 会经常用到,它可以为某一个类型自定义名称。和#define比较类似。但是又有不同的地方。 typedef 创建的符号只能用于数据类型,不能用于值。而#define 创建的符号可以用于值。typedef 是由编译器来解释,而不是预…

typedef介绍

[20210330更新]:这篇博客写的时间有点久了:)。本次更新修改了博客内容中的错误和表述不当的地方。 本文介绍C语言中的关键字 typedef 的用法。 1 概述 typedef 为C语言的关键字,作用是为一种数据类型定义一个新名字,这里的数据类型包括内部…