STC32物联网项目-基本定时器

基本定时器

本次实验使用STM32F103ZET6的基本定时器6作定时,在中断中每隔1秒翻转LED电平状态

1.CubeMX初始化定时器

先开启定时器6

在这里插入图片描述

再对定时器6的参数进行配置,将定时器6定时时间配置为5ms,在中断中再累计到1秒钟,实现LED翻转功能

Prescaler配置为7199,因为溢出时间 = ((psc+1)/fCK_PSC) * (arr+1),fCK_PSC是72MHz,7199+1 = 7200,7200/72MHz = 7200/72000000Hz = 0.0001s = 0.1ms;Counter Period(装载值)配置为49,根据公式,(49+1)*0.1ms = 5ms

Counter Mode配置为Up,向上计数模式

在这里插入图片描述

开启NVIC,使能定时器6全局中断,抢占优先级和响应优先级都设为1,因为只有一个中断,所以这里设什么值没太大关系

在这里插入图片描述

最后生成代码

2.keil代码编写

在CubeMX生成的工程中,多了tim.x和tim.h这两个文件,tim.c中就有定时器6的初始化函数,同时在main.c文件中也被自动调用,定时器6已经被初始化,但还没开启,要自己开启定时器

TIM_HandleTypeDef htim6:就是定时器6的初始化句柄,类似结构体的用法

/* Includes ------------------------------------------------------------------*/
#include "tim.h"

/* USER CODE BEGIN 0 */

/* USER CODE END 0 */

TIM_HandleTypeDef htim6;

/* TIM6 init function */
void MX_TIM6_Init(void)
{
  TIM_MasterConfigTypeDef sMasterConfig = {0};

  htim6.Instance = TIM6;
  htim6.Init.Prescaler = 7199;
  htim6.Init.CounterMode = TIM_COUNTERMODE_UP;
  htim6.Init.Period = 49;
  htim6.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_ENABLE;
  if (HAL_TIM_Base_Init(&htim6) != HAL_OK)
  {
    Error_Handler();
  }
  sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
  sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
  if (HAL_TIMEx_MasterConfigSynchronization(&htim6, &sMasterConfig) != HAL_OK)
  {
    Error_Handler();
  }
}

增加Timer6.c和Timer6.h文件

Timer6.h

定义定时时间的枚举类型,比如定时10ms是5ms的两倍,所以枚举值是2

#ifndef __TIMER6_H_
#define __TIMER6_H_

#include "MyApplication.h"

//定义定时时间枚举类型
typedef enum
{
    TIMER_10ms     = (uint16_t)2,
    TIMER_50ms     = (uint16_t)10,
    TIMER_100ms    = (uint16_t)20,
    TIMER_200ms    = (uint16_t)40,
    TIMER_500ms    = (uint16_t)100,
    TIMER_1s       = (uint16_t)200,
    TIMER_2s       = (uint16_t)400,
    TIMER_3s       = (uint16_t)600,
    TIMER_5s       = (uint16_t)1000,
    TIMER_10s      = (uint16_t)2000,
    TIMER_3min     = (uint16_t)3600,
}TIMER_Value_t;

//定义结构体类型
typedef struct
{
    uint16_t volatile usMCU_Run_Timer;     //系统运行定时器
    void (*Timer6_Start_IT)(void);
}Timer6_t;

/* extern variables-----------------------------------------------------------*/
extern Timer6_t Timer6;
/* extern function prototypes-------------------------------------------------*/ 

#endif
/********************************************************
  End Of File
********************************************************/

Timer6.c

源文件中对系统的定时器6以中断模式启动函数进行了封装

/* Includes ------------------------------------------------------------------*/
#include <MyApplication.h>

/* Private define-------------------------------------------------------------*/

/* Private variables----------------------------------------------------------*/
static void Timer6_Start_IT(void);    //定时器6以中断模式启动
/* Public variables-----------------------------------------------------------*/
Timer6_t Timer6 = 
{
  0,
  Timer6_Start_IT
};
/* Private function prototypes------------------------------------------------*/

/*
* @name   Timer6_Start_IT
* @brief  定时器6以中断模式启动
* @param  None
* @retval None   
*/
static void Timer6_Start_IT()
{
    HAL_TIM_Base_Start_IT(&htim6);    //调用系统的定时器6以中断模式启动
}

/********************************************************
  End Of File
********************************************************/

MyInit.c

自己定义的初始化函数中调用定时器6的启动函数

/*
* @name   Peripheral_Set
* @brief  外设设置
* @param  None
* @retval None   
*/
static void Peripheral_Set()
{
  Timer6.Timer6_Start_IT();   //启动定时器6
}

Peripheral_Set函数在main.c中被调用,进入while循环前已将定时器6初始化,并启动

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_TIM6_Init();			//初始化定时器6,系统自动生成
  /* USER CODE BEGIN 2 */
  MyInit.Peripheral_Set();	//自己的初始化函数,调用定时器6启动函数
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
 while (1)
  {
    System.Run();
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */

CallBack.c

在回调文件中重写HAL_TIM_PeriodElapsedCallback函数,该函数是定时器的中断回调函数,重写后,调用的就是下面这个让LED灯电平翻转的函数,并不是系统原本的空函数

Timer6.usMCU_Run_Timer在结构体中被初始化为0,因为1s等于200*5ms,当Timer6.usMCU_Run_Timer大于200时,就完成了定时1秒,然后就让LED灯电平翻转

因为该函数是中断回调函数,所以不需要在主函数中调用,定时器溢出后便会自动到该函数处执行

/*
* @name   HAL_TIM_PeriodElapsedCallback
* @brief  定时器中断回调函数
* @param  *htim:处理定时器的结构体指针
* @retval None   
*/
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
  if(htim->Instance == htim6.Instance)
  {
    //定时器6每隔5ms进入一次中断,usMCU_Run_Timer就加1,当计时时间达到1s时,翻转LED灯的状态
    if(++Timer6.usMCU_Run_Timer >= TIMER_1s)
    {
      Timer6.usMCU_Run_Timer = 0;
      LED.LED_Fun(LED1,LED_Flip);
      LED.LED_Fun(LED2,LED_Flip);
      LED.LED_Fun(LED3,LED_Flip);
    }
  }
}

HAL_TIM_PeriodElapsedCallback函数体在stm32f1xx.hal_tim.c文件中被定义,_weak修饰了函数,说明这个函数是个弱函数,当没有被重构时,如果函数被调用,调用的是系统的这个没有具体功能的函数体,因为生成的代码并不知道开发者要干嘛,所以是个空函数;当该函数被重构后,调用的就是开发者重构后的函数

/**
  * @brief  Period elapsed callback in non-blocking mode
  * @param  htim TIM handle
  * @retval None
  */
__weak void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
  /* Prevent unused argument(s) compilation warning */
  UNUSED(htim);  /* NOTE : This function should not be modified, when the callback is needed,
            the HAL_TIM_PeriodElapsedCallback could be implemented in the user file
   */
}
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值