找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 5490|回复: 0
打印 上一主题 下一主题
收起左侧

红外接收头解码实验程序 带详细注释 lcd1602显示

[复制链接]
跳转到指定楼层
楼主
ID:71259 发表于 2014-12-30 01:41 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
  1. //采用11.0592MHZ晶振
  2. #include<reg52.h>       //包含单片机寄存器的头文件
  3. #include<intrins.h>  //包含_nop_()函数定义的头文件
  4. sbit IR=P3^2;           //将IR位定义为P3.2引脚
  5. sbit RS=P2^0;    //寄存器选择位,将RS位定义为P2.0引脚
  6. sbit RW=P2^1;    //读写选择位,将RW位定义为P2.1引脚
  7. sbit E=P2^2;     //使能信号位,将E位定义为P2.2引脚
  8. sbit BF=P0^7;    //忙碌标志位,,将BF位定义为P0.7引脚
  9. sbit BEEP = P3^6; //蜂鸣器控制端口P36
  10. unsigned char a[4];    //储存用户码、用户反码与键数据码、键数据反码
  11. unsigned int LowTime,HighTime; //储存高、低电平的宽度
  12. unsigned char code string[ ]= {"1602IR-CODE TEST"};
  13. /*****************************************************
  14. 函数功能:延时1ms
  15. ***************************************************/
  16. void delay1ms(unsigned int n)
  17. {
  18.    unsigned char i,j;
  19.    while(n--)
  20.   for(i=0;i<10;i++)
  21.    for(j=0;j<33;j++);  
  22. }
  23. /*********************************************************/
  24. void beep()  //蜂鸣器响一声函数
  25. {
  26.   unsigned char i;
  27.   for (i=0;i<100;i++)   //500HZ的声音  持续100ms
  28.    {
  29.    delay1ms(1);
  30.    BEEP=!BEEP;       //BEEP取反  方波信号
  31.    }
  32.    BEEP=1;           //关闭蜂鸣器
  33.    delay1ms(250);       //延时   
  34. }

  35. /*****************************************************
  36. 函数功能:判断液晶模块的忙碌状态
  37. 返回值:result。result=1,忙碌;result=0,不忙
  38. ***************************************************/
  39. unsigned char BusyTest(void)
  40.   {
  41.     bit result;
  42. RS=0;       //根据规定,RS为低电平,RW为高电平时,可以读状态
  43.     RW=1;
  44.     E=1;        //E=1,才允许读写
  45.     _nop_();   //空操作
  46.     _nop_();
  47.     _nop_();
  48.     _nop_();   //空操作四个机器周期,给硬件反应时间
  49.     result=BF;  //将忙碌标志电平赋给result
  50. E=0;
  51.     return result;
  52.   }
  53. /*****************************************************
  54. 函数功能:将模式设置指令或显示地址写入液晶模块
  55. 入口参数:dictate
  56. ***************************************************/
  57. void WriteInstruction (unsigned char dictate)
  58. {  
  59.     while(BusyTest()==1); //如果忙就等待
  60.   RS=0;                  //根据规定,RS和R/W同时为低电平时,可以写入指令
  61.   RW=0;  
  62.   E=0;                   //E置低电平(根据表8-6,写指令时,E为高脉冲,
  63.                              // 就是让E从0到1发生正跳变,所以应先置"0"
  64.   _nop_();
  65.   _nop_();             //空操作两个机器周期,给硬件反应时间
  66.   P0=dictate;            //将数据送入P0口,即写入指令或地址
  67.   _nop_();
  68.   _nop_();
  69.   _nop_();
  70.   _nop_();               //空操作四个机器周期,给硬件反应时间
  71.   E=1;                   //E置高电平,读指令
  72.   _nop_();
  73.   _nop_();
  74.   _nop_();
  75.   _nop_();               //空操作四个机器周期,给硬件反应时间
  76.    E=0;                  //当E由高电平跳变成低电平时,液晶模块开始执行命令
  77. }

  78. /*****************************************************
  79. 函数功能:指定字符显示的实际地址
  80. 入口参数:x
  81. ***************************************************/
  82. void WriteAddress(unsigned char x)
  83. {
  84.      WriteInstruction(x|0x80); //显示位置的确定方法规定为"80H+地址码x"
  85. }

  86. /*****************************************************
  87. 函数功能:将数据(字符的标准ASCII码)写入液晶模块
  88. 入口参数:y(为字符常量)
  89. ***************************************************/
  90. void WriteData(unsigned char y)
  91. {
  92.     while(BusyTest()==1);
  93.    RS=1;           //RS为高电平,RW为低电平时,可以写入数据
  94.    RW=0;
  95.    E=0;            //E置低电平(根据表8-6,写指令时,E为高脉冲,
  96.                        // 就是让E从0到1发生正跳变,所以应先置"0"
  97.    P0=y;           //将数据送入P0口,即将数据写入液晶模块
  98.    _nop_();
  99.    _nop_();
  100.     _nop_();
  101.      _nop_();       //空操作四个机器周期,给硬件反应时间
  102.    E=1;          //E置高电平 读指令
  103.    _nop_();
  104.    _nop_();
  105.    _nop_();
  106.   _nop_();        //空操作四个机器周期,给硬件反应时间
  107.   E=0;            //当E由高电平跳变成低电平时,液晶模块开始执行命令
  108. }

  109. /*****************************************************
  110. 函数功能:对LCD的显示模式进行初始化设置
  111. ***************************************************/
  112. void LcdInitiate(void)
  113. {
  114.    delay1ms(15);             //延时15ms,首次写指令时应给LCD一段较长的反应时间
  115.    WriteInstruction(0x38);  //显示模式设置:16×2显示,5×7点阵,8位数据接口
  116. delay1ms(5);   //延时5ms 
  117. WriteInstruction(0x38);
  118. delay1ms(5);
  119. WriteInstruction(0x38);
  120. delay1ms(5);
  121. WriteInstruction(0x0C);  //显示模式设置:显示开,有光标,光标闪烁
  122. delay1ms(5);
  123. WriteInstruction(0x06);  //显示模式设置:光标右移,字符不移
  124. delay1ms(5);
  125. WriteInstruction(0x01);  //清屏幕指令,将以前的显示内容清除
  126. delay1ms(5);
  127. }
  128. /************************************************************
  129. 函数功能:对4个字节的用户码和键数据码进行解码
  130. 说明:解码正确,返回1,否则返回0
  131. 出口参数:dat
  132. *************************************************************/
  133. bit DeCode(void)      
  134. {
  135.    
  136.    unsigned char  i,j;
  137. unsigned char temp;    //储存解码出的数据
  138. for(i=0;i<4;i++)      //连续读取4个用户码和键数据码
  139.    {
  140.    for(j=0;j<8;j++)  //每个码有8位数字
  141.     {
  142.           temp=temp>>1;  //temp中的各数据位右移一位,因为先读出的是低位数据         
  143.       TH0=0;         //定时器清0
  144.       TL0=0;         //定时器清0
  145.       TR0=1;         //开启定时器T0
  146.         while(IR==0)   //如果是低电平就等待
  147.                 ;       //低电平计时
  148.         TR0=0;         //关闭定时器T0
  149.       LowTime=TH0*256+TL0;    //保存低电平宽度
  150.       TH0=0;         //定时器清0
  151.       TL0=0;         //定时器清0
  152.       TR0=1;         //开启定时器T0
  153.       while(IR==1)   //如果是高电平就等待
  154.           ;     
  155.       TR0=0;        //关闭定时器T0
  156.       HighTime=TH0*256+TL0;   //保存高电平宽度
  157.       if((LowTime<370)||(LowTime>640))
  158.            return 0;        //如果低电平长度不在合理范围,则认为出错,停止解码   
  159.       if((HighTime>420)&&(HighTime<620))   //如果高电平时间在560微秒左右,即计数560/1.085=516次
  160.               temp=temp&0x7f;       //(520-100=420, 520+100=620),则该位是0
  161.       if((HighTime>1300)&&(HighTime<1800)) //如果高电平时间在1680微秒左右,即计数1680/1.085=1548次
  162.               temp=temp|0x80;       //(1550-250=1300,1550+250=1800),则该位是1
  163.        }               
  164.     a[i]=temp; //将解码出的字节值储存在a[i]                     
  165.     }         
  166.   if(a[2]=~a[3])  //验证键数据码和其反码是否相等,一般情况下不必验证用户码
  167.   return 1;     //解码正确,返回1
  168. }

  169. /*------------------二进制码转换为压缩型BCD码,并显示---------------*/
  170. void two_2_bcd(unsigned char date)
  171. {

  172.    unsigned char temp;
  173.    temp=date;
  174.    date&=0xf0;
  175.    date>>=4;                    //右移四位得到高四位码
  176.    date&=0x0f;                  //与0x0f想与确保高四位为0
  177.    if(date<=0x09)
  178.    {               
  179.      WriteData(0x30+date);            //lcd显示键值高四位
  180.    }
  181.    else
  182.    {
  183.      date=date-0x09;
  184.   WriteData(0x40+date);
  185.    }
  186.    date=temp;
  187.    date&=0x0f;
  188.    if(date<=0x09)
  189.    {
  190.      WriteData(0x30+date);            //lcd显示低四位值
  191.    }
  192.    else
  193.    {
  194.      date=date-0x09;
  195.   WriteData(0x40+date);
  196.    }
  197.    WriteData(0x48);                 //显示字符'H'
  198. }
  199. /************************************************************
  200. 函数功能:1602LCD显示
  201. *************************************************************/
  202. void Disp()
  203. {
  204.     WriteAddress(0x40);  // 设置显示位置为第2行的第1个字
  205.       two_2_bcd(a[0]);
  206.    WriteData(0x20); //空格
  207.    two_2_bcd(a[1]);
  208.     WriteData(0x20);
  209.    two_2_bcd(a[2]);
  210.      WriteData(0x20);
  211.    two_2_bcd(a[3]);

  212. }

  213. /************************************************************
  214. 函数功能:主函数
  215. *************************************************************/
  216. void main()
  217. {
  218.     unsigned char i;
  219. LcdInitiate();         //调用LCD初始化函数
  220.     delay1ms(10);
  221.      WriteInstruction(0x01);//清显示:清屏幕指令
  222.   WriteAddress(0x00);  // 设置显示位置为第一行的第1个字
  223.     i = 0;
  224.   while(string[i] != '\0')    //'\0'是数组结束标志
  225.    {      // 显示字符
  226.     WriteData(string[i]);
  227.     i++;
  228.    }
  229. EA=1;        //开启总中断
  230.    EX0=1;       //开外中断0 P3^2口
  231.    ET0=1;       //定时器T0中断允许
  232.    IT0=1;       //外中断的下降沿触发
  233.     TMOD=0x01;   //使用定时器T0的模式1
  234. TR0=0;       //定时器T0关闭

  235.    while(1);   //等待红外信号产生的中断
  236.   
  237. }

  238. /************************************************************
  239. 函数功能:红外线触发的外中断处理函数
  240. *************************************************************/
  241. void Int0(void) interrupt 0
  242.   {
  243.      EX0=0;      //关闭外中断0,不再接收二次红外信号的中断,只解码当前红外信号
  244.    TH0=0;      //定时器T0的高8位清0
  245.    TL0=0;      //定时器T0的低8位清0
  246.    TR0=1;     //开启定时器T0
  247.    while(IR==0);          //如果是低电平就等待,给引导码低电平计时
  248.    TR0=0;                //关闭定时器T0   
  249.    LowTime=TH0*256+TL0;  //保存低电平时间
  250.    TH0=0;      //定时器T0的高8位清0
  251.    TL0=0;      //定时器T0的低8位清0
  252.    TR0=1;     //开启定时器T0
  253.    while(IR==1);  //如果是高电平就等待,给引导码高电平计时
  254.    TR0=0;        //关闭定时器T0
  255.    HighTime=TH0*256+TL0; //保存引导码的高电平长度
  256.      if((LowTime>7800)&&(LowTime<8800)&&(HighTime>3600)&&(HighTime<4700))
  257.    {
  258.       //如果是引导码,就开始解码,否则放弃,引导码的低电平计时
  259.         //次数=9000us/1.085=8294, 判断区间:8300-500=7800,8300+500=8800.
  260.        if(DeCode()==1) // 执行遥控解码功能
  261.    {
  262.   
  263.     Disp();//调用1602LCD显示函数
  264.     beep();//蜂鸣器响一声 提示解码成功
  265.    }
  266.    }
  267.    EX0=1;   //开启外中断EX0
  268.   }

复制代码


分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏2 分享淘帖 顶 踩
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表