您当前的位置:五五电子网电子知识单片机-工控设备源码-程序单片机解码万能红外遥控器的C51程序 正文
单片机解码万能红外遥控器的C51程序

单片机解码万能红外遥控器的C51程序

点击数:7550 次   录入时间:03-04 11:39:48   整理:http://www.55dianzi.com   源码-程序

    使用方法:打开串口调试助手,设置为9600 bps 单片机这边用11.0592MHz的晶振,使用sm0038或者其他型号的红外接收头按下面的电路连接好,其中out直接与单片机的p3.2脚相连.按下遥控器,串口调试助手便会出现解码值.

点击浏览下一页

    /******************************************************************/

    /* 本程序的蓝本从网上搜集,经修改并注释,万能遥控器解码成功 */

    /* 晶振:11.0592MHz */

    /* 整理与测试:单片机教程网 http://www.51hei.com 胡琴 2012.5.15 */

    /************************* 说 明 *********************************/

    /* 以一个9ms的低电平和4.5ms的高电平为引导码,后跟32位二进制代码 */

    /* 前16位为8位用户码及其反码,后16位为8位的操作码及其反码 */

    /* 以脉宽为0.565ms、间隔0.56ms、周期为1.125ms的组合表示"0"; */

    /* 以脉宽为0.565ms、间隔1.685ms、周期为2.25ms的组合表示"1"。 */

    /* 注意:接收码的脉宽与间隔是对发射码取反的,即间隔是0.565ms */

    /* 解码后共有四个十六进制码,本程序取第三个作为识别码 */

    /*******************************************************************/

    #include

    #define uchar unsigned char

    uchar data IRcode[4]; //定义一个4字节的数组用来存储代码

    uchar CodeTemp; //编码字节缓存变量

    uchar i,j,k; //延时用的循环变量

    sbit IRsignal=P3^2; //HS0038接收头OUT端直接连P3.2(INT0)

    sbit P0_0=P0^0; //P0连接到 LED 上

    sbit P0_1=P0^1;

    sbit P0_2=P0^2;

    /**************************延时0.9ms子程序**********************/

    void Delay0_9ms(void)

    {

    uchar j,k;

    for(j=18;j>0;j--)

    for(k=20;k>0;k--)

    ;

    }

    /***************************延时1ms子程序**********************/

    void Delay1ms(void)

    {

    uchar i,j;

    for(i=2;i>0;i--)

    for(j=230;j>0;j--)

    ;

    }

    /***************************延时4.5ms子程序**********************/

    void Delay4_5ms(void)

    {

    uchar i,j;

    for(i=10;i>0;i--)

    for(j=225;j>0;j--)

    ;

    }

    /**************************** 延时子程序 ************************/

    void Delay(void)

    {

    uchar i,j,k;

    for(i=200;i>0;i--)

    for(j=200;j>0;j--)

    for(k=3;k>0;k--)

    ;

    }

    /******************** 中断0解码服务子程序 ********************/

    void int0(void) interrupt 0 using 2

    {

    EA = 0; //??? 可以这样,跳入中断,但仍可对P3.2(INT0)进行电平变化的读取

    for(k=0;k<10;k++)

    {

    Delay0_9ms();

    if (IRsignal==1) //如果0.9ms后IRsignal=1,说明不是引导码

    {

    k=10;

    break;

    }

    else if(k==9) //如果 持续了10×0.9ms=9ms的低电平,说明是引导码

    {

    while(IRsignal==0);

    Delay4_5ms(); //跳过持续4.5ms的高电平

    for(i=0;i<4;i++) //分别读取4个字节

    {

    for(j=1;j<=8;j++) //每个字节8个bit的判断

    {

    while(IRsignal==0); //等待上升沿 此处用得很好:因为0.56ms的低电平(接收时)是代码0与1的相同部分

    Delay0_9ms(); //从上升沿那一时刻开始延时0.9ms(因为0.9介于0.56(=1.125-0.56)与1.69(=2.25-0.56)之间),再判断IRsignal

    if(IRsignal==1) //如果IRsignal是"1",高位置"1",并向右移一位

    {

    Delay1ms(); //为什么要延时1ms呢?因为要使IRsignal跳至低电平(即0.56ms的0与1相同部分上)

    CodeTemp=CodeTemp|0x80; //此处的算法很好

    if(j<8) CodeTemp=CodeTemp>>1;

    }

    else

    if(j<8)

    CodeTemp=CodeTemp>>1;//如果IRsignal是"0",则直接向右移一位,自动补"0"

    }

    IRcode=CodeTemp;

    CodeTemp=0;

    }

    for(i=0;i<4;i++) //通过串口将代码发出

    {

    SBUF=IRcode;

    while(!TI); //等待一个字节发送完毕

    TI=0;

    }

    Delay();

    }

    }

    EA = 1;

    }

    /***********************串口初始化程序*********************/

    void initUART(void)

    {

    TMOD |= 0x20; //

    SCON = 0x50; //

    PCON |= 0x80; //

    TH1 = 250; // 9600 bps @ 11.0592MHz

    TL1 = 250;

    TR1 = 1;

    }

    /**************************主程序*************************/

    void main()

    {

    P0=0XFF;

    initUart();

    IT0 = 1; //INT0为负边沿触发, (1:负边沿触发,0:低电平触发)

    EX0 = 1; //外部中断INT0开, (1:开, 0:关 )

    EA = 1; //开所有中断

    CodeTemp = 0; //初始化红外编码字节缓存变量

    Delay();

    while(1)

    {

    switch(IRcode[2])

    {

    case 0x42:P0=0XFF;P0_0=0;break;

    case 0x4e:P0=0XFF;P0_1=0;break;

    case 0x52:P0=0XFF;P0_2=0;break;

    }

    }

    }




本文关键字:单片机  程序  遥控器  源码-程序单片机-工控设备 - 源码-程序