STM32 最简单的边沿触发脉冲计数程序

本文详细介绍了STM32微控制器中定时器的配置与使用,包括TIM3和TIM5的初始化过程,以及如何通过中断进行精确的时间测量和控制。探讨了定时器的预分频、自动重装载寄存器、中断优先级配置等关键概念。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

 

#include "led.h"
#include "delay.h"
#include "key.h"
#include "sys.h"
#include "usart.h"
#include "timer.h"
#include "lcd.h"


extern u8 sign;
extern u16 TIM5CH1_CAPTURE_COUNT;
 int main(void)
 {		 
	delay_init();	    	 //延时函数初始化	  
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); 	 //设置NVIC中断分组2:2位抢占优先级,2位响应优先级
	uart_init(115200);	 //串口初始化为115200
 	LED_Init();			     //LED端口初始化
	LCD_Init();			   	    //初始化LCD 	
	KEY_Init();         	//初始化与按键连接的硬件接口
  TIM3_Int_Init(4999,7199);	 //72000000/(4999+1)(7199+1)=2Hz----500ms
  TIM5_Int_Init(4999,7199);	 //

	 
	 
	 while(1)
	 {
		 TIM_SetCompare2(TIM3,2500);		   
	     LCD_ShowNum(10,80,TIM5CH1_CAPTURE_COUNT,24,16);
		 LCD_ShowNum(10,120,sign,24,16);
	 }

 }

#include "timer.h"
#include "led.h"
#include "usart.h"

void TIM3_Int_Init(u16 arr,u16 psc)
{
    TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
	NVIC_InitTypeDef NVIC_InitStructure;

	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //时钟使能

	TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值	 计数到5000为500ms
	TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为TIMx时钟频率除数的预分频值  10Khz的计数频率  
	TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim
	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式
	TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位
 
	TIM_ITConfig(  //使能或者失能指定的TIM中断
		TIM3, //TIM2
		TIM_IT_Update  |  //TIM 中断源
		TIM_IT_Trigger,   //TIM 触发中断源 
		ENABLE  //使能
		);
	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_InitStruct中指定的参数初始化外设NVIC寄存器

	TIM_Cmd(TIM3, ENABLE);  //使能TIMx外设
							 
}




void TIM5_Int_Init(u16 arr,u16 psc)
{	  
	TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
  	TIM_ICInitTypeDef TIM5_ICInitStructure;
	NVIC_InitTypeDef NVIC_InitStructure;
	

	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM5, ENABLE); //tim5时钟使能
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //pa端口使能
	
	

	TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值	
	TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为TIMx时钟频率除数的预分频值
	TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //设置时钟分割:TDTS = Tck_tim
	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式
	TIM_TimeBaseInit(TIM5, &TIM_TimeBaseStructure); //根据指定的参数初始化TIMx的时间基数单位
	

	TIM5_ICInitStructure.TIM_Channel=TIM_Channel_1;//通道1----CH1----pa0
	TIM5_ICInitStructure.TIM_ICFilter=0x00;//不滤波
	TIM5_ICInitStructure.TIM_ICPolarity=TIM_ICPolarity_Rising;//上升沿检测
	TIM5_ICInitStructure.TIM_ICPrescaler=TIM_ICPSC_DIV1;
	TIM_ICInit(TIM5,&TIM5_ICInitStructure);
	
	TIM_ITConfig(TIM5,TIM_IT_CC1,ENABLE);//允许更新中断和捕获中断
	

	//中断分组初始化
	NVIC_InitStructure.NVIC_IRQChannel = TIM5_IRQn;  //TIM5中断
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;  //先占优先级2级
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;  //从优先级0级
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能
	NVIC_Init(&NVIC_InitStructure);  //根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器 
	
	TIM_Cmd(TIM5, ENABLE);  //使能TIMx					 
}

u16 TIM5CH1_CAPTURE_COUNT=0;//输入捕捉数

//定时器5中断服务程序
void TIM5_IRQHandler(void)   //TIM3中断
{
	if (TIM_GetITStatus(TIM5, TIM_IT_CC1) != RESET)  //检查TIM5更新中断发生与否
		{
	    TIM5CH1_CAPTURE_COUNT++;
			LED1=!LED1;
		}
		 TIM_ClearITPendingBit(TIM5, TIM_IT_CC1|TIM_IT_Update); //清除中断标志位
}


u8 sign=0;
//定时器3中断服务程序
void TIM3_IRQHandler(void)   //TIM3中断
{
	if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET)  //检查TIM3更新中断发生与否
		{
			if(sign++==10)
			{
					TIM5CH1_CAPTURE_COUNT=0;
                    sign=0;				
			}
		}
			TIM_ClearITPendingBit(TIM3, TIM_IT_Update  );  //清除TIMx的中断待处理位:TIM 中断源 
}

 

### 回答1: 您可以使用STM32F030C8T6的定时器来测量脉冲频率。您需要将定时器设置为计数模式和输入捕获模式,并使用外部引脚来捕获脉冲上升沿和下降沿。您可以根据捕获到的两个沿之间的时间差来计算频率。下面是一个简单的示例代码: ``` #include "stm32f0xx.h" int main(void) { // Enable clock for GPIO port and timer RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA, ENABLE); RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); // Configure GPIO pin as input GPIO_InitTypeDef gpio; gpio.GPIO_Pin = GPIO_Pin_0; gpio.GPIO_Mode = GPIO_Mode_AF; gpio.GPIO_PuPd = GPIO_PuPd_UP; GPIO_Init(GPIOA, &gpio); // Configure timer for input capture TIM_ICInitTypeDef ic; ic.TIM_Channel = TIM_Channel_1; ic.TIM_ICPolarity = TIM_ICPolarity_Rising; ic.TIM_ICSelection = TIM_ICSelection_DirectTI; ic.TIM_ICPrescaler = TIM_ICPSC_DIV1; ic.TIM_ICFilter = 0; TIM_ICInit(TIM2, &ic); // Start the timer TIM_Cmd(TIM2, ENABLE); while (1) { // Wait for input capture while (TIM_GetFlagStatus(TIM2, TIM_FLAG_CC1) == RESET) {} TIM_ClearFlag(TIM2, TIM_FLAG_CC1); // Get the time difference between rising and falling edges uint16_t rise_time = TIM_GetCapture1(TIM2); while (TIM_GetFlagStatus(TIM2, TIM_FLAG_CC1) == RESET) {} uint16_t fall_time = TIM_GetCapture1(TIM2); uint16_t period = fall_time - rise_time; // Calculate frequency float frequency = 1.0 / ((float)period / (float)SystemCoreClock); } } ``` 这段代码使用TIM2的通道1来捕获GPIOA的引脚0上升沿和下降沿的时间戳,并计算出频率。需要注意的是,您需要在“stm32f0xx.h”中包含适当的头文件,并确保正确配置系统时钟。 ### 回答2: 要使用stm32f030c8t6来测量脉冲频率,我们可以利用该微控制器的输入捕获功能和计数器来实现。 首先,我们需要将外部脉冲源连接到stm32f030c8t6的一个可用输入引脚上,例如GPIO引脚。然后,我们需要配置这个引脚作为输入模式。 接下来,我们可以使用stm32f030c8t6的输入捕获功能来捕获脉冲上升沿和下降沿。配置输入捕获时,我们可以选择计数器的工作模式和时钟源。常见的选择包括计数器在上升沿捕获、下降沿捕获或同时捕获上升沿和下降沿。 一旦配置好输入捕获,我们可以开始计算脉冲频率。输入捕获会记录每次脉冲上升沿和下降沿的计数器值。根据这些值的差异,我们可以计算出每个脉冲的时间间隔。通过将时间间隔倒数,我们就可以得到脉冲频率。 最后,我们可以将脉冲频率输出到一个或多个显示设备上,例如LCD显示屏或串口调试窗口。我们还可以根据需要加入其他功能,比如设置频率上下限报警或记录历史频率数据。 需要注意的是,以上是一个大致的步骤,具体的实现还取决于具体的应用场景和使用的开发工具。因此,在实际操作中,我们需要查阅相应的参考手册和文档,以确保正确地配置和使用stm32f030c8t6的输入捕获功能。 ### 回答3: 要测量脉冲频率,我们可以使用STM32F030C8T6单片机的定时器功能。首先,我们需要配置定时器的输入捕获功能,以便捕获脉冲信号的边沿。 在配置定时器之前,需要设置GPIO引脚的模式为输入模式,并启用输入捕获功能。接下来,配置定时器的模式为输入捕获模式,并设置触发边沿上升沿或下降沿。然后,启动定时器开始计数。 当脉冲信号到达时,定时器会自动捕获当前的计数值。我们可以使用这个值来计算脉冲的周期或频率。 为了获得更准确的结果,我们可以通过使用中断的方式来处理捕获事件。当定时器捕获到输入信号时,可以触发一个中断,我们在中断处理函数中可以读取捕获值,并进行进一步的处理。 在处理捕获值时,可以使用公式: 频率 = 定时器时钟频率 / (捕获值 * 定时器分频系数) 通过以上步骤,我们可以使用STM32F030C8T6单片机来测量脉冲信号的频率。请注意,需要根据具体的应用场景和定时器设置进行适当的参数调整和错误处理。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

南叔先生

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值