51单片机红外接收端的程序怎么写,我想知道写的方法和原理,最好有一个具体的模版,好让我参考.

2025-04-12 17:30:16
推荐回答(3个)
回答1:

这是采用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
;;***************************************************

回答2:

遥控器解码非常简单,我给你个参考程序,你直接下载我上传的附近程序,直接用你也可以研究研究,看看我的设计程序思路!


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

*******   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();  

}


回答3:

额,这个程序我这倒是有,你要不?