找回密码
 立即注册

QQ登录

只需一步,快速开始

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

IR中断方式的单片机红外解码程序

[复制链接]
跳转到指定楼层
楼主
ID:110653 发表于 2016-3-28 23:23 | 只看该作者 回帖奖励 |正序浏览 |阅读模式
本帖最后由 hubaba 于 2016-3-28 23:31 编辑


1电路图

1号单片机与红外接收头连接电路如图所示,红外接收头一般都可互换使用。




电路和介绍详见:http://www.51hei.com/bbs/dpj-47282-1.html (在这贴附件中可下载完整的4个红外解码的源码压缩包)
程序如下:
  1. /****************《51单片机轻松入门-基于STC15W4K系列》配套例程 *************
  2. ★★★★★★★★★★★★★★★★★★★★★★★★
  3. 《51单片机轻松入门-基于STC15W4K系列》 一书已经由北航出版社正式出版发行。
  4.   作者亲手创作的与教材配套的51双核实验板(2个MCU)对程序下载、调试、仿真方便,不需要外部
  5.   仿真器与编程器,这种设计方式彻底解决了系统中多个最高优先级谁也不能让谁的中断竞争问题。
  6.   QQ群:STC51-STM32(3) :515624099 或 STC51-STM32(2):99794374。
  7.         验证信息:STC15单片机
  8.   邮箱:xgliyouquan@126.com
  9.   ★★★★★★★★★★★★★★★★★★★★★★★★*/

  10. ////////////////////////////   main.c  ////////////////////////////////////////
  11. // 红外接收数据,中断方式,并通过串口发送接收到的4字节,晶振:22.118400 MHz
  12. // 接收头信号引脚P5.5,串口波特率9600
  13. ///////////////////////////////////////////////////////////////////////////
  14. #include "STC15W4K.H"
  15. sbit Ir_Pin = P3^6;                        // 红外接收头信号输出脚
  16. unsigned char Ir_Buf[4];   // 用于保存解码结果
  17. bit IRflag = 0;            // 红外接收标志,收到一帧正确数据后置1
  18. void UART_init(void)                      // 9600@22.1184MHz
  19. {                  
  20.         //下面代码设置定时器1
  21.         TMOD = 0x20;           // 0010 0000 定时器1工作于方式2(8位自动重装方式)
  22.         TH1  = 0xFA;           // 波特率:9600 /22.1184MHZ
  23.         TL1  = 0xFA;           // 波特率:9600 /22.1184MHZ
  24.         TR1  = 1;
  25.         //下面代码设置定串口
  26.         AUXR = 0x00;             // 很关键,使用定时器1作为波特率发生器,S1ST2=0
  27.         SCON = 0x50;         // 0101 0000 SM0.SM1=01(最普遍的8位通信),REN=1(允许接收)
  28. }        

  29. //  UART发送一字节        
  30. void UART_Send_Byte(unsigned char dat)
  31. {
  32.         SBUF = dat;
  33.         while (TI == 0);
  34.         TI = 0;             // 此句可以不要,不影响后面数据的发送,只供代码查询数据是否发送完成
  35. }
  36. // 获取低电平时间 (其实是16位计数器的计数值,STC15系列定时器默认为16位自动重装方式)
  37. unsigned int Ir_Get_Low()
  38. {
  39.         TL0 = 0;                 // 清空16位计数器0
  40.         TH0 = 0;                 // 清空16位计数器0
  41.         TR0 = 1;                 // 计数器0开始运行
  42.         while (!Ir_Pin && (TH0<0x80));          // 信号引脚变成高或低电平时间>17ms退出(只要>12ms即可)
  43.                                       // 0x8000=32768,  32768*0.54253uS=17777.62 uS            
  44.         TR0 = 0;                          // 这里 ! 优先级大于&&     
  45.         return (TH0 * 256 + TL0);                  // 返回16位计数器的计数值。
  46. }

  47. // 获取高电平时间(其实是16位计数器的计数值,STC15系列定时器默认为16位自动重装方式)
  48. unsigned int Ir_Get_High()
  49. {
  50.         TL0 = 0;                    // 清空16位计数器0
  51.         TH0 = 0;                    // 清空16位计数器0
  52.         TR0 = 1;
  53.         while (Ir_Pin && (TH0<0x40));           // 信号引脚变成低电平或高电平时间>17ms退出
  54.         TR0 = 0;
  55.         return (TH0 * 256 + TL0);
  56. }  

  57. // 外部中断初始化
  58. void int2_init()      // P3.6引脚即为外部中断2
  59. {
  60.         INT_CLKO|=0x10;          // 开启外中断2
  61.         EA = 1;                          // 总开关
  62. }

  63. void int2_isr() interrupt 10      // 外部中断2中断函数
  64. {
  65.         unsigned char i,j;
  66.         unsigned char DAT;                // 临时存放接收到的字节,接收字节无误后再存入数组使用
  67.         unsigned int time;

  68.         // 接收并判定引导码的9ms 低电平
  69.         time = Ir_Get_Low();
  70.         if ((time < 15667) || (time > 17510))   
  71.         {                         // 引导脉冲低电平8500~9500us,T=12/22.1184=0.54253uS
  72.                                  // 8500/0.54253uS=15667.3    9500/0.54253uS=17510.5
  73.                 IE1 = 0; // 退出前清除可能新的一次INT1 外中断引发的标志
  74.                 return;
  75.         }

  76.         // 接收并判定引导码的4.5ms 高电平
  77.         time = Ir_Get_High();
  78.         if ((time < 7372) || (time > 9216))   //引导脉冲高电平4000~5000us
  79.         {                                // 4000/0.54253uS=7372.8    5000/0.54253uS=9216
  80.         IE1 = 0;    // 退出前清除可能新的一次INT1 外中断引发的标志
  81.                 return;
  82.         }

  83.         // 接收后续的4 字节数据
  84.         for (i=0; i<4; i++) // 循环接收4 个字节
  85.         {
  86.                 for (j=0; j<8; j++)   // 每个字节8位
  87.                 {
  88.                         time = Ir_Get_Low();     // 接收每位560us 低电平
  89.                         if ((time < 626) || (time > 1438))  //  340~780us
  90.                         {                                // 340/0.54253uS=626.7    780/0.54253uS=1437.7
  91.                 IE1 = 0;    // 退出前清除可能新的一次INT1 外中断引发的标志
  92.                                 return;
  93.                         }

  94.                         time = Ir_Get_High();         // 接收每位560us或1690us高电平时间
  95.             if ((time>626) && (time<1438))  // 时间范围为340-780us(中心值560us)   
  96.             {
  97.                  DAT >>= 1;   // 因低位在先,所以数据右移,移入的最高位为0
  98.                         }
  99.             else if ((time>2728) && (time<3502)) // 时间判定范围为1480~1900us(中心值1690us)
  100.             {                                  // 1480/0.54253uS=2727.9   1900/0.54253uS=3502.1
  101.                                 DAT >>= 1;    // 因低位在先,所以数据右移,移入的最高位为0
  102.                                 DAT |= 0x80;  // 最高位置1
  103.                         }
  104.             else              // 不在上述范围内则说明为误码,直接退出
  105.                         {
  106.                                 IE1 = 0;      // 退出前清除可能新的一次INT1 外中断引发的标志
  107.                                 return;
  108.                         }                                    
  109.                 }                          // 单个字节处理完毕
  110.                 Ir_Buf[i]=DAT;                  // 传输正确后才放入数组
  111.         }
  112.         IRflag = 1;               // 接收完毕后设置标志
  113.         IE1 = 0;                  // 退出前清除可能新的一次INT1 外中断引发的标志
  114. }

  115. void main()
  116. {
  117.         UART_init();;    // 串口初始化
  118.         int2_init();         // 外中断 2 初始化(红外接收引脚)  
  119.         while (1)
  120.         {
  121.                 if (IRflag)  // 接收到红外数据时发计算机显示
  122.                 {
  123.                         IRflag = 0;
  124.                         UART_Send_Byte(Ir_Buf[0]);           // 用户码低字节
  125.                         UART_Send_Byte(Ir_Buf[1]);           // 用户码高字节
  126.                         UART_Send_Byte(Ir_Buf[2]);           // 键码
  127.                         UART_Send_Byte(Ir_Buf[3]);           // 键反码
  128.                 }
  129.         }
  130. }
复制代码


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

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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