在单片机应用中,红外遥控以其简单易用而受到欢迎,在一些简单的室内控制中,红外遥控是首选。

    红外遥控就是一种通讯过程,他也有自己的通讯协议,9ms的引导码+32位信息码。

    信息码:8位用户码+8位用户码+8位功能码+8位功能反码。

    以上都是用2进制来表示。

    0---高电平:低电平=1:1,都大概为0.56ms

    1---高电平:低电平=1:3,高电平为0.56,低电平则为1.5ms左右。

    利用一个定时器0中断,方式是8位自动重载方式,目的是利用计数值获得时间基准。相当于是一个手表的作用。因为按照12M晶振,每递增一次,是1us,定时器加到256自动重载并产生中断,则每次产生中断的时间间隔是256us,在中断服务程序中让一个标志位递增。

   再利用外部中断0,每次来个下降沿,就读一下标志位的数值,如果是10,则时间长度是10*0.256=2.56ms,所以每次下降沿就读一下,就可以判断引导码、0与1了。

    在使用一个新的遥控器时,必须首先要解码,并显示出来。利用下面的程序来进行:

   以下为源码:

#include<reg52.h>
#define uchar unsigned char 
#define uint unsigned int
#define DATAPORT P0
uchar irtime;
uchar startflag;
uchar irdata[33];
uchar bitnumber;
uchar irreceiveok;
uchar ircode[4];
uchar irprosok;
uchar disp[8];
sbit LATCH1=P2^2;//定义锁存使能端口 段锁存
sbit LATCH2=P2^3;//                 位锁存
sbit IR=P3^2;  unsigned char code DuanMa[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};
// 显示段码值01234567
unsigned char code WeiMa[]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};
//分别对应相应的数码管点亮,即位码  void time0init(void)//定时器中断初始化
{
    TMOD=0x02;     //方式2,8位自动重载
    TH0=0X00;      //初值
    TL0=0X00;
    ET0=1;         //打开定时器
    EA=1;          //开总中断
    TR0=1;        
}
void int0init(void)//外部中断初始化
{
    IT0=1;     //下降沿
    EX0=1;
    EA=1;    
}
void delay_50ms(uint t)
{
    uint j;
    for(;t>0;t--)
    for(j=100;j>0;j--);
}
void irwork(void)         //把获得的16进制数进行分解,
{
      disp[0]=ircode[0]/16;
    disp[1]=ircode[0]%16;
     disp[2]=ircode[1]/16;
    disp[3]=ircode[1]%16;
    disp[4]=ircode[2]/16;
    disp[5]=ircode[2]%16;
    disp[6]=ircode[3]/16;
    disp[7]=ircode[3]%16;
}
void display(void)
{    
    uchar i;
    for(i=0;i<8;i++)
    {          DATAPORT=WeiMa[i];
        LATCH2=1;
        LATCH2=0;
        DATAPORT=DuanMa[disp[i]];
        LATCH1=1;
        LATCH1=0;
        delay_50ms(2);
    }
}
void  irpros(void)    //提取获得的信息码
{
    uchar k,i,j,value;
    k=1;//不读第一个引导码
    for(j=0;j<4;j++)
    {
        for(i=0;i<8;i++)         //因为每个字节8位 
        {
            value=value>>1;      //右移7次,第一次进来因为都是0,所以移位没有任何影响。
            if(irdata[k]>6)        //判断8次,0为4.4,1为8.8,这里选择6,若大于6,说明是1
            {
                value=value|0x80;  //则或上1000 0000,最高位置1,然后进行右移(因为低位在前)        
            }
            k++;
        }
        ircode[j]=value;       //最终得到码
    }
    irprosok=1;                //解码成功标志位
}
void main()
{
    time0init();
    int0init();
    while(1)
    {
        if(irreceiveok)
        {
            irpros();
            irreceiveok=0;//判断完后,清零
        }
        if(irprosok)
        {
            irwork();
            irprosok=0;
        }
        display();
    }
}
void time0() interrupt 1//定时器中断0的ISR
{
    irtime++;//255
}
void int0() interrupt 0//外部中断0的ISR
{
    if(startflag)    //刚开始进来,检查启动接受标志位,肯定不为1
    {
        if(irtime>32)  //说明检测到引导码
        {
            bitnumber=0;
         }            
        irdata[bitnumber]=irtime; //把每次进来的irtime都存起来,最后再判断 
        irtime=0;
        bitnumber++;           //第几位标志
        if(bitnumber==33)   //遥控器发出高电平,一体化红外接收头接收后产生低电平。
                            //每帧数据为32位,再加上引导码,总共33位
        {
        bitnumber=0;
        irreceiveok=1;         //红外接收完毕
        }
    }
    else
    {
        startflag=1;//如果不为1,则置1
        irtime=0;   //清零
    }
}