NEC协议红外遥控器_第1页
NEC协议红外遥控器_第2页
NEC协议红外遥控器_第3页
NEC协议红外遥控器_第4页
NEC协议红外遥控器_第5页
已阅读5页,还剩9页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

NEC协议红外遥控器家电遥控器通信距离往往要求不高,而红外的成本比其他无线设备要低的多,所以家电遥控器应用中红外始终占据着一席之地。遥控器的基带通信协议许多,也许有几十种,常用的就有ITT协议、NEC协议、Sharp协议、PhilipsRC-5协议、SonySIRC协议等。用的最多的就是NEC协议了,因此我们KST-51开发板随板的遥控器直接采纳NEC协议,我们这节课也以NEC协议标准来讲解一下。

NEC协议的数据格式包括了引导码、用户码、用户码(或者用户码反码)、按键键码和键码反码,最终一个停止位,停止位主要起隔离作用,一般不进行推断,编程时我们也不予理睬。其中数据编码总共是4个字节32位,如图1所示。第一个字节是用户码,其次个字节可能也是用户码,或者是用户码的反码,详细由生产商打算,第三个字节就是当前按键的键数据码,而第四个字节是键数据码的反码,可用于对数据的纠错。

图1NEC协议数据格式

这个NEC协议,表示数据的方式不像我们之前学过的比如uart那样直观,而是每一位数据本身也需要进行编码,编码后再进行载波调制。

引导码:9ms的载波+4.5ms的空闲。

比特值“0”:560us的载波+560us的空闲。

比特值“1”:560us的载波+1.68ms的空闲。

结合图1我们就能看明白了,最前面黑乎乎的一段,是引导码的9ms载波,紧接着是引导码的4.5ms的空闲,而后边的数据码,是众多载波和空闲交叉,它们的长短就由其要传递的详细数据来打算。我们的HS0038B这个红外一体化接收头,当收到有载波的信号的时候,会输出一个低电平,空闲的时候会输出高电平,我们用规律分析仪抓出来一个红外按键通过HS0038解码后的图形来了解一下,如图2所示。

图2红外遥控器按键编码

从图上可以看出,先是9ms载波加4.5ms空闲的起始码,数据码是低位在前,高位在后,数据码第一个字节是8组560us的载波加560us的空闲,也就是0x00,其次个字节是8组560us的载波加1.68ms的空闲,可以看出来是0xFF,这两个字节就是用户码和用户码的反码。按键的键码二进制是0x0B,反码就是0xF3,最终跟了一个560us载波停止位。对于我们的遥控器来说,不同的按键,就是键码和键码反码的区分,用户码是一样的。这样我们就可以通过单片机的程序,把当前的按键的键码给解出来。

我们前边学习中断的时候,学到51单片机有外部中断0和外部中断1这两个外部中断。我们的红外接收引脚接到了P3.3引脚上,这个引脚的其次功能就是外部中断1。在寄存器TCON中的bit3和bit2这两位,是和外部中断1相关的两位。其中IE1是外部中断标志位,当外部中断发生后,这一位被自动置1,和定时器中断标志位TF相像,进入中断后会自动清零,也可以软件清零。bit2位是设置外部中断类型的,假如bit2位为0,那么只要P3.3为低电平就可以触发中断,假如bit2位为1,那么P3.3从高电平到低电平的下降沿发生才可以触发中断。此外,外部中断1使能位是EX1。那下面我们就把程序写出来,使用数码管把遥控器的用户码和键码显示出来。

Infrared.c文件主要是用来检测红外通信的,当发生外部中断后,进入外部中断,通过定时器1定时,首先对引导码推断,而后对数据码的每个位逐位猎取凹凸电平的时间,从而得知每一位是0还是1,最终把数据码解出来。

/***********************infrared.c文件程序源代码*************************/

#includereg52.h

sbitIR_INPUT=P3^3;//红外接收引脚

bitirflag=0;//红外接收标志,收到一帧正确数据后置1

unsignedcharircode[4];//红外代码接收缓冲区

voidInitInfrared(void)//红外功能的初始化函数

{

TMOD=0x0F;//清零T1的掌握位

TMOD|=0x10;//配置T1为模式1

TR1=0;//停止T1计数

ET1=0;//禁止T1中断

IT1=1;//设置INT1为负边沿触发

EX1=1;//使能INT1中断

}

unsignedintGetHighTime(void)//猎取高电平常间

{

TH1=0;//清零T1计数初值

TL1=0;

TR1=1;//启动T1计数

while(IR_INPUT)//红外输入引脚为1时循环检测等待,变为0时则结束本循环

{

if(TH1=0x40)

{//当T1计数值大于0x4000,即高电平持续时间超过约18ms时,

break;//强制退出循环,是为了避开信号特别时,程序假死在这里。

}

}

TR1=0;//停止T1计数

return(TH1*256+TL1);//返回T1的计数值

}

unsignedintGetLowTime(void)//猎取低电平常间

{

TH1=0;//清零T1计数初值

TL1=0;

TR1=1;//启动T1计数

while(!IR_INPUT)//红外输入引脚为0时循环检测等待,变为1时则结束本循环

{

if(TH1=0x40)

{//当T1计数值大于0x4000,即低电平持续时间超过约18ms时,

break;//强制退出循环,是为了避开信号特别时,程序假死在这里。

}

}

TR1=0;//停止T1计数

return(TH1*256+TL1);//返回T1的计数值

}

voidEXINT1_ISR()interrupt2//INT1中断服务函数,执行红外接收及解码

{

unsignedchari,j;

unsignedcharbyt;

unsignedinttime;

//接收并判定引导码的9ms低电平

time=GetLowTime();

if((time7833)||(time8755))//时间判定范围为8.5~9.5ms,

{//超过此范围则说明为误码,直接退出

IE1=0;//退出前清零INT1中断标志

return;

}

//接收并判定引导码的4.5ms高电平

time=GetHighTime();

if((time3686)||(time4608))//时间判定范围为4.0~5.0ms,

{//超过此范围则说明为误码,直接退出

IE1=0;

return;

}

//接收并判定后续的4字节数据

for(i=0;i4;i++)//循环接收4个字节

{

for(j=0;j8;j++)//循环接收判定每字节的8个bit

{

//接收判定每bit的560us低电平

time=GetLowTime();

if((time313)||(time718))//时间判定范围为340~780us,

{//超过此范围则说明为误码,直接退出

IE1=0;

return;

}

//接收每bit高电平常间,判定该bit的值

time=GetHighTime();

if((time313)(time718))//时间判定范围为340~780us,

{//在此范围内说明该bit值为0

byt=1;//因低位在先,所以数据左移,高位为0

}

elseif((time1345)(time1751))//时间判定范围为1460~1900us,

{//在此范围内说明该bit值为1

byt=1;//因低位在先,所以数据左移,

byt|=0x80;//高位置1

}

else//不在上述范围内则说明为误码,直接退出

{

IE1=0;

return;

}

}

ircode[i]=byt;//接收完一个字节后保存到缓冲区

}

irflag=1;//接收完毕后设置标志

IE1=0;//退出前清零INT1中断标志

}

大家在阅读这个文件里的代码时,会发觉我们在猎取凹凸电平常间的时候做了超时推断if(TH1=0x40),这个超时推断一方面是应对空间突发的红外干扰信号,假如我们不做超时推断,程序有可能会始终等待下一个跳变才会停止检测,造成程序假死。另外一个方面,遥控器的单按按键和持续按住按键发出来的信号是不同的。我们先来对比一下两种按键方式的信号状态,如图3和4所示。

图3红外单次按键时序图

图4红外持续按键时序图

单次按键的结果3和我们之前的图2是一样的,这个不需要再解释。而持续按键,首先会发出一个和单次按键一样的波形出来,经过也许40ms后,会产生一个9ms载波加2.25ms空闲,再跟一个停止位的波形,而后只要你还在按住按键,每经过也许96ms就会产生9ms载波加2.25ms空闲加停止位这样的重复波形。我们人为按下按键的时候,很难掌握按下的时间,因此后边的很简单消失这种连续波形,我们加上超时推断也可以有效的避开进入连续波形的死循环中去。

/***********************main.c文件程序源代码*************************/

#includereg52.h

sbitADDR3=P1^3;//LED选择地址线3

sbitENLED=P1^4;//LED总使能引脚

unsignedcharcodeLedChar[]={//数码管显示字符转换表

0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,

0x80,0x90,0x88,0x83,0xC6,0xA1,0x86,0x8E

};

unsignedcharLedBuff[6]={//数码管显示缓冲区

0xFF,0xFF,0xFF,0xFF,0xFF,0xFF

};

unsignedcharT0RH=0;//T0重载值的高字节

unsignedcharT0RL=0;//T0重载值的低字节

externbitirflag;

externunsignedcharircode[4];

voidConfigTimer0(unsignedintms);

externvoidInitInfrared(void);

voidmain()

{

P0=0xFF;//P0口初始化

ADDR3=1;//选择数码管

ENLED=0;//LED总使能

InitInfrared();//初始化红外功能

ConfigTimer0(1);//配置T0定时1ms

EA=1;//开总中断

//PT0=1;//配置T0中断为高优先级

while(1)

{

if(irflag)//接收到红外数据时刷新显示

{

irflag=0;

LedBuff[5]=LedChar[ircode[0]4];//用户码显示

LedBuff[4]=LedChar[ircode[0]0x0F];

LedBuff[1]=LedChar[ircode[2]4];//键码显示

LedBuff[0]=LedChar[ircode[2]0x0F];

}

}

}

voidConfigTimer0(unsignedintms)//T0配置函数

{

unsignedlongtmp;

tmp=11059200/12;//定时器计数频率

tmp=(tmp*ms)/1000;//计算所需的计数值

tmp=65536-tmp;//计算定时器重载值

tmp=tmp+15;//修正中断响应延时造成的误差

T0RH=(unsignedchar)(tmp8);//定时器重载值拆分为凹凸字节

T0RL=(unsignedchar)tmp;

TMOD=0xF0;//清零T0的掌握位

TMOD|=0x01;//配置T0为模式1

TH0=T0RH;//加载T0重载值

TL0=T0RL;

ET0=1;//使能T0中断

TR0=1;//启动T0

}

voidInterruptTimer0()interrupt1//T0中断服务函数

{

staticunsignedchariled=0;

TH0=T0RH;//定时器重新加载重载值

TL0=T0RL;

//LED数码管动态扫描

P0=0xFF;//关闭全部段选位,显示消隐

P1=(P10xF8)|iled;//位选索引值赋值到P1口低3位

P0=LedBuff[iled];//相应显示缓冲区的值赋值到P0口

if(iled5)//位选索引0-5循环,因有6个数码管

il

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论