找回密码
 立即注册

QQ登录

只需一步,快速开始

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

只用一个单片机定时器的红外接收解码程序

[复制链接]
跳转到指定楼层
楼主
ID:110653 发表于 2016-3-28 23:29 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
本帖最后由 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. ///*************        功能说明        **************        
  11. //红外接收程序。适用于市场上用量最大的HT6121/6122及其兼容IC的编码。
  12. //当遥控器用户码与程序定义的用户码不同时,程序会将遥控器的用户码一起从串口输出。
  13. //使用模拟串口发送用户码与键码,波特率9600,接收端要求使用字符格式显示。        
  14. /******************************************/        
  15. #include        "STC15W4K.H"
  16. #define MAIN_Fosc                22.1184        // 定义主时钟, 红外接收会自动适应,5~36MHZ,但模拟串口输出部分不能自动适应。

  17. #define        User_code                0xFD02                // 定义红外接收用户码  
  18. sbit        Ir_Pin = P3^6;                        // 定义红外接收输入端口
  19. sbit        TXD1 = P3^1;                          // 定义模拟串口发送脚
  20. /*************        用户系统配置        **************/
  21. #define TIME                125                              // 选择定时器时间125us, 红外接收要求在60us~250us之间
  22. /*************        以下宏定义用户请勿修改        **************/
  23. #define Timer0_Reload (unsigned int)(65536 - (TIME * MAIN_Fosc /12.0))  // 定时器初值
  24.                                    
  25. /*************        本地变量声明        **************/
  26. unsigned char        IR_SampleCnt;                // 采样次数计数器,通用定时器对红外口检测次数累加记录
  27. unsigned char        IR_BitCnt;                        // 记录位数
  28. unsigned char        IR_UserH;                        // 用户码(地址)高字节
  29. unsigned char        IR_UserL;                        // 用户码(地址)低字节
  30. unsigned char        IR_data;                        // 键原码
  31. unsigned char        IR_DataShit;                // 键反码
  32. unsigned char        IR_code;                        // 红外键码                 
  33. bit                Ir_Pin_temp;                        // 记录红外引脚电平的临时变量
  34. bit                IR_Sync;                                // 同步标志(1——已收到同步信号,0——没收到)
  35. bit                IrUserErr;                            // 用户码错误标志
  36. bit                IR_OK;                        // 完成一帧红外数据接收的标志(0,未收到,1,收到一帧完整数据)

  37. /******************** 红外采样时间宏定义, 用户不要随意修改        *******************/
  38. //数据格式: Synchro,AddressH,AddressL,data,/data, (total 32 bit).
  39. #if ((TIME <= 250) && (TIME >= 60))                        // TIME决定测量误差,TIME太大防错能力降低,TIME太小会干扰其它中断函数执行。
  40.         #define        IR_sample                        TIME                // 定义采样时间,在60us~250us之间,
  41. #endif

  42. #define IR_SYNC_MAX                (15000/IR_sample)        // 同步信号SYNC 最大时间 15ms(标准值9+4.5=13.5ms)
  43. #define IR_SYNC_MIN                (9700 /IR_sample)        // 同步信号SYNC 最小时间 9.5ms,(连发信号标准值9+2.25=11.25ms)
  44. #define IR_SYNC_DIVIDE        (12375/IR_sample)        // 区分13.5ms同步信号与11.25ms连发信号,11.25+(13.5-11.25)/2=12.375ms
  45. #define IR_DATA_MAX                (3000 /IR_sample)        // 数据最大时间3ms    (标准值2.25 ms)
  46. #define IR_DATA_MIN                (600  /IR_sample)        // 数据最小时,0.6ms   (标准值1.12 ms)
  47. #define IR_DATA_DIVIDE        (1687 /IR_sample)        // 区分数据 0与1,1.12+ (2.25-1.12)/2 =1.685ms
  48. #define IR_BIT_NUMBER                32                                // 32位数据

  49. //****************************  红外接收模块  ********************************************
  50. // 信号第1个下降沿时刻清零计数器并让计数器从0开始计数,第2个下降沿时刻计算计数器运行时间
  51. // 因此检测的是每个信号从低电平开始到高电平结束这段时间,也就是脉冲周期。
  52. void IR_RX_HT6121(void)
  53. {
  54.         unsigned char        SampleTime;                                // 信号周期
  55.         IR_SampleCnt++;                                                        // 定时器对红外口检测次数

  56.         F0 = Ir_Pin_temp;                                                // 保存前一次此程序扫描到的红外端口电平         
  57.         Ir_Pin_temp = Ir_Pin;                                        // 读取当前红外接收输入端口电平
  58.         if(F0 && !Ir_Pin_temp)                                        // 前一次采样高电平并且当前采样低电平,说明出现了下降沿
  59.         {
  60.                 SampleTime = IR_SampleCnt;                        // 脉冲周期
  61.                 IR_SampleCnt = 0;                                        // 出现了下降沿则清零计数器
  62.                 //******************* 接收同步信号 **********************************
  63.                 if(SampleTime > IR_SYNC_MAX)                IR_Sync = 0;        // 超出最大同步时间, 错误信息。
  64.                 else if(SampleTime >= IR_SYNC_MIN)                // SYNC
  65.                 {
  66.                         if(SampleTime >= IR_SYNC_DIVIDE)    // 区分13.5ms同步信号与11.25ms连发信号
  67.                         {
  68.                                 IR_Sync = 1;                                    // 收到同步信号 SYNC
  69.                                 IR_BitCnt = IR_BIT_NUMBER;          // 赋值32(32位有用信号)
  70.                         }
  71.                 }
  72.                 //********************************************************************
  73.                 else if(IR_Sync)                                                // 已收到同步信号 SYNC
  74.                 {
  75.                         if((SampleTime < IR_DATA_MIN)|(SampleTime > IR_DATA_MAX)) IR_Sync=0;        // 数据周期过短或过长,错误
  76.                         else
  77.                         {
  78.                                 IR_DataShit >>= 1;                                        // 键反码右移1位(发送端是低位在前,高位在后的格式)
  79.                                 if(SampleTime >= IR_DATA_DIVIDE)        IR_DataShit |= 0x80;        // 区别是数据 0还是1

  80.                                 //***********************  32位数据接收完毕 ****************************************
  81.                                 if(--IR_BitCnt == 0)                                 
  82.                                 {
  83.                                         IR_Sync = 0;                                        // 清除同步信号标志
  84.                                         if(~IR_DataShit == IR_data)                // 判断数据正反码
  85.                                         {
  86.                                                 if((IR_UserH == (User_code / 256)) && IR_UserL == (User_code % 256))
  87.                                                 {
  88.                                                                 IrUserErr = 0;            // 用户码正确
  89.                                                 }
  90.                                                 else        IrUserErr = 1;            // 用户码错误
  91.                                                         
  92.                                                 IR_code      = IR_data;                // 键码值
  93.                                                 IR_OK   = 1;                            // 数据有效
  94.                                         }
  95.                                 }
  96.                                 // 格式:  用户码L —— 用户码H —— 键码 —— 键反码
  97.                                 // 功能:  将 “用户码L —— 用户码H —— 键码”通过3次接收交换后存入对应字节,
  98.                                 //         这样写代码可以节省内存RAM占用,但是不如用数组保存好理解。        
  99.                                 //         键反码前面部分代码已保存好了        :IR_DataShit |= 0x80;                  
  100.                                 //**************************** 将接收的************************************************
  101.                                 else if((IR_BitCnt & 7)== 0)                // 1个字节接收完成
  102.                                 {
  103.                                         IR_UserL = IR_UserH;                        // 保存用户码高字节
  104.                                         IR_UserH = IR_data;                                // 保存用户码低字节
  105.                                         IR_data  = IR_DataShit;                        // 保存当前红外字节
  106.                                 }
  107.                         }
  108.                 }  
  109.         }
  110. }

  111. /**************** Timer0初始化函数 ******************************/
  112. void InitTimer0(void)
  113. {
  114.         TMOD = 0x01;                         // 16位计数方式.
  115.         TH0 = Timer0_Reload / 256;         
  116.         TL0 = Timer0_Reload % 256;
  117.         ET0 = 1;                                         
  118.         TR0 = 1;
  119.         EA  = 1;
  120. }  

  121. /********************** Timer0中断函数************************/
  122. void timer0 (void) interrupt 1
  123. {
  124.         IR_RX_HT6121();
  125.         TH0 = Timer0_Reload / 256;           // 重装定时器初值   
  126.         TL0 = Timer0_Reload % 256;           // 重装定时器初值   
  127. }

  128. /********************** 模拟串口相关函数************************/
  129. void delay104us(void)
  130. {
  131.    unsigned char i,j,k;
  132.    for(i=1;i>0;i--)       // 注意后面没分号
  133.    for(j=3;j>0;j--)       // 注意后面没分号
  134.    for(k=189;k>0;k--);    // 注意后面有分号        101
  135. }

  136. //模拟串口发送
  137. void Tx1Send(unsigned char dat)                //9600,N,8,1                发送一个字节
  138. {
  139.         unsigned char        i;
  140.         EA = 0;
  141.         TXD1 = 0;
  142.         delay104us();
  143.         for(i=0; i<8; i++)
  144.         {
  145.                 if(dat & 1)                TXD1 = 1;
  146.                 else                        TXD1 = 0;
  147.                 dat >>= 1;
  148.                 delay104us();
  149.         }
  150.         TXD1 = 1;
  151.         EA = 1;
  152.         delay104us();
  153.         delay104us();
  154. }

  155. void PrintString(unsigned char code *puts)                    // 发送一串字符串
  156. {
  157.     for (; *puts != 0;        puts++)  Tx1Send(*puts);         // 遇到停止符0结束
  158. }

  159. /********************* 十六进制转ASCII函数 *************************/
  160. unsigned char        HEX2ASCII(unsigned char dat)
  161. {
  162.         dat &= 0x0f;
  163.         if(dat <= 9)        return (dat + '0');        //数字0~9
  164.         return (dat - 10 + 'A');                        //字母A~F
  165. }

  166. /********************* 主函数 *************************/
  167. void main(void)
  168. {
  169.         InitTimer0();                                    // 初始化Timer0        
  170.         PrintString("定时器0初始化完毕\r\n");        // 上电后串口发送一条提示信息

  171.         while(1)
  172.         {
  173.                 if(IR_OK)                                     // 接收到一帧完整的红外数据
  174.                 {
  175.                         PrintString("红外键码: 0x");                // 提示红外键码
  176.                         Tx1Send(HEX2ASCII(IR_code >> 4));        // 键码高半字节
  177.                         Tx1Send(HEX2ASCII(IR_code));                // 键码低半字节
  178.                         if(IrUserErr)                                                // 用户码错误,则发送用户码
  179.                         {
  180.                                 Tx1Send(' ');                                        // 发空格
  181.                                 Tx1Send(' ');                                        // 发空格
  182.                                 PrintString("用户码: 0x");                // 提示用户码
  183.                                 Tx1Send(HEX2ASCII(IR_UserH >> 4));        // 用户码高字节的高半字节
  184.                                 Tx1Send(HEX2ASCII(IR_UserH));                // 用户码高字节的低半字节
  185.                                 Tx1Send(HEX2ASCII(IR_UserL >> 4));        // 用户码低字节的高半字节
  186.                                 Tx1Send(HEX2ASCII(IR_UserL));                // 用户码低字节的低半字节
  187.                         }
  188.                         Tx1Send(0x0d);                          // 发回车
  189.                         Tx1Send(0x0a);                          // 发回车
  190.                         IR_OK = 0;                              // 清除IR键按下标志
  191.                 }
  192.         }
  193. }
复制代码


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

使用道具 举报

沙发
ID:154878 发表于 2016-12-14 23:38 | 只看该作者
请问一下,移值到C51 只要修改一下引脚和晶振就可以了么?
回复

使用道具 举报

板凳
ID:365500 发表于 2018-8-10 17:27 | 只看该作者
好厉害,,,,不知道什么时候才有这个水平啊!
回复

使用道具 举报

地板
ID:60263 发表于 2019-4-6 00:11 | 只看该作者
STC官方网站上有这个例程,一样的,可以下载,我用的STC15F104实测可用。
回复

使用道具 举报

5#
ID:363812 发表于 2019-10-17 18:25 | 只看该作者
大哥  我用两个新唐的8位单片机做红外对射去解码   套用这个怎么弄?   谢谢大哥回下
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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