历史上的今天
返回首页

历史上的今天

今天是:2025年01月25日(星期六)

2018年01月25日 | 红外遥控器软件解码原理和程序

2018-01-25 来源:eefocus

红外一开始发送一段13。5ms的引导码,引导码由9ms的高电平和4。5ms的低电平组成,跟着
引导码是系统码,系统反码,按键码,按键反码,如果按着键不放,则遥控器则发送一段重
复码,重复码由9ms的高电平,2。25ms的低电平,跟着是一个短脉冲,本程序是免费给大
家,版权所有,不得用于商业目的,如需用到本程序到商业上请与本人联系
jiang_xi_sheng@163.com,经本人同意后方可用于商业目的,本程序经过试用,能解大部分


遥控器的编码!

#include    "at89x52.h"
#define     NULL       0x00//数据无效
#define     RESET      0X01//程序复位
#define     REQUEST    0X02//请求信号
#define     ACK        0x03//应答信号,在接收数据后发送ACK信号表示数据接收正确,
也位请求信号的应答信号
#define     NACK       0x04//应答信号,表示接收数据错误
#define     BUSY       0x05//忙信号,表示正在忙
#define     FREE       0x06//空闲信号,表示处于空闲状态
#define     READ_IR    0x0b//读取红外
#define     STORE_IR   0x0c//保存数据
#define     READ_KEY   0x0d//读取键值
#define     RECEIVE    0Xf400//接收缓冲开始地址
#define     SEND       0xfa00//发送缓冲开始地址
#define     IR         0x50//红外接收缓冲开始地址
#define     HEAD       0xaa//数据帧头
#define     TAIL       0x55//数据帧尾
#define     SDA        P1_7
#define     SCL        P1_6



unsigned char xdata *buf1;  //接受数据缓冲
unsigned int  buf1_length;  //接收到的数据实际长度
unsigned char xdata *buf2;  //发送数据缓冲
unsigned int  buf2_length;  //要发送的数据实际长度
bit buf1_flag;    //接收标志,1表示接受到一个数据帧,0表示没有接受到数据帧或数据
帧为空
bit buf2_flag;    //发送标志,1表示需要发送或没发送完毕,0表示没有要发送的数据或
发送完毕
unsigned char state1,state2;         //用来标志接收字符的状态,state1用来表示接
收状态,state2用来表示发送状态
unsigned char data *ir;
union{
    unsigned char a[2];
    unsigned int b;
    unsigned char data *p1[2];
    unsigned int data *p2[2];
    unsigned char xdata *p3;    //红外缓冲的指针
    unsigned int xdata *p4;
}p;
//union{                       //
  //  unsigned char a[2];           //
  //  unsigned int b;
  //  unsigned char data *p1[2];
  //  unsigned int data *p2[2];
  //  unsigned char xdata *p3;
  //  unsigned int xdata *p4;       //地址指针
//}q;                        //

union{
   unsigned char a[2];
   unsigned int b;
}count;
union{
   unsigned char a[2];
   unsigned int b;
}temp;
union{
   unsigned char a[4];
   unsigned int b[2];
   unsigned long c;
}ir_code;

union{
   unsigned char a[4];
   unsigned int b[2];
   unsigned long c;
   unsigned char data *p1[4];
   unsigned int data *p2[4];
   unsigned char xdata *p3[2];
   unsigned int xdata *p4[2];
}i;
unsigned char ir_key;
bit ir_flag;        //红外接收标志,0为缓冲区空,1为接收成功,2为缓冲溢出
void sub(void);
void delay(void);
void ie_0(void);
void tf_0(void);
void ie_1(void);
void tf_1(void);
void tf_2(void);
void read_ir(void);
void ir_jiema(void);
void ir_init(void);
void ir_exit(void);
void store_ir(void);
void read_key(void);
void reset_iIC(void);
unsigned char read_byte_ack_iic(void);
unsigned char read_byte_nack_iic(void);
bit write_byte_iic(unsigned char a);
void send_ack_iic(void);
void send_nack_iic(void);
bit receive_ack_iic(void);
void start_iic(void);
void stop_iic(void);
void write_key_data(unsigned char a);
unsigned int read_key_data(unsigned char a);
void ie0(void)   interrupt 0{ie_0();}
void tf0(void)   interrupt 1{tf_0();}
void ie1(void)   interrupt 2{ie_1();}
void tf1(void)   interrupt 3{tf_1();tf_2();}
void tf2(void)   interrupt 5{            //采用中断方式跟查询方式相结合的办法解

   EA=0;                                 //禁止中断
   if(TF2){                              //判断是否是溢出还是电平变化产生的中断
        TF2=0;                           //如果是溢出产生的中断则清除溢出位,重
新开放中断退出
        EA=1;
        goto end;
    }
   EXF2=0;                               //清除电平变化产生的中断位
   *ir=RCAP2H;                            //把捕捉的数保存起来
   ir++;
   *ir=RCAP2L;
   *ir++;
   F0=1;
   TR0=1;                                 //开启计数器0
loop:
   TL0=0;  //将计数器0重新置为零
   TH0=0;
   while(!EXF2){                         //查询等待EXF2变为1
        if(TF0)goto exit;                //检查有没超时,如果超时则退出
   };
   EXF2=0;                               //将EXF2清零
   if(!TH0)                            //判断是否是长低电平脉冲过来了
   {                                     //不是长低电平脉冲而是短低电平
      if(F0)count.b++;                      //短脉冲数加一
      temp.a[0]=RCAP2H;                  //将捕捉数临时存放起来
      temp.a[1]=RCAP2L;
      goto loop;                         //返回继续查询
   }
   else{                                 //是低电平脉冲,则进行处理

       F0=0;
       *ir=temp.a[0];       //把连续的短脉冲总时间记录下来
       ir++;
       *ir=temp.a[1];
       ir++;
       *ir=RCAP2H;          //把长电平脉冲时间记录下来
       ir++;
       *ir=RCAP2L;
       ir++;
       if(ir>=0xda) {
                 goto exit;    //判断是否溢出缓冲,如果溢出则失败退出
       }
       goto loop;         //返回继续查询
       }
exit:
       ir_flag=1;       //置ir_flag为1表示接收成功
end:
       ;
}


void rs232(void)   interrupt  4{
     static unsigned char sbuf1,sbuf2,rsbuf1,rsbuf2;      //sbuf1,sbuf2用来接收
发送临时用,rsbuf1,rsbuf2用来分别用来存放接收发送的半字节
     EA=0;                                         //禁止中断
     if(RI){
         RI=0;                                     //清除接收中断标志位
         sbuf1=SBUF;                               //将接收缓冲的字符复制到sbuf1
         if(sbuf1==HEAD){                                  //判断是否帧开头
                         state1=10;                 //是则把state赋值为10
                         buf1=RECEIVE;              //初始化接收地
址                        
         }
         else{
         switch(state1){
         case 10:sbuf2=sbuf1>>4;                   //把高半字节右移到的半字节
                 sbuf2=~sbuf2;                     //把低半字节取反
                 if((sbuf2&0x0f)!=(sbuf1&0x0f))    //判断接收是否正确
                      {                            //接收错误,有可能接收的是数
据帧尾,也有可能是接收错误
                       if(sbuf1==TAIL)             //判断是否接收到数据帧尾
                            {                      //是接收到数据帧尾
                                buf1=RECEIVE;      //初始化接收的地址
                                if(*buf1==RESET)   //判断是否为复位命令
                                     {
                                        ES=0;
                                        sbuf2=SP+1;
                                        for(p.p1[0]=SP-0x10;p.p1[0]<=sbuf2;p.p1
[0]++)*p.p1[0]=0;
                                     }
                                state1=0;          //将接收状态标志置为零,接收
下一个数据帧
                                buf1_flag=1;       //置接收标志为1,表示已经接收
到一个数据帧
                                REN=0;             //禁止接收
                            }
                       else
                           {                       //不是接受到数据帧尾,表明接
收错误
                               state1=0;           // 将接收状态标志置为零,重新
接收
                               buf1=RECEIVE;       //初始化发送的地址
                               *buf1=NACK;         //把NACK信号存入接收缓冲里
                               buf1_flag=1;        //置标志位为1,使主程序能对接
收错误进行处理
                               REN=0;              //禁止接收
                           }

                      }
                 else
                 {                                 //接收正确
                     rsbuf1=~sbuf1;                //按位取反,使高半字节变原码
                     rsbuf1&=0xf0;                 //仅保留高半字节,低半字节去

                     state1=20;                    //将状态标志置为20,准备接收
低半字节
                 }
                 break;
         case 20:sbuf2=sbuf1>>4;                   //把高半字节右移到的半字节
                 sbuf2=~sbuf2;                     //将低半字节取反
                 if((sbuf2&0x0f)!=(sbuf1&0x0f))    //判断接收是否正确
                    {                              //接受错误
                        state1=0;                  // 将接收状态标志置为零,重新
接收
                        buf1=RECEIVE;              //初始化接收的地址
                        *buf1=NACK;                //把NACK信号存入发送缓冲里
                        buf1_flag=1;               //置标志位为1,使主程序能对接
收错误进行处理
                        REN=0;                     //禁止接收
                    }
                 else
                    {
                    sbuf1&=0x0f;                   //仅保留低半字节,去掉高半字

                    rsbuf1|=sbuf1;                 //高低半字节合并
                    *buf1++=rsbuf1;                //将接收的数据保存至接收缓冲
里,并且数据指针加一
                    buf1_length++;                 //接收数据长度加一
                    state1=10;                     //将state1置为10,准备接收下
个字节的高半字节
                    }
                 break;

         }
         }



     }
  else{

       TI=0;                                       //清除发送中断标志
       if(buf2_length)                             //判断发送长度是否为零
               {                                   //发送长度不为零
               if(state2==0)                       //判断是否发送高半字节
                   {                               //发送高半字节
                       sbuf2=*buf2;                //将要发送的字节送到sbuf2
                       rsbuf2=~sbuf2;              //取反,使高半字节变为反码
                       sbuf2>>=4;                  //将高半字节右移到低半字节
                       rsbuf2&=0xf0;               //保留高半字节,去掉低半字节
                       sbuf2&=0x0f;                //保留低半字节,去掉高半字节
                       rsbuf2|=sbuf2;              //合并高低半字节
                       SBUF=rsbuf2;                //发送出去
                       state2=10;                  //将state2置为10准备发送下半
字节
                    }
                else
                    {                              //发送低半字节
                       sbuf2=*buf2;                //将要发送的字节送到sbuf2
                       buf2++;                     //指针加一
                       buf2_length--;              //发送数据长度减一
                       rsbuf2=~sbuf2;              //取反,使低半字节变为反码
                       rsbuf2<<=4;                 //将低半字节反码左移到高半字

                       rsbuf2&=0xf0;               //保留高半字节,去掉低半字节
                       sbuf2&=0x0f;                //保留低半字节,去掉高半字节
                       rsbuf2|=sbuf2;              //合并高低半字节
                       SBUF=rsbuf2;                //发送出
                       state2=0;
                     }
                }
         else
                {                                  //如果发送数据长度为零则发送
数据帧尾
                    if(buf2_flag){                 //判断是否发过数据帧尾
                    SBUF=TAIL;                     //将数据帧尾发送出去
                    while(TI==0);
                    TI=0;
                    buf2_flag=0;                   //置发送标志为零,表示发送完

                    }
                }
  }
  EA=1;                                             //开放中断
}


推荐阅读

史海拾趣

EBG RESISTORS LLC公司的发展小趣事

为了进一步扩大市场份额,EBG RESISTORS LLC公司开始实施国际化战略。他们积极参加国际电子展会,与全球各地的客户建立联系。同时,公司还积极寻求与国际知名企业的合作,共同开发新产品。在XXXX年,公司成功打入欧洲市场,并在随后几年内陆续进入亚洲和北美市场。

Amprobe公司的发展小趣事

作为一家有社会责任感的企业,EBG RESISTORS LLC公司积极履行环保责任。他们采用环保材料和生产工艺,减少生产过程中的废弃物排放。同时,公司还积极参与公益活动,回馈社会。这种负责任的态度赢得了客户和社会的广泛赞誉。

Communications & Power Industries公司的发展小趣事

随着全球化进程的加速,CPI意识到要想在电子行业中取得更大的发展,必须积极拓展国际市场。公司积极寻求与国际知名企业的合作机会,通过技术共享和市场合作,成功打开了多个海外市场。同时,CPI还加强了与国际研究机构的合作,不断引进先进的技术和管理经验,提升了自身的竞争力。这些国际合作不仅为CPI带来了更多的商业机会,也推动了公司在全球范围内的品牌传播和市场拓展。

Heidenhain Corp公司的发展小趣事

随着电子行业的快速发展,对高精度电子元件的需求日益增长。海德汉公司凭借其iTNC系列数控系统的高精度控制能力,成为众多电子元件制造商的首选合作伙伴。某知名电子元件生产商在引入海德汉iTNC 530数控系统后,成功实现了微型电子元件的精密加工,将加工精度提升至微米级,大大提高了产品的质量和市场竞争力。这一合作不仅巩固了海德汉在高精度加工领域的领先地位,也推动了电子行业向更高精度、更高质量方向发展。

Boundary Devices公司的发展小趣事

Boundary Devices公司成立于2003年,总部位于美国亚利桑那州。创立之初,公司便专注于嵌入式系统硬件的开发与生产,凭借对技术的深刻理解和市场需求的敏锐洞察,Boundary Devices迅速在行业中崭露头角。其推出的Boundary Devices插座和适配器,因其兼容性强、性能稳定而备受市场青睐。同时,公司与NXP/Freescale建立了紧密的合作关系,共同推进i.MX系列处理器的应用与发展,为公司的后续发展奠定了坚实的基础。

Altmustech公司的发展小趣事

为了进一步扩大市场份额,Altmustech积极寻求与行业内外的战略合作。公司与多家知名企业签订了长期合作协议,共同研发新产品,开拓新市场。通过这些合作,Altmustech不仅获得了更多的资源支持,还提高了品牌知名度,进一步巩固了其在电子行业的地位。

问答坊 | AI 解惑

谁能帮我找到10102A 芯片的PDF

谁能帮我找到10102A 芯片的PDF,非常感谢!请联系我QQ:645652168…

查看全部问答>

CDMA模块

CDMA模块基于CDMA平台的通信模块,它将通信芯片、存储芯片等集成在一块电路板上,使其具有发送通过CDMA平台收发短消息、语音通话、数据传输等功能。CDMA模块可以实现普通CDMA手机的主要通信功能,也可以说是一个“精简版”的手机。电脑、单片机、AR ...…

查看全部问答>

在6F877,18F1320,18F1220上通过的18B20程序

这是在6F877,18F1320,18F1220上通过的18B20程序,18B20主要是延时问题,这个解决了,什么都可以通过。 # include <pic18f1220.h> # define uch unsigned char # define unint unsigned int # define DQ        R ...…

查看全部问答>

编译uclinux内核时候遇到的问题,帮我看看吧。

由于毕业设计需要和本身兴趣,要构造ARM+uclinux,本机Linux版本:UBUNTU6 .06. 在本机上安装了www.uclinux.org上下的arm-elf-tools, 然后arm-elf-gcc-v,显示如下 Reading specs from /usr/local/lib/gcc-lib/arm-elf/2.95.3/specs gcc ver ...…

查看全部问答>

LED照明灯饰技术发展特点剖析

replyreload += \',\' + 746877;Timson,如果您要查看本帖隐藏内容请回复…

查看全部问答>

怎样程序烧写到指定的flash地址里?

在项目中遇到一个问题:需要将一个变量放到一个指定的flash地址里,即flash的绝对地址中。我查看了IAR的complier手册,上面的例子说以下面的方式定义即可。const int bootKey @ 0x0801FFFC = 3; /* O ...…

查看全部问答>

关于选折NANDFLASH,还是SPIFLASH?

所用STM32F103ZET6, 由于,512K Flash, 64 RAM,已经能够满足系统的需要。 所以就不加NORFLASH, 和SRAM了, 但是由于要存储数据,所以就想到NandFlash,和SPI Flash. 而NandFlash容量大,但是引脚多,(NAND128W3A, 128M) SPI Flash容 ...…

查看全部问答>

谁来做一次9B96的使用教程?

期待EEWORLD和TI能够为我们这些菜鸟提供一些学习的方法和使用介绍!我想以后使用这个9B96但是这个入门貌似还不懂!…

查看全部问答>

G2231没有定时器B吗?

如题啊,突然发现没有TB,那只有WD和TA这2个定时器了???…

查看全部问答>

根据LM3S811评估板画的扩展板,求意见

上次在论坛秒杀了LM3S811评估板,点亮了灯,试了试按键,就再也没动过了,最近刚开学没事,就根据LM3S811评估板画了一个扩展板,含有LED、按键、24C02、串口、1602、OLED、诺基亚5110屏、模拟比较器、DS18B20、ADC,希望大家能提点意见来。 有兴趣 ...…

查看全部问答>