红外解码.docx
《红外解码.docx》由会员分享,可在线阅读,更多相关《红外解码.docx(27页珍藏版)》请在冰豆网上搜索。
红外解码
红外线遥控系统原理及软件解码实例
红外线遥控是目前使用最广泛的一种通信和遥控手段。
由于红外线遥控装置具有体积小、功耗低、功能强、成本低等特点,因而,继彩电、录像机之后,在录音机、音响设备、空凋机以及玩具等其它小型电器装置上也纷纷采用红外线遥控。
工业设备中,在高压、辐射、有毒气体、粉尘等环境下,采用红外线遥控不仅完全可靠而且能有效地隔离电气干扰。
1、红外遥控系统
通用红外遥控系统由发射和接收两大部分组成。
应用编/解码专用集成电路芯片来进行控制操作,如图1所示。
发射部分包括键盘矩阵、编码调制、LED红外发送器;接收部分包括光、电转换放大器、解调、解码电路。
图1红外线遥控系统框图
2、遥控发射器及其编码
遥控发射器专用芯片很多,根据编码格式可以分成两大类,这里我们以运用比较广泛,解码比较容易的一类来加以说明,现以日本NEC的uPD6121G组成发射电路为例说明编码原理(一般家庭用的DVD、VCD、音响都使用这种编码方式)。
当发射器按键按下后,即有遥控码发出,所按的键不同遥控编码也不同。
这种遥控码具有以下特征:
采用脉宽调制的串行码,以脉宽为0.565ms、间隔0.56ms、周期为1.125ms的组合表示二进制的“0”;以脉宽为0.565ms、间隔1.685ms、周期为2.25ms的组合表示二进制的“1”,其波形如图2所示。
图2遥控码的“0”和“1”(注:
所有波形为接收端的与发射相反) 上述“0”和“1”组成的32位二进制码经38kHz的载频进行二次调制以提高发射效率,达到降低电源功耗的目的。
然后再通过红外发射二极管产生红外线向空间发射,如图3所示。
图3遥控信号编码波形图
UPD6121G产生的遥控编码是连续的32位二进制码组,其中前16位为用户识别码,能区别不同的电器设备,防止不同机种遥控码互相干扰。
该芯片的用户识别码固定为十六进制01H;后16位为8位操作码(功能码)及其反码。
UPD6121G最多额128种不同组合的编码。
遥控器在按键按下后,周期性地发出同一种32位二进制码,周期约为108ms。
一组码本身的持续时间随它包含的二进制“0”和“1”的个数不同而不同,大约在45~63ms之间,图4为发射波形图。
图4遥控连发信号波形
当一个键按下超过36ms,振荡器使芯片激活,将发射一组108ms的编码脉冲,这108ms发射代码由一个引导码(9ms),一个结果码(4.5ms),低8位地址码(9ms~18ms),高8位地址码(9ms~18ms),8位数据码(9ms~18ms)和这8位数据的反码(9ms~18ms)组成。
如果键按下超过108ms仍未松开,接下来发射的代码(连发码)将仅由起始码(9ms)和结束码(2.25ms)组成。
图5引导码 图6连发码
3、遥控信号接收
接收电路可以使用一种集红外线接收和放大于一体的一体化红外线接收器,不需要任何外接元件,就能完成从红外线接收到输出与TTL电平信号兼容的所有工作,而体积和普通的塑封三极管大小一样,它适合于各种红外线遥控和红外线数据传输。
接收器对外只有3个引脚:
Out、GND、Vcc与单片机接口非常方便,如图7所示。
图7
①脉冲信号输出接,直接接单片机的IO口。
②GND接系统的地线(0V);
③Vcc接系统的电源正极(+5V);
4遥控信号的解码
下面是一个对51实验板配套的红外线遥控器的解码程序,它可以把红外遥控器每一个按键的键值读出来,并且通过实验板上P1口的8个LED显示出来,在解码成功的同时并且能发出“嘀嘀嘀”的提示音。
;=================================================
;红外遥控接收
;=================================================
; ew51仿真编程器 配套实验板学习例程
;
; 中山单片机学习网 智佳科技 逸风
;
; http:
//www.zsMCU.com
; E-mail:
info@
;=================================================
ORG0000H
MAIN:
JNB P2.2,IR ;遥控扫描
LJMP MAIN ;在正常无遥控信号时,一体化红外接收头输出是高电平,程序一直在循环。
;=================================================
; 解码程序
IR:
;以下对遥控信号的9000微秒的初始低电平信号的识别,波形见图5。
MOV R6,#10
IR_SB:
ACALLDELAY882 ;调用882微秒延时子程序
JB P2.2,IR_ERROR;延时882微秒后判断P2.2脚是否出现高电平如果有就退出解码程序
DJNZ R6,IR_SB ;重复10次,目的是检测在8820微秒内如果出现高电平就退出解码程序
;识别连发码,和跳过4.5ma的高电平。
JNB P2.2,$ ;等待高电平避开9毫秒低电平引导脉冲
ACALLDELAY2400
JNB P2.2,IR_Rp ;这里为低电平,认为是连发码信号,见图6。
ACALLDELAY2400 ;延时4.74毫秒避开4.5毫秒的结果码
;以下32数据码的读取 ,0和1的识别 请看图2
MOV R1,#1AH ;设定1AH为起始RAM区
MOV R2,#4
IR_4BYTE:
MOV R3,#8
IR_8BIT:
JNB P2.2,$ ;等待地址码第一位的高电平信号
LCALLDELAY882 ;高电平开始后用882微秒的时间尺去判断信号此时的高低电平状态
MOV C,P2.2 ;将P2.2引脚此时的电平状态0或1存入C中
JNC IR_8BIT_0 ;如果为0就跳转到IR_8BIT_0
LCALLDELAY1000
IR_8BIT_0:
MOV A,@R1 ;将R1中地址的给A
RRC A ;将C中的值0或1移入A中的最低位
MOV @R1,A ;将A中的数暂时存放在R1中
DJNZ R3,IR_8BIT ;接收地址码的高8位
INC R1 ;对R1中的值加1,换下一个RAM
DJNZ R2,IR_4BYTE ;接收完16位地址码和8位数据码和8位数据,
;存放在1AH/1BH/1CH/1DH的RAM中
;解码成功
JMP IR_GOTO
IR_Rp:
;这里为重复码执行处
;按住遥控按键时,每过108ms就到这里来
JMP IR_GOTO
IR_ERROR:
;错语退出
LJMP MAIN;退出解码子程序
;=================================================
;遥控执行部份
IR_GOTO:
;这里还要判断1AH和1BH 两个系统码或用户码,用于识别不同的遥控器
;MOV A,1AH
;CJNEA,#xxH,IR_ERROR ;用户码1不对则退出
;MOV A,1BH
;CJNEA,#xxH,IR_ERROR ;用户码2不对则退出
;判断两个数据码是否相反
MOV A,1CH
CPL A
CJNE A,1DH,IR_ERROR ;两个数据码不相反则退出
;遥控执行部份
;MOV A,1DH ;判断对应按键
;CJNEA,#xxH,$+6
;LJMP-à跳到对应按键执行处
;CJNEA,#xxH,$+6
;LJMP-à跳到对应按键执行处 ;.
MOV P1,1DH ;将按键的键值通过P1口的8个LED显示出来!
CLR P2.3 ;蜂鸣器鸣响-嘀嘀嘀-的声音,表示解码成功
LCALLDELAY2400
LCALLDELAY2400
LCALLDELAY2400
SETB P2.3 ;蜂鸣器停止
;清除遥控值使连按失效
MOV 1AH,#00H
MOV 1BH,#00H
MOV 1CH,#00H
MOV 1DH,#00H
LJMP MAIN
;=================================================
;延时子程序
;=============================882
DELAY882:
;1.085x((202x4)+5)=882
MOVR7,#202
DELAY882_A:
NOP
NOP
DJNZ R7,DELAY882_A
RET
;=============================1000
DELAY1000:
;1.085x((229x4)+5)=999.285
MOVR7,#229
DELAY1000_A:
NOP
NOP
DJNZ R7,DELAY1000_A
RET
;=============================2400
DELAY2400:
;1.085x((245x9)+5)=2397.85
MOVR7,#245
DELAY2400_A:
NOP
NOP
NOP
NOP
NOP
NOP
NOP
DJNZ R7,DELAY2400_A
RET
END
/*测频率原理*/
/*定时器T0作为定时装置,编程设定定时1s,定时器T1对外部信号计数,每1S中对计数结果处理*/
#include
#include
#defineuintunsignedint//宏定义
#defineucharunsignedchar
externvoidDISPLAY(uintnumber);
#defineSEG_AXBYTE[0x9ff8]
#defineSEG_BXBYTE[0x9ff9]
#defineSEG_CXBYTE[0x9ffa]
#defineSEG_DXBYTE[0x9ffb]
#defineSEG_EXBYTE[0x9ffc]
#defineSEG_FXBYTE[0x9ffd]
voidmain()
{
TMOD=0x51;/*01010001T1计数,T0定时*/
TH1=0;
TL1=0;
//晶振22.1184M
TH0=(65536-20023)/256;
TL0=(65536-20023)%256;
TR0=1;//开定时器0
TR1=1;//启动计时器1
EA=1;//开总中断
ET0=1;//开定时器0中断
while
(1);//原地踏步,等待中断
}
//T0定时中断,晶振22.1184M
voidT0_int(void)interrupt1
{
staticcharcounter=0;
TH0=(65536-20023)/256;
TL0=(65536-20023)%256;//此处延时不是理论计算的1秒,是根据仿真调节出来的
if(counter++==91)
{
counter=0;
TR1=0;
DISPLAY(256*TH1+TL1);
TH1=TL1=0;
TR1=1;
}
}
//数码管显示函数,传入数据0~999999
voidDISPLAY(uintnumber)
{
ucharcodetab1[10]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90};
uchartemp=0,flag=0;
if(number<1000000)
{
//十万位数码管
temp=number/100000%10;
if(temp)
{
SEG_A=tab1[temp];
flag=1;
}
else
{
SEG_A=0xFF;//数码管熄灭
flag=0;
}
//万位数码管
temp=number/10000%10;
if(flag|temp)
{
SEG_B=tab1[temp];
flag=1;
}
else
{
SEG_B=0xFF;
flag=0;
}
//千位数码管
temp=number/1000%10;
if(flag|temp)
{
SEG_C=tab1[temp];
flag=1;
}
else
{
SEG_C=0xFF;
flag=0;
}
//百位数码管
temp=number/100%10;
if(flag|temp)
{
SEG_D=tab1[temp];
flag=1;
}
else
{
SEG_D=0xFF;
flag=0;
}
//十位数码管
temp=number/10%10;
if(flag|temp)SEG_E=tab1[temp];
elseSEG_E=0xFF;
//个位数码管
temp=number%10;
SEG_F=tab1[temp];
}
else
{
;
}
}
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
#include
#include
#include
#include
floatf;//频率
floatp;//周期
floatsj;//闸门时间
charidatabuff[20];
charflag=0;//频率\周期选择标志位
charxs=0;//设置闸门时间结束后是否显示结果的标志位
unsignedcharm=0,n=0,yichu=0,fenpin;//m定时中断次数n计数中断次数yichu判断是定时器还是计数器溢出
#defineKey_SetP1
#defineK10xbf
#defineK20x7f
#defineNO_Set0xff
#defineFreq0
#definePeri1
sbitB153=P2^4;
sbitA153=P2^3;
sbitP17=P1^7;
sbitP16=P1^6;
sbitP35=P3^5;
sbitSet=P3^2;
unsignedcharLCD_Wait(void);
voidLCD_Write(bitstyle,unsignedcharinput);
voidLCD_SetDisplay(unsignedcharDisplayMode);
voidLCD_SetInput(unsignedcharInputMode);
voidLCD_Initial();
voidGotoXY(unsignedcharx,unsignedchary);
voidPrint(unsignedchar*str);
voidC52_Initial();
voidDelay(unsignedintt);
voiddisplay(floatf);
voidcepin();
voidpanduan();
voidtimedisplay(floatsj);
voidTime_Set1();
voidTime_Set2();
voidt0();
voidt1();
/*****************************************************
*模块名称:
LCD1602显示程序*
**
*Author:
ZhaoKaiWen*
*Created:
2008/4/28*
****************************************/
/***********************PortDefinitions**********************/
sbitLcdRs=P2^0;
sbitLcdRw=P2^1;
sbitLcdEn=P2^2;
sfrDBPort=0x80;//P0=0x80,P1=0x90,P2=0xA0,P3=0xB0.数据端口
/************************内部等待函数************/
unsignedcharLCD_Wait(void)
{
LcdRs=0;//寄存器选择输入端1:
数据0:
指令
LcdRw=1;_nop_();//RW:
为0:
写状态;为1:
读状态;
LcdEn=1;_nop_();//使能输入端,读状态,高电平有效;写状态,下降沿有效
LcdEn=0;
returnDBPort;
}
/***********************向LCD写入命令或数据******/
#defineLCD_COMMAND0//Command
#defineLCD_DATA1//Data
#defineLCD_CLEAR_SCREEN0x01//清屏
#defineLCD_HOMING0x02//光标返回原点
voidLCD_Write(bitstyle,unsignedcharinput)
{
LcdEn=0;
LcdRs=style;
LcdRw=0;_nop_();
DBPort=input;_nop_();//注意顺序
LcdEn=1;_nop_();//注意顺序
LcdEn=0;_nop_();
LCD_Wait();
}
/********************设置显示模式*****************/
#defineLCD_SHOW0x04//显示开
#defineLCD_HIDE0x00//显示关
#defineLCD_CURSOR0x02//显示光标
#defineLCD_NO_CURSOR0x00//无光标
#defineLCD_FLASH0x01//光标闪动
#defineLCD_NO_FLASH0x00//光标不闪动
voidLCD_SetDisplay(unsignedcharDisplayMode)
{
LCD_Write(LCD_COMMAND,0x08|DisplayMode);
}
/*********************设置输入模式****************/
#defineLCD_AC_UP0x02
#defineLCD_AC_DOWN0x00//default
#defineLCD_MOVE0x01//画面可平移
#defineLCD_NO_MOVE0x00//default
voidLCD_SetInput(unsignedcharInputMode)
{
LCD_Write(LCD_COMMAND,0x04|InputMode);
}
/******初始化LCD*****************************