上一篇介绍了通用定时器的输出比较部分,这一篇再来介绍一下输入捕获的相关内容。
输入捕获,见名知意,就用来对输入信号进行捕获的,说到捕获输入信号,之前介绍过一个叫做外部中断的片上外设,它的作用也是捕获输入;它们的不同在于,外部中断捕获的只是边沿,而定时器的输入捕获,捕获的是信号的时间信息,可以用来测试脉冲宽度、高电平时间、低电平时间等,还可以利用这个输入捕获的功能来获取一些模块采集的数据,典型代表就是HC-SR04超声波模块。
老规矩还是看看AI对于输入捕获的简介:
上面提到的四个部分都是在配置过程中需要进行编程来实现的,
输入捕获:捕获输入到芯片中的脉冲宽度,捕获的是单个脉冲的宽度
捕获的原理:通过边沿触发捕获
可以通过计算得到:上升沿和下降之间的时间
输出比较:调整脉冲宽度
输入捕获:计算脉冲宽度 捕获一个脉冲的有效时间
对输入捕获有了一个大致的概念后,来看看框图,简易版的框图再上上篇中提到过,那个还看不出具体的配置流程,所以这里再将详细的框图做个分析,和输出比较一样,输入捕获部分的详细框图也很复杂,具体的如下图所示:
这里还是将框图拆分开了看。
首先来看框图的前半段,输入信号进来后,第一个需要经过的就是数字滤波器,如下图红框中的位置。
该滤波器是通过TIMx_CCMR1的ICF[3:0]位控制的,功能是“每N个事件视为一个有效边沿”,举个栗子来说就是,假设编程了这个N为4,而我们检测的高电平有效,那么就需要四个高电平才会生成一次有效边沿。
具体的控制寄存器描述如下:
需要注意的是,这里的频率分为了两部分,一部分是fck_INT也就是初始化的时钟,另一部分是fDTS,这里的f都是频率的意思,是对应的周期分之一;这里的fdts与fck_int的关系取决于TIMx 控制寄存器 1 (TIMx_CR1)的第八第九位的配置,这里配置的是周期的关系。t=1/f.
经过数字滤波器后,信号需要经过一个边沿计测器,检测对应的上升沿还是下降沿,注意上方或门以及后面的一个向上的箭头都是到从模式控制器的,与输入捕获无关,所以可以忽略。
进一步简化后如下图所示,通道1的输入信号经过边沿检测器后,会有一个数据选择器,这个选择器的控制寄存器是CC1P/CC1NP。
对应的是TIMx_CCER寄存器的第一位与第三位,这里需要特别注意,虽然也是两位进行配合,来控制此处的是上升沿有效、下降沿有效还是双边沿都有效,但是是间隔开的两位,位2是不做配置的,在编写代码的时候需要特别小心。
然后就是后面橙色框内的大的数据选择器,这个选择器是用来选输入信号的,一共三个输入,根据上上篇的简略框图,我们知道这三个输入分别是来自通道一的输入信号、来自通道二的输入信号,以及从模式控制器来的信号;此时配置的是输入捕获,而且为了避免交叉使用通道导致配置出错,后两个输入信号在此处是不考虑的,可以直接不看。也就是说,在配置过程中,需要将TIMx_CCMR1的CC1s的两位配置为“01”——选择通道一作为输入。
最后就是蓝色框内的分频器了,这个分频器的作用于前面的数字滤波器的效果类似,假设配置为2分频,则需要有两个对应事件产生才会生成一个有效边沿。一般都配置为了不分频,检测到一个边沿就进行捕获。他的控制寄存器对应的是TIMx_CCMR1的IC1PSC的两位:
最后还有一个控制器,TIMx_CCER的CC1E控制位,这个想必大家也都猜到了,是使能位的意思,细心的同学可能发现了,上一篇的PWM输出,在输出通道最后也是这个CC1E位做的使能,其实他们使用一个寄存器,只是在不同模式中有着不同的含义。
在输入捕获中,除了上面的通道配置以外还有两个需要配置的,如下图,第一个框内CCR1比较捕获寄存器需要写入对应的比较值,这与PWM模式下的CCR1的写法其实一样。
还有一个是CC1G,这个对应的是TIMx 事件生成寄存器 (TIMx_EGR)的第1位,这里后面是个或门,由于CC1E和IC1PS已经配置了,上面的那个输入是真,所以这一位其实不配置也可以。
至于时基部分,与之前的一样,所以在此就不做赘述了,接下来总结一下具体使用到的寄存器有哪些。
1.TIMx 控制寄存器 1 (TIMx_CR1):
写法:TIMx->CR1
位 9:8 CKD:时钟分频 (Clock division)
和滤波器的采样频率有关
2.TIMx 从模式控制寄存器 (TIMx_SMCR)
写法:TIMx->SMCR
时钟源选择
3.TIMx DMA/中断使能寄存器 (TIMx_DIER)
写法:TIMx->DIER
配置更新中断
配置捕获中断
4.TIMx 事件生成寄存器 (TIMx_EGR)
写法:TIMx->EGR
人为生成更新事件
5.TIMx 捕获/比较模式寄存器 1 (TIMx_CCMR1)
写法:TIMx->CCMR1
位 1:0 CC1S:捕获/比较 1 选择 (Capture/Compare 1 selection)
选择输入还是输出
选择输入中的第一个信号源
位 3:2 IC1PSC:输入捕获 1 预分频器 (Input capture 1 prescaler)
配置无预分频
位 7:4 IC1F:输入捕获 1 滤波器 (Input capture 1 filter)
选择对应采样频率 1111
6.TIMx 捕获/比较使能寄存器 (TIMx_CCER)
写法:TIMx->CCER
位 0 CC1E:捕获/比较 1 输出使能 (Capture/Compare 1 output enable)。
使能位
位 1 CC1P:捕获/比较 1 输出极性 (Capture/Compare 1 output Polarity)。
位 3 CC1NP:捕获/比较 1 输出极性 (Capture/Compare 1 output Polarity)。
可读可写
00:上升沿触发
01:下降沿触发
11:双边触发 ----- 通过进入中断的次数来判断
根据上面的框图解析以及寄存器介绍,我们可以大致总结出配置为输入捕获是时的代码思路,
伪代码:
输入捕获初始化函数
{
//打开时钟 GPIO TIM5
GPIO控制器配置
//GPIO模式
复用功能寄存器
定时器的时基单元
//更新禁止
//更新请求源
//关闭单脉冲
//方向
//重装载值得影子寄存器
//分频 4分频
//选择时钟源
//预分频器配置
//重装载值配置
//产生更新事件
定时器得输入部分
//选择输入模式 TI1
//输入捕获分频器 无分频器
//采样频率配置 1111
//边沿触发
//使能捕获
中断配置
//更新中断使能
//捕获中断使能
//NVIC控制器
//计数器使能
}
而对应的功能则需要到对应的中断服务函数去进行,中断服务函数的思路如下:
中断服务函数
{
判断位更新中断
{
N++;//记录产生更新事件的次数,依次来计算时间
}
判断为捕获中断
{
判断为上升沿捕获
{
N=0;
或者这时刻得CCR得值
切换下降沿捕获 //flag=1
}
判断为下降沿捕获
{
直接计算整个时间
切换回上升沿捕获 下一次进来还是先捕获上升沿
}
}
}
上面总结了输入捕获的编程流程,这里借助两个小需求,来具体编程实现对应的功能。
需求1:
检测开发板上Key_UP的按下时间,并通过串口输出。
需求分析:
首先,要检测按键按下的时间,肯定需要使用到上面的输入捕获功能了,具体怎么实现呢,第一步查看原理图找到KEY_UP按下时的输入信号是什么样子的。
如上图所示,在按键输入的时候使用过这个按键,当时是配置为了下拉,使得空闲状态是低电平,按下按键会产生一个脉冲,我们需要捕获时间长度的就是这个高脉冲的时间。具体的捕获过程应该是:
举个栗子:假设定时器的分频系数是84分频,每秒会计数84M/84=1 000 000次,重装载数是1000;也就是每1ms计数1000次,每一次计数时间等于1us;
此时按键按下时对应的计数值是800,从按下按键到松手期间一共产生了6个更新事件,松手时的计数值为400;则此次按键按下的时间为:
TIME = 6*1000+400-800=5600us=5.6ms
分析清楚了捕获过程后,我们来看看具体的配置流程,由于定时器是片上外设,捕获按键按下时常需要有GPIO的参与,所以对应的GPIO需要配置为复用模式。前面提到过,通用定时器至少都有两个通道,怎么知道KEY_UP具体的定时器和通道是哪个呢,这就有需要使用到引脚复用表了。通过查询原理图,或者根据经验,KEY_UP的引脚应该是GPIOA0;在引脚复用表中可以看出它对应的是TIM5的CH1。
根据以前的配置经验,第一步应该打开对应的片上外设的时钟,GPIOA的对应的时钟线是AHB1;TIM5对应的时钟线是APB1,然后到对应的编程手册找到对应位,写入1即可,具体的配置请看后面的代码。
打开时钟后,就需要将GPIOA的0号管脚配置为复用模式,且要对照复用表映射到对应的功能,TIM5的CH1,GPIOA0对应的应该是低位的复用功能寄存器的0-3四位,写入AF2对应的0010即可配置为复用TIM5的通道1。
紧接着就是定时器的时基部分了,这个也是很熟悉的了,前面配置过多次,需要配置CR1、SMCR、预分频和重装载而且还需要手动操作EGR产生一次更新事件,让数据写入影子寄存器已生效。
再这后就是输入通道的配置了,具体的配置流程参考上面的框图流程,主要配置CCMR1、CCER。
这需要使用到两个中断,一个是更新事件的中断,用来记录脉冲的期间内的更新事件个数,还有一个是捕获中断,利用捕获中断来获取对应边沿的计数器值。
再就是中断源使能,优先级配置,最后的最后,一定不要忘记使能计数器。
/*******************************
函数名:Time5_Interrupt
函数功能:Time5_上升沿捕获
函数形参:u32 arr u16 psc预分频系数
函数返回值:void
备注:
预分频系数 重装载值
********************************/
void Time5_Capture(u16 psc, u32 arr)
{
/*-----------开始时钟使能-----------------------------------------------------------------*/
RCC->AHB1ENR |=(1<<0);//开启GPIOA对应的时钟
RCC->APB1ENR |=(1<<3);//开启TIM5对应的时钟
/*-----------初始化GPIO-----------------------------------------------------------------*/
GPIOA->MODER &=~(3<<0);
GPIOA->MODER |=(1<<1);//GPIOA为复用模式
GPIOA->AFR[0] &=~(0xf<<0);
GPIOA->AFR[0] |=(0X2<<0);//GPIOA复用为TIM5
/*-----------时基配置-----------------------------------------------------------------*/
TIM5->CR1 &=~(0Xf<<1); //更新禁止,更新请求源,关闭单脉冲,向上计数
TIM5->CR1 |=(1<<7); //重装载影子寄存器
TIM5->CR1 &=~(3<<8); //不分频
// TIM5->CR1 |=(2<<8); //四分频
TIM5->SMCR &=~(7<<0); //选择内部时钟
TIM5->PSC = psc-1; //预分频值
TIM5->ARR = arr-1; //重装载值
TIM5->EGR |= (1<<0);//更新事件,写入预分频和重装载值
/*-----------定时器输入部分-----------------------------------------------------------------*/
TIM5->CCMR1 &=~(0X3<<0);//
TIM5->CCMR1 |=(1<<0);//模式选择输入 TI1线
TIM5->CCMR1 &=~(3<<2);//输入捕获无预分频
TIM5->CCMR1 &=~(0XF<<4);
TIM5->CCMR1 |=(0xf<<4);//采样频率为最低1111
TIM5->CCER &=~(1<<1);//上升沿触发
TIM5->CCER &=~(1<<3);//上升沿触发
TIM5->CCER |=(1<<0);//捕获使能
/*-----------中断配置-----------------------------------------------------------------*/
TIM5->DIER |=(1<<0);//更新中断使能
TIM5->DIER |=(1<<1);//捕获中断使能
/*-----------NVIC配置-----------------------------------------------------------------*/
u32 pri=NVIC_EncodePriority(7-2,2,3);
NVIC_SetPriority(TIM5_IRQn,pri);
NVIC_EnableIRQ(TIM5_IRQn);
TIM5->CR1 |=(1<<0);//使能计数器
}
实现功能的中断服务函数:
/*******************************
函数名:TIM5_IRQHandler
函数功能:定时器5中断服务函数函数
函数形参:无
函数返回值:void
备注:
********************************/
void TIM5_IRQHandler(void)
{
static u32 cnt;
static u32 cnt_data;
static u32 timer=0;
if(TIM5->SR & (1<<0))//更新中断
{
TIM5->SR &=~(1<<0);
cnt++;//更新中断的次数
}
if(TIM5->SR &(1<<1))//捕获中断
{
TIM5->SR &=~(1<<1);
if(!(TIM5->CCER&(1<<1)))
{
cnt=0;//让更新中断的次数清零
cnt_data = TIM5->CCR1;//获取当前值
TIM5->CCER |=(1<<1);//下降升沿触发
}
else if(TIM5->CCER & (1<<1))
{
timer =cnt*1000-cnt_data+TIM5->CCR1;
if(timer/1000>30)//将抖动的误触发舍弃
{
printf("按键按下时间为%d\r\n",timer/1000);//单位是MS
}
TIM5->CCER &=~(1<<1);//上降升沿触发
}
}
}
利用HC-SR04实现超声波测距。
需求分析:
这里使用到了超声波模块,首先一定要看看他对应的手册,如下图,简介里面我们需要注意的描述,一个是测量周期,最低最低50ms,也就是说,采集一次后最低要给它50ms的休息时间才可以开始下一次采集,然后就是说它支持UART、IIC、单总线的协议,但是需要修改对应的电阻,这里我们不改,就用它最原始的通信方式。
在手册的后面有一个GPIO模式的时序图,这里就告诉了我们测距流程,中间红色框的那一条线是模块内部的,不需要我们操作,也就是说,要驱动这个模块,我们需要使用到的就是两个脚一个触发信号一个输出响应信号。
注意距离的计算是根据模块输出的高电平脉冲信号的时间T来计算的,说白了就是采集超声波模块输出的高电平时间,利用这个时间与音速的关系就可计算出对应的距离。高电平时间的采集与上一个需求中的按键按下时间是一样的。
只是在这个模块需要我们提供一个触发信号后才会输出对应的脉冲,刚刚的按键是按下就会有对应的脉冲。
这个模块由于是外接,所以可以由我们自己选择管脚来实现,其中需要一个GPIO配置为通用推挽输出为Trlg提供触发信号;还需要一个具有定时器捕获通道GPIO来采集超声波模块Echo脉冲输出脚的脉冲持续的时间。
还是打开引脚复用表进行查询,
这里笔者选用了PD12作为输入捕获,为了方便,选择了PD13作为脉冲触发信号。
具体的配置代码如下:
#include "Ultrasonic.h"
/*******************************************
*函数名 :Ult_Init
*函数功能 :超声波初始化函数
*函数参数 :void
*函数返回值:无
*函数描述 :
PD12-------检测电平时间
PD13-------发送起始信号通用模式
*********************************************/
void Ult_Init(void)
{
Time4_Capture(84,1000);//定时器4的输入捕获(用于超声波检测)
GPIOD->MODER &=~(3<<26);
GPIOD->MODER |=(1<<26);//GPIOD13为通用模式
/*端口输出推挽模式*/
GPIOD->OTYPER &= ~(1<<13);
/*端口输出速度2MHz*/
GPIOD->OSPEEDR &= ~(1<<26);//清零OSPEEDR
GPIOD->PUPDR &= ~(1<<26);//默认无上下拉
/* 端口输出寄存器*/
GPIOD->ODR &=~(1<<13);//置零拉低
}
/*******************************
函数名:Time4_Interrupt
函数功能:Time4_上升沿捕获
函数形参:u16 arr u16 psc预分频系数
函数返回值:void
备注:
预分频系数 重装载值
********************************/
void Time4_Capture(u16 psc, u16 arr)
{
/*-----------开始时钟使能-----------------------------------------------------------------*/
RCC->AHB1ENR |=(1<<3);//开启GPIOD对应的时钟
RCC->APB1ENR |=(1<<2);//开启TIM4对应的时钟
/*-----------初始化GPIO-----------------------------------------------------------------*/
GPIOD->MODER &=~(3<<24);
GPIOD->MODER |=(1<<25);//GPIOD12为复用模式
GPIOD->AFR[1] &=~(0xf<<16);
GPIOD->AFR[1] |=(0X2<<16);//GPIOD12复用为TIM4
/*-----------时基配置-----------------------------------------------------------------*/
TIM4->CR1 &=~(0Xf<<1); //更新禁止,更新请求源,关闭单脉冲,向上计数
TIM4->CR1 |=(1<<7); //重装载影子寄存器
TIM4->CR1 &=~(3<<8); //不分频
// TIM5->CR1 |=(2<<8); //四分频
TIM4->SMCR &=~(7<<0); //选择内部时钟
TIM4->PSC = psc-1; //预分频值
TIM4->ARR = arr-1; //重装载值
TIM4->EGR |= (1<<0);//更新事件,写入预分频和重装载值
/*-----------定时器输入部分-----------------------------------------------------------------*/
TIM4->CCMR1 &=~(0X3<<0);//
TIM4->CCMR1 |=(1<<0);//模式选择输入 TI1线
TIM4->CCMR1 &=~(3<<2);//输入捕获无预分频
TIM4->CCMR1 &=~(0XF<<4);
TIM4->CCMR1 |=(0xf<<4);//采样频率为最低1111 84 000 000/32/8
TIM4->CCER &=~ (1<<1);//上升沿触发
TIM4->CCER &=~ (1<<3);//上升沿触发
TIM4->CCER |=(1<<0);//捕获使能
/*-----------中断配置-----------------------------------------------------------------*/
TIM4->DIER |=(1<<0);//更新中断使能
TIM4->DIER |=(1<<1);//捕获中断使能
/*-----------NVIC配置-----------------------------------------------------------------*/
u32 pri=NVIC_EncodePriority(7-2,2,3);
NVIC_SetPriority(TIM4_IRQn,pri);
NVIC_EnableIRQ(TIM4_IRQn);
TIM4->CR1 |=(1<<0);//使能计数器
}
//头文件
#ifndef _ULTRASONIC_H
#define _ULTRASONIC_H
#include "stm32f4xx.h"
#define TRIG_OFF GPIOD->ODR &=~(1<<13);//置零拉低对应端口
#define TRIG_ON GPIOD->ODR |= (1<<13);//置1拉高对应端口
void Ult_Init(void);
void Time4_Capture(u16 psc, u16 arr);
#endif
主函数中产生触发信号:每隔50ms产生一次触发信号。
中断服务函数实现功能:
/*******************************
函数名:TIM4_IRQHandler
函数功能:定时器4中断服务函数函数
函数形参:无
函数返回值:void
备注:
********************************/
void TIM4_IRQHandler(void)
{
static u32 cnt;
static u32 cnt_data;
static float timer=0;
if(TIM4->SR & (1<<0))//更新中断
{
TIM4->SR &=~(1<<0);
cnt++;//更新中断的次数
}
if(TIM4->SR &(1<<1))//捕获中断
{
TIM4->SR &=~(1<<1);
if(!(TIM4->CCER&(1<<1)))
{
cnt=0;//让更新中断的次数清零
cnt_data = TIM4->CCR1;//获取当前值
TIM4->CCER |=(1<<1);//下降沿触发
}
else if(TIM4->CCER & (1<<1))
{
timer =cnt*1000-cnt_data+TIM4->CCR1;
printf("物体的距离为:%.2fcm\r\n",timer/58.0f);
TIM4->CCER &=~(1<<1);//上升沿触发
}
}
}
关于定时器输入捕获与输出比较的介绍就记录到这,从这开始,配置都会比之前的复杂,写起来也会比较麻烦,所以更新速度会变慢,大家谅解,然后就是文中如有不足欢迎批评指正。