stm32f103输入捕获

stm32的定时器具有捕获功能,能够捕获上升沿或者下降沿然后触发中断。

定时器框图:

细化框图:

滤波器:判断在捕获到边沿信号的时候,以Fdts的频率连续采集N次该引脚上电平判断电平是否稳定。其中Fdts由控制寄存器 1(TIMx_CR1)的CKD位控制而采集频率又受到IC1F位的控制。

注意:这里有两个频率,一个Fdts一个采样频率分开理解。采样频率可能受Fdts有可能受FINT的控制所以要设置CDK是中分频因子,而N为采集次数。

例如:IC1F设置为011:N=8且上升沿捕获,当我们捕获到上升沿后滤波器以采样频率采集8次io上的电平如果8次都是1那么说明是有效触发就会触发中断反之不会触发中断。

边沿检测器:检测是上升沿还是下降沿。

输入捕获映射通道:我们通过配置CCMR1寄存器可以使得TI1的输入捕获映射到TI1或者TI2上

 

输入捕获分频器:实际就是配置捕获到几次有效边沿触发一次中断,例如ICPS=10也就是4分频,当捕获到4次有效边沿时才会触发一次中断请求。

捕获到有效信号可以开启中断

输入捕获过程总结:得到io边沿---->滤波------>检测是上升沿还是下降沿------->是否要开启通道映射------->分频(几次有效触发产生一起中断请求)

stm32库函数配置过程:

① 初始化定时器和通道对应IO的时钟。

② 初始化IO口,模式为输入:GPIO_Init();

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD; //PA0 输入

③初始化定时器ARRPSC

   TIM_TimeBaseInit();

④初始化输入捕获通道

   TIM_ICInit();

⑤如果要开启捕获中断,

    TIM_ITConfig();

    NVIC_Init();

⑥使能定时器:TIM_Cmd();

⑦编写中断服务函数:TIMx_IRQHandler();

注意:

通道极性设置独立函数:

void TIM_OCxPolarityConfig(TIM_TypeDef* TIMx, uint16_t TIM_OCPolarity)

获取通道捕获值:

uint32_t TIM_GetCapture1(TIM_TypeDef* TIMx)

输入捕获通道初始化函数:

void TIM_ICInit(TIM_TypeDef* TIMx, TIM_ICInitTypeDef* TIM_ICInitStruct);

typedef struct

{

  uint16_t TIM_Channel; //捕获通道1-4  

  uint16_t TIM_ICPolarity; //捕获极性

  uint16_t TIM_ICSelection; //映射关系

  uint16_t TIM_ICPrescaler; //分频系数

  uint16_t TIM_ICFilter;  //滤波器

} TIM_ICInitTypeDef;

例如:

TIM5_ICInitStructure.TIM_Channel = TIM_Channel_1; TIM5_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising;

TIM5_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI;

TIM5_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;

TIM5_ICInitStructure.TIM_ICFilter = 0x00;

TIM_ICInit(TIM5, &TIM5_ICInitStructure);

这里我们参考原子历程说说如何捕获高电平所占时间:

首先我们初始化捕获中断时设置捕获上升沿触发中断。当我们捕获到上升沿后开启下降沿捕获模式,并且当定时器发生溢出事件CAPTURE_STA++直到我们捕获到下降沿时标志着捕获完成此时的高电平所占时间=CAPTURE_STA&0x3f*定时间一出一次的时间+当前拘束寄存器中的值。由于我们设置捕获频率为1Mhz也就是1us中断一次表达式变为:

高电平所占时间=+CAPTURE_STA&0x3f*65536+当前拘束寄存器中的值(的道德实践单位为us)

其中:CAPTURE_STA为8位高两位为捕获标志位:bit7带包捕获完成,bit6代表捕获到下降沿,其余的为溢出次数。

注意:当溢出次数达到3f时代表高电平持续时间太长,捕获失败。

void TIM2_IRQHandler(void)

     if((TIM2CH1_CAPTURE_STA&0X80)==0)//还未成功捕获    
    {      
        if (TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET)
         
        {        
            if(TIM2CH1_CAPTURE_STA&0X40)//已经捕获到高电平了
            {
                if((TIM2CH1_CAPTURE_STA&0X3F)==0X3F)//高电平太长了
                {
                    TIM2CH1_CAPTURE_STA|=0X80;//标记成功捕获了一次
                    TIM2CH1_CAPTURE_VAL=0XFFFF;
                }else TIM2CH1_CAPTURE_STA++;
            }     
        }
    if (TIM_GetITStatus(TIM2, TIM_IT_CC1) != RESET)//捕获1发生捕获事件
        {    
            if(TIM2CH1_CAPTURE_STA&0X40)        //捕获到一个下降沿         
            {                  
                TIM2CH1_CAPTURE_STA|=0X80;        //标记成功捕获到一次上升沿
                TIM2CH1_CAPTURE_VAL=TIM_GetCapture1(TIM2);
                   TIM_OC1PolarityConfig(TIM2,TIM_ICPolarity_Rising); //CC1P=0 设置为上升沿捕获
            }else                                  //还未开始,第一次捕获上升沿
            {
                TIM2CH1_CAPTURE_STA=0;            //清空
                TIM2CH1_CAPTURE_VAL=0;
                 TIM_SetCounter(TIM2,0);
                TIM2CH1_CAPTURE_STA|=0X40;        //标记捕获到了上升沿
                   TIM_OC1PolarityConfig(TIM2,TIM_ICPolarity_Falling);        //CC1P=1 设置为下降沿捕获
            }            
        }                                                
     }
 
    TIM_ClearITPendingBit(TIM2, TIM_IT_CC1|TIM_IT_Update); //清除中断标志位

#include "led.h" #include "delay.h" #include "key.h" #include "sys.h" #include "exti.h" #include "timer.h" #include "usart.h" #include "IWDG.h" //int main(void) //中断 //{ ////delay_init(); //LED_Init(); ////KEY_Init(); ////NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); ////EXTIX_Init(); //LED=0; //while(1); //} //int main(void) //定时器中断 //{ // NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); // delay_init(); // LED_Init(); // // TIM3_Int_Init(1999,7199);//((1+7199)/72M)*(1+9999)=1秒*/反 // while(1); //} //int main(void) //pwm //{ // u16 ledpwmval=0; // u8 dir=1; // delay_init(); // NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); // uart_init(115200); // LED_Init (); // // TIM3_PWM_Init(199,7199);//50Hz // while(1) // { // delay_ms(10); // if(dir)ledpwmval++; // else ledpwmval --; // if(ledpwmval >1000) // dir=0; // if(ledpwmval ==0) // dir=1; // TIM_SetCompare2(TIM3,5); // delay_ms(500); // TIM_SetCompare2(TIM3,10); // delay_ms(500); // TIM_SetCompare2(TIM3,15); // delay_ms(500); // TIM_SetCompare2(TIM3,20); // delay_ms(500); // TIM_SetCompare2(TIM3,25); // delay_ms(500); // } //} //int main() //串口 //{ // u16 t; // u16 len; // u16 times=0; // delay_init(); // NVIC_PriorityGroupConfig (NVIC_PriorityGroup_2 ); // uart_init(115200); // LED_Init(); // KEY_Init(); // while(1) // { // if(USART_RX_STA&0x8000) // { // len=USART_RX_STA&0x3fff;//得到此次接收的数据长度 // printf("\r\n您发送的消息为:\r\n\r\n"); // for(t=0;t<len;t++) // { // USART_SendData(USART1,USART_RX_BUF[t]); // while(USART_GetFlagStatus(USART1,USART_FLAG_TC)!=SET); // } // printf("\r\n\r\n");//插入换行 // USART_RX_STA=0; // } // else // { // times++; // if(times%500000==0) // { // LED=!LED; // } // } // } //} extern void TIM4_Cap_Init(u16 arr,u16 psc); extern u8 TIM4CH1_CAPTURE_STA; //输入捕获状态 输入捕获实验 extern u16 TIM4CH1_CAPTURE_VAL; //输入捕获值 int main(void) { u32 temp=0; delay_init(); //延时函数初始化 NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置NVIC中断分组2:2位抢占优先级,2位响应优先级 uart_init(115200); //串口初始化为115200 LED_Init(); //LED端口初始化 // TIM3_PWM_Init(899,0); //不分频。PWM频率72000/(899+1)=80Khz TIM4_Cap_Init(0XFFFF,72-1); //以1Mhz的频率计数 while(1) { delay_ms(10); TIM_SetCompare2(TIM3,TIM_GetCapture2(TIM3)+1); if(TIM_GetCapture2(TIM3)==300) TIM_SetCompare2(TIM3,0); if(TIM4CH1_CAPTURE_STA&0X80)//成功捕获到了一次上升沿 { temp=TIM4CH1_CAPTURE_STA&0X3F; temp*=65536; //溢出时间总和 temp+=TIM4CH1_CAPTURE_VAL;//得到总的高电平时间 printf("HIGH:%d us\r\n",temp);//打印总的高点平时间 TIM4CH1_CAPTURE_STA=0;//开启下一次捕获 } } }
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值