1、红外遥控系统
通用红外遥控系统由发射和接收两大部分组成,应用编/解码专用集成电路芯片来进行控制操作,如图1所示。 发射部分包括键盘矩阵、编码调制、LED红外发送器; 接收部分包括光、电转换放大器、解调、解码电路。
下面,我们将使用下面两种设备:
另外,使用51单片机进行解码。 2、原理图
从原理图看出,IR的data脚与51的PD2(P3.2)相连。 2、红外发射原理 要对红外遥控器所发的信号进行解码,必须先理解这些信号。 a) 波形 首先来看看,当我们按下遥控器时,红外发射器是发送了一个什么样的信号波形,如下图:
由上图所示,当一个键按下超过22ms,振荡器使芯片激活,将发射一组108ms的编码脉冲(由位置1所示)。如果键按下超过108ms仍未松开,接下来发射的代码(连发代码由位置3所示)将仅由起始码(9ms)和结束码(2.5ms)组成。下面把位置1的波形放大:
由位置1的波形得知,这108ms发射代码由一个起始码(9ms),一个结果码(4.5ms),低8位地址码(用户编码)(9ms~18ms),高8位地址码(用户编码)(9ms~18ms),8位数据码(键值数据码)(9ms~18ms)和这8位数据的反码(键值数据码反码)(9ms~18ms)组成。 b) 编码格式 遥控器发射的信号由一串0和1的二进制代码组成.不同的芯片对0和1的编码有所不同。通常有曼彻斯特编码和脉冲宽度编码。XS-091遥控板的0和1采用PWM方法编码,即脉冲宽度调制。下图为一个发射波形对应的编码方法: 放大0和1的波形如下图: 这种编码具有以下特征:以脉宽为0.565ms、间隔0.56ms、周期为1.125ms的组合表示二进制的“0”;以脉宽为0.565ms、间隔1.685ms、周期为2.25ms的组合表示二进制的“1”。 3、红外接收原理 a) 波形 红外接收头将38K载波信号过虑,接收到的波形刚好与发射波形相反:放大,位定义0和位定义1波形如下:
4、解码原理及算法
注:代码宽度算法:
16位地址码的最短宽度:1.12×16=18ms 16位地址码的最长宽度:2.24ms×16=36ms
可以得知8位数据代码及其8位反代码的宽度和不变:(1.12ms+2.24ms)×8=27ms
所有32位代码的宽度为(18ms+27ms)~(36ms+27ms) 对于红外线遥控对于很多电子爱好者来讲,都感觉到非常神奇,看不到,摸不着,但能实现无线遥控,其实控制的关键就是我们要用单片机芯片来识别红外线遥控器发出红外光信号,即我们通常所说的解码。单片机得知发过来的是什么信号,然后再做出相应的判断与控制,如我们按电视机遥控器的频道按钮,则单片机会控制更换电视频道,如按的是遥控器音量键,则单片机会控制增减音量。 解码的关键是如何识别“0”和“1” !! 从位的定义我们可以发现“0”、“1”均以 0.56ms的低电平开始,不同的是高电平的宽度不同!,“0”为0.56ms,“1”为1.68ms,所以必须根据高电平的宽度区别“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的结果码完成后才能读码。 5、实例代码:#include "include.h"
sbit LED1 = P3^0;
sbit LED2 = P3^1;
sbit IRIN = P3^2;
sbit LED3 = P1^0;
unsigned char KeyValue; //机器码
unsigned char MaValue; //键值码;
unsigned char num; //所记录IR数值
unsigned char flag;
/***********************************************
*延时程序
*ms:延时时间
**********************************************/
void delay_ms(uint ms)
{
for(;ms>0;ms--)
{
unsigned char i, j;
_nop_();
_nop_();
_nop_();
i = 11;
j = 190;
do
{
while (--j);
} while (--i);
}
}
/*********************************************
*延时1 us
**********************************************/
void delay_us(uint us)
{
for(;us>0;us--)
{
_nop_();
_nop_();
_nop_();
}
}
/**
* 读码
*/
unsigned char GetCode()
{
unsigned char n;
static temp = 0;
for( n = 0; n < 8; n++ )
{
while(!IRIN); // 等待高电平,开始解码
delay_us(840); // 延时0.84ms
if(IRIN) // 若仍然为高电平,则为1,否则为0
{
temp = (0x80|(temp>>1)); // 1
while(IRIN); //等待跳变成低电平
}
else {
temp=(0x00|(temp>>1)); // 0
}
}
return temp;
}
void main()
{
IT0 = 1; //设置为下降沿触发
EX0 = 1; //打开外部触发中断1
UART_Init();
EA = 1; //打开总中断
while(1)
{
//LED1 = !LED1;
if(flag == 1)
{
EX0 =0;
USART1_send(num);
USART1_sendchar('\r');
USART1_sendchar('\n');
flag = 0;
EX0 =1;
}
}
}
void IR_read() interrupt 0 using 1
{
/**
* 用户码和机器码
*/
unsigned char addrl,addrh,num1,num2;
EA = 0; //先关闭外部中断0
delay_ms(9); // 检测9ms开始码
if (IRIN) { // 检测是否为干扰信号
EA = 1; // 重新开启外部中断0
return ; // 退出解码
}
while(!IRIN); // 等待跳为高电平
//delay_ms(4); // 检测4.5ms结果码
delay_us(4500);
if (IRIN) { // 检测是否为干扰信号
EA = 1; // 重新开启外部中断0
return ; // 退出解码
}
// 读码
addrl=GetCode(); // 用户编码高位
addrh=GetCode(); // 用户编码低位
num1=GetCode(); // 机器码
num2=GetCode(); // 机器码反码
LED3 = !LED3;
//校验是否为错码
/*
if(num1!=~num2)
{
KeyValue=14;
EA=1;
return;
}
*/
// KeyValue=num2;
//MaValue=addrh;
flag = 1;
num= addrl;
num = addrh;
num = num1;
num = num2;
EA=1;
}
6、程序的实验结果
我们使用遥控器对着红外接头,依次按动,我们从串口接收到的数据为。后边的红框中的是“\r\n”的十六制显示。
7、完整程序