这是采用STC12C5A60S2单片机的红外解码程序及其说明。
;采用脉宽调制的串行码,以脉宽为0.565ms、间隔0.56ms、周期为1.125ms的组合表示二进制的"0";
;以脉宽为0.565ms、间隔1.685ms、周期为2.25ms的组合表示二进制的"1
;上述"0"和"1"组成的32位二进制码经38kHz的载频进行二次调制以提高发射效率,
;达到降低电源功耗的目的。然后再通过红外发射二极管产生红外线向空间发射
;遥控编码是连续的32位二进制码组,其中前16位为用户识别码,能区别不同的电器设备,
;防止不同机种遥控码互相干扰。该芯片的用户识别码固定为十六进制01H
;后16位为8位操作码(功能码)及其反码。
;当一个键按下超过36ms,振荡器使芯片激活,将发射一组108ms的编码脉冲,这108ms发射代码由一个起始码(9ms),
;一个结果码(4.5ms),低8位地址码(9ms~18ms),高8位地址码(9ms~18ms),8位数据码(9ms~18ms)
;和这8位数据的反码(9ms~18ms)组成。如果键按下超过108ms仍未松开,
;接下来发射的代码(连发代码)将仅由起始码(9ms)和结束码(2.5ms)组成。
;
;解码的关键是如何识别"0"和"1",接收端而言,"0"是0.56ms的高+0.56ms的低。"1"是1.68ms的高+0.56ms的低。
;所以可以根据高电平的宽度区别"0"和"1"。当高电平出现时开始延时,0.56ms以后,若读到的电平为低,
;说明该位为"0",反之则为"1",为了可靠起见,延时必须比0.56ms长些,但又不能超过1.12ms,否则如果该位为"0",
;读到的已是下一位的高电平,因此取(1.12ms+0.56ms)/2=0.84ms最为可靠,一般取0.84ms左右均可。
;为了共用引导部分延时程序,这里用0.9ms延时。
;-------------红外解码程序---------------------------
EXINT0:
PUSH ACC
PUSH PSW
PUSH 1
PUSH 2
PUSH 6
CLR EA ;暂时关闭中断请求
MOV R6,#10
EXINT10:
LCALL DELAY09MS ;调用900us延时子程序
JB IRIN,INTOUT1 ;判断P3.2是否有高电平,如果有就退出解码程序
DJNZ R6,EXINT10 ;循环10次,检测在900微妙中是否存在高电平。以上完成对遥控信号的9000微秒的初始低电平信号的识别。
JNB IRIN,$ ;等待高电平避开9毫秒低电平引导脉冲
LCALL DELAY45MS ;延时4.5毫秒
;-------------接受32位代码--------------------------
MOV R1,#IRUSERL
MOV R2,#04H
EXINT101:
MOV R6,#08H ;每组数据位8位
EXINT102:
JNB IRIN,$ ;等待地址码第一组数据的高电平信号
LCALL DELAY09MS ;高电平开始后延时判断信号此时的高/低状态
MOV C,IRIN ;将P3.2引脚此时的电平状态0或1存入C中
JNC INT1OUT ;如果为0跳出
LCALL DELAY1MS
INT1OUT:
MOV A,@R1
RRC A ;将C中的数据0/1移入A中最低位
MOV @R1,A ;将A中的数据暂存在R1
DJNZ R6,EXINT102 ;接受完8位代码
INC R1
DJNZ R2,EXINT101 ;接受完4组32位代码
;--------------数据码比较-------------------------------
MOV A,IRDATAL
; LCALL SENDRXDAT
MOV A,IRDATAL
CPL A
CJNE A,IRDATAH,INTOUT1 ;判断数码正误,不等退出
MOV IR_DAT,IRDATAL ;相等则保存正确数据
MOV A,IR_DAT
; LCALL SENDRXDAT
SETB IRBIT
INTOUT1:
LCALL DELAY45MS
SETB EA ;允许中断
POP 6
POP 2
POP 1
POP PSW
POP ACC
RETI
;;*****************11.0592*900=9953******************
DELAY09MS: ;6
PUSH 4 ;4
PUSH 3 ;4
MOV R4,#20 ;2
DLY900:
MOV R3,#122 ;2
DJNZ R3,$ ;4
DJNZ R4,DLY900 ;4
MOV R4,#11 ;2
DJNZ R4,$ ;4
POP 3 ;3
POP 4 ;3
RET ;4
;TOTAL=9952
;;*****************11.0592*560=6193******************
DELAY056: ;6
PUSH 4 ;4
PUSH 3 ;4
MOV R4,#12 ;2
DLY5600:
MOV R3,#122 ;2
DJNZ R3,$ ;4
DJNZ R4,DLY5600 ;4
MOV R4,#71 ;2
DJNZ R4,$ ;4
POP 3 ;3
POP 4 ;3
RET ;4
;TOTAL=6194
;;*****************11.0592*4500=49766****************
DELAY45MS: ;6
PUSH 4 ;4
PUSH 3 ;4
MOV R4,#52 ;2
DLY45:
MOV R3,#236 ;2
DJNZ R3,$ ;4
DJNZ R4,DLY45 ;4
MOV R4,#85 ;2
DJNZ R4,$ ;4
POP 3 ;3
POP 4 ;3
RET ;4
;;TOTAL=49768
;;*****************11.0592*1000=11059****************
DELAY1MS: ;6
PUSH 4 ;4
PUSH 3 ;4
MOV R4,#20 ;2
DLY1MS:
MOV R3,#136 ;2
DJNZ R3,$ ;4
DJNZ R4,DLY1MS ;4
MOV R4,#8 ;2
DJNZ R4,$ ;4
POP 3 ;3
POP 4 ;3
RET ;4
;TOTAL=11060
;;***************************************************
DELAY100US: ;6
PUSH 4 ;4
MOV R4,#140 ;2
DJNZ R4,$ ;4
MOV R4,#131 ;2
DJNZ R4,$ ;4
POP 4 ;3
RET ;4
;TOTAL=1105
;;***************************************************
遥控器解码非常简单,我给你个参考程序,你直接下载我上传的附近程序,直接用你也可以研究研究,看看我的设计程序思路!
/***************************************************************************
******* NEC型遥控器协议解码
***************************************************************************/
#include "STC89C5X.H"
#include "NEC_IR.H"
#include "TYPEDEF.H"
/*
* 宏定义红外端口
*/
#define NEC_IRinput P3_2 //红外数据输入端口
/*
* 宏定义蜂鸣器端口
*/
#define NEC_BEEP P2_3 //蜂鸣器数据输入端口
/*
* 变量
*/
INT8U CAIy_Byte; //计算接收码变量
INT16U CAIy_Data; //接收码暂存变量
INT8U NEC_IRBUF[4]; //接收码保存数组
/*
* 初始化T0定时器
*/
void NEC_SystemInit(void)
{
// T0/INT0中断配置
TMOD = 0x01;
//
EX0 = 1;
//
// ET0 = 1;
IT0 = 1;
// IE0 = 1;
// 开启总中断
EA = 1;
}
/*
* NEC遥控器按键音子函数
*/
void NEC_BeepByte(void)
{
unsigned char i;
unsigned int t = 2000;
for(i = 200; i > 0; i--)
{
NEC_BEEP = 0;
}
while(t > 0)t--;
NEC_BEEP = 1;
}
/*
* 采样红外触发外部中断低电平数据
*/
unsigned int INT0_INTERRUPT_LSB()
{
TL0 = 0;
TH0 = 0;
TR0 = 1;
while(~NEC_IRinput && (TH0 & 0x80) == 0);
TR0 = 0;
return (TH0 * 256 + TL0);
}
/*
* 采样红外没有触发外部中断高电平数据
*/
unsigned int INT0_INTERRUPT_MSB()
{
TL0 = 0;
TH0 = 0;
TR0 = 1;
while(NEC_IRinput && (TH0 & 0x80) == 0);
TR0 = 0;
return (TH0 * 256 + TL0);
}
/*
* INT0外部中断0服务程序
*/
void INT0_IRQ(void) interrupt 0
{
unsigned char i;
unsigned char k = 3;
// unsigned char NEC_CHAN;
/* 拉高红外数据端口准备采样有效数据 */
NEC_IRinput = 1;
/* 接收红外低电平数据采样 */
CAIy_Data = INT0_INTERRUPT_LSB();
/* 判断接收到的红外采样数据是否在9ms引导码范围内 */
if((CAIy_Data < NEC_8500) || (CAIy_Data > NEC_9500))
{
/* 如果超出或者低于引导码低电平9ms程序直接跳出 */
return;
}
/* 清除采样变量 */
CAIy_Data &= 0x00;
/* 接收红外高电平数据采样 */
CAIy_Data = INT0_INTERRUPT_MSB();
/* 判断接收到的红外采样数据是否在4.5ms引导码范围内 */
if((CAIy_Data < NEC_4300) || (CAIy_Data > NEC_4600))
{
/* 如果超出或者低于引导码高电平4.5ms程序直接跳出 */
return;
}
/*
* NEC引导接收正确后,进入8用户码和8按键码接收
*/
for(i = 0; i < 32; i++)
{
CAIy_Byte >>= 1; /* 接收数据位左移一位 */
/* 接收红外低电平数据采样 */
CAIy_Data = INT0_INTERRUPT_LSB();
/* 判断低电平是否在NEC协议0.56ms范围以内 */
if((CAIy_Data < NEC_200) || (CAIy_Data > NEC_800))
{
/* 如果超出NEC协议0.56ms低电平直接跳出接收 */
return;
}
/* 接收红外高电平数据采样 */
CAIy_Data = INT0_INTERRUPT_MSB();
/* 判断高电平是否在NEC协议1.68ms范围以内 */
if((CAIy_Data < NEC_200) || (CAIy_Data > NEC_2000))
{
/* 如果超出NEC协议1.68ms低电平直接跳出接收 */
return;
}
/* 判断高电平是否在NEC协议1.68范围内 */
if(CAIy_Data >= NEC_1120)
{
/* 如果符合NEC协议1.68ms为高电平1变量高位制1 */
CAIy_Byte |= 0x80;
}
/* 用来区分8位用户正码、8位用户反码、8位按键正码、8位按键反码 */
if(i == 7 || i == 15 || i == 23 || i == 31)
{
k++;
if(k == 4)
{
k = 0;
}
NEC_IRBUF[k] = CAIy_Byte; // 把接收到的数据存放到数组
CAIy_Byte = 0x00; // 清除接收数据暂存变量
}
}
/* 判断NEC协议用户码是否正确 */
if((NEC_IRBUF[0] != 0x00) || (NEC_IRBUF[0] != ~NEC_IRBUF[1]))
{
/* 如果NEC协议用户码不对程序直接跳出 */
return;
}
/* 把接收到的十进制数据转换成十六进制数据 */
/* NEC_CHAN = NEC_IRBUF[2] / 16;
NEC_IRBUF[2] = NEC_IRBUF[2] % 16;
NEC_IRBUF[2] = NEC_IRBUF[2] + NEC_CHAN * 10; */
/* NEC遥控器按键音 */
NEC_BeepByte();
}
额,这个程序我这倒是有,你要不?