您的位置:首页 > 健康 > 养生 > STM32的通用定时器中断编程

STM32的通用定时器中断编程

2024/10/12 21:51:18 来源:https://blog.csdn.net/qq_59757948/article/details/139695497  浏览:    关键词:STM32的通用定时器中断编程

如果遇到需要单片机产生严格时序的场景(比如DAC输出特定模拟信号,GPIO口控制模拟开关),延时函数可能就无法胜任了。最近在工作时公司上级教会了我使用“门票”思维(中断标志位)编写单片机裸机程序,今天写一篇简单的验证实验报告来记录一下。这里我设计了一个简单的实验,单片机开发板上有LED0和LED1,我们让LED0亮500ms之后熄灭,随后LED1亮100ms之后熄灭,以此循环。

1.首先设定挂载定时器的时钟总线频率,我这里使用了默认的16M。

1e1616742a9d4e1da52adc6104ab1aac.png

2.使能定时器并且设置定时器的预分频系数计数周期(自动重装载计数器的值)

5b6e21c409894bcea49d77a52003c19c.png

3.使能中断函数并且调整中断优先级

63b876a9356242d780e7e272f630544b.png

4.生成代码验证实验

 (1)在定时器初始化函数中开启定时器7中断

/* TIM7 init function */
void MX_TIM7_Init(void)
{/* USER CODE BEGIN TIM7_Init 0 *//* USER CODE END TIM7_Init 0 */TIM_MasterConfigTypeDef sMasterConfig = {0};/* USER CODE BEGIN TIM7_Init 1 *//* USER CODE END TIM7_Init 1 */htim7.Instance = TIM7;htim7.Init.Prescaler = 3999;htim7.Init.CounterMode = TIM_COUNTERMODE_UP;htim7.Init.Period = 199;htim7.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;if (HAL_TIM_Base_Init(&htim7) != HAL_OK){Error_Handler();}sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;if (HAL_TIMEx_MasterConfigSynchronization(&htim7, &sMasterConfig) != HAL_OK){Error_Handler();}/* USER CODE BEGIN TIM7_Init 2 */HAL_TIM_Base_Start_IT(&htim7);//开启定时器7中断/* USER CODE END TIM7_Init 2 */}

(2)在stm32f4xx_it.c中找到这个中断处理函数

d1eadf0db63c4f6991a6c9fecf140b11.png

(3)跳转定义查看函数底层逻辑,找到中断回调函数

1067e76ec54e49e6892aa54a25f36b5b.png

(4)跳转定义,发现属于虚函数可以进行重写

__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*/
}

(5)在stm32f4xx_it.c的代码沙盒段内进行虚函数重写,并且定义所需全局变量

/* USER CODE BEGIN EV */
uint8_t g_timTick=0;               //定时器中断节拍
uint8_t g_timSwitch=0;             //定时器中断“门票”
/* USER CODE END EV */
/* USER CODE BEGIN 1 */
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{if(htim->Instance == TIM7){                              //涉及到多个定时器中断的时候检查就很有必要了const uint8_t increment=1;                         //计数步长为常量1g_timTick = (g_timTick + increment) % 5;               //5:1的GPIO口时序}g_timSwitch++;                                         //回调函数发放门票
}
/* USER CODE END 1 */

(6)在stm32f4xx_it.中声明全局变量

/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */
#include "main.h"            //防止uint8_t无法被编译器识别
/* USER CODE END Includes *//* Exported types ------------------------------------------------------------*/
/* USER CODE BEGIN ET *//* USER CODE END ET *//* Exported constants --------------------------------------------------------*/
/* USER CODE BEGIN EC */
extern uint8_t g_timTick;          
extern uint8_t g_timSwitch;
/* USER CODE END EC */

(7)主函数编写功能函数

			if(g_timSwitch){g_timSwitch--;                             //进入主循环则收回门票,防止CPU重复判断if(g_timTick==0){HAL_GPIO_WritePin(GPIOB,GPIO_PIN_0,GPIO_PIN_RESET);	HAL_GPIO_WritePin(GPIOB,GPIO_PIN_1,GPIO_PIN_SET);	}else{HAL_GPIO_WritePin(GPIOB,GPIO_PIN_0,GPIO_PIN_SET);	HAL_GPIO_WritePin(GPIOB,GPIO_PIN_1,GPIO_PIN_RESET);}	}

(8)下载验证

9c315ac46ed8438eaa04645774a4329c.jpeg

观察到单片机开发板的LED灯进行频闪,试验成功。不过从波形上来看会有较大的误差,因为只有500ms的时序是用中断严格实现的。想要精准控制原理就是上述的那些,感兴趣的同学可以自己试一试。

补档:高级定时器的内部时钟计时操作

从模式选择触发模式(Reset Mode),触发源选择内部触发(ITRx即Internal Trigger Input),时钟源选择内部时钟,其他和通用定时器基本相同。最后请大家注意每一个操作的timer挂载的始终总线(APB1之类的)。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com