STM32 定时器输入捕获2——捕获高电平时长

由上图我们可以知道,高电平时间=t2-t1。在代码中,可以记录此时t1的时间然后再记录t2的时间,t2-t1,就是我们所想要的答案。

但是,还有更简单一点点的,当到达t1的时候,我们把定时器清零,然后直接读出t2,就是这个高电平的时间。

  • 1.当TIM3的TIM_FLAG_CC1==1时,清除TIM3
cpp 复制代码
void TIM3_IRQHandler(void)
{
	if(TIM_GetITStatus(TIM3,TIM_FLAG_CC1)==SET)
	{
		TIM_SetCounter(TIM3,0);
	}
	TIM_ClearITPendingBit(TIM3,TIM_FLAG_CC1);
}
  • 2.从波形图可以看出,刚开始检测的是上升沿,所以检测TIM_FLAG_CC1的时候是SET,到了第二步,波形开始下降,变为下降沿,此时我们就要更改一下检测的方式了,我们可以用到下面这个函数。
cpp 复制代码
TIM_OC1PolarityConfig(TIM3,TIM_OCPolarity_Low);
//更改下次检测的极性为low(低电平)
cpp 复制代码
void TIM3_IRQHandler(void)
{
	static unsigned char high_flag = 0;
	if(TIM_GetITStatus(TIM3,TIM_FLAG_CC1)==SET)
	{
		
		if(high_flag == 0)//上升沿
		{
			high_flag = 1;
			TIM_SetCounter(TIM3,0);
			TIM_OC1PolarityConfig(TIM3,TIM_OCPolarity_Low);
		}
		else//下降沿
		{
			high_value = TIM_GetCounter(TIM3);
			TIM_OC1PolarityConfig(TIM3,TIM_OCPolarity_High);
			high_flag = 0;
		}
	}
	TIM_ClearITPendingBit(TIM3,TIM_FLAG_CC1);
}
  • 3.每次到下一个波形的时候,都重新初始化一下high_value.
cpp 复制代码
if(high_flag == 0)//上升沿
{
    high_flag = 1;
	high_value = 0;
	TIM_SetCounter(TIM3,0);
	TIM_OC1PolarityConfig(TIM3,TIM_OCPolarity_Low);
}
  • 4.如果在main中,去使用了high_value,此时高电平还在不断输出,定时器也在运行,high_value仍会该改变,对于high_value的使用就不方便。所以我们需要限制一个条件,当我们使用high_value的时候,不要去重新写high_value了。
cpp 复制代码
void TIM3_IRQHandler(void)
{
	static unsigned char high_flag = 0;
	
	if(high_complete_flag == 0)
	{
		if(TIM_GetITStatus(TIM3,TIM_FLAG_CC1)==SET)
		{
			if(high_flag == 0)//上升沿
			{
				high_flag = 1;
				high_value = 0;
				TIM_SetCounter(TIM3,0);
				TIM_OC1PolarityConfig(TIM3,TIM_OCPolarity_Low);
			}
			else//下降沿
			{
				high_value = TIM_GetCounter(TIM3);
				TIM_OC1PolarityConfig(TIM3,TIM_OCPolarity_High);
				high_flag = 0;
				high_complete_flag = 1;
			}
		}
	}
	TIM_ClearITPendingBit(TIM3,TIM_FLAG_CC1);
}

在使用high_value之前,让high_complete_flag = 0;这样之后时,high_value就不会变了

main

cpp 复制代码
int main(void)
{
	TIM2_PWM_Configuration();//PWM周期为20ms
	
	TIM3_InputCaputure_Configuration();//PA1:发出《------------》PA6:接收
	
	Uart1_Configuration();
	
	TIM_SetCompare2(TIM2,10000);//占空比为50%,即10ms高电平
	
	while(1)
	{
		if(high_complete_flag == 1)
		{
			printf("%d us",high_value);
			high_complete_flag = 0;
		}
		Delay_s(10);
	}
	
}
相关推荐
榆榆欸14 分钟前
14.主从Reactor+线程池模式,Connection对象引用计数的深入分析
linux·服务器·网络·c++·tcp/ip
lzb75924 分钟前
积分赛——读取实时时间
单片机
编程侦探1 小时前
【设计模式】原型模式:用“克隆”术让对象创建更灵活
c++·设计模式·原型模式
KangkangLoveNLP1 小时前
手动实现一个迷你Llama:使用SentencePiece实现自己的tokenizer
人工智能·深度学习·学习·算法·transformer·llama
基极向上的三极管1 小时前
【51单片机】2-5【I/O口】433无线收发模块控制继电器
单片机·51单片机
Once_day2 小时前
Linux错误(6)X64向量指令访问地址未对齐引起SIGSEGV
linux·c++·sse·x64·sigsegv·xmm0
浪淘沙jkp2 小时前
大模型学习二:DeepSeek R1+蒸馏模型组本地部署与调用
学习·deepseek
JhonKI2 小时前
【从零实现Json-Rpc框架】- 项目实现 - 客户端注册主题整合 及 rpc流程示意
c++·qt·网络协议·rpc·json
__lost2 小时前
为什么new分配在堆上,函数变量在栈上+递归调用时栈内存的变化过程
c++·内存分配
m0_613607012 小时前
数据集(Dataset)和数据加载器(DataLoader)-pytroch学习3
学习