当前位置: 首页 > news >正文

11.STM32F4 输入捕获

一、输入捕获概念

       输入捕获模式可以用来测量脉冲宽度或者测量频率。我们以测量脉宽为例,用一个简图来说明输入捕获的原理,如图1所示:

图1:输入捕获脉宽测量原理图

        STM32F4的输入捕获,简单的说就是通过检测TIMx_CHx上的边沿信号,在边沿信号发生跳变(比如上升沿/下降沿)的时候,将当前定时器的值(TIMx_CNT)存放到对应的通道的捕获/比较寄存器(TIMx_CCRx)里面,完成一次捕获。同时还可以配置捕获时是否触发中断/DMA等。

  本文我将用TIM5_CH1来捕获高电平脉宽,捕获原理如图1所示,我就不多介绍了。

二、输入捕获功能及应用

       捕获波形:功能,把模拟信号的边沿捕获,得到波形时间。如图2所示:

图2:输入捕获波形图解

   遥控器:一般两种(蓝牙遥控器),(红外遥控器),(有线遥控器)

  发送装置如图3所示:

图3

   接收装置如图4所示:

图4

        输入捕获:可以捕获红外发送过来的波形,通过每个波形的高低电平时间,知道你是什么样的键值

       遥控器:每个按键都不一样、每个按键都有不同的键值,键值会通过红外发送,就使用红外接收进行输入捕获。

       因为我的硬件设备上没有遥控器,也没有接收。所以在我的硬件设备这里没办法实现。但是可以使用按键,可以捕获按键的按下时间,从而达到捕获的效果。

        可以用于捕获按键高低电平持续时间,捕获波形时间。

三、输入捕获模式

       在输入捕获模式下,当相应的ICx信号检测到跳变沿后,将使用捕获/比较寄存器(TIMx_CCRx)来锁存计数器的值。发生捕获事件时,会将相应的CCXIF标志(TIMx_SR寄存器)置1,并可发送中断或DMA请求(如果已使能)。如果发生捕获事件时CCxIF标志已处于高位,则会将重复捕获标志CCxOF(TIMx_SR寄存器)置1。可通过软件向CCxIF写入0来给CCxIF清零,或读取存储在TIMx_CCRx寄存器中的已捕获数据。向CCxOF写入0后会将

其清零。具体内容如下图所示:

四、输入捕获函数配置步骤

        1、开启TIM5时钟,配置PA0为复用功能(AF2),并开启下拉电阻;

        2、初始化TIM5,设置TIM5的ARR和PSC;

        3、设置TIM5的输入捕获参数,开启输入捕获;

        4、使能捕获和更新中断(设置TIM5的DIER寄存器);

        5、设置中断优先级,编写中断服务函数;

        6、使能定时器(设置TIM5的CR1寄存器)。

五、TIM_ICInitTypeDef的定义及结构体中参数说明

    typedef struct{

        uint16_t TIM_Channel;//通道

        uint16_t TIM_ICPolarity;//捕获极性

        uint16_t TIM_ICSelection;//映射

        uint16_t TIM_ICPrescaler;//分频系数

        uint16_t TIM_ICFilter; //滤波器长度

} TIM_ICInitTypeDef;

六、代码示例

1、tim5.c
//定时器5通道1输入捕获配置
//arr:自动重装值(TIM2,TIM5是32位的!!)psc:时钟预分频数TIM_ICInitTypeDef  TIM5_ICInitStructure;  
void TIM5_CH1_Cap_Init(u32 arr,u16 psc){GPIO_InitTypeDef GPIO_InitStructure;TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;NVIC_InitTypeDef NVIC_InitStructure;RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM5,ENABLE);  //TIM5时钟使能        RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE); //使能PORTA时钟GPIO_InitStructure.GPIO_Pin         = GPIO_Pin_0; //GPIOA0 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;//复用功能GPIO_InitStructure.GPIO_Speed       = GPIO_Speed_100MHz;//速度100MHzGPIO_InitStructure.GPIO_OType       = GPIO_OType_PP; //推挽复用输出GPIO_InitStructure.GPIO_PuPd        = GPIO_PuPd_DOWN; //下拉GPIO_Init(GPIOA,&GPIO_InitStructure); //初始化PA0GPIO_PinAFConfig(GPIOA,GPIO_PinSource0,GPIO_AF_TIM5); //PA0复用位定时器5TIM_TimeBaseStructure.TIM_Prescaler     = psc;  //定时器分频        TIM_TimeBaseStructure.TIM_CounterMode   = TIM_CounterMode_Up; //向上计数模式    TIM_TimeBaseStructure.TIM_Period        = arr;   //自动重装载值TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;TIM_TimeBaseInit(TIM5,&TIM_TimeBaseStructure);TIM5_ICInitStructure.TIM_Channel     = TIM_Channel_1; //选择输入端IC1映射到TI1上TIM5_ICInitStructure.TIM_ICPolarity  = TIM_ICPolarity_Rising;//上升沿捕获TIM5_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI; //映射到TI1上            TIM5_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;//配置输入分频,不分频    TIM5_ICInitStructure.TIM_ICFilter    = 0x00;//IC1F=0000配置输入滤波器不滤波TIM_ICInit(TIM5, &TIM5_ICInitStructure);//初始化TIM5输入捕获参数TIM_ITConfig(TIM5,TIM_IT_Update|TIM_IT_CC1,ENABLE);//允许更新和捕获中断TIM_Cmd(TIM5,ENABLE );//使能定时器5NVIC_InitStructure.NVIC_IRQChannel                   = TIM5_IRQn;NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;//抢占优先级2     NVIC_InitStructure.NVIC_IRQChannelSubPriority        = 0;//响应优先级0 NVIC_InitStructure.NVIC_IRQChannelCmd                = ENABLE;//IRQ通道使能NVIC_Init(&NVIC_InitStructure);//根据指定的参数初始化VIC寄存器、
}/**************************************************************************
//捕获状态//[7]:0,没有成功的捕获;1,成功捕获到一次.
//[6]:0,还没捕获到低电平;1,已经捕获到低电平了.
//[5:0]:捕获低电平后溢出的次数(对于32位定时器来说,1us计数器加1,溢出时间:4294秒)
u8 TIM5CH1_CAPTURE_STA=0;//输入捕获状态
u32TIM5CH1_CAPTURE_VAL;//输入捕获值(TIM2/TIM5是32位)
**************************************************************************/
//定时器5中断服务程序
void TIM5_IRQHandler(void){//还未成功捕获if((TIM5CH1_CAPTURE_STA&0X80)==0){//溢出if(TIM_GetITStatus(TIM5, TIM_IT_Update) != RESET){//已经捕获到高电平了if(TIM5CH1_CAPTURE_STA&0X40){//高电平太长了if((TIM5CH1_CAPTURE_STA&0X3F)==0X3F){TIM5CH1_CAPTURE_STA |= 0X80;//标记成功捕获了一次TIM5CH1_CAPTURE_VAL = 0XFFFFFFFF;}else TIM5CH1_CAPTURE_STA++;}}//捕获1发生捕获事件if(TIM_GetITStatus(TIM5, TIM_IT_CC1) != RESET){//捕获到一个下降沿if(TIM5CH1_CAPTURE_STA&0X40){TIM5CH1_CAPTURE_STA |= 0X80;//标记成功捕获到一次高电平脉宽                        TIM5CH1_CAPTURE_VAL = TIM_GetCapture1(TIM5);//获取当前的捕获值.TIM_OC1PolarityConfig(TIM5,TIM_ICPolarity_Rising); //设置上升沿捕获}//还未开始,第一次捕获上升沿else {TIM5CH1_CAPTURE_STA = 0;//清空TIM5CH1_CAPTURE_VAL=0;TIM5CH1_CAPTURE_STA |= 0X40;//标记捕获到了上升沿TIM_Cmd(TIM5,ENABLE );//使能定时器5TIM_SetCounter(TIM5,0);//计数器清空TIM_OC1PolarityConfig(TIM5,TIM_ICPolarity_Falling);//设置下降沿捕获            TIM_Cmd(TIM5,ENABLE );//使能定时器5}}}TIM_ClearITPendingBit(TIM5, TIM_IT_CC1|TIM_IT_Update); //清除中断标志位
}
2、tim5.h
#ifndef _TIM5_H_
#define _TIM5_H_#include "stm32f4xx.h"
#include "stdio.h"void TIM5_CH1_Cap_Init(u32 arr,u16 psc);#endif
3、main.c
extern u8  TIM5CH1_CAPTURE_STA; //输入捕获状态
extern     u32TIM5CH1_CAPTURE_VAL;//输入捕获值
int main(void){long long temp = 0;NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置系统中断优先级分组2SysTick_Init(168);  //初始化延时函数uart_init(115200);//初始化串口波特率为115200TIM14_PWM_Init(500-1,84-1);//84M/84=1Mhz的计数频率计数到500,频率为1M/500=2Khz TIM5_CH1_Cap_Init(0XFFFFFFFF,84-1);//以84M/84=1Mhz的频率计数while(1){Delay_ms(10);TIM_SetCompare1(TIM14,TIM_GetCapture1(TIM14)+1);if(TIM_GetCapture1(TIM14)==300)TIM_SetCompare1(TIM14,0);//成功捕获到了一次高电平if(TIM5CH1_CAPTURE_STA&0X80){temp = TIM5CH1_CAPTURE_STA&0X3F;temp *= 0XFFFFFFFF;//溢出时间总和temp += TIM5CH1_CAPTURE_VAL;  //得到总的高电平时间printf("HIGH:%lld us\r\n",temp);//打印总的高点平时间TIM5CH1_CAPTURE_STA = 0;//开启下一次捕获}}
}

七、下载验证结果

        在完成代码编写之后,将编译好的文件下载到我们的硬件开发板上,此时,可以看到DS0的状态由暗到亮的循环。即可说明程序已正常在执行了,然后再打开串口调试助手,选择对应的串口,按下KEY_UP按键,可以看到串口打印的高电平持续时间,如下图所示:

        从上图中可看出,其中有2次高电平在50us以内的,这种就是按键按下时发生的抖动。这就是为什么我们按键输入的时候,一般都需要做防抖处理,防止类似的情况干扰正常输入。大家还可以用杜邦线连接PA0和PF9,具体的内容可参考STM32F4开发指南库函数版本中第十四章PWM输出实验

八、总结

         输入捕获是一种用于测量和捕获外部信号的功能,常用于测量信号的频率、周期、占空比等参数。通过配置定时器和GPIO引脚,并设置输入捕获通道的工作模式和触发条件,来实现信号的捕获和测量。通过中断处理程序读取输入捕获寄存器的值,可以获取外部信号的相关参数。学习了输入捕获需要了解定时器和GPIO引脚的配置,掌握了输入捕获通道的设置和中断处理方法,并参考相关的芯片手册和开发环境文档进行实际应用。

九、其他说明

      《STM32F4开发指南—库函数版本》可到我的资源进行下载。

       TIM14_PWM_Init 函数代码编写可参考9.PWM简单介绍章程。

注意:本人所写文章内容均用于记录自己在学习嵌入式的成长过程!!!!!

http://www.lryc.cn/news/288089.html

相关文章:

  • opencv#30 线性滤波
  • 如何使用iPhone或iPad上的二维码共享Wi-Fi密码?这里有详细步骤
  • 在游戏里开公司!基于ERNIE SDK的多智能体游戏应用
  • 【SpringCloud Nacos】 微服务治理介绍及Nacos引入初体验
  • JavaEE进阶(6)SpringBoot 配置文件(作用、格式、properties配置文件说明、yml配置文件说明、验证码案例)
  • 面包屑是什么
  • C++ 设计模式之责任链模式
  • HTML5与App封装技术将网站一键打包成App
  • openssl3.2/test/certs - 060 - any.bad.com is excluded by CA2.
  • SpringBoot整理-数据库操作
  • IP被封怎么办?访问网站时IP被阻止?解决IP禁令全方法
  • 利用aiohttp异步爬虫实现网站数据高效抓取
  • navicat连接postgresql、人大金仓等数据库报错
  • AUTOSAR看门狗篇 -看门狗驱动(Wdg)
  • 数字图像处理:图像内插
  • Cantor表(刷题)(C语言)
  • linux install nvm
  • HTML 炫酷进度条
  • Windows10上使Git Bash支持rsync命令操作步骤
  • rust for循环里的所有权 - into_iter / iter / iter_mut
  • GitHub README-Template.md - README.md 模板
  • 【文本到上下文 #6】Word2Vec、GloVe 和 FastText
  • yolov5 opencv dnn部署自己的模型
  • Cortex-M4处理器 电源管理
  • Linux 驱动开发基础知识——编写LED驱动程序(三)
  • YOLOv8 视频识别
  • elementplus Dialog 对话框设置距离页面顶部的距离
  • 便捷接口调测:API 开发工具大比拼 | 开源专题 No.62
  • openssl3.2/test/certs - 008 - root-nonca trust variants: +serverAuth +anyEKU
  • cg插画设计行业怎么样,如何学习插画设计