STM32F103标准库学习——输入捕获实验

东方欲晓,莫道君行早。
踏遍青山人未老,风景这边独好。——毛泽东《清平乐·会昌》

一、实验内容

STM32PA0(TIM5的CH1) 配置为 输入捕获模式 ,由于 PA0KEY3 相连接,编写程序实现以下功能:
(1)当按下按键 KEY3 时,捕获低电平持续的时间;
(2)将按键 KEY3 低电平持续的时间转换为 毫秒(ms) 为单位的数值;
(3)将低电平的持续时间通过 UART1 发送到计算机;
(4)通过串口助手查看按键 KEY3 低电平持续的时间。

所用工具:

  1. 芯片:STM32F103RCT6
  2. IDE :MDK-Keil 软件
  3. SMT32F10x 固件库函数
  4. TIM5 CH1 :通用定时器5 通道1(CH1)

二、实验原理

2.1、输入捕获简介

  • 输入捕获是定时器通用定时器许多功能中的一种。输入捕获可以对输入的信号的上升沿,下降沿或者双边沿进行捕获,通常用于测量输入信号的脉宽、测量 PWM 输入信号的频率及占空比。
  • 输入捕获的工作原理比较简单,在输入捕获模式下,当相应的 ICx 信号检测到跳变沿后,将使用捕获/比较寄存器 (TIMx_CCRx) 来锁存计数器的值。简单的说就是通过检测 TIMx_CHx 上的边沿信号,在边沿信号发生跳变(比如上升沿/下降沿)的时候,将当前定时器的值(TIMx_CNT)存放到对应的通道的捕获/比较寄存 (TIMx_CCRx) 里面,完成一次捕获。同时还可以配置捕获时是否触发 中断/DMA 等。下面我们以输入捕获测量脉宽为例,通过一个简图来介绍输入捕获的工作原理,如图 所示:

在这里插入图片描述
从上图可以看出,t1 - t2 时间就是我们需要测量的高电平时间,假如定时器工作在向上计数模式,测量方法是:首先设置定时器通道 x 为上升沿捕获,这样在 t1 时刻,就会捕获到当前的 CNT 值,然后立即清零 CNT,并设置通道 x 为下降沿捕获,这样到 t2 时刻,又会发生捕获事件,得到此时的 CNT 值,记为 CCRx2。根据定时器的计数频率,我们就可以算出 t1 - t2 的时间,从而得到高电平脉宽。在 t1 - t2 时间内可能会出现 N 次定时器溢出,因此我们还需要对定时器溢出进行处理,防止因高电平时间过长发生溢出导致测量数据不准。CNT计数的次数等于:N * ARR + CCR x 2,有了这个计数次数,再乘以 CNT 的计数周期,即可得到 t2 - t1 的时间长度,即高电平持续时间。)
通用定时器框图

2.2、输入捕获实验流程图分析

  • 输入捕获实验中断服务函数流程图:
    在这里插入图片描述
  • 输入捕获实验应用层流程图:
    在这里插入图片描述

2.3、通用定时器部分寄存器

本实验涉及的通用定时器寄存器除了包括定时器基本使用的控制寄存器1(TIMx_CR1)、控制寄存器2(TIMx_CR2)、DMA/中断使能寄存器(TIMx_DIER)、状态寄存器(TIMx_SR)、事件产生寄存器(TIMx_EGR)、计数器(TIMx_CNT)、预分频器(TIMx_PSC)和自动重载寄存器(TIMx_ARR),还包括捕获/比较模式寄存器1(TIMx_CCMR1)、捕获/比较使能寄存器(TIMx_CCER)和捕获/比较寄存器1(TIMx_CCR1)。
在这里插入图片描述

2.4、通用定时器部分固件库函数

本实验涉及的通用定时器固件库函数包括TIM_TimeBaseInitTIM_Cmd、TIM_ITConfigTIM_ClearITPendingBitTIM_GetITStatusTIM_SelectOutputTrigger,还包括TIM_ICInit()TIM_OC1PolarityConfig()

  • TIM_ICInit() 函数:的功能是根据TIM_ICInitStruct中指定的参数初始化外设TIMx;
  • TIM_OC1PolarityConfig() 函数:的功能是设置TIMx通道1极性。
    在这里插入图片描述
    在这里插入图片描述

三、代码编写

3.1、复制并编译原始工程

链接: https://pan.baidu.com/s/1SPBUBYaOGNuDIkTp_6HnkA
提取码: if52
在这里插入图片描述

3.2、添加 Capture 文件对 (.c/.h)

3.3、完善 Capture.h 文件

#ifndef _CAPTURE_H_
#define _CAPTURE_H_

/*********************************************************************************************************
*                                              包含头文件
*********************************************************************************************************/
#include "DataType.h"

/*********************************************************************************************************
*                                              宏定义
*********************************************************************************************************/

/*********************************************************************************************************
*                                              枚举结构体定义
*********************************************************************************************************/

/*********************************************************************************************************
*                                              API函数声明
*********************************************************************************************************/
void  InitCapture(void);              //初始化Capture模块
u8    GetCaptureVal(i32* pCapVal);    //获取捕获时间,返回值为1表示捕获成功,此时*pCapVal才有意义
 
#endif

3.4、完善 Capture.c 文件

/*********************************************************************************************************
*                                              包含头文件
*********************************************************************************************************/
#include "Capture.h"
#include "stm32f10x_conf.h"

/*********************************************************************************************************
*                                              内部变量
*********************************************************************************************************/
//s_iCaptureSts中的bit7为捕获完成的标志,bit6为捕获到下降沿标志,bit5-bit0为捕获到下降沿后定时器溢出的次数
static  u8  s_iCaptureSts = 0;     //捕获状态 
static  u16 s_iCaptureVal;         //捕获值
 
/*********************************************************************************************************
*                                              内部函数声明
*********************************************************************************************************/
static  void ConfigTIM5ForCapture(u16 arr, u16 psc);    //配置TIM3
/*********************************************************************************************************
* 函数名称:ConfigTIM5ForCapture
* 函数功能:配置TIM3 
* 输入参数:arr-自动重装值,psc-预分频器值
* 输出参数:void
* 返 回 值:void
* 创建日期:2018年01月01日 
* 注    意:此处暂定使用定时器TIM5的CH1(PA0)来做输入捕获,捕获PA0(KEY3)上低电平的脉宽
*********************************************************************************************************/
static  void ConfigTIM5ForCapture(u16 arr, u16 psc)
{ 
  GPIO_InitTypeDef        GPIO_InitStructure;    //GPIO_InitStructure用于存放GPIO的参数
  TIM_TimeBaseInitTypeDef TIMx_TimeBaseStructure;//TIM_TimeBaseStructure用于存放定时器的基本参数
  TIM_ICInitTypeDef       TIMx_ICInitStructure;  //TIMx_ICInitStructure用于存放定时器的通道参数
  NVIC_InitTypeDef        NVIC_InitStructure;    //NVIC_InitStructure用于存放NVIC的参数

  //使能RCC相关时钟
  RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM5, ENABLE);  //使能TIM5的时钟
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //使能捕获的GPIOA的时钟
  
  //配置PA0,对应TIM5的CH1
  GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_0;            //设置引脚
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;         //设置输入模式 
  GPIO_Init(GPIOA, &GPIO_InitStructure);                //根据参数初始化GPIO
  GPIO_SetBits(GPIOA, GPIO_Pin_0);  //将捕获对应的引脚置为高电平
  
  //配置TIM5
  TIMx_TimeBaseStructure.TIM_Period        = arr;                     //设定计数器自动重装值 
  TIMx_TimeBaseStructure.TIM_Prescaler     = psc;                     //设置TIMx时钟频率除数的预分频值  
  TIMx_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;            //设置时钟分割
  TIMx_TimeBaseStructure.TIM_CounterMode   = TIM_CounterMode_Up;      //设置定时器TIMx为向上计数模式
  TIM_TimeBaseInit(TIM5, &TIMx_TimeBaseStructure); //根据指定的参数初始化TIMx的时间基数单位
  
  //配置TIM5的CH1为输入捕获
  //CC1S = 01,CC1通道被配置为输入,输入通道IC1映射到定时器引脚TI1上
  TIMx_ICInitStructure.TIM_Channel        = TIM_Channel_1;            //设置输入通道通道 
  TIMx_ICInitStructure.TIM_ICPolarity     = TIM_ICPolarity_Falling;   //设置为下降沿捕获
  TIMx_ICInitStructure.TIM_ICSelection    = TIM_ICSelection_DirectTI; //设置为直接映射到TI1
  TIMx_ICInitStructure.TIM_ICPrescaler    = TIM_ICPSC_DIV1;           //设置为每一个边沿都捕获,捕捉不分频 
  TIMx_ICInitStructure.TIM_ICFilter       = 0x08;                     //设置输入滤波器
  TIM_ICInit(TIM5, &TIMx_ICInitStructure);         //根据参数初始化TIM5的CH1
  
  //配置NVIC
  NVIC_InitStructure.NVIC_IRQChannel                   = TIM5_IRQn;     //中断通道号   
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;             //设置抢占优先级   
  NVIC_InitStructure.NVIC_IRQChannelSubPriority        = 0;             //设置子优先级   
  NVIC_InitStructure.NVIC_IRQChannelCmd                = ENABLE;        //使能中断   
  NVIC_Init(&NVIC_InitStructure);                                       //根据参数初始化NVIC    
  
  TIM_ITConfig(TIM5, TIM_IT_Update | TIM_IT_CC1, ENABLE); //使能定时器的更新中断和CC1IE捕获中断 
  
  TIM_Cmd(TIM5, ENABLE);  //使能TIM5
}

/*********************************************************************************************************
* 函数名称:TIM5_IRQHandler
* 函数功能:TIM5中断服务函数
* 输入参数:void
* 输出参数:void
* 返 回 值:void
* 创建日期:2018年01月01日 
* 注    意:
*********************************************************************************************************/
void TIM5_IRQHandler(void)
{ 
  if((s_iCaptureSts & 0x80) == 0) //最高位为0,表示捕获还未完成
  {  
    //高电平,定时器TIMx发生了溢出事件
    if(TIM_GetITStatus(TIM5, TIM_IT_Update) != RESET)        
    {    
      if(s_iCaptureSts & 0x40)    //发生溢出,并且前一次已经捕获到低电平
      {
        //TIM_APR 16位预装载值,即CNT > 65536-1(2^16 - 1)时溢出。
        //若不处理,(s_iCaptureSts & 0x3F)++等于0x40 ,溢出数等于清0
        if((s_iCaptureSts & 0x3F) == 0x3F)  //达到多次溢出,低电平太长
        {
          s_iCaptureSts |= 0x80;  //强制标记成功捕获了一次
          s_iCaptureVal = 0xFFFF; //捕获值为0xFFFF
        } 
        else
        {
          s_iCaptureSts++;        //标记计数器溢出一次
        }
      }
    }
    
    if (TIM_GetITStatus(TIM5, TIM_IT_CC1) != RESET) //发生捕获事件
    { 
      if(s_iCaptureSts & 0x40)  //bit6为1,即上次捕获到下降沿,那么这次捕获到上升沿
      {
        s_iCaptureSts |= 0x80;  //完成捕获,标记成功捕获到一次上升沿
        s_iCaptureVal = TIM_GetCapture1(TIM5);  //s_iCaptureVa记录捕获比较寄存器的值
        //CC1P=1 设置为下降沿捕获,为下次捕获做准备
        TIM_OC1PolarityConfig(TIM5, TIM_ICPolarity_Falling);   
      }
      else  //bit6为0,表示上次没捕获到下降沿,这是第一次捕获下降沿
      {
        s_iCaptureSts = 0;  //清空溢出次数
        s_iCaptureVal = 0;  //捕获值为0
                                                             
        TIM_SetCounter(TIM5, 0);  //设置寄存器的值为0
                                                             
        s_iCaptureSts |= 0x40;    //bit6置为1,标记捕获到了下降沿
        
        TIM_OC1PolarityConfig(TIM5, TIM_ICPolarity_Rising);  //CC1P=0 设置为上升沿捕获
      }    
    } 
  }
 
  TIM_ClearITPendingBit(TIM5, TIM_IT_CC1 | TIM_IT_Update);    //清除中断标志位
}

/*********************************************************************************************************
* 函数名称:GetCaptureVal
* 函数功能:获取捕获时间,返回值为1表示捕获成功,此时*pCapVal才有意义 
* 输入参数:void
* 输出参数:pCalVal,捕获到的值的地址
* 返 回 值:ok:1-获取成功
* 创建日期:2018年01月01日 
* 注    意:
*********************************************************************************************************/
u8   GetCaptureVal(i32* pCapVal)
{
  u8  ok = 0;
  
  if(s_iCaptureSts & 0x80)              //最高位为1,表示成功捕获到了上升沿(获取到按键弹起标志)
  {
    ok = 1;                             //捕获成功
    (*pCapVal)  = s_iCaptureSts & 0x3F; //取出低6位计数器的值赋给(*pCapVal),得到溢出次数
    (*pCapVal) *= 65536;//计数器计数次数为2^16=65536,乘以溢出次数,得到溢出时间总和(以1/1MHz=1us为单位)
    (*pCapVal) += s_iCaptureVal;        //加上最后一次比较捕获寄存器的值,得到总的低电平时间

    s_iCaptureSts = 0;                  //设置为0,开启下一次捕获
  }

  return(ok);                           //返回是否捕获成功的标志
}

3.5、完善 main.c 文件

/*********************************************************************************************************
* 函数名称:Proc2msTask
* 函数功能:2ms处理任务 
* 输入参数:void
* 输出参数:void
* 返 回 值:void
* 创建日期:2018年01月01日
* 注    意:
*********************************************************************************************************/
static  void  Proc2msTask(void)
{
  static i16 s_iCnt5 = 0;                       //10ms计数器
  i32 captureVal;                               //捕获到的值
  float captureTime;                            //将捕获值转换成时间
  
  if(Get2msFlag())                              //判断2ms标志状态
  {  
//    LEDFlicker(250);                            //调用闪烁函数  
    
    if(s_iCnt5 >= 4)                            //计数器数值大于或等于4
    {
      if(GetCaptureVal(&captureVal))            //成功捕获
      {
        captureTime = captureVal / 1000.0;
        printf("H-%0.2fms\r\n", captureTime);   //打印出捕获值     
      }
      
      s_iCnt5 = 0;                              //重置计数器的计数值为0 
    }
    else
    {
      s_iCnt5++;                                //计数器的计数数值加1
    }
    
    Clr2msFlag();                               //清除2ms标志
  }    
}


四、实验结果

在这里插入图片描述

谢谢阅读!!!

  • 12
    点赞
  • 41
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

W_oilpicture

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

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

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

打赏作者

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

抵扣说明:

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

余额充值