1、红外线遥控器软件解码原理和程序(C 语言) UPD6121G 产生的遥控编码是连续的 32 位二进制码组,其中前 16 位为用户识别码,能区别不同的电器设备,防止不同机种遥控码互相干扰。该芯片的用户识别码固定为十六进制 01H;后 16 位为 8 位操作码(功能码)及其反码。UPD6121G 最多额 128 种不同组合的编码。遥控器在按键按下后,周期性地发出同一种 32 位二进制码,周期约为108ms。一组码本身的持续时间随它包含的二进制“0”和“1”的个数不同而不同,大约在 4563ms 之间,图 4 为发射波形图。当一个键按下超过 36ms,振荡器使芯片激活,将发射一组 108ms 的编码
2、脉冲,这 108ms 发射代码由一个起始码(9ms),一个结果码(4.5ms),低 8 位地址码(9ms18ms),高 8 位地址码(9ms18ms),8 位数据码(9ms18ms)和这 8位数据的反码(9ms18ms)组成。如果键按下超过 108ms 仍未松开,接下来发射的代码(连发代码)将仅由起始码(9ms)和结束码(2.5ms)组成。代码格式(以接收代码为准,接收代码与发射代码反向)位定义 单发代码格式 连发代码格式 注:代码宽度算法:16 位地址码的最短宽度:1.1216=18ms 16 位地址码的最长宽度:2.24ms16=36ms 易知 8 位数据代码及其 8 位反代码的宽度和不变
3、:(1.12ms+2.24ms)8=27ms所以 32 位代码的宽度为(18ms+27ms)(36ms+27ms)1 解码的关键是如何识别“0”和“1”,从位的定义我们可以发现“0”、“1”均以 0.56ms 的低电平开始,不同的是高电平的宽度不同,“0”为0.56ms,“1”为 1.68ms,所以必须根据高电平的宽度区别“0”和“1”。如果从 0.56ms 低电平过后,开始延时,0.56ms 以后,若读到的电平为低,说明该位为“0”,反之则为“1”,为了可靠起见,延时必须比 0.56ms 长些,但又不能超过 1.12ms,否则如果该位为“0”,读到的已是下一位的高电平,因此取(1.12ms+
4、0.56ms)/2=0.84ms 最为可靠,一般取 0.84ms 左右均可。2 根据码的格式,应该等待 9ms 的起始码和 4.5ms 的结果码完成后才能读码。接收器及解码一体化红外线接收器是一种集红外线接收和放大于一体,不需要任何外接元件,就能完成从红外线接收到输出与 TTL 电平信号兼容的所有工作,而体积和普通的塑封三极管大小一样,它适合于各种红外线遥控和红外线数据传输。下面是一个对 51 实验板配套的红外线遥控器的解码程序,它可以把上图 32 键的红外遥控器每一个按键的键值读出来,并且通过实验板上 P1 口的 8 个 LED 显示出来,在解码成功的同时并且能发出“嘀嘀嘀”的提示音。红外线
5、一开始发送一段 13.5ms 的引导码,引导码由 9ms 的高电平和 4.5ms 的低电平组成,跟着引导码是系统码,系统反码,按键码,按键反码,如果按着键不放,则遥控器则发送一段重复码,重复码由 9ms 的高电平,2.25ms 的低电平,跟着是一个短脉冲。本程序经过试用,能解大部分遥控器的编码!#include “at89x52.h“#define NULL 0x00/数据无效#define RESET 0X01/程序复位#define REQUEST 0X02/请求信号#define ACK 0x03/应答信号,在接收数据后发送 ACK 信号表示数据接收正确,也位请求信号的应答信号#defi
6、ne 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 SD
7、A P1_7#define SCL P1_6unsigned 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;
8、/用来标志接收字符的状态,state1 用来表示接收状态,state2 用来表示发送状态unsigned char data *ir;unionunsigned char a2;unsigned int b;unsigned char data *p12;unsigned int data *p22;unsigned char xdata *p3; /红外缓冲的指针unsigned int xdata *p4;p;/union / unsigned char a2; / unsigned int b;/ unsigned char data *p12;/ unsigned int data *
9、p22;/ unsigned char xdata *p3;/ unsigned int xdata *p4; /地址指针/q; /unionunsigned char a2;unsigned int b;count;unionunsigned char a2;unsigned int b;temp;unionunsigned char a4;unsigned int b2;unsigned long c;ir_code;unionunsigned char a4;unsigned int b2;unsigned long c;unsigned char data *p14;unsigned
10、int data *p24;unsigned char xdata *p32;unsigned int xdata *p42;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
11、);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
12、_iic(void);void write_key_data(unsigned char a);unsigned int read_key_data(unsigned char a);void ie0(void) interrupt 0ie_0();void tf0(void) interrupt 1tf_0();void ie1(void) interrupt 2ie_1();void tf1(void) interrupt 3tf_1();tf_2();void tf2(void) interrupt 5 /采用中断方式跟查询方式相结合的办法解码EA=0; /禁止中断if(TF2) /判断
13、是否是溢出还是电平变化产生的中断TF2=0; /如果是溢出产生的中断则清除溢出位,重新开放中断退出EA=1;goto end;EXF2=0; /清除电平变化产生的中断位*ir=RCAP2H; /把捕捉的数保存起来ir+;*ir=RCAP2L;*ir+;F0=1;TR0=1; /开启计数器 0loop:TL0=0; /将计数器 0 重新置为零TH0=0;while(!EXF2) /查询等待 EXF2 变为 1if(TF0)goto exit; /检查有没超时,如果超时则退出;EXF2=0; /将 EXF2 清零if(!TH0) /判断是否是长低电平脉冲过来了 /不是长低电平脉冲而是短低电平if(
14、F0)count.b+; /短脉冲数加一temp.a0=RCAP2H; /将捕捉数临时存放起来temp.a1=RCAP2L;goto loop; /返回继续查询else /是低电平脉冲,则进行处理F0=0;*ir=temp.a0; /把连续的短脉冲总时间记录下来ir+;*ir=temp.a1;ir+;*ir=RCAP2H; /把长电平脉冲时间记录下来ir+;*ir=RCAP2L;ir+;if(ir=0xda) goto exit; /判断是否溢出缓冲,如果溢出则失败退出goto loop; /返回继续查询exit:ir_flag=1; /置 ir_flag 为 1 表示接收成功end:;voi
15、d rs232(void) interrupt 4static unsigned char sbuf1,sbuf2,rsbuf1,rsbuf2; /sbuf1,sbuf2 用来接收发送临时用,rsbuf1,rsbuf2 用来分别用来存放接收发送的半字节EA=0; /禁止中断if(RI)RI=0; /清除接收中断标志位sbuf1=SBUF; /将接收缓冲的字符复制到 sbuf1if(sbuf1=HEAD) /判断是否帧开头state1=10; /是则把 state 赋值为 10buf1=RECEIVE; /初始化接收地址 elseswitch(state1)case 10:sbuf2=sbuf1
16、4; /把高半字节右移到的半字节sbuf2=sbuf2; /把低半字节取反if(sbuf2 /初始化接收的地址if(*buf1=RESET) /判断是否为复位命令ES=0;sbuf2=SP+1;for(p.p10=SP-0x10;p.p104; /把高半字节右移到的半字节sbuf2=sbuf2; /将低半字节取反if(sbuf2 / 将接收状态标志置为零,重新接收buf1=RECEIVE; /初始化接收的地址*buf1=NACK; /把 NACK 信号存入发送缓冲里buf1_flag=1; /置标志位为 1,使主程序能对接收错误进行处理REN=0; /禁止接收elsesbuf1 /仅保留低半字
17、节,去掉高半字节rsbuf1|=sbuf1; /高低半字节合并*buf1+=rsbuf1; /将接收的数据保存至接收缓冲里,并且数据指针加一buf1_length+; /接收数据长度加一state1=10; /将 state1 置为 10,准备接收下个字节的高半字节break;elseTI=0; /清除发送中断标志if(buf2_length) /判断发送长度是否为零 /发送长度不为零if(state2=0) /判断是否发送高半字节 /发送高半字节sbuf2=*buf2; /将要发送的字节送到 sbuf2rsbuf2=sbuf2; /取反,使高半字节变为反码sbuf2=4; /将高半字节右移到
18、低半字节rsbuf2 /保留高半字节,去掉低半字节sbuf2 /保留低半字节,去掉高半字节rsbuf2|=sbuf2; /合并高低半字节SBUF=rsbuf2; /发送出去state2=10; /将 state2 置为 10 准备发送下半字节else /发送低半字节sbuf2=*buf2; /将要发送的字节送到 sbuf2buf2+; /指针加一buf2_length-; /发送数据长度减一rsbuf2=sbuf2; /取反,使低半字节变为反码rsbuf2=4; /将低半字节反码左移到高半字节rsbuf2 /保留高半字节,去掉低半字节sbuf2 /保留低半字节,去掉高半字节rsbuf2|=sbuf2; /合并高低半字节SBUF=rsbuf2; /发送出state2=0;else /如果发送数据长度为零则发送数据帧尾if(buf2_flag) /判断是否发过数据帧尾SBUF=TAIL; /将数据帧尾发送出去while(TI=0);TI=0;buf2_flag=0; /置发送标志为零,表示发送完毕EA=1; /开放中断