在单片机应用中,红外遥控以其简单易用而受到欢迎,在一些简单的室内控制中,红外遥控是首选。
红外遥控就是一种通讯过程,他也有自己的通讯协议,9ms的引导码+32位信息码。
信息码:8位用户码+8位用户码+8位功能码+8位功能反码。
以上都是用2进制来表示。
0---高电平:低电平=1:1,都大概为0.56ms
1---高电平:低电平=1:3,高电平为0.56,低电平则为1.5ms左右。
利用一个定时器0中断,方式是8位自动重载方式,目的是利用计数值获得时间基准。相当于是一个手表的作用。因为按照12M晶振,每递增一次,是1us,定时器加到256自动重载并产生中断,则每次产生中断的时间间隔是256us,在中断服务程序中让一个标志位递增。
再利用外部中断0,每次来个下降沿,就读一下标志位的数值,如果是10,则时间长度是10*0.256=2.56ms,所以每次下降沿就读一下,就可以判断引导码、0与1了。
在使用一个新的遥控器时,必须首先要解码,并显示出来。利用下面的程序来进行:
以下为源码:
#include<reg52.h>
#define uchar unsigned char
#define uint unsigned int
#define DATAPORT P0
uchar irtime;
uchar startflag;
uchar irdata[33];
uchar bitnumber;
uchar irreceiveok;
uchar ircode[4];
uchar irprosok;
uchar disp[8];
sbit LATCH1=P2^2;//定义锁存使能端口 段锁存
sbit LATCH2=P2^3;// 位锁存
sbit IR=P3^2; unsigned char code DuanMa[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};
// 显示段码值01234567
unsigned char code WeiMa[]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};
//分别对应相应的数码管点亮,即位码 void time0init(void)//定时器中断初始化
{
TMOD=0x02; //方式2,8位自动重载
TH0=0X00; //初值
TL0=0X00;
ET0=1; //打开定时器
EA=1; //开总中断
TR0=1;
}
void int0init(void)//外部中断初始化
{
IT0=1; //下降沿
EX0=1;
EA=1;
}
void delay_50ms(uint t)
{
uint j;
for(;t>0;t--)
for(j=100;j>0;j--);
}
void irwork(void) //把获得的16进制数进行分解,
{
disp[0]=ircode[0]/16;
disp[1]=ircode[0]%16;
disp[2]=ircode[1]/16;
disp[3]=ircode[1]%16;
disp[4]=ircode[2]/16;
disp[5]=ircode[2]%16;
disp[6]=ircode[3]/16;
disp[7]=ircode[3]%16;
}
void display(void)
{
uchar i;
for(i=0;i<8;i++)
{ DATAPORT=WeiMa[i];
LATCH2=1;
LATCH2=0;
DATAPORT=DuanMa[disp[i]];
LATCH1=1;
LATCH1=0;
delay_50ms(2);
}
}
void irpros(void) //提取获得的信息码
{
uchar k,i,j,value;
k=1;//不读第一个引导码
for(j=0;j<4;j++)
{
for(i=0;i<8;i++) //因为每个字节8位
{
value=value>>1; //右移7次,第一次进来因为都是0,所以移位没有任何影响。
if(irdata[k]>6) //判断8次,0为4.4,1为8.8,这里选择6,若大于6,说明是1
{
value=value|0x80; //则或上1000 0000,最高位置1,然后进行右移(因为低位在前)
}
k++;
}
ircode[j]=value; //最终得到码
}
irprosok=1; //解码成功标志位
}
void main()
{
time0init();
int0init();
while(1)
{
if(irreceiveok)
{
irpros();
irreceiveok=0;//判断完后,清零
}
if(irprosok)
{
irwork();
irprosok=0;
}
display();
}
}
void time0() interrupt 1//定时器中断0的ISR
{
irtime++;//255
}
void int0() interrupt 0//外部中断0的ISR
{
if(startflag) //刚开始进来,检查启动接受标志位,肯定不为1
{
if(irtime>32) //说明检测到引导码
{
bitnumber=0;
}
irdata[bitnumber]=irtime; //把每次进来的irtime都存起来,最后再判断
irtime=0;
bitnumber++; //第几位标志
if(bitnumber==33) //遥控器发出高电平,一体化红外接收头接收后产生低电平。
//每帧数据为32位,再加上引导码,总共33位
{
bitnumber=0;
irreceiveok=1; //红外接收完毕
}
}
else
{
startflag=1;//如果不为1,则置1
irtime=0; //清零
}
}