一、红外通信原理 红外遥控有发送和接收两个组成部分。发送端采用单片机将待发送的二进制信号编码调制为一系列的脉冲串信号,通过红外发射管发射红外信号。红外接收完成对红外信号的接收、放大、检波、整形,并解调出遥控编码脉冲。为了减少干扰,采用的是价格便宜性能可靠的一体化红外接收头(HS0038, 它接收红外信号频率为38kHz,周期约26μs) 接收红外信号,它同时对信号进行放大、检波、整形得到TTL 电平的编码信号,再送给单片机,经单片机解码并执行去控制相关对象。如图1 所示: 红外发送部分由51单片机、键盘、红外发光二极管和7段数码管组成。键盘用于输入指令, 51单片机检测键盘上按键的状态,并对红外信号进行调制,发光二极管产生红外线,数码管用来显示发送的键值。图2红外发射电路 红外接收部分由51单片机、一体化红外接收头HS0038和7段数码管组成。51单片机检测HS0038,并对HS0038接收到的数据解码,通过数码管显示接收到的键值。图 3红外接收电路 二、编码、解码 (1) 二进制信号的调制 二进制信号的调制由单片机来完成,它把编码后的二进制信号调制成频率为38kHz 的间断脉冲串,相当于用二进制信号的编码乘以频率为38kHz 的脉冲信号得到的间断脉冲串,即是调制后用于红外发射二极管发送的信号如图4 二进制码的调制所示 (2) 红外接收需先进行解调,解调的过程是通过红外接收管进行接收的。其基本工作过程为:当接收到调制信号时,输出高电平,否则输出为低电平,是调制的逆过程(图5 解调)。HS0038是一体化集成的红外接收器件,直接就可以输出解调后的高低电平信号;红外接收器HS0038的应用电路(图6)。 (3)红外遥控发射芯片采用 PPM 编码方式,当发射器按键按下后 ,将发射一组 108ms 的编码脉冲。遥控编码脉冲由前导码、16位地址码(8 位地址码、 8 位地址码的反码)和16位操作码(8 位操作码、 8 位操作码的反码)组成。通过对用户码的检验,每个遥控器只能控制一个设备动作,这样可以有效地防止多个设备之间的干扰。编码后面还要有编码的反码,用来检验编码接收的正确性,防止误操作,增强系统的可靠性。前导码是一个遥控码的起始部分,由一个 9ms 的高电平 ( 起始码 ) 和一个 4. 5ms 的低电平 ( 结果码 ) 组成,作为接受数据的准备脉冲。以脉宽为 0. 56ms 、周期为 1. 12ms 的组合表示二进制的 “0” ;以脉宽为 1. 68ms 、周期为 2. 24ms 的组合表示二进制的 “1” 。 (4)单片机采用外部中断 INT0 管脚和红外接收头的信号线相连,中断方式为边沿触发方式。计算中断的间隔时间,来区分前导码、二进制的 “1” 、 “0” 码。并将 8 位操作码提取出来在数码管上显示。 红外接收头输出的原始遥控数据信号,正好和发射端倒向.也就是以前发射端原始信号是高电平,那接收头输出的就是低电平,反之. 软件原理: 开始时发射一个特定的同步码头,对于接收端而言就是一个9ms的低电平,和一个4.5ms的高电平,这个同步码头可以使程序知道从这个同步码头以后可以开始接收数据。 采用脉宽调制的串行码,以脉宽为0.565ms、间隔0.56ms、周期为1.125ms的组合表示二进制的“0”;以脉 宽为0.565ms、间隔1.685ms、周期为2.25ms 的组合表示二进制的“1”。 解码的关键是如何识别“0”和“1”,从位的定义我们可以发现“0”、“1”均以0.56ms的高电平开始,不同的是低电平的宽度不同,“0”为0.56ms,“1”为1.685ms,所以必须根据高电平的宽度区别“0”和“1”。如果从0.56ms低电平过后,开始延时,0.56ms以后,若读到的电平为低,说明该位为“0”,反之则为“1”,为了可靠起见,延时必须比0.56ms长些,但又不能超过1.12ms,否则如果该位为“0”,读到的已是下一位的高电平,因此取(1.12ms+0.56ms)/2=0.84ms最为可靠,一般取0.84ms 左右即可。根据红外编码的格式,程序应该等待9ms的起始码和4.5ms的结果码完成后才能读码。 HS0038红外接收器,接收红外遥控器发射的信号,输出DATA口和单片机的外部中断0P3.2口相连。当有红外信号时,触发中断查询中断时间,并和红外起始码,“0”、“1”、终止码的时间进行比较。从而检测红外的操作码。 51可参考的程序 (1)发送程序 #include static bit OP; //红外发射管的亮灭 static unsigned int count; //延时计数器 static unsigned int endcount; //终止延时计数 static unsigned int temp; //按键 static unsigned char flag; //红外发送标志 static unsigned char num; sbit ir_in=P3^4; char iraddr1; //十六位地址的第一个字节 char iraddr2; //十六位地址的第二个字节 unsigned char code table[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8, 0x80,0x90,0x88,0x83,0xc6,0xa1,0x86,0x8e}; //共阳数码管 0~~f void SendIRdata(char p_irdata); void delay(unsigned int); void keyscan(); void main(void) { num=0; P2=0x3f; count = 0; flag = 0; OP = 0; ir_in= 0; EA = 1; //允许CPU中断 TMOD = 0x11; //设定时器0和1为16位模式1 ET0 = 1; //定时器0中断允许 TH0 = 0xFF; TL0 = 0xE6; //设定时值0为38K 也就是每隔26us中断一次 TR0 = 1;//开始计数 iraddr1=3;//00000011 iraddr2=252;//11111100 do{keyscan(); }while(1); } void timeint(void) interrupt 1 { TH0=0xFF; TL0=0xE6; //设定时值为38K 也就是每隔26us中断一次 count++; if (flag==1) { OP=~OP; } else { OP = 0; } ir_in= OP; } void SendIRdata(char p_irdata) { char irdata=p_irdata; //发送9ms的起始码 endcount=223; flag=1; count=0; do{}while(count endcount=117; flag=0; count=0; do{}while(count irdata=iraddr1; for(i=0;i<> { endcount=10; flag=1; count=0; do{}while(count if(irdata-(irdata/2)*2) //判断二进制数个位为1还是0 { endcount=41; //1为宽的高电平 } else { endcount=15; //0为窄的高电平 } flag=0; count=0; do{}while(count irdata=irdata>>1; } irdata=iraddr2; for(i=0;i<> { endcount=10; flag=1; count=0; do{}while(count if(irdata-(irdata/2)*2) { endcount=41; } else {endcount=15; } flag=0; count=0; do{}while(count irdata=irdata>>1; } irdata=p_irdata; for(i=0;i<> { endcount=10; flag=1; count=0; do{}while(count if(irdata-(irdata/2)*2) { endcount=41; } else { endcount=15; } flag=0; count=0; do{}while(count irdata=irdata>>1; } irdata=~p_irdata; for(i=0;i<> { endcount=10; flag=1; count=0; do{}while(count if(irdata-(irdata/2)*2) { endcount=41; } else { endcount=15; } flag=0; count=0; do{}while(count irdata=irdata>>1; } endcount=10; flag=1; count=0; do{}while(count flag=0; } void delay(unsigned int z) { unsigned char x,y; for(x=z;x>0;x--) for(y=110;y>0;y--); } (2)接收程序 #include'reg52.h' #define uchar unsigned char #define uint unsigned int uchar dis_num,num,num1,num2,num3; sbit led=P1^0; unsigned char code table[]={ 0xc0,0xf9,0xa4,0xb0, 0x99,0x92,0x82,0xf8, 0x80,0x90,0x88,0x83, 0xc6,0xa1,0x86,0x8e}; //共阳数码管 0~~f sbit prem =P3^2; //定义遥控头的接收脚 uchar ram[4]={0,0,0,0};//存放接受到的4个数据 地址码16位+按键码8位+按键码取反的8位 void delaytime(uint time) //延迟90uS { uchar a,b; for(a=time;a>0;a--) { for(b=40;b>0;b--); } } void rem()interrupt 0 //中断函数 { uchar ramc=0; //定义接收了4个字节的变量 uchar count=0; //定义现在接收第几位变量 uint i=0; //此处变量用来在下面配合连续监测9MS 内是否有高电平 prem=1; for(i=0;i<1100;i++)> { if(prem) //进入遥控接收程序首先进入引导码的前半部判断,即:是否有9MS左右的低电平 return; //引导码错误则退出 } while(prem!=1); //等待引导码的后半部 4.5 MS 高电平开始的到来。 delaytime(50); //延时大于4.5MS时间,跨过引导码的后半部分,来到真正遥控数据32位中 //第一位数据的0.56MS开始脉冲 for(ramc=0;ramc<4;ramc++)> { for(count=0;count<8;count++)> { while(prem!=1); //开始判断现在接收到的数据是0或者1 ,首先在这行本句话时, //保已经进入数据的0.56MS 低电平阶段 //等待本次接受数据的高电平的到来。 delaytime(9);//高电平到来后,数据0 高电平最多延续0.56MS,而数据1,高电平可 //延续1.66MS大于0.8MS 后我们可以再判断遥控接收脚的电平, if(prem) //如果这时高电平仍然在继续那么接收到的数据是1的编码 { ram[ramc]=(ram[ramc]<1)+1;> delaytime(11); //如果本次接受到的数据是1,那么要继续延迟1MS,这样才能跨入//下个位编码的低电平中(即是开始的0.56MS中) } else //否则目前接收到的是数据0的编码 ram[ramc]=ram[ramc]<1;> } //本次接收结束,进行下次位接收,此接收动作进行32次,正好完成4个字节的接收 } if(ram[2]!=(~(ram[3]&0x7f))) //本次接收码的判断 { for(i=0;i<4;i++)> ram[i]=0; return ; } dis_num=ram[2]; //将接收到的按键数据赋给显示变量 } |
|
来自: 随风入夜1qkk89 > 《单片机》