stm32红外遥控的定时器实现

我用的红外遥控是使用的NEC协议,即使用PWM来调制发送的信息    NEC协议,其特征如下:

 1、8位地址和8位指令长度;

 2、地址和命令2次传输(确保可靠性)

 3、PWM脉冲位置调制,以发射红外载波的占空比代表“0”和“1”;

 4、载波频率为38Khz;

 5、位时间为1.125ms或2.25ms;

NEC码的位定义:一个脉冲对应560us的连续载波,一个逻辑1传输需要2.25ms(560us脉冲+1680us低电平),一个逻辑0的传输需要1.125ms(560us脉冲+560us低电平)。而遥控接收头在收到脉冲的时候为低电平,在没有脉冲的时候为高电平,这样,我们在接收头端收到的信号为:逻辑1应该是560us低+1680us高,逻辑0应该是560us低+560us高。

NEC遥控指令的数据格式为:同步码头、地址码、地址反码、控制码、控制反码。同步码由一个9ms的低电平和一个4.5ms的高电平组成,地址码、地址反码、控制码、控制反码均是8位数据格式。按照低位在前,高位在后的顺序发送。采用反码是为了增加传输的可靠性(可用于校验)。

stm32红外遥控的定时器实现
红外接收头与stm32连接如上图所示,既然是PWM调制,很容易想到了stm32的通用定时器的输入捕获和输出比较功能,这里由于stm32是接收红外遥控发送的信息,所以与红外接收头连接的IO口要设置位输入模式,因为在空闲状态的时候输入始终要保持高电平,所以要配置成上拉输入。

              

RCC->APB2ENR|=1<<3;

       GPIOB->CRH&=0xffffff0f; 

       GPIOB->CRH|=0x00000080; 

       GPIOB->ODR|=1<<9;

因为PB.9是通用定时器的通道四,所以还要对定时器进行配置,额。。。好长时间没有用定时器了,都忘得差不多了,又得重新拾起来             

 void time4_init()

{

 RCC->APB1ENR|=1<<2;//开启定时器四的时钟

 TIM4->SR=0;//其实复位值就是0,多此一举了

 TIM4->DIER|=1<<4;//允许定时器四的捕获中断

 TIM4->PSC=71;//计数频率设置为1M CNT每增加一 时间为1us

 TIM4->ARR=10000;//计数器每隔10ms溢出一次

 TIM4->CCMR2|=1<<8;//CC4通道被配置为输入,IC4映射在TI4上;

 TIM4->CCER&=~(1<<13);//通道四配置为上升沿捕获

 TIM4->CCMR2|=3<<12;//进行滤波处理

 TIM4->CCER|=1<<12;//通道四捕获使能

  TIM4->CR1|=1<<0;//定时器四计数使能

}

因为红外接收头接收的信号第一个数据必然是同步码,首先低电平保持9ms,然后一个跳变,高电平保持4.5ms,而我们判断接收的数据是逻辑0还是逻辑1,或者是同步码,都是要根据高电平的持续时间来判定的,所以要关心高电平保持时间,故定时器四初始化时要配置为上升沿捕获,好了,定时器也设置好了,接下来该设置定时器四的中断处理函数啦

对啦,要先把NVIC中的TIM4中断打开

void nvic_init()

{

 NVIC->ISER[0]|=1<<30;//TIM4的中断编号为30

}

void TIM4_IRQHandler(void)

{

 if(TIM4->SR&0X10)

 {

  if(CS==1)//发生上升沿捕获

  {

   TIM4->CNT=0;

   TIM4->CCER|=1<<13;

   TIM4->SR=0;

   dcb=1;

  }

  if(CS==0)//发生下降沿捕获

  {

     TIM4->SR=0;

     if(dcb==1)  

    {

  dcb=0;

  temp=TIM4->CCR4;

  TIM4->CCER&=~(1<<13);//改为上升沿捕获

  if(temp>4200&&temp<4700)

  {

   led1=~led1;

   OK1=1;

  }

  if(OK1==1)

  { 

  if(temp>1400&&temp<1800)

  {

data=(data<<1);

data|=1<<0;

ray_flag++;

  }

  if(temp>300&&temp<800)

  {

data=(data<<1);

data&=~(1<<0);

ray_flag++;

  }

  

  }

   if(ray_flag==32)

 {

    OK2=1;

led0=~led0;

  }

}

   }

 }

 }

  

   

 if(ray_flag>=32)//NEC协议 一次发送的数据位为32位

    OK2=1;

     TIM4->SR=0;

  }

   }

 }

 }

中断服务程序配置好了,接下来就是中程序啦

int main()

{

 Stm32_Clock_Init(9);

 delay_init(72);

 gpio_init();

 nvic_init();

 time4_init();

 usart1_init();

 while(1)

 {

  if(OK1==1&&OK2==1)

  {

   usart1_senddata(temp);

   OK1=0;

   OK2=0;

   ray_flag=0;

  }

 }

使用的是串口打印数据,串口配置程序就不写啦

最大的教训就是 if(temp>4200&&temp<4700)和if(4200

猜你喜欢

转载自blog.csdn.net/weixin_42314225/article/details/82657253