当前位置: 首页 > article >正文

STM32物联网实战开发(4)——基本定时器

        我使用的是正点原子的阿波罗F429开发板,他有14个定时器,本次实验使用STM32F429的基本定时器6作定时,在中断中每隔1秒翻转LED电平状态。

1.CubeMX初始化定时器

先开启定时器6

         再对定时器6的参数进行配置,将定时器6定时时间配置为5ms,在中断中再累计到1秒钟(200次),实现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,因为只有一个中断,所以这里设什么值没太大关系。(stm32单片机将抢占优先级和响应优先级,一共赋予四位,我们可以给抢占优先级和响应优先级各占两位,那么他们的优先级都是(0~3),其中优先级数字越小,优先级越大

最后生成代码

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();//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灯电平翻转

(其中usMCU_Run_Timer被volatile修饰代表他是一个随时变化的变量

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

/*
* @name   HAL_TIM_PeriodElapsedCallback
* @brief  定时器中断回调函数
* @param  *htim:处理定时器的结构体指针
* @retval None   
*/
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
  if(htim->Instance == htim6.Instance)//判断是否为定时器6
  {
    //定时器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函数体在stm32f4xx.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
   */
}

总结

        综上所述,我们明白啦,利用了HAL库,已经帮我们生成了大部分的代码,我们只需要定义一个结构体(一个变量,一个函数指针(用来打开定时器))。后期只需要调用结构体就可实现定时的功能,最后我们还需要写一个回调函数来实现我们想要实现的功能,因为hal中已经构建了一个弱函数,我们只需要重新构建一个同名的函数即可(定时器中断函数大功告成)。


http://www.kler.cn/a/16485.html

相关文章:

  • WebGIS三维地图框架--Cesium
  • 车-路-站-网”信息耦合的汽车有序充电
  • C#发票识别、发票查验接口集成、电子发票(航空运输电子行程单)
  • INQUIRE:一个包含五百万张自然世界图像,涵盖10,000个不同物种的专为专家级文本到图像检索任务设计的新型基准数据集。
  • MacOS 本地生成SSH key并关联Github
  • AcWing 302 任务安排 斜率优化的dp
  • 32k*16 薪,3年自动化测试历经3轮面试成功拿下华为Offer....
  • 【Java笔试强训 7】
  • GEE:MODIS计算遥感指数(NDVI、BSI、NDSI、EVI、LSWI、SIPI、EBI等)
  • 吉布斯采样方法
  • 设计模式-单例模式
  • 一文搞懂PMP挣值管理那些让你头疼的公式
  • mockjs学习笔记
  • maven中的 type ,scope的作用
  • 2335. 装满杯子需要的最短总时长
  • 83. map函数()-通过函数实现对可迭代对象的操作(适合零基础)
  • 「SQL面试题库」 No_55 销售分析 I
  • ramfs, rootfsinitramfs
  • HTML(四) -- 多媒体设计
  • CCD视觉检测设备如何选择光源
  • 【面试长文】HashMap的数据结构和底层原理以及在JDK1.6、1.7和JDK8中的演变差异
  • Blender启动场景的修改
  • 资讯汇总230503
  • 哈希表企业应用-DNA的字符串检测
  • CKA/CKS/CKAD认证考试攻略
  • 【五一创作】( 字符串) 409. 最长回文串 ——【Leetcode每日一题】