STM32使用定时器更新中断使LED亮灭交替1S闪烁

在向上计数模式中,计数器从0计数到自动加载值(TIMx_ARR(自动装载寄存器)计数器的内容),然后重新从0开始计数并且产生一个计数器溢出事件。每次计数器溢出时可以产生更新事件,当使能了更新中断以后,计数器溢出时则产生更新中断。

上图源自一位b站up主的定时器中断图及参考手册里的图。

可见,计数器时钟可由下列时钟源提供:

● 内部时钟(CK_INT)

● 外部时钟模式1:外部输入脚(TIx)(图中没有截取到)

● 外部时钟模式2:外部触发输入(ETR)

● 内部触发输入(ITRx)

定时器2内部时钟且使其更新中断使LED亮灭交替1S闪烁(通俗说就是定时器定时1S)为例,选用RCC内部时钟模式,则不需要配置GPIO(外部触发输入模式等则涉及GPIO的端口复用)。在使用内部时钟时首先进行定时器2的使能, 配置时基单元。配置完定时器2后,开启定时器2的更新中断,配置其优先级。在使能计数器后且当计数器溢出时,就会产生更新中断,并且计数器重新开始计数。

定时1s且系统频率为72MHz时,重装值和预分频值的关系:(重装值+1)*(预分频值+1)=72*10^6

频率(1s的震动次数)和时间的关系是:nMHz则震动n次(Hz)历经1us(1MHz=10^6Hz)。

Timer.c

#include "timer.h"

void Timer_Init(u16 arr,u16 psc)  //RCC内部时钟模式
{
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);
	
	TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
	TIM_TimeBaseInitStructure.TIM_ClockDivision = 0;
	TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
	TIM_TimeBaseInitStructure.TIM_Period = arr;  //重装
	TIM_TimeBaseInitStructure.TIM_Prescaler = psc;  //预分频
	TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStructure);
	
	NVIC_InitTypeDef NVIC_InitStructure;
	NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;
	NVIC_Init(&NVIC_InitStructure);
	
	TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE);  //配置定时器更新中断
	
	TIM_Cmd(TIM2,ENABLE);
}

Timer.h

#ifndef __TIMER_H
#define __TIMER_H

#include "stm32f10x.h"

void Timer_Init(u16 arr,u16 psc);

#endif

led.c

#include "led.h"

/*我这里板载灯是PC13*/
/*开启时钟,初始化端口引脚,再进行相关操作*/

void LED_Init(void){
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);
	
	GPIO_InitTypeDef GPIO_InitIntructrue;
	GPIO_InitIntructrue.GPIO_Mode = GPIO_Mode_Out_PP;
	GPIO_InitIntructrue.GPIO_Pin = GPIO_Pin_13;
	GPIO_InitIntructrue.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOC,&GPIO_InitIntructrue);
}

void LED_ON(void){
	GPIO_SetBits(GPIOC,GPIO_Pin_13);
}

void LED_OFF(void){
	GPIO_ResetBits(GPIOC,GPIO_Pin_13);
}

led.h

#ifndef __LED_H
#define __LED_H

#include "stm32f10x.h"

void LED_Init(void);

void LED_ON(void);

void LED_OFF(void);

#endif

main.c

#include "stm32f10x.h"
#include "led.h"
#include "timer.h"

u16 i=0;

int main(void)
{
	LED_Init();
	Timer_Init(7199,9999);
	while(1);
}

void TIM2_IRQHandler()  //TIM2的中断服务函数名不能有误
{
	switch(i)
	{
		case 0:LED_ON();i++;break;
		case 1:LED_OFF();i=0;break;
	}
	TIM_ClearFlag(TIM2,TIM_FLAG_Update); //清除更新中断标志位
}

效果是:LED交替亮灭1S。 

  • 4
    点赞
  • 41
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
您可以使用STM32定时器来控制两个LED亮灭。下面是一个示例代码,使用定时器2来控制两个LED亮灭LED1连接到GPIOA的Pin0,LED2连接到GPIOA的Pin1。 首先,您需要初始化定时器2以及相关的GPIO引脚: ```c #include "stm32f4xx.h" void LED_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; // 打开GPIOA时钟 RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); // 配置GPIOA的Pin0和Pin1为推挽输出模式 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_Init(GPIOA, &GPIO_InitStructure); } void TIM2_Init(void) { TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; // 打开TIM2时钟 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); // 填充TIM2初始化结构体 TIM_TimeBaseStructure.TIM_Period = 100 - 1; // 定时器周期为100个时钟周期 TIM_TimeBaseStructure.TIM_Prescaler = (84000000 / 1000000) - 1; // 定时器时钟预分频为84MHz/1MHz-1 TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; // 初始化TIM2 TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); // 使能TIM2更新中断 TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE); // 启动TIM2计数器 TIM_Cmd(TIM2, ENABLE); } int main(void) { // 初始化LED定时器 LED_Init(); TIM2_Init(); while (1) { // 等待定时器溢出中断发生 while (TIM_GetFlagStatus(TIM2, TIM_FLAG_Update) == RESET); // 清除定时器溢出中断标志位 TIM_ClearFlag(TIM2, TIM_FLAG_Update); // 切换LED状态 GPIO_ToggleBits(GPIOA, GPIO_Pin_0 | GPIO_Pin_1); } } ``` 在上述代码中,定时器2的周期为100个时钟周期,预分频为84MHz/1MHz-1,即每1微产生一次定时器溢出中断。在主循环中,等待定时器溢出中断发生后,切换LED的状态。 请注意,您需要根据您的具体硬件连接和需求进行适当的修改。此外,您还需要根据您使用的具体STM32系列芯片和开发环境进行相关的配置。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值