【STM32】HAL库 CubeMX 教程 --- 通用定时器 TIM2 定时


实验目标:

通过CUbeMX+HAL,配置TIM2,1s中断一次,闪烁LED。


一、常用型号的TIM时钟频率

1. STM32F103系列:

  • 所有 TIM 的时钟频率都是72MHz;
  • F103C8不带基本定时器,F103RC及以上才带基本定时器。

2、STM32F407系列:

  • TIM1、8、9、10、11的时钟频率是168MHz;
  • 其它TIM的时钟频率是84MHz。

二、CubeMX配置

这里使用现有的CubeMX工程,略过工程配置部分。

如果学习如何新建CubeMX工程, 可以参考下面文章:

STM32CubeMX新建一个工程_STM32F103

STM32CubeMX新建一个工程_STM32F407

使用CubeMX对定时器进行定时配置,过程相当简单便捷。

我们只需配置几个重要参数:时钟源、PSC、ARR、中断。

1、主要参数配置

主要参数讲解:

Prescaler(预分频系数):即PSC寄存器值。简单地理解:内部时钟分频后再给CNT计数器使用,用于控制每一计数脉冲的时长。按上文内容得知,STM32F407的TIM2时钟是84M,在设置预分频为84后,计数脉冲频率:84MHz÷84=1MHz,即每秒产生1百万个计数信号,每个计数脉冲的时长:1秒/1M=1us。

Counter Mode (计数方式): 向上计数

Counter Period (计数周期): 即ARR寄存器值,多少个脉冲作为1个波形周期。

auto-reload preload(自动重载值的预装载): 当改变周期值ARR时,是否等到下一个更新事件再写入数值,使得数值的更改不影响执行中的波形。

2、打开中断

通用定时器,在只用定时功能的情况下,只有一种中断,如下图, 打勾即可。

当计数器(CNT)的值,达到周期值(ARR)时,就会触发中断。

注意:打勾中断后,生成的代码,只是帮我们添加了中断的配置; 而中断的开启,需要使用代码"手动"开启。


三、编写代码

代码共为3部分:初始化配置、开启TIM、中断回调函数。

1、初始化配置

CubeMX根据我们的配置,已生成好了初始化代码,我们不用管它。

如果想查看它的初始化实现过程,可以双击 "tim.c"。

2、开启TIM、中断

谨记一个,当我们使用CubeMX配置外设功能时,它只是根据参数的配置生成初始化代码(即上面的第1步),而不会主动开启外设功能,特别是中断的开启,如TIM、ADC、UART等。

所以,我们需要在main.c中,"手动"开启TIM2。

如下图,在while前,添加:

HAL_TIM_Base_Start_IT(&htim2);

调用HAL_TIM_Base_Start_IT(&htim2)后,TIM2就会开始工作。

下面是它的工作原理,硬件自动运行,不用干预,我们只需大概地知道工作过程、原理:

① TIM会自动检测脉冲信号,每来一个脉冲,计数器CNT就会硬件自动加1。

② 当寄存器CNT的计数值==设定的自动重装载值ARR,就会产生溢出中断。

③ 发生溢出中断时,CNT值被硬件置0,开始新一轮计数; 硬件自动找寻中断服务函数入口。

使用CubeMX生成的HAL代码,我们不需要像使用标准库时那样自己编写中断函数,CubeMX已帮我们整理好中断函数的跳转,只需要重写中断回调函数。

3、重写TIM2的中断回调函数

如上所述,通用定时器,只使用定时功能时,只有一种中断(CNT向上计数,达到ARR值)。

所以也只有一个中断回调函数: HAL_TIM_PeriodElapsedCallback();

当CNT值==周期值(ARR),即1周期结束,硬件自动调用中断服务函数,继而调用回调函数。

我们可以在工程中任意一个c文件中编写它。

习惯上,会在main.c的尾部编写这个函数。

如下图,我们在main.c的 注释行 /* USER CODE BEGIN 4 */ 下面编写它。

如果工程中已有这个函数,我们只需在函数内增加TIM2的判断部分;

如果工程中没有这个函数,增加即可。

注意:所有TIM的计数器溢出中断(即1周期结束), 都会统一调用它。

在这个回调函数中,我们执行的动作是:每1000ms, 反转PB2电平(LED)。

如果你的开发板上,PB2引脚接的是 LED, 而且也初始化成输出模式,它就会每1秒闪烁1次。

4、思路优化:

上述操作,将会每1s进入1000次中断,当发现变量cnt累加到1000后,才反转LED。

频繁地出入中断,大大地浪费着芯片运行资源。本篇如此设计,仅为了直观地展示和理解。

我们在设计时,应该尽量地节省运行资源。

上面的定时操作,有多种优化的方法,这里使用下面的操作:

psc改为8400(计数脉冲0.0001s),ARR改为10000(1万个脉冲组成1周期);

然后,修改回调函数,取消计数判断那部分。

这样修改后,将会每1s进入1次中断,调用回调函数1次,能有效地节省运行资源。

相关推荐
时光飞逝的日子5 小时前
stm32进入睡眠模式的几个注意点
stm32·单片机·嵌入式硬件
weixin_4629019710 小时前
STM32F103C8T6裸机多任务编程的问题
stm32·单片机·嵌入式硬件
Jumbuck_1012 小时前
基于OpenMV+STM32+OLED与YOLOv11+PaddleOCR的嵌入式车牌识别系统开发笔记
笔记·stm32·嵌入式硬件
网易独家音乐人Mike Zhou16 小时前
【Linux应用】交叉编译环境配置,以及最简单粗暴的环境移植(直接从目标板上复制)
linux·stm32·mcu·物联网·嵌入式·iot
憧憬一下18 小时前
stm32之EXIT外部中断详解
stm32·单片机·嵌入式·中断
The Mr.Nobody1 天前
STM32MPU开发之旅:从零开始构建嵌入式Linux镜像
linux·stm32·嵌入式硬件
阿川!1 天前
嵌入式软件--stm32 DAY 3
stm32·单片机·嵌入式硬件
#金毛1 天前
STM32的定时器输出PWM时,死区时间(DTR)如何计算
stm32·单片机·嵌入式硬件
无脑学c++1 天前
STM32串口重定向:MDK与GCC重定向需重写的不同函数
stm32·单片机·物联网
Invinciblenuonuo1 天前
STM32八股【6】-----CortexM3的双堆栈(MSP、PSP)设计
stm32·单片机·嵌入式硬件