stc8a8k64s4a12红外解码LCD1602显示
在原来解码数码管的程序上修复了8位LED灯初始显示任意值bug,新增加了解码显示到LCD屏幕与pc串口端完全同步,可以通过三种有效解码方式更加精确的对支持NEC协议的所有红外遥控设备解码,此外对于其他型号及平台移植更为方便,希望有用到的 积极下载共享,有好的独特想法的可以提出来共同讨论,积极指正批评。
单片机源程序如下:
- ////////////////////////////////////////////////////////////
- // 红外接收数据,查询方式,并通过串口发送输出
- // 晶振:22.1184MHz ,波特率:9600
- ////////////////////////////////////////////////////////////
- #include <STC8.H>
- #include "LCD1602.h"
- #define LED P0
- sbit Ir_Pin = P2^7; // 红外接收头信号输出脚
- extern delay(uint time);
- uchar code lhc_table[16]="0123456789ABCDEF";
- uchar tempdata[16];
- uchar Ir_Buf[4]; // 用于保存解码结果(Ir_Buf[0]--用户码L,Ir_Buf[3] --键反码)
- // 获取低电平时间 (其实是16位计数器的计数值,STC15系列定时器默认为16位自动重装方式)
- uint Ir_Get_Low()
- {
- TL0 = 0; // 清空16位计数器0
- TH0 = 0; // 清空16位计数器0
- TR0 = 1; // 计数器0开始运行
- while (!Ir_Pin && (TH0<0x80)); // 信号引脚变成高或低电平时间>17ms退出(只要>12ms即可)
- // 0x8000=32768, 32768*0.54253uS=17777.62 uS
- TR0 = 0; // 这里 ! 优先级大于&&
- return (TH0 * 256 + TL0); // 返回16位计数器的计数值。
- }
- // 获取高电平时间(其实是16位计数器的计数值,STC15系列定时器默认为16位自动重装方式)
- uint Ir_Get_High()
- {
- TL0 = 0; // 清空16位计数器0
- TH0 = 0; // 清空16位计数器0
- TR0 = 1;
- while (Ir_Pin && (TH0<0x40)); // 信号引脚变成低电平或高电平时间>17ms退出
- TR0 = 0;
- return (TH0 * 256 + TL0);
- }
- void UART_init(void) // 9600@22.1184MHz
- {
- //下面代码设置定时器1
- TMOD = 0x20; // 0010 0000 定时器1工作于方式2(8位自动重装方式)
- TH1 = 0xFA; // 波特率:57600 /22.1184MHZ
- TL1 = 0xFA; // 波特率:57600 /22.1184MHZ
- TR1 = 1;
- //下面代码设置定串口
- AUXR = 0x00; // 很关键,使用定时器1作为波特率发生器,S1ST2=0
- SCON = 0x50; // 0101 0000 SM0.SM1=01(最普遍的8位通信),REN=1(允许接收)
- }
-
- // UART发送一字节
- void UART_Send_Byte(uchar dat)
- {
- SBUF = dat;
- while (TI == 0);
- TI = 0; // 此句可以不要,不影响后面数据的发送,只供代码查询数据是否发送完成
- }
- void IR_work() //红外解码程序
- {
- tempdata[0]=lhc_table[Ir_Buf[0]/16]; //处理客户码
- tempdata[1]=lhc_table[Ir_Buf[0]%16];
- tempdata[2]='-';
- tempdata[3]=lhc_table[Ir_Buf[1]/16]; //处理客户码
- tempdata[4]=lhc_table[Ir_Buf[1]%16];
- tempdata[5]='-';
- tempdata[6]=lhc_table[Ir_Buf[2]/16]; //处理数据码
- tempdata[7]=lhc_table[Ir_Buf[2]%16];
- tempdata[8]='-';
- tempdata[9]=lhc_table[Ir_Buf[3]/16]; //处理数据反码
- tempdata[10]=lhc_table[Ir_Buf[3]%16];
- LCD_write_string(5,1,tempdata);
- LED=~Ir_Buf[2];
- }
- void main()
- {
- uint time;
- uchar i,j;
- LED=0xff;
- UART_init();
- UART_Send_Byte(0X55); // 测试串口工作是否正常
- LCD_init();
- delay(20); //延时有助于稳定
- LCD_clear();
-
- LCD_write_string(0,0,"LHC-DIY-STC8");
- LCD_write_string(0,1,"code:");
- while (1)
- {
- start:
- ///////////// 接收同步信号 ///////////
- while (Ir_Pin); // 等待低电平出现
- time = Ir_Get_Low(); // 低电平区间16位计数器的计数值
- if ((time < 15667) || (time > 17510)) goto start;
- // 引导脉冲低电平8500~9500us,T=12/22.1184=0.54253uS
- // 8500/0.54253uS=15667.3 9500/0.54253uS=17510.5
- time = Ir_Get_High();
- if ((time < 7372) || (time > 9216)) goto start; // 引导脉冲高电平4000-5000us
- // 4000/0.54253uS=7372.8 5000/0.54253uS=9216
- ////////// 接收后续的4 字节数据 ////////
- for (i=0; i<4; i++) // 4个字节
- {
- for (j=0; j<8; j++) // 每个字节8位
- {
- time = Ir_Get_Low(); // 接收每位560us 低电平
- if ((time < 626) || (time > 1438)) goto start; // 340-780us
- // 340/0.54253uS=626.7 780/0.54253uS=1437.7
- time = Ir_Get_High(); // 接收每位560us或1690us高电平时间
- if ((time>626) && (time<1438)) // 时间范围为340-780us(中心值560us),
- {
- Ir_Buf[i] >>= 1; // 因低位在先,所以数据右移,移入的最高位为0
- }
- else if ((time>2728) && (time<3502))
- { // 时间判定范围为1480~1900us(中心值1690us)
- // 1480/0.54253uS=2727.9 1900/0.54253uS=3502.1
- Ir_Buf[i] >>= 1; // 因低位在先,所以数据右移,移入的最高位为0
- Ir_Buf[i] |= 0x80; // 最高位置1
- }
- else // 不在上述范围内则说明为误码,直接退出
- {
- goto start;
- }
- }
- }
- UART_Send_Byte(Ir_Buf[0]); // 用户码低字节
- UART_Send_Byte(Ir_Buf[1]); // 用户码高字节
- UART_Send_Byte(Ir_Buf[2]); // 键码
- UART_Send_Byte(Ir_Buf[3]); // 键反码
- IR_work();
- }
- }
复制代码
所有资料51hei提供下载:
03 LCD1602显示.7z
(33.15 KB, 下载次数: 144)
|