首页 > 其他分享 >STM32单片机芯片与内部12 TIM-基本定时器TIM67 -定时计数功能、库函数配置、HAL库配置、多定时器中断

STM32单片机芯片与内部12 TIM-基本定时器TIM67 -定时计数功能、库函数配置、HAL库配置、多定时器中断

时间:2024-12-10 09:30:54浏览次数:11  
标签:12 HAL void htim TIM 定时器 BASIC 库函数

目录

一、功能

二、库函数工程模板

1、NVIC_InitTypeDef与TIM_TimeBaseInitTypeDef

2、时钟使能

3、初始化

4、清除中断

5、开启/关闭中断

6、使能/失能计数器

三、库函数API

1、初始化的封装

2、中断服务函数

四、HAL库工程模板

1、TIM_HandleTypeDef

2、TIM_MasterConfigTypeDef

3、时钟使能

4、初始化

5、启动定时器

五、HAL库API

1、初始化的封装

2、中断服务函数

六、用户侧

1、初始化全流程

2、多定时器中断

(1)、标准库

(2)、HAL库


一、功能

        前文提到,TIM6、TIM7基本定时器可以实现常规的定时功能,因此可以进行定时器,然后进行LED的反转。

二、库函数工程模板

1、NVIC_InitTypeDef与TIM_TimeBaseInitTypeDef

        NVIC_InitTypeDef之前EXTI的章节介绍过了,主要是设定通道与优先级及使能。

        TIM_TimeBaseInitTypeDef在stm32f10x_tim.h中,其包括了基本定时器、通用定时器、高级定时器的共用,但是例如TIM_ClockDivision、TIM_CounterMode只存在于通用定时器,TIM_RepetitionCounter则只存在于高级定时器。预分频系数、计数模式、自动重装载寄存器、时钟分频因子、重复计数器。

  • TIM_Prescaler:定时器预分频器设置,时钟源经该预分频器才是定时器计数时钟 CK_CNT,它设定 PSC 寄存器的值。计算公式为:计数器时钟频率 (fCK_CNT) 等于 fCK_PSC / (PSC[15:0] + 1),可实现 1 至 65536 分频。
  • TIM_CounterMode:定时器计数方式,可设置为向上计数、向下计数以及中心对齐。通用和高级控定时器允许选择任意一种。
  • TIM_Period:定时器周期,实际就是设定自动重载寄存器 ARR 的值, ARR 为要装载到实际
  • 动重载寄存器(即影子寄存器)的值,可设置范围为 0 至 65535。
  • TIM_ClockDivision:时钟分频,设置定时器时钟 CK_INT 频率与死区发生器以及数字滤波器采样时钟频率分频比。通用和高级可以选择 1、 2、 4 分频。
  • TIM_RepetitionCounter:重复计数器,只有 8 位,只存在于高级定时器。
typedef struct
{
  uint16_t TIM_Prescaler;         /*!< Specifies the prescaler value used to divide the TIM clock.
                                       This parameter can be a number between 0x0000 and 0xFFFF */

  uint16_t TIM_CounterMode;       /*!< Specifies the counter mode.
                                       This parameter can be a value of @ref TIM_Counter_Mode */

  uint16_t TIM_Period;            /*!< Specifies the period value to be loaded into the active
                                       Auto-Reload Register at the next update event.
                                       This parameter must be a number between 0x0000 and 0xFFFF.  */ 

  uint16_t TIM_ClockDivision;     /*!< Specifies the clock division.
                                      This parameter can be a value of @ref TIM_Clock_Division_CKD */

  uint8_t TIM_RepetitionCounter;  /*!< Specifies the repetition counter value. Each time the RCR downcounter
                                       reaches zero, an update event is generated and counting restarts
                                       from the RCR value (N).
                                       This means in PWM mode that (N+1) corresponds to:
                                          - the number of PWM periods in edge-aligned mode
                                          - the number of half PWM period in center-aligned mode
                                       This parameter must be a number between 0x00 and 0xFF. 
                                       @note This parameter is valid only for TIM1 and TIM8. */
} TIM_TimeBaseInitTypeDef; 

2、时钟使能

        可以看到其中基本定时器TIM6、TIM7都是挂载在APB1上。

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM6,ENABLE);
#define RCC_APB1Periph_TIM6              ((uint32_t)0x00000010)
#define RCC_APB1Periph_TIM7              ((uint32_t)0x00000020)

3、初始化

        可以看到最终计数为:72MHz/72/1000=1KHz。中断服务函数为TIM6_IRQHandler,优先级为0,3。


    NVIC_InitTypeDef NVIC_InitStructure; 
    // 设置中断组为0
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);		
		// 设置中断来源
    NVIC_InitStructure.NVIC_IRQChannel = TIM6_IRQn;	
		// 设置主优先级为 0
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;	 
	  // 设置抢占优先级为3
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;	
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    NVIC_Init(&NVIC_InitStructure);

    TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
		// 开启定时器时钟,即内部时钟CK_INT=72M
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM6, ENABLE);
		// 自动重装载寄存器的值,累计TIM_Period+1个频率后产生一个更新或者中断
    TIM_TimeBaseStructure.TIM_Period = (1000-1);	
	  // 时钟预分频数为
    TIM_TimeBaseStructure.TIM_Prescaler= 71;
	  // 初始化定时器
    TIM_TimeBaseInit(TIM6, &TIM_TimeBaseStructure);

4、清除中断

        其中在定时器初始化后需要进行一次,再每次中断服务函数结束后需要再次进行。

  • TIM_ClearFlag:清除定时器的事件标志,适用于事件驱动的应用。
  • TIM_ClearITPendingBit:清除定时器的中断待处理位,适用于中断驱动的应用。
void TIM_ClearFlag(TIM_TypeDef* TIMx, uint16_t TIM_FLAG)
{  
  /* Check the parameters */
  assert_param(IS_TIM_ALL_PERIPH(TIMx));
  assert_param(IS_TIM_CLEAR_FLAG(TIM_FLAG));
  /* Clear the flags */
  TIMx->SR = (uint16_t)~TIM_FLAG;
}
void TIM_ClearITPendingBit(TIM_TypeDef* TIMx, uint16_t TIM_IT)
{
  /* Check the parameters */
  assert_param(IS_TIM_ALL_PERIPH(TIMx));
  assert_param(IS_TIM_IT(TIM_IT));
  /* Clear the IT pending Bit */
  TIMx->SR = (uint16_t)~TIM_IT;
}

5、开启/关闭中断

void TIM_ITConfig(TIM_TypeDef* TIMx, uint16_t TIM_IT, FunctionalState NewState)
{  
  /* Check the parameters */
  assert_param(IS_TIM_ALL_PERIPH(TIMx));
  assert_param(IS_TIM_IT(TIM_IT));
  assert_param(IS_FUNCTIONAL_STATE(NewState));
  
  if (NewState != DISABLE)
  {
    /* Enable the Interrupt sources */
    TIMx->DIER |= TIM_IT;
  }
  else
  {
    /* Disable the Interrupt sources */
    TIMx->DIER &= (uint16_t)~TIM_IT;
  }
}

6、使能/失能计数器

void TIM_Cmd(TIM_TypeDef* TIMx, FunctionalState NewState)
{
  /* Check the parameters */
  assert_param(IS_TIM_ALL_PERIPH(TIMx));
  assert_param(IS_FUNCTIONAL_STATE(NewState));
  
  if (NewState != DISABLE)
  {
    /* Enable the TIM Counter */
    TIMx->CR1 |= TIM_CR1_CEN;
  }
  else
  {
    /* Disable the TIM Counter */
    TIMx->CR1 &= (uint16_t)(~((uint16_t)TIM_CR1_CEN));
  }
}

三、库函数API

1、初始化的封装

// 中断优先级配置
static void BASIC_TIM_NVIC_Config(void)
{
    NVIC_InitTypeDef NVIC_InitStructure; 
    // 设置中断组为0
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);		
		// 设置中断来源
    NVIC_InitStructure.NVIC_IRQChannel = BASIC_TIM_IRQ ;	
		// 设置主优先级为 0
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;	 
	  // 设置抢占优先级为3
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;	
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    NVIC_Init(&NVIC_InitStructure);
}
static void BASIC_TIM_Mode_Config(void)
{
    TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
		
		// 开启定时器时钟,即内部时钟CK_INT=72M
    BASIC_TIM_APBxClock_FUN(BASIC_TIM_CLK, ENABLE);
	
		// 自动重装载寄存器的值,累计TIM_Period+1个频率后产生一个更新或者中断
    TIM_TimeBaseStructure.TIM_Period = BASIC_TIM_Period;	

	  // 时钟预分频数为
    TIM_TimeBaseStructure.TIM_Prescaler= BASIC_TIM_Prescaler;
	
		// 时钟分频因子 ,基本定时器没有,不用管
    //TIM_TimeBaseStructure.TIM_ClockDivision=TIM_CKD_DIV1;
		
		// 计数器计数模式,基本定时器只能向上计数,没有计数模式的设置
    //TIM_TimeBaseStructure.TIM_CounterMode=TIM_CounterMode_Up; 
		
		// 重复计数器的值,基本定时器没有,不用管
		//TIM_TimeBaseStructure.TIM_RepetitionCounter=0;
	
	  // 初始化定时器
    TIM_TimeBaseInit(BASIC_TIM, &TIM_TimeBaseStructure);
		
		// 清除计数器中断标志位
    TIM_ClearFlag(BASIC_TIM, TIM_FLAG_Update);
	  
		// 开启计数器中断
    TIM_ITConfig(BASIC_TIM,TIM_IT_Update,ENABLE);
		
		// 使能计数器
    TIM_Cmd(BASIC_TIM, ENABLE);	
}
void BASIC_TIM_Init(void)
{
	BASIC_TIM_NVIC_Config();
	BASIC_TIM_Mode_Config();
}

        可以看到,除此之外在h文件中进行了额外的define定义,使得c文件中不再出现TIM6的标识,这样的好处是,如果要修改为TIM7,可以统一在h中修改不会遗漏错漏。

#define BASIC_TIM6 // 如果使用TIM7,注释掉这个宏即可

#ifdef  BASIC_TIM6 // 使用基本定时器TIM6
#define            BASIC_TIM                   TIM6
#define            BASIC_TIM_APBxClock_FUN     RCC_APB1PeriphClockCmd
#define            BASIC_TIM_CLK               RCC_APB1Periph_TIM6
#define            BASIC_TIM_Period            (1000-1)
#define            BASIC_TIM_Prescaler         71
#define            BASIC_TIM_IRQ               TIM6_IRQn
#define            BASIC_TIM_IRQHandler        TIM6_IRQHandler

#else  // 使用基本定时器TIM7
#define            BASIC_TIM                   TIM7
#define            BASIC_TIM_APBxClock_FUN     RCC_APB1PeriphClockCmd
#define            BASIC_TIM_CLK               RCC_APB1Periph_TIM7
#define            BASIC_TIM_Period            1000-1
#define            BASIC_TIM_Prescaler         71
#define            BASIC_TIM_IRQ               TIM7_IRQn
#define            BASIC_TIM_IRQHandler        TIM7_IRQHandler

#endif

2、中断服务函数

        中断服务函数进行的time++,相当于1ms加一次,一秒钟加1000次。中断服务函数最好不要写过多的代码,如果单纯让LED闪烁,这个地方也可以加入,千万不要加入时延等操作。

void  BASIC_TIM_IRQHandler (void)
{
	if ( TIM_GetITStatus( BASIC_TIM, TIM_IT_Update) != RESET ) 
	{	
		time++;
		TIM_ClearITPendingBit(BASIC_TIM , TIM_FLAG_Update);  		 
	}		 	
}

四、HAL库工程模板

1、TIM_HandleTypeDef

        TIM_HandleTypeDef在stm32f1xx_hal_tim.h中。之前说过HAL库就是高级封装的库函数,节省了很多步骤,可以看到结构体,对于基本定时器而言仅可以使用Instance、Prescaler、Period。

typedef struct
#endif /* USE_HAL_TIM_REGISTER_CALLBACKS */
{
  TIM_TypeDef                 *Instance;     /*!< Register base address             */
  TIM_Base_InitTypeDef        Init;          /*!< TIM Time Base required parameters */
  HAL_TIM_ActiveChannel       Channel;       /*!< Active channel                    */
  DMA_HandleTypeDef           *hdma[7];      /*!< DMA Handlers array
                                                  This array is accessed by a @ref DMA_Handle_index */
  HAL_LockTypeDef             Lock;          /*!< Locking object                    */
  __IO HAL_TIM_StateTypeDef   State;         /*!< TIM operation state               */

#if (USE_HAL_TIM_REGISTER_CALLBACKS == 1)
  void (* Base_MspInitCallback)(struct __TIM_HandleTypeDef *htim);              /*!< TIM Base Msp Init Callback                              */
  void (* Base_MspDeInitCallback)(struct __TIM_HandleTypeDef *htim);            /*!< TIM Base Msp DeInit Callback                            */
  void (* IC_MspInitCallback)(struct __TIM_HandleTypeDef *htim);                /*!< TIM IC Msp Init Callback                                */
  void (* IC_MspDeInitCallback)(struct __TIM_HandleTypeDef *htim);              /*!< TIM IC Msp DeInit Callback                              */
  void (* OC_MspInitCallback)(struct __TIM_HandleTypeDef *htim);                /*!< TIM OC Msp Init Callback                                */
  void (* OC_MspDeInitCallback)(struct __TIM_HandleTypeDef *htim);              /*!< TIM OC Msp DeInit Callback                              */
  void (* PWM_MspInitCallback)(struct __TIM_HandleTypeDef *htim);               /*!< TIM PWM Msp Init Callback                               */
  void (* PWM_MspDeInitCallback)(struct __TIM_HandleTypeDef *htim);             /*!< TIM PWM Msp DeInit Callback                             */
  void (* OnePulse_MspInitCallback)(struct __TIM_HandleTypeDef *htim);          /*!< TIM One Pulse Msp Init Callback                         */
  void (* OnePulse_MspDeInitCallback)(struct __TIM_HandleTypeDef *htim);        /*!< TIM One Pulse Msp DeInit Callback                       */
  void (* Encoder_MspInitCallback)(struct __TIM_HandleTypeDef *htim);           /*!< TIM Encoder Msp Init Callback                           */
  void (* Encoder_MspDeInitCallback)(struct __TIM_HandleTypeDef *htim);         /*!< TIM Encoder Msp DeInit Callback                         */
  void (* HallSensor_MspInitCallback)(struct __TIM_HandleTypeDef *htim);        /*!< TIM Hall Sensor Msp Init Callback                       */
  void (* HallSensor_MspDeInitCallback)(struct __TIM_HandleTypeDef *htim);      /*!< TIM Hall Sensor Msp DeInit Callback                     */
  void (* PeriodElapsedCallback)(struct __TIM_HandleTypeDef *htim);             /*!< TIM Period Elapsed Callback                             */
  void (* PeriodElapsedHalfCpltCallback)(struct __TIM_HandleTypeDef *htim);     /*!< TIM Period Elapsed half complete Callback               */
  void (* TriggerCallback)(struct __TIM_HandleTypeDef *htim);                   /*!< TIM Trigger Callback                                    */
  void (* TriggerHalfCpltCallback)(struct __TIM_HandleTypeDef *htim);           /*!< TIM Trigger half complete Callback                      */
  void (* IC_CaptureCallback)(struct __TIM_HandleTypeDef *htim);                /*!< TIM Input Capture Callback                              */
  void (* IC_CaptureHalfCpltCallback)(struct __TIM_HandleTypeDef *htim);        /*!< TIM Input Capture half complete Callback                */
  void (* OC_DelayElapsedCallback)(struct __TIM_HandleTypeDef *htim);           /*!< TIM Output Compare Delay Elapsed Callback               */
  void (* PWM_PulseFinishedCallback)(struct __TIM_HandleTypeDef *htim);         /*!< TIM PWM Pulse Finished Callback                         */
  void (* PWM_PulseFinishedHalfCpltCallback)(struct __TIM_HandleTypeDef *htim); /*!< TIM PWM Pulse Finished half complete Callback           */
  void (* ErrorCallback)(struct __TIM_HandleTypeDef *htim);                     /*!< TIM Error Callback                                      */
  void (* CommutationCallback)(struct __TIM_HandleTypeDef *htim);               /*!< TIM Commutation Callback                                */
  void (* CommutationHalfCpltCallback)(struct __TIM_HandleTypeDef *htim);       /*!< TIM Commutation half complete Callback                  */
  void (* BreakCallback)(struct __TIM_HandleTypeDef *htim);                     /*!< TIM Break Callback                                      */
#endif /* USE_HAL_TIM_REGISTER_CALLBACKS */
} TIM_HandleTypeDef;
  TIM_HandleTypeDef htimx;
  htimx.Instance = TIM6;
  htimx.Init.Prescaler = 71;
  htimx.Init.Period = 1000;

        对于NVIC而言,前文EXTI已经阐述过,不再具有结构体,而是直接进行配置:

    /* 外设中断配置 */
    HAL_NVIC_SetPriority(TIM6_IRQn, 1, 0);
    HAL_NVIC_EnableIRQ(TIM6_IRQn);

2、TIM_MasterConfigTypeDef

        TIM_MasterConfigTypeDef是相较于库函数而言额外需要配置的内容,它决定了主从输出,

typedef struct
{
  uint32_t  MasterOutputTrigger;   /*!< Trigger output (TRGO) selection
                                        This parameter can be a value of @ref TIM_Master_Mode_Selection */
  uint32_t  MasterSlaveMode;       /*!< Master/slave mode selection
                                        This parameter can be a value of @ref TIM_Master_Slave_Mode
                                        @note When the Master/slave mode is enabled, the effect of
                                        an event on the trigger input (TRGI) is delayed to allow a
                                        perfect synchronization between the current timer and its
                                        slaves (through TRGO). It is not mandatory in case of timer
                                        synchronization mode. */
} TIM_MasterConfigTypeDef;
  TIM_MasterConfigTypeDef sMasterConfig;
  sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
  sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;

3、时钟使能

__HAL_RCC_TIM6_CLK_ENABLE();
__HAL_RCC_TIM6_CLK_DISABLE();
__HAL_RCC_TIM7_CLK_ENABLE();
__HAL_RCC_TIM7_CLK_DISABLE();

4、初始化

TIM_HandleTypeDef htimx;
  HAL_NVIC_SetPriority(BASIC_TIM_IRQ, 1, 0);
  HAL_NVIC_EnableIRQ(BASIC_TIM_IRQ);

  TIM_MasterConfigTypeDef sMasterConfig;

  htimx.Instance = BASIC_TIMx;
  htimx.Init.Prescaler = BASIC_TIMx_PRESCALER;
  htimx.Init.CounterMode = TIM_COUNTERMODE_UP;
  htimx.Init.Period = BASIC_TIMx_PERIOD;
  HAL_TIM_Base_Init(&htimx);

  sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
  sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
  HAL_TIMEx_MasterConfigSynchronization(&htimx, &sMasterConfig);

5、启动定时器

HAL_StatusTypeDef HAL_TIM_Base_Start_IT(TIM_HandleTypeDef *htim)
{
  uint32_t tmpsmcr;

  /* Check the parameters */
  assert_param(IS_TIM_INSTANCE(htim->Instance));

  /* Enable the TIM Update interrupt */
  __HAL_TIM_ENABLE_IT(htim, TIM_IT_UPDATE);

  /* Enable the Peripheral, except in trigger mode where enable is automatically done with trigger */
  tmpsmcr = htim->Instance->SMCR & TIM_SMCR_SMS;
  if (!IS_TIM_SLAVEMODE_TRIGGER_ENABLED(tmpsmcr))
  {
    __HAL_TIM_ENABLE(htim);
  }

  /* Return function status */
  return HAL_OK;
}

五、HAL库API

1、初始化的封装

        可以看到,同时在h中进行了相应名字的define。

TIM_HandleTypeDef htimx;

void HAL_TIM_Base_MspInit(TIM_HandleTypeDef* htim_base)
{
   /* 基本定时器外设时钟使能 */
    BASIC_TIM_RCC_CLK_ENABLE();

    /* 外设中断配置 */
    HAL_NVIC_SetPriority(BASIC_TIM_IRQ, 1, 0);
    HAL_NVIC_EnableIRQ(BASIC_TIM_IRQ);
 
}
void BASIC_TIMx_Init(void)
{
  TIM_MasterConfigTypeDef sMasterConfig;

  htimx.Instance = BASIC_TIMx;
  htimx.Init.Prescaler = BASIC_TIMx_PRESCALER;
  htimx.Init.Period = BASIC_TIMx_PERIOD;
  HAL_TIM_Base_Init(&htimx);

  sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
  sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
  HAL_TIMEx_MasterConfigSynchronization(&htimx, &sMasterConfig);
}

2、中断服务函数

        相较于标准库而言,这个地方改动的比较大,其分为强相关与弱相关。

HAL_TIM_IRQHandler 是一个由 HAL 库提供的函数,它的作用是处理定时器的中断,并且执行一些基础操作,例如清除中断标志,调用回调函数等。后续几乎所有的定时器中断都要调用这个函数,在这个函数内会进行判断是什么定时器引起的中断并执行相应的服务函数。

void TIM6_IRQHandler(void)
{
  HAL_TIM_IRQHandler(&htimx);
}
void HAL_TIM_IRQHandler(TIM_HandleTypeDef *htim)
{
...
...

  /* TIM Update event */
  if (__HAL_TIM_GET_FLAG(htim, TIM_FLAG_UPDATE) != RESET)
  {
    if (__HAL_TIM_GET_IT_SOURCE(htim, TIM_IT_UPDATE) != RESET)
    {
      __HAL_TIM_CLEAR_IT(htim, TIM_IT_UPDATE);
#if (USE_HAL_TIM_REGISTER_CALLBACKS == 1)
      htim->PeriodElapsedCallback(htim);
#else
      HAL_TIM_PeriodElapsedCallback(htim);
#endif /* USE_HAL_TIM_REGISTER_CALLBACKS */
    }
  }
...
...
}

void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
  time++;
}

HAL_TIM_PeriodElapsedCallback用户定义的回调函数,由 HAL_TIM_IRQHandler 在处理完定时器中断后调用。此回调函数会在定时器周期到达时触发。

整体工作流程:

  1. 定时器中断触发:当定时器(如 TIM6)达到预设的周期或计数条件时,定时器产生中断,STM32的中断控制器触发 TIM6_IRQHandler 函数。

  2. TIM6_IRQHandler 中断处理:在这个中断处理函数里,HAL_TIM_IRQHandler(&htimx) 被调用,HAL 库会进一步处理定时器中断。

  3. HAL_TIM_IRQHandler 进一步处理

    • HAL_TIM_IRQHandler 会检查定时器中断的来源(比如溢出或其他事件)。
    • 它会清除中断标志位,防止中断标志位被重复触发。
    • 如果有配置回调函数(如 HAL_TIM_PeriodElapsedCallback),它会调用此回调函数。
  4. HAL_TIM_PeriodElapsedCallback 执行业务逻辑:回调函数 HAL_TIM_PeriodElapsedCallback 执行用户自定义的操作,例如更新计数器、执行任务等。在代码中,time++ 每次定时器溢出时都会执行,表示定时器事件的计数。

六、用户侧

1、初始化全流程

volatile uint32_t time = 0; // ms 计时变量 
int main(void)
{
	/* led 端口配置 */ 
  LED_GPIO_Config();
  BASIC_TIM_Init();
  while(1)
  {
    if ( time == 1000 ) /* 1000 * 1 ms = 1s 时间到 */
    {
            time = 0;
			/* LED1 取反 */      
			LED1_TOGGLE; 
    }        
  }
}
volatile uint32_t time = 0; // ms 计时变量 
int main(void)
{   
  /* 配置系统时钟为72 MHz */
  SystemClock_Config();
  /* 初始化USART1 配置模式为 115200 8-N-1 */
  DEBUG_USART_Config();
  LED_GPIO_Config();
  /* 基本定时器初始化:1ms中断一次 */
  BASIC_TIMx_Init();
   /* 在中断模式下启动定时器 */
  HAL_TIM_Base_Start_IT(&htimx);
  while(1)
  {
    if ( time == 1000 ) /* 1000 * 1 ms = 1s 时间到 */
    {
      time = 0;
			/* LED1 取反 */      
	  LED1_TOGGLE; 
    }        
  }
}

2、多定时器中断

        初始化基本上不影响,额外定义即可,主要是中断服务函数的编写。

(1)、标准库

        可以看到定义两个中断服务函数,并判断TIM6和TIM7即可。

​void  TIM6_IRQHandler(void)
{
	if ( TIM_GetITStatus( TIM6, TIM_IT_Update) != RESET ) 
	{	
		time++;
		TIM_ClearITPendingBit(TIM6, TIM_FLAG_Update);  		 
	}		 	
}
void  TIM7_IRQHandler(void)
{
	if ( TIM_GetITStatus( TIM7, TIM_IT_Update) != RESET ) 
	{	
		time++;
		TIM_ClearITPendingBit(TIM7, TIM_FLAG_Update);  		 
	}		 	
}

(2)、HAL库

        可知同一类型的定时器都是由HAL_TIM_IRQHandle统一处理的。

void TIM6_IRQHandler(void)
{
  /* 通过 TIM6 的句柄调用 HAL 库的中断处理函数 */
  HAL_TIM_IRQHandler(&htim6);
}

void TIM7_IRQHandler(void)
{
  /* 通过 TIM7 的句柄调用 HAL 库的中断处理函数 */
  HAL_TIM_IRQHandler(&htim7);
}
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
  if (htim->Instance == TIM6)
  {
    // TIM6 的定时器溢出处理
    // 例如:增加 TIM6 的计数值
    tim6_counter++;
  }
  else if (htim->Instance == TIM7)
  {
    // TIM7 的定时器溢出处理
    // 例如:增加 TIM7 的计数值
    tim7_counter++;
  }
}

        而HAL_TIM_IRQHandler 是无法区分TIM6还是TIM7导致的,因此需要在回调服务函数进行判断。

标签:12,HAL,void,htim,TIM,定时器,BASIC,库函数
From: https://blog.csdn.net/qq_39376872/article/details/144289334

相关文章

  • C#/.NET/.NET Core技术前沿周刊 | 第 16 期(2024年12.01-12.08)
    前言C#/.NET/.NETCore技术前沿周刊,你的每周技术指南针!记录、追踪C#/.NET/.NETCore领域、生态的每周最新、最实用、最有价值的技术文章、社区动态、优质项目和学习资源等。让你时刻站在技术前沿,助力技术成长与视野拓宽。欢迎投稿、推荐或自荐优质文章、项目、学习资源等......
  • 0元试用!NXP S32K312 新品系列核心板免费体验
    NXPS32K312是一颗汽车通用型MCU,可应用在汽车和工业等领域,包括车身、域控制器、以及电气化等应用。此次大大通联合世平集团推出基于NXPS32K312系列核心板免费试用活动,我们将招募10位试用者来体验!无论您是学生、电子爱好者、还是资深工程师,我们都欢迎您来参加试用,体验......
  • 11.12微课
    微课定义微课的时长特点:微课通常时长较短,一般为5-10分钟,适合快速学习和碎片化时间吸收。微课的互动性:微课鼓励学习者参与,通过在线讨论、测验等形式增强学习体验和效果。微课的教学目标:微课聚焦于单一知识点或技能点,旨在通过精炼内容实现高效教学。微课教学优势灵活性高:微课可......
  • 20241209打卡
    UML:现代软件开发中的建模利器引言在现代软件开发过程中,随着系统复杂度的增加,如何高效地描述、设计和沟通系统架构变得尤为重要。统一建模语言(UnifiedModelingLanguage,UML)作为一种图形化建模语言,提供了一种标准化的方法来可视化、构建和记录软件系统的结构和行为。本文将围绕UM......
  • Diary - 2024.12.09
    周五根本不敢查分,都是看云斗榜的时候顺带看到的分数。顺带看到了自己的排名,感觉释怀了也没救了。我的2log跑不过,我破防了我破防了我破防了我破防了我破防了。问题不大,不是最后一年,我还有得救(确信)。更重要的应该是放平心态继续前进。但是我现在被whk拿下了搞不了oi,罢了罢......
  • 2024.12.10讲座
    总体概览主题:嵌入式领域#非阻塞式编程属性:经验分享、进阶教程##之前单片机赛道的同学,学的大部分知识都是对于外设怎么操作、通信协议如何使用。这一讲的内容将让我们的主程序逻辑更加清晰、代码运行更加流畅功能:让程序更高效、清晰、严谨内容阻塞?阻塞,执行某段程序时,CPU......
  • 2024.12.9~2024.12.14
    2024.12.9早上有点小困,多睡了半个小时,上午把矩阵快速幂写完了,感觉效率有点小低然后中午去外面屯了一点食物下午开始写CDQ分治,迅速的切掉了一道题,然后下一道题就开始了漫长的调题,然后一直调调不过,情绪有点崩溃了晚上准备出去打乒乓球放松一下,结果一直赢,把一直霸台的老师都给打......
  • 上周热点回顾(12.2-12.8)
    热点随笔:· 网站刚上线,就被DDoS攻击炸了! (程序员鱼皮)· 终于解决了.net在线客服系统总是被360误报的问题(对软件进行数字签名) (升讯威在线客服系统)· 翻到了我2016年的面试经历,那是一个互联网的黄金时代。 (why技术)· 瞧瞧别人家的参数校验,那叫一个优雅! (苏三说技术)......
  • linux-12 关于shell(十一)ls
       登录系统输入用户名和密码以后,会显示给我们一个命令提示符,就意味着我们在这里就可以输入命令了,给一个命令,这个命令必须要可执行,那问题是我的命令怎么去使用,命令格式有印象吗?在命令提示符下,我们首先是命令吧?command,后面可以带什么?参数对吗?options,再后面是arguments,我们......
  • F12抓包
    一、下载postmanhttps://www.postman.com/downloads/安装好后,登录postman二、F12复制请求信息找到要抓包的请求,copyascURL,如下图:2.生产postman请求:打开postman,新建一个collection,选中collection后,import-粘贴剪切板的内容,工具会自动识别。 识别后浮层展示,可以编辑请求......