stm32的输入捕获实验程序问题(测量脉冲宽度)
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了stm32的输入捕获实验程序问题(测量脉冲宽度)相关的知识,希望对你有一定的参考价值。
初始化函数没有问题,计数时间单位为1us,编程思路是定义两个全局变量一个32位的一个8位的,32位用来存放输入捕获值,8位的最高位用来表示捕获到高电平(1表示捕获)次高位表示脉冲是否捕获完成(捕获完成置1),低六位用来表示计数器溢出次数的,每溢出一次自加一,通过定义的i/o口结合按键手动输入高电平,本来是不会出现溢出情况的(32位定时器溢出需要4千多秒)但是按一下按键有的时候就会出现溢出的那种情况,实在是找不出来有什么逻辑问题了,求大神指教,本人虚心求教万望各位指点一二
参考技术A 1、脉冲宽度如下图所示,采集该高电平脉冲的宽度,只需要进入输入捕获上升沿检测,记录当前的发生上升沿时的CNT值,再进行输入捕获下降沿检测,也记录当前发生下降沿时的CNT值,两次CNT值的差值再根据计数的频率就可以算出脉冲的宽度。
上升沿及下降沿捕获的程序具体实现如下:
TIM8_Cap_Init(0XFFFF,72-1); //以1Mhz的频率计数
void TIM8_UP_IRQHandler(void)
if((TIM8CH4_CAPTURE_STA&0X80)==0) //还未成功捕获
if (TIM_GetITStatus(TIM8,TIM_IT_Update) != RESET)
if(TIM8CH4_CAPTURE_STA&0X40) //已经捕获到高电平了
if((TIM8CH4_CAPTURE_STA&0X3F)==0X3F) //高电平太长了
TIM8CH4_CAPTURE_STA"=0X80; //标记成功捕获了一次
TIM8CH4_CAPTURE_VAL=0XFFFF;
else
TIM8CH4_CAPTURE_STA++; //捕获高电平后定时器溢出的次数++
TIM_ClearITPendingBit(TIM8,TIM_IT_Update); //清除中断标志位
void TIM8_CC_IRQHandler(void)
if((TIM8CH4_CAPTURE_STA&0X80)==0)
if(TIM_GetITStatus(TIM8,TIM_IT_CC4) != RESET) //捕获1发生捕追问
首先感谢您耐心的回答,但是我真找不出我的程序哪里有错误,没有问题的程序我这也有就是想不明白我自己写的这个有什么问题
参考技术B具体的捕获流程我不太了解,但你的代码里有几个明显的问题。
1.图一中你的判断条件是STA>=0x80进入第一层if后,不可能出现STA==0x3f的情况,因为此时STA总是大于0x3f的。还有STA++会导致一个后果,会可能修改掉你的bit7和bit6,正确的此处加加的方式应该是
//1.达到最大计数后重新开始计数if(STA & 0x3F == 0x3F)
STA &= 0xC0;
STA++;
//2.或者保持最大计数
if(STA & 0x3F == 0x3F)
STA |= 0x3F;
else
STA++;
2.图三里为什么要乘以0xFFFFFFFF啊?还有while(a==40)完全没意义啊,这个循环体里把a=0了呀
STM32的输入捕获实验
一、什么是输入捕获?
输入捕获模式可以用来测量脉冲宽度或者测量频率。STM32的定时器,除了TIM6和TIM7,
其他定时器都有输入捕获功能。STM32 的输入捕获,简单的说就是通过检测 TIMx_CHx 上的
边沿信号,在边沿信号发生跳变(比如上升沿/下降沿)的时候,将当前定时器的值(TIMx_CNT)
存放到对应的通道的捕获/比较寄存器(TIMx_CCRx)里面,完成一次捕获。同时还可以配置
捕获时是否触发中断/DMA 等。
二、使用TIM5的输入高电平捕获
1.捕获TIM5的CH1
TIM5的通道1在PA0引脚上,我们将捕获 TIM5_CH1(PA0)上的高电平脉
宽,通过PA0上的按键输入高电平,并从串口打印高电平脉宽。同时我们保留上节的 PWM 输
出,大家也可以通过用杜邦线连接 PB5 和 PA0,来测量 PWM 输出的高电平脉宽。
2.代码解读
从这个工程,我们的难度有点大了,因为我们要接触到核心部分了,我们在使用输入捕获时,我们先确定我们是使用上升沿捕获还是下降沿捕获,我们在这儿使用的是上升沿捕获,就是通过按键产生一个上升沿,然后定时器开始计时,我们用这个捕获看高电平时间,通过串口显示高电平时间。
三、代码示例
timer.c文件
#include "timer.h"
#include "led.h"
//定时器5通道1输入捕获配置
TIM_ICInitTypeDef TIM5_ICInitStructure;
void TIM5_Cap_Init(u16 arr,u16 psc)
{
GPIO_InitTypeDef GPIO_InitStructure;
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
NVIC_InitTypeDef NVIC_InitStructure;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM5, ENABLE); //使能TIM5时钟
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //使能GPIOA时钟
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; //PA0 清除之前设置
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD; //PA0 输入
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_ResetBits(GPIOA,GPIO_Pin_0); //PA0 下拉
//初始化定时器5 TIM5
TIM_TimeBaseStructure.TIM_Period = arr; //设定计数器自动重装值
TIM_TimeBaseStructure.TIM_Prescaler =psc; //预分频器
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //设置时钟分割:TDTS = Tck_tim
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式
TIM_TimeBaseInit(TIM5, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位
//初始化TIM5输入捕获参数
TIM5_ICInitStructure.TIM_Channel = TIM_Channel_1; //CC1S=01 选择输入端 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);
//中断分组初始化
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_ITConfig(TIM5,TIM_IT_Update|TIM_IT_CC1,ENABLE);//允许更新中断 ,允许CC1IE捕获中断
TIM_Cmd(TIM5,ENABLE ); //使能定时器5
}
u8 TIM5CH1_CAPTURE_STA=0; //输入捕获状态
u16 TIM5CH1_CAPTURE_VAL; //输入捕获值
//定时器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=0XFFFF;
}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); //CC1P=0 设置为上升沿捕获
}else //还未开始,第一次捕获上升沿
{
TIM5CH1_CAPTURE_STA=0; //清空
TIM5CH1_CAPTURE_VAL=0;
TIM_SetCounter(TIM5,0);
TIM5CH1_CAPTURE_STA|=0X40; //标记捕获到了上升沿
TIM_OC1PolarityConfig(TIM5,TIM_ICPolarity_Falling); //CC1P=1 设置为下降沿捕获
}
}
}
TIM_ClearITPendingBit(TIM5, TIM_IT_CC1|TIM_IT_Update); //清除中断标志位
}
main.c文件
#include "delay.h"
#include "key.h"
#include "usart.h"
#include "timer.h"
extern u8 TIM5CH1_CAPTURE_STA; //输入捕获状态
extern u16 TIM5CH1_CAPTURE_VAL; //输入捕获值
int main(void)
{
u32 temp=0;
delay_init(); //延时函数初始化
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置NVIC中断分组2:2位抢占优先级,2位响应优先级
uart_init(115200); //串口初始化为115200
TIM5_Cap_Init(0XFFFF,72-1); //以1Mhz的频率计数
while(1)
{
delay_ms(10);
if(TIM5CH1_CAPTURE_STA&0X80)//成功捕获到了一次上升沿
{
temp=TIM5CH1_CAPTURE_STA&0X3F;
temp*=65536;//溢出时间总和
temp+=TIM5CH1_CAPTURE_VAL;//得到总的高电平时间
printf("HIGH:%d us\\r\\n",temp);//打印总的高点平时间
TIM5CH1_CAPTURE_STA=0;//开启下一次捕获
}
}
}
实验现象:
这里值得注意的是,WAKE_UP按键(引脚PA0)是接到高电平上的,所以一开始PA0是低电平,才会有上升沿。
相关的资料代码都在这里,作者使用的是正点原子开发板,所以资料是正点原子的。
总结
输入捕获在测量频率的时候用到的,所以我们可以用输入捕获做一个简易示波器,这里作者没有做简易的示波器,因为比较麻烦,作者最近工作太忙,没有来得及做,所以着这里只是提一下,我们可以采用输入捕获,来作为触摸按键的开关确定触摸开关是否按下,那个东西下个文章讲。
代码资料如下:
链接:https://pan.baidu.com/s/1-Arnzh4sAodYidIa-Xj56g
提取码:4pi2
以上是关于stm32的输入捕获实验程序问题(测量脉冲宽度)的主要内容,如果未能解决你的问题,请参考以下文章