完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
这个程序,当我把jiema程序放在外部中断里执行时,可以正确解码;但如果外部中断只检测下降沿,退出外部中断后再去红外解码就会出错。关大好几圈了,求助啊!!!!!!!!!!!!!!!!!!!!
//实例96:用P1口显示红外遥控器的按键值 #include #include #define uint unsigned int #define uchar unsigned char ***it wei=P2^1; ***it duan=P2^2; ***it beep=P1^0; ***it led=P1^1; ***it IR=P3^2; //将IR位定义为P3.2引脚 bit c=1; uchar no; unsigned char a[4]; //储存用户码、用户反码与键数据码、键数据反码 unsigned int Lowtime,HighTime; //储存高、低电平的宽度 uchar code table[16]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71}; /************************************************************ //延时函数 ************************************************************/ void delayms(uint xms) { uint aa,bb; for(aa=xms;aa>0;aa--) for(bb=55;bb>0;bb--); } void delayms1(uint xms) { uint aa,bb; for(aa=xms;aa>0;aa--) for(bb=55;bb>0;bb--); } void display() { uchar shi,ge; shi=no>>4&0x0f;//a[2]>>4&0x0f; ge=no&0x0f;//a[2]&0x0f; duan=1; P0=table[shi]; duan=0; P0=0XFF; wei=1; P0=0xfe; wei=0; delayms(5); duan=1; P0=table[ge]; duan=0; P0=0XFF; wei=1; P0=0xfd; wei=0; delayms(5); //************************************************* shi=a[2]>>4&0x0f;//a[2]>>4&0x0f; ge=a[2]&0x0f;//a[2]&0x0f; duan=1; P0=table[shi]; duan=0; P0=0XFF; wei=1; P0=0xfb; wei=0; delayms(5); duan=1; P0=table[ge]; duan=0; P0=0XFF; wei=1; P0=0xf7; wei=0; delayms(5); //***************************************************** shi=a[3]>>4&0x0f;//a[2]>>4&0x0f; ge=a[3]&0x0f;//a[2]&0x0f; duan=1; P0=table[shi]; duan=0; P0=0XFF; wei=1; P0=0xef; wei=0; delayms(5); duan=1; P0=table[ge]; duan=0; P0=0XFF; wei=1; P0=0xdf; wei=0; delayms(5); } /************************************************************ 函数功能:执行遥控功能 *************************************************************/ void Function(void) { beep=0; //将按键数据码送P1口显示 delayms(5); beep=1; delayms(5); //display(); } void Function1(void) { beep=0; //将按键数据码送P1口显示 delayms1(5); beep=1; delayms1(5); //display(); } /************************************************************ 函数功能:对4个字节的用户码和键数据码进行解码 说明:解码正确,返回1,否则返回0 出口参数:dat *************************************************************/ bit decode(void) { uchar i,j; uchar temp; //储存解码出的数据 for(i=0;i<4;i++) //连续读取4个用户码和键数据码 { for(j=0;j<8;j++) //每个码有8位数字 { temp=temp>>1; //temp中的各数据位右移一位,因为先读出的是高位数据 TH0=0; //定时器清0 TL0=0; //定时器清0 TR0=1; //开启定时器T0 while(IR==0) //如果是低电平就等待 { LowTime=TH0*256+TL0; if(LowTime>700) { return 0; } } //低电平计时 TR0=0; //关闭定时器T0 LowTime=TH0*256+TL0; //保存低电平宽度 TH0=0; //定时器清0 TL0=0; //定时器清0 TR0=1; //开启定时器T0 while(IR==1) //如果是高电平就等待 { HighTime=TH0*256+TL0; if(HighTime>2000) { return 0; } } TR0=0; //关闭定时器T0 HighTime=TH0*256+TL0; //保存高电平宽度 if((LowTime<370)||(LowTime>640)) return 0; //如果低电平长度不在合理范围,则认为出错,停止解码 if((HighTime>420)&&(HighTime<620)) //如果高电平时间在560微秒左右,即计数560/1.085=516次 temp=temp&0x7f; //(520-100=420, 520+100=620),则该位是0 if((HighTime>1300)&&(HighTime<1800)) //如果高电平时间在1680微秒左右,即计数1680/1.085=1548次 temp=temp|0x80; //(1550-250=1300,1550+250=1800),则该位是1 } a=temp; //将解码出的字节值储存在a // if(i==0) // led=0; } if(a[2]=~a[3]) //验证键数据码和其反码是否相等,一般情况下不必验证用户码 return 1; //解码正确,返回1 } void jiema()//引导码检测 { if(c==0) { IE0=0; TH0=0; //定时器T0的高8位清0 TL0=0; //定时器T0的低8位清0 TR0=1; //开启定时器T0 while(IR==0) //如果是低电平就等待,给引导码低电平计时 { LowTime=TH0*256+TL0; if(LowTime>9000) { TR0=0; c=1; IE0=0; EA=1;//EX0=1; return; } } TR0=0; //关闭定时器T0 LowTime=TH0*256+TL0; //保存低电平时间 TH0=0; //定时器T0的高8位清0 TL0=0; //定时器T0的低8位清0 TR0=1; //开启定时器T0 IE0=0; while(IR==1) //如果是高电平就等待,给引导码高电平计时 { HighTime=TH0*256+TL0; if(HighTime>65500) { TR0=0; c=1; IE0=0; EA=1; //EX0=1; //led=!led; return; } } IE0=0; TR0=0; //关闭定时器T0 HighTime=TH0*256+TL0; //保存引导码的高电平长度*/ //a[2]=TH0; //a[3]=TL0; if((LowTime>7800)&&(LowTime<8800)&&(HighTime>3600)&&(HighTime<5500))//7800,8800,3600,5000 { //如果是引导码,就开始解码,否则放弃,引导码的低电平计时 //次数=9000us/1.085=8294, 判断区间:8300-500=7800,8300+500=8800. //led=!led; if(decode()==1) //decode(); Function1(); //如果满足条件,执行遥控功能 //while(1)display(); } TR0=0; c=1; led=!led; IE0=0; EA=1; //EX0=1; //开启外中断EX0 } } /************************************************************ 函数功能:红外线触发的外中断处理函数 *************************************************************/ void Int0(void) interrupt 0 using 0 { IE0=0; EA=0; //EX0=0; //关闭外中断0,不再接收二次红外信号的中断,只解码当前红外信号 c=0; no++; // jiema(); // IE0=0; // EX0=1; //led=!led; } /************************************************************ 函数功能:主函数 *************************************************************/ void main() { c=1; TMOD=0x01; //使用定时器T0的模式1 IT0=1; //外中断的下降沿触发 EX0=1; //开外中断0 ET0=0; //定时器T0中断不允许 TR0=0; //定时器T0关闭*/ EA=1; //开启总中断 while(1) //等待红外信号产生的中断 { display(); //jiema(); //IE0=0; } } |
|
相关推荐
1 个讨论
|
|
只有小组成员才能发言,加入小组>>
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2024-11-17 17:18 , Processed in 0.804168 second(s), Total 53, Slave 40 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191 工商网监 湘ICP备2023018690号