其余的问题,都是根据红外接收时序来编的程序,以下总结以下红外收发时序供参考,你读懂就可以理解了。
采用脉宽调制的串行码,以脉宽为0.565ms、间隔0.56ms、周期为1.125ms的组合表示二进制的"0";以脉宽为0.565ms、间隔1.685ms、周期为2.25ms的组合表示二进制的"1“上述"0"和"1"组成的32位二进制码经38kHz的载频进行二次调制以提高发射效率,达到降低电源功耗的目的。然后再通过红外发射二极管产生红外线向空间发射遥拿让者控编码是连续的32位二进制码组,其中前16位为用户识别码,能区别不同的电器设备,防止不同机种遥控码互相干扰。该芯片的用户识别码固定为十六进制01H后16位为8位 *** 作码(功能码)及其反码。
发射代码由一个起始码(9ms),一个结果码(4.5ms),低8位地址码(9ms~18ms),高8位地址码(9ms~18ms),8位数据码(9ms~18ms)和这8位数据的反码(9ms~18ms)组成。
解码的关键是如何识别"0"滑衫和"1",接收端而言,"0"是0.56ms的高+0.56ms的低。"1"是1.68ms的高+0.56ms的低。所以可以根据高电平消薯的宽度区别"0"和"1"。当高电平出现时开始延时,0.56ms以后,若读到的电平为低,说明该位为"0",反之则为"1",为了可靠起见,延时必须比0.56ms长些,但又不能超过1.12ms,否则如果该位为"0",读到的已是下一位的高电平,因此取(1.12ms+0.56ms)/2=0.84ms最为可靠,一般取0.84ms左右均可。为了共用引导部分延时程序,一般用0.9ms延时。
由此可见,有效数据是4字节(32位)。前两个字节可定义用户编码,后两个字节分别是真正的数据及其反码。
给你一个我编的红外解码的程序,这个程序会将接收到的红外编码中的8位码通过串口发送到上位机,你可以通过串口调试助手来看看,程序是正确的,希望对你有参考价值。 ////////////////////////////////////////////////////////////////// //////////////////////////yorkWorldDream//////////////////////搭咐/// ///////////////////////////////////////////////////////////////// #include"reg52.h" #define uchar unsigned char #define uint unsigned int //tc9012 uchar flag=0//是否接到起始位标志 未接到为0 接到为1 uchar traflag=0//翻译是否完成标记 完成为1 uchar rec[33]//接收遥控器发送的所有数据 用来记录两个下降沿之间的时间 uchar recok=0//是否接收完一帧数据标记 接收完为1 uchar sendok=0//是否发送完毕标记 发送完成为1 uchar num=0//变相记录时间 uchar n=0//rec[]中的数组定位 void timerinit()//定时器0初始化 void ruptinit()//外部中断0初始化 void tradata()//一帧数据的翻译 即把时间记录转换成0/1 void send()//把收到的数据发给上位机 void delay()//////////////////////////定时器 中断初始化///////////////////////////////// void timerinit()//定时器0 { TMOD=0x02//定时器0使用方式2 TH0=0x00 TL0=0x00 EA=1 ET0=1//定时器中断开 TR0=1//计时开 } void ruptinit()//外部中断0 { EA=1 EX0=1//外部中断开 IT0=1//下降沿触发 0是电平触发 } ////////////////////////////定时器 中断函数////////////////////////////// void timer() interrupt 1 //系统从0x00记到0xff 每记一次时间大概为1us 记256次 即256us中断一次 { num++//num记录的是256us的个数 总时间=num*256us } void rupt() interrupt 0 //外部中断0 当遇到下降沿时触发 { if(flag==1)//flag为1 说明收到了起始位 接下来要开始进行记录了 { if(num>32)//时间为32*256us=8.2ms是整个起始位的时间 { n=0 } rec[n]=num//记录两个下降沿之间的num值 这样也就相当于记录了其间的时间 num=0 n++ if(n==33)//因为前面是n++ 所以当n=33时 数组已经从0记到了32 已经记满了 { recok=1//标记 接收一帧数据完成 n=0 } } else//首次得到下降沿到达这里 使flag变1 用来说明收到了起始位 { flag=1 } } /////////////////////////////时间翻译函数/////////////////////////////////////// void tradata()//一帧数据的翻译 即把时间记录转换成0/1 { uchar i for(i=1i/0的数据时间是num=4.4 1的数据时间是num=8.8 { //记住要从rec[1]开始 因为rec[0]记录的是引导码的num值 if(rec[i]>6)//使用6做中间值 小于则为0 大于为1 用0/1替换rec中的数据 这样就进行了翻译 { rec[i]='1' }//因为要以字符串发送 所以这里用字符形式 else rec[i]='0' } traflag=1//翻译完成标记 } /////////////////////////////发送数据函数//////////////////////////////////// void send()//把收到的数据发送给上位机 具体细节不懂看串口通信 mcu to pc { uchar i TMOD=0x20 TH1=0xfd TL1=0xfd TR1=1 SM0=0 SM1=1 REN=1 for(i=25i/选择要发送出去的数据 这里是从25开始发送的 { SBUF=rec[i] while(!TI) TI=0 } sendok=1} 搏燃 //////////////////////////////////////////////////////////////////////// void delay()//1ms { unsigned char a,b,c for(c=1c>0c--) for(b=142b>0b--) for(a=2a>0a--)} void main() { timerinit()//定时器0初始化 ruptinit()//外部中断0初始化 while(1) { if(recok==1)//接收一帧数据完成 { P1=0xfe//灯亮一下 为了指示是否收到了数据 tradata()//翻译数据 } delay() if(traflag==1)//翻译完成 { send()//发送 基枝虚 } delay() if(sendok==1)//发送完成 { flag=0 traflag=0 recok=0 sendok=0 P1=0xff timerinit() ruptinit() } //全部恢复初始 }欢迎分享,转载请注明来源:内存溢出
评论列表(0条)