stm32实现定时器中断

13.2 硬件设计
本实验用到的硬件资源有:
1) 指示灯 DS0 和 DS1
2) 定时器 TIM3
本章将通过 TIM3 的中断来控制 DS1 的亮灭,DS1 是直接连接到 PE5 上的,这个前面已经
有介绍了。而 TIM3 属于 STM32 的内部资源,只需要软件设置即可正常工作。
13.3 软件设计
软件设计我们直接打开我们光盘实验 8 定时器中断实验即可。我们可以看到我们的工程中
的 HARDWARE 下面比以前多了一个 time.c 文件(包括头文件 time.h),这两个文件是我们自己
编写。同时还引入了定时器相关的固件库函数文件 stm32f10x_tim.c 和头文件 stm32f10x_tim.h。
下面我们来看看我们的 time.c 文件。
time.c 文件代码:
#include “timer.h”
#include “led.h”
//通用定时器 3 中断初始化
//这里时钟选择为 APB1 的 2 倍,而 APB1 为 36M
//arr:自动重装值。
//psc:时钟预分频数
//这里使用的是定时器 3!
void TIM3_Int_Init(u16 arr,u16 psc)
{
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
NVIC_InitTypeDef NVIC_InitStructure;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //①时钟 TIM3 使能
//定时器 TIM3 初始化
TIM_TimeBaseStructure.TIM_Period = arr; //设置自动重装载寄存器周期的值
TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置时钟频率除数的预分频值
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //设置时钟分割
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM 向上计数
TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //②初始化 TIM3
TIM_ITConfig(TIM3,TIM_IT_Update,ENABLE ); //③允许更新中断
//中断优先级 NVIC 设置
NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn; //TIM3 中断
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //先占优先级 0 级
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //从优先级 3 级
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ 通道被使能
NVIC_Init(&NVIC_InitStructure); //④初始化 NVIC 寄存器
TIM_Cmd(TIM3, ENABLE); //⑤使能 TIM3
}
//定时器 3 中断服务程序⑥
void TIM3_IRQHandler(void) //TIM3 中断
{
if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET) //检查 TIM3 更新中断发生与否
{
TIM_ClearITPendingBit(TIM3, TIM_IT_Update ); //清除 TIM3 更新中断标志
LED1=!LED1;
}
}
该文件下包含一个中断服务函数和一个定时器 3 中断初始化函数,中断服务函数比较简单,
在每次中断后,判断 TIM3 的中断类型,如果中断类型正确(溢出中断),则执行 LED1(DS1)
的取反。
TIM3_Int_Init()函数就是执行我们上面 13.1 节介绍的那 6 个步骤,我们分别用标号①~⑥来
标注,该函数的 2 个参数用来设置 TIM3 的溢出时间。在前面时钟系统部分我们讲解过,系统
初始化的时候在默认的系统初始化函数 SystemInit函数里面已经初始化 APB1 的时钟为2 分频,
所以 APB1 的时钟为 36M,而从 STM32 的内部时钟树图得知:当 APB1 的时钟分频数为 1 的
时候,TIM2~7 的时钟为 APB1 的时钟,而如果 APB1 的时钟分频数不为 1,那么 TIM2~7 的时
钟频率将为 APB1 时钟的两倍。因此,TIM3 的时钟为 72M,再根据我们设计的 arr 和 psc 的值,
就可以计算中断时间了。计算公式如下:
Tout= ((arr+1)(psc+1))/Tclk;
其中:
Tclk:TIM3 的输入时钟频率(单位为 Mhz)。
Tout:TIM3 溢出时间(单位为 us)。
timer.h 文件的代码非常简单,一些函数申明,这里就不讲解。
最后,我们在主程序里面输入如下代码:
int main(void)
{
delay_init(); //延时函数初始化
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置 NVIC 中断分组 2
uart_init(115200); //串口初始化波特率为 115200
LED_Init(); //LED 端口初始化
TIM3_Int_Init(4999,7199); //10Khz 的计数频率,计数到 5000 为 500ms
while(1)
{
LED0=!LED0;
delay_ms(200);
}
}
这里的代码和之前大同小异,此段代码对 TIM3 进行初始化之后,进入死循环等待 TIM3
溢出中断,当 TIM3_CNT 的值等于 TIM3_ARR 的值的时候,就会产生 TIM3 的更新中断,然
后在中断里面取反 LED1,TIM3_CNT 再从 0 开始计数。根据上面的公式,我们可以算出中断
溢出时间为 500ms,即 Tout= ((4999+1)
( 7199+1))/72=500000us=500ms。

你可能感兴趣的:(嵌入式,stm32)