如何利用stm32的中断和计数器即上升沿下降沿捕获电平时长

发布者:CaptivatingEyes最新更新时间:2015-09-21 来源: eefocus关键字:stm32  中断  计数器  电平时长 手机看文章 扫描二维码
随时随地手机看文章
2.jpg1.jpg

/*! @file
********************************************************************************


模块名       : 红外解码文件
文件名       : IR_REMOTE.c
相关文件     : IR_REMOTE.h
文件实现功能 : 定时器红外解码 
作者         : 李

版本         : 
--------------------------------------------------------------------------------
硬件平台     : STM32F107
软件开发平台 : iar
--------------------------------------------------------------------------------
修改记录     : 2015-8-17 建立
日 期        版本   修改人         修改内容
2015/8/17   0.1               创建

********************************************************************************
*******************************************************************************/
//该类遥控器的引导码为4MS 高电平 4MS低电平,bit'1'为2ms H,bit'0'为0.5ms L


#include "IRM_Remote.h"
#include

static  u8 user_code[3]={0};//用户码
static  u8 lead_flg=0;//引导码状态变量
static  u8 ir_cnt;//定时器电平时间计数量
static  u8 high_cnt=0;//24位
static  u8  REM_FLG=0;//接收一个码标志位
static  u8  Remote_Scan();
/*************************************************************
名称: Remote_Key()
功能:遥控码值转换
输出参数:返回键值 重新编码
作者:v
******************************************************************/
u8  Remote_Key()
{
u16 keyvalue=0;
u8 ret=0;
//Remote_Scan();
if(REM_FLG==1)
{
REM_FLG=0;
keyvalue=user_code[1]*256+user_code[2];
switch(keyvalue)
{
      case  IR_1:
ret = 1;
break;
 
case  IR_2:
ret = 2;
    break;
 
case  IR_3:
ret = 3;
    break;
 
case  IR_4:
ret = 4;
    break;
 
case  IR_5:
ret = 5;
    break; 

case  IR_6:
ret = 6;
    break;
 
case  IR_7:
ret = 7;
    break;
 
case  IR_8:
ret = 8;
    break; 

case  IR_9:
ret = 9;
    break;
 
case  IR_10:
ret = 10;
    break;
 
case  IR_11:
ret = 11;
    break; 

case  IR_12:
ret = 12;
    break;
 
case  IR_13:
ret = 13;
    break;
 
case  IR_14:
ret = 14;
    break;
 
case  IR_15:
 ret = 15;
     break;
 
case  IR_16:
ret = 16;
    break;
 
case  IR_17:
ret = 17;
    break; 


  case  IR_18:
ret = 18;
    break;
 
case  IR_19:
ret = 19;
    break;
 
case  IR_20:
ret = 20;
    break; 

case  IR_21:
ret = 21;
    break;
 
case  IR_22:
ret = 22;
    break;
 
case  IR_23:
ret = 23;
    break; 

case  IR_24:
ret = 24;
    break;
 
case  IR_25:
ret = 25;
    break;
 
case  IR_26:
ret = 26;
    break; 

case  IR_27:
ret = 27;
    break;
 
case  IR_28:
ret = 28;
    break;
 
case  IR_29:
ret = 29;
    break; 

case  IR_30:
ret = 30;
    break;
 
case  IR_31:
ret = 31;
    break;
 
case  IR_32:
ret = 32;
    break; 
 
case  IR_33:
ret = 33;
    break;


case  IR_34:
ret = 34;
    break;
 
case  IR_35:
ret = 35;
    break; 
 
case  IR_36:
ret = 36;
    break;
 
case  IR_37:
ret = 37;
    break; 
default:
ret=0;
break;
}
    return ret;
}
return ret;
}
[page]
/*************************************************************
名称: Timer4_CFG()  72M时钟晶振
功能: 初始化timer4    100us定时发生器
输出参数:
作者:v
******************************************************************/

 void Timer4_CFG()


TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE);
GPIO_InitTypeDef GPIO_InitStructure; 





TIM_TimeBaseStructure.TIM_Prescaler = 3599; //TIM_CKD_DIV1 72MHz/(3599+1)=20 000  HZ  50uS
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //
TIM_TimeBaseStructure.TIM_Period = 200;       // ARR基准值
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure);

//TIM_PrescalerConfig(TIM4,35,TIM_PSCReloadMode_Immediate);//
TIM_ARRPreloadConfig(TIM4,DISABLE);

TIM_ITConfig(TIM4,TIM_IT_Update,ENABLE);
TIM_ITConfig(TIM4,TIM_IT_Trigger,ENABLE);
TIM_Cmd(TIM4, DISABLE);

}
/*************************************************************
名称 ;外部中断函数  启用第12个中断源 作为中断引脚
功能:    中端配置初始化 引脚初始化
输出参数:无
作者:v
******************************************************************/

void EXTIX_Init(void)
{
 
EXTI_InitTypeDef EXTI_InitStructure;
    NVIC_InitTypeDef NVIC_InitStructure;

GPIO_InitTypeDef GPIO_InitStructure; 

RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOE | RCC_APB2Periph_AFIO , ENABLE);
 
GPIO_InitStructure.GPIO_Pin = IR_LED_PIN; 
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; 
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; 
GPIO_Init(IR_LED_PORT, &GPIO_InitStructure); 

GPIO_EXTILineConfig(GPIO_PortSourceGPIOE ,GPIO_PinSource12); 
EXTI_InitStructure.EXTI_Line = EXTI_Line12;
EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; 
EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising_Falling; //上升下降沿都触发中断

EXTI_InitStructure.EXTI_LineCmd = ENABLE; 
EXTI_Init(&EXTI_InitStructure); 

NVIC_InitStructure.NVIC_IRQChannel = EXTI15_10_IRQn;       //中断通道
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;//强占优先级
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;       //次优先级
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;          //通道中断不使能,有切换需要的时候使能
    NVIC_Init(&NVIC_InitStructure); 
 
}



#if 0
/*************************************************************
名称: Remote_Scan()
功能: 
输出参数:
作者:v
******************************************************************/
static u8  Remote_Scan()

u16 TIMt=0;
u16  waitquit;//超时退出
u8 i,ia;


if(RDATA()!=0)//先判断是否为0  排除干扰
{
return  0;
}
          
TIM_SetCounter(TIM4,0);//TIM4->CNT=0;
TIM_Cmd(TIM4, ENABLE);


waitquit=60000;
do

waitquit--;

while((!RDATA())&&(waitquit));//检查高电平的时间4ms

TIM_Cmd(TIM4, DISABLE); 
TIMt=TIM_GetCounter(TIM4);
if((TIMt<70)||(TIMt>120))
{
return 0;

}

TIM_SetCounter(TIM4,0);  
TIM_Cmd(TIM4, ENABLE);
waitquit=60000;
do
{
waitquit--;

while((RDATA())&&( waitquit));//接收高电平
TIMt=TIM_GetCounter(TIM4);
TIM_Cmd(TIM4, DISABLE); 
if((TIMt>120)||(TIMt<60))//同样 4MS
{
return 0;
}

////////////引导码接收完毕///////开始接收客户码//////////////////////////////
 
for(i=0;i<3;i++)//接收3个字节 24位码
{
for(ia=0;ia<8;ia++)
{
TIM_Cmd(TIM4, ENABLE);
while(!RDATA());//等待高
TIM_SetCounter(TIM4,0);
waitquit=60000;//可根据实测调整
do{
waitquit--;

while((RDATA())&&( waitquit));//接收高电平
TIM_Cmd(TIM4, DISABLE);
TIMt=TIM_GetCounter(TIM4); 
////////////////////////////高电平结束/////////////////////////
user_code[i]>>=1;
if((TIMt>45))
{
return 0;
}

if((TIMt>35)&&(TIMt<45))//经过实测 在2ms以上为1 小于一般为 9 为0
{
user_code[i]|=0x80;

}
}

}

REM_FLG=1;
return 1;
}

#endif
/*************************************************
  Function:       EXTI15_10_IRQHandler
  Description:    F103切换信号中断
  Input:          无
  Output:         无
  Return:         无
  Others:         需要切换时,打开此中断
*************************************************/
#if 1
u8 EXTI15_10_IRQHandler()
{


//NVIC_InitTypeDef NVIC_InitStructure;
if(EXTI_GetITStatus(EXTI_Line14)!= RESET)
{
EXTI_ClearITPendingBit(EXTI_Line14);

//调用康工切换函数,g_SDISrc,g_SDIDst
//UserSwitch(g_SW_Src,g_SW_Dst); //2013.05.16
//GPIO_SetBits(GPIOE,GPIO_Pin_6);

/* Enable the EXTI15 Interrupt for 控制板F103*/
//NVIC_InitStructure.NVIC_IRQChannel = EXTI15_10_IRQn; //中断通道
//NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;//强占优先级
//NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;   //次优先级
//NVIC_InitStructure.NVIC_IRQChannelCmd = DISABLE; //通道中断不使能,有切换需要的时候使能
//NVIC_Init(&NVIC_InitStructure); 

}
if(EXTI_GetITStatus(EXTI_Line12)!= RESET)
{
if(RDATA()==0)//说明是下降沿到来
{
  
        TIM_Cmd(TIM4, ENABLE);
if(lead_flg==0)//说明是第一次接收引导码
{
TIM_SetCounter(TIM4,0);//TIM4->CNT=0; (1)
            lead_flg=1;
}
else if(lead_flg==2)
{
        ir_cnt= TIM_GetCounter(TIM4); 
            TIM_SetCounter(TIM4,0);
lead_flg=3;
if((ir_cnt<70)||(ir_cnt>90))//测试表明为79 或80
{
 lead_flg=0; //超出范围  退出
              return 0;
               
}
}
else if(lead_flg==3)//获得高电平时间
{
ir_cnt= TIM_GetCounter(TIM4); 
TIM_SetCounter(TIM4,0);
//接收3个字节 24位码
                high_cnt++;//加到24位   每8次为1个字节
            if(high_cnt<=8)
{
user_code[0]>>=1;
}
                if(high_cnt<=16)
{
user_code[1]>>=1;
}
                if(high_cnt<=24)
{
user_code[2]>>=1;
}
           if((ir_cnt>35)&&(ir_cnt<45))//我们认为是 bit '1'  2ms
{
                if(high_cnt<=8)
{
user_code[0]|=0x80;

}
                    if(high_cnt<=16)
{
user_code[1]|=0x80;
}
                    if(high_cnt<=24)
{
user_code[2]|=0x80;}
    }
       if(high_cnt>24)//说明接受完毕
{
lead_flg = 0;
high_cnt = 0;
REM_FLG = 1;//采集完毕 清除标志位

TIM_Cmd( TIM4,DISABLE);
}
       

}

}
else if(RDATA()!=0)//说明是第一次接收引导码的高电平
{
       if(lead_flg==0)
       {
       return 0;
       }
       else if(lead_flg==1)//我们只接受一次低电平即可   2
      {
ir_cnt= TIM_GetCounter(TIM4);
TIM_SetCounter(TIM4,0);
lead_flg=2;
if((ir_cnt<70)||(ir_cnt>90))
{
 lead_flg=0;//不是我们要的 退出
              return 0;
                
}

}
else if(lead_flg==3)//非引导码进入中断   上升沿触发的
{   
   ir_cnt= TIM_GetCounter(TIM4);
TIM_SetCounter(TIM4,0);

if((ir_cnt<8)||(ir_cnt>20))//测试数据为 ir_cnt =9 10 稳定值  0.5ms
{
 lead_flg=0;
              return 0;//不是我们要的  退出
                
}


}
}

}
EXTI_ClearITPendingBit(EXTI_Line12);
}

关键字:stm32  中断  计数器  电平时长 引用地址:如何利用stm32的中断和计数器即上升沿下降沿捕获电平时长

上一篇:投币器的输出信号脉冲检测
下一篇:单片机红外遥控器解码例程源码新说

推荐阅读最新更新时间:2024-03-16 14:32

如何在STM32中得到最佳的ADC精度
STM32家族中的所有芯片都内置了逐次逼近寄存器型ADC模块.内部大致框架如下: 每次ADC转换先进行采样保持,然后分多步执行比较输出,步数等于ADC的位数,每个ADC时钟产生一个数据位。说到这里,用过STM32 ADC的人是不是想到了参考手册中关于12位ADC转换时间的公式: ST官方就如何保障或改善ADC精度写了一篇应用笔记AN2834。该应用笔记旨在帮助用户了解ADC误差的产生以及如何提高ADC的精度。主要介绍了与ADC设计的相关内容,比如外部硬件设计参数,不同类型的ADC误差来源分析等,并提出了一些如何减小误差的设计上建议。 当我们在做STM32的ADC应用遇到转换结果不如意时,常有人提醒或建议你对采样时间或
[单片机]
如何在<font color='red'>STM32</font>中得到最佳的ADC精度
STM32串口DMA容易忽视的问题
博主昨天晚上在STM32串口DMA的问题上纠结了好长时间,所以今天上午写篇博客来谈谈我对串口DMA发送的理解→_→今天主要讨论三个问题:1、什么叫串口DMA 请求;2、串口简要复习;3、串口DMA发送流程。 1、什么叫串口DMA 请求(博主用的是战舰STM32开发板) 说这个问题之前先简单回顾DMA的基本特性。先导出原子哥的PPT内容: DMA全称Direct Memory Access,即直接存储器访问。 DMA传输将数据从一个地址空间复制到另一个地址空间。当CPU初始化这个传输动作,传输动作本身是由DMA控制器来实现和完成的。 STM32有两个DMA控制器(DMA2只存在于大容量产品中),DMA1有7个通道
[单片机]
<font color='red'>STM32</font>串口DMA容易忽视的问题
STM32-FSMC-NANDFLASH
STM32 FSMC 支持两个NAND闪存块,支持硬件ECC并可检测多达8K字节数据 其地址映射如下图所示 图161 FSMC存储块 NAND和PC卡地址映射 表88 存储器映像和时序寄存器  对于NAND闪存存储器,通用和属性空间又可以在低256K字节部分划分为3个区(见表89) ● 数据区(通用/属性空间的前64K字节区域) ● 命令区(通用/属性空间的第2个64K字节区域) ● 地址区(通用/属性空间的第2个128K字节区域) 表89 NAND存储块选择 应用软件使用这3个区访问NAND闪存存储器: ● 发送命令到NAND闪存存储器:软件只需对命令区的任意一个地址写入命令即可。 ● 指定操作NAND
[单片机]
STM32-FSMC-NANDFLASH
STM32的启动堆栈初始化
有几个问题,众多博文中抄来抄去,内容一样,却没有解释清楚 上电初始化堆栈,在进入_main后又说初始化堆栈,有什么不同 堆栈的地址是怎么得出来的 关于这两个问题,先借用一下要标准的启动流程 一般而言,系统上电后第一个执行的是由汇编所编写的启动文件,其主要工作为一下五部分: (1)、初始化堆栈指针SP=_initial_sp (2)、初始化PC指针,令其=Reset_Handler (3)、初始化中断向量表 (4)、配置系统时钟 (5)、调用C库函数_main初始化用户堆栈,从而最终调用main函数进入C的世界 STM32的中断向量表规定每一行必须是SP地址,第二行是复位中断入口地址,上电后,C
[单片机]
keil库的运用,进阶版
看一个图片就ok了。看起来是不是很清晰脱俗,非常舒服。下载运行时ok的。 用这个库新建工程也是ok的,实测通过。
[单片机]
keil库的运用,进阶版
stm32外部中断实验
// 上一篇是关于串口通信的,用到GPIO的复用,将GPIO复用为usart串口; // 此处是利用按键进行中断处理,这里配置GPIO模式为输入,因为要接收按键的状态; //GPIO端口有很多,ABCD....但是中断只有22个,其中0~15个中断中断线与IO端口一一对应,需要配置GPIO与中断线的映射关系,(类似于端口复用)这里利用了函数:SYSCFG_EXTILineConfig(EXTI_PortSourceGPIOA,EXTI_PinSource0); // 这里需要注意,使用的外部中断,先打开SYSCFG时钟,不然没法实现GPIO与中断线的映射。 #include stm32f4xx.h void init_led
[单片机]
STM32学习笔记11——HardFault_Handler处理方法
根据网络资料及自己调试经验总结如下: STM32 出现 HardFault_Handler 故障的原因主要有两个方面: 1、内存溢出或者访问越界。这个需要自己写程序的时候规范代码,遇到了需要慢慢排查。 2、堆栈溢出。增加堆栈的大小。 排查方法: 发生异常之后可首先查看 LR 寄存器中的值,确定进入异常前一刻使用的堆栈为 MSP 或 PSP,然后找到相应堆栈的指针? 注:在 HardFault_Handler(void)中断里第一条语句打断点,进入中断后,查看 LR 寄存器的值,如果是 0XFFFFFFF9,那么中断前使用的是 MSP,如果是 0XFFFFFFFD,那么中断前使用的是 PSP; 根据找到的堆栈指针, 在内存中查
[单片机]
<font color='red'>STM32</font>学习笔记11——HardFault_Handler处理方法
s3c2410汇编启动代码与中断跳转分析
最开始的代码,大家都很熟悉 ,最基本的中断跳转 b ResetHandler b HandlerUndef ;handler for Undefined mode b HandlerSWI ;handler for SWI interrupt b HandlerPabort ;handler for PAbort b HandlerDabort ;handler for DAbort b . ;reserved b HandlerIRQ ;handler for IRQ interrupt b HandlerFIQ ;handler for FIQ interrupt 跳转的标号是一系列的宏 LTORG HandlerFIQ
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved