红外通信
红外通信原理:
普中科技51单片机使用的是PPM,脉时调制(即脉冲的时间间隔)
实例:按下遥控器按键,数码管显示红外遥控键值数据
扩展实验:红外线控制蜂鸣器、直流电机、数码管等等
代码:
/************************************************************************************** * 红外通信实验 * 实现现象:下载程序后,数码管显示红外遥控键值数据 注意事项: 红外遥控器内的电池绝缘片一定要抽掉 ***************************************************************************************/ #include <reg52.h> #define u16 unsigned int #define u8 unsigned char // 红外接口的数据管脚,P3^2管脚是外部中断0管脚 sbit IRIN=P3^2; // 38译码器管脚 sbit LSA=P2^2; sbit LSB=P2^3; sbit LSC=P2^4; u8 IrValue[4]; // 数据:用户码、用户吗、数据码、数据反码,为4字节32位的数据格式 u8 Time; // PPM,脉时调制,高电平持续的时间长度 u8 DisplayData[4]; // 数码管要显示的数据 // 0、1、2、3、4、5、6、7、8、9、A、b、C、d、E、F、H的数码管显示码 u8 code smgduan[17]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07, 0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71,0X76}; /******************************************************************************* * 函 数 名 : delay * 函数功能 : 延时函数,i=1时,大约延时10us *******************************************************************************/ void delay(u16 i) { while(i--); } /******************************************************************************* * 函数名 :DigDisplay() * 函数功能 :数码管显示函数 * 输入 : 无 * 输出 : 无 *******************************************************************************/ void DigDisplay() { u8 i; for(i=0;i<3;i++) { switch(i) // 位选,选择点亮的数码管, { case(0): LSA=0;LSB=0;LSC=0; break;// 显示第0位 case(1): LSA=1;LSB=0;LSC=0; break;// 显示第1位 case(2): LSA=0;LSB=1;LSC=0; break;// 显示第2位 } P0=DisplayData[2-i];// 发送数据 delay(100); // 间隔一段时间扫描 P0=0x00; // 消隐 } } /******************************************************************************* * 函数名 : IrInit() * 函数功能 : 初始化红外线接收 * 输入 : 无 * 输出 : 无 *******************************************************************************/ void IrInit() { IT0=1; // 下降沿触发,因为红外发射端常态下是高电平,所以使用下降沿触发 EX0=1; // 打开中断0允许 EA=1; // 打开总中断 IRIN=1; // 初始化端口 } // 主函数 void main() { while(1) { // 红外通信初始化 IrInit(); // 数据拆分处理,IrValue[2]是数据码,IrValue[0]才是用户码 DisplayData[0] = smgduan[IrValue[2]/16]; // 取十六进制的个位 DisplayData[1] = smgduan[IrValue[2]%16]; // 取十六进制的十位 DisplayData[2] = smgduan[16]; // 显示H,表示十六进制数 // 数码管数据显示 DigDisplay(); } } /******************************************************************************* * 函数名 : ReadIr(),外部中断0函数 * 函数功能 : 读取红外数值的中断函数 * 输入 : 无 * 输出 : 无 *******************************************************************************/ void ReadIr() interrupt 0 { /* 因为要发送的数据是4字节32位的数据,而单片机是一位一位数据发送的, 所以需要两层for循环,j是外层循环,k是内层循环,外层循环用于4字节,内层循环用于每个字节的位 */ u8 j,k; // 这是延迟时间的变量 u16 err; // 这是PPM类型的数据位时间形式,表示高电平持续的时间长度 Time=0; /* 查单片机红外数据格式资料可知,在起始码阶段低电平持续时间为9ms,但不一定就是9ms, 所以这里先延迟7ms,然后再判断是否真的是红外信号 */ delay(700); // 7ms if(IRIN==0) // 判断是否真的接收到正确的信号 { /* 当两个条件都为真是循环,如果有一个条件为假的时候跳出循环,免得程序出错的时 侯,程序死在这里,1000*10us=10ms,超过说明接收到错误的信号 */ err=1000; while((IRIN==0)&&(err>0)) // 等待前面9ms的低电平过去 { delay(1); err--; } // 9ms的低电平过去之后就是高电平 if(IRIN==1) { // 查单片机红外数据格式资料可知,在起始码阶段高电平持续时间为4.5ms,这里延迟5ms>4.5ms err=500; while((IRIN==1)&&(err>0)) { delay(1); err--; } // 4.5ms的起始码的高电平过去之后,就是真正的数据传输,包括:用户码、用户码、数据码、数据反码 for(k=0;k<4;k++) // 共有4组数据 { for(j=0;j<8;j++) // 接收每一组数据 { /* 根据数据位的格式,位0和位1的区别和联系是: 位0和位1低电平持续的时间都是0.56ms, 位0高电平持续的时间是:t0=总时间-低电平时间=1.125ms-0.56ms=0.565ms 位1高电平持续的时间是:t1=总时间-低电平时间=2.25ms-0.56ms=1.69ms 等待信号前面的560us低电平过去 */ // 这里延迟60*10us=600us=0.6ms,等待低电平的0.56ms过去 err=60; while((IRIN==0)&&(err>0)) { delay(1); err--; } /* 计算高电平的时间长度: 如果是位0的高电平持续时间,那么持续0.565ms后就会退出循环 如果是位1的高电平持续时间,那么就会在500次循环之内,达到最大的高电平持续时间Time 理论上是上面计算的t1=1.69ms */ err=500; while((IRIN==1)&&(err>0)) { delay(10); // 0.1ms Time++; err--; if(Time>30) { return; } } // 到这里,不管是位0或者是位1的高电平,都会计算出来一个高电平持续时间Time /* IrValue[k]>>=1; 因为各组数据中是从第一组的低位(即最右边那一位)开始传输的, 所以右移1位之后补零,这个补上来的零就是要传输的数据位置,接着看下面就会明白 */ IrValue[k]>>=1; // k表示第几组数据 /* 如果高电平出现大于565us,那么是1,这里的8,是从上面计算出来的t1来的 为了更加准确地区分位0和位1,这里取位1的高电平持续时间长度的平均数来作为判断依据 即:1.69ms/2=0.845ms,约等于0.8ms */ if(Time>=8) // 这里进来的都是高电平1 { /* 原始数据和0x80相或,就是为了改变上面右移1位的那个数据位的数据的,具体看下面: 比如原始数据为:0000 0000, 第一次: 0000 0000右移1位还是0000 0000, 0x80转成二进制数是:1000 0000,相或之后是:1000 0000,即IrValue[0]=1000 0000 第二次: 1000 0000右移1位是:0100 0000, 与1000 0000相或之后是:1100 0000 第三次: 1100 0000右移1位是:0110 0000, 与1000 0000相或之后是:1110 0000 .... 这样经过每个字节的8次内部循环之后,所有数据位都是1,即最后变成:1111 1111 如果没有进入这个if之中,那么就是低电平0,就不是1111 1111了 */ IrValue[k]|=0x80; } Time=0; // 用完时间要重新赋值 // 到这里,一个字节数据就传输完成了,接下来是下一组1字节的8位数据的传输,直至全部传完 } } } } }
现象:受手机相机信号的干涉,无法拍照!