找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 4435|回复: 10
收起左侧

基于51单片机的空调红外解码程序

  [复制链接]
ID:499469 发表于 2019-6-4 20:43 | 显示全部楼层 |阅读模式
基于51单片机的红外解码程序   1602显示屏  可以显示8位16进制 38k的遥控都可以,制作万能遥控器的话这个程序可以起到作用

制作出来的实物图如下:
1.jpg

单片机源程序如下:
  1. /*********************************************************************************************
  2. 程序名:    DHT11-1602
  3. 编写人:    fby
  4. 编写时间:  2019年3月21日
  5. 硬件支持:  STC89C52RC  DHT11  LCD1602
  6. 接口说明:  DHT11:VCC DATA NC GND
  7. *********************************************************************************************
  8. 说明:1、单片机是STC89C52RC,晶振频率11.0592MHz,实验成功。
  9.           2、好东西要共享,大家随意折腾~~
  10. *********************************************************************************************/
  11. #include<reg51.h>
  12. #include<intrins.h>
  13. #include<stdio.h>

  14. #define uchar unsigned char        //无符号字符型类型数据
  15. #define uint unsigned int //无符号整型数据
  16.         #define ulong unsigned long //无符号整型数据
  17.         #define xiao float //xiaoshu
  18. #define IODATA  P0
  19. unsigned char IrValue[12]; //接收收据
  20. unsigned char Time;

  21. sbit RS=P2^6; //定义rs端口
  22. sbit RW=P2^5; //定义rw端口
  23. sbit E=P2^7;
  24.         
  25. sbit IRIN=P3^2;//红外数据io

  26.               
  27. /*******************************************************************************
  28. * 函数名         : DelayMs()
  29. * 函数功能                   : 延时
  30. * 输入           : x
  31. * 输出                  : 无
  32. *******************************************************************************/

  33. void DelayMs(unsigned int x)   //0.14ms误差 0us
  34. {
  35. unsigned char i;
  36.   while(x--)
  37. {
  38.   for (i = 0; i<13; i++)
  39. {}
  40. }
  41. }
  42. /*******************************************************************************
  43. * 函数名         : IrInit()
  44. * 函数功能                   : 初始化红外线接收
  45. * 输入           : 无
  46. * 输出                  : 无
  47. *******************************************************************************/

  48. void IrInit()
  49. {
  50.         IT0=1;//下降沿触发
  51.         EX0=1;//打开中断0允许
  52.         EA=1;        //打开总中断

  53.         IRIN=1;//初始化端口
  54. }
  55. /*******************************************************************************
  56. * 函数名         : ReadIr()
  57. * 函数功能                   : 读取红外数值的中断函数
  58. * 输入           : 无
  59. * 输出                  : 无
  60. *******************************************************************************/

  61. void ReadIr() interrupt 0
  62. {
  63.         unsigned char j,k;
  64.         unsigned int err;
  65.         Time=0;                                         
  66.         DelayMs(70);

  67.         if(IRIN==0)                //确认是否真的接收到正确的信号
  68.         {         
  69.                
  70.                 err=1000;                                //1000*10us=10ms,超过说明接收到错误的信号
  71.                 /*当两个条件都为真是循环,如果有一个条件为假的时候跳出循环,免得程序出错的时
  72.                 侯,程序死在这里*/        
  73.                 while((IRIN==0)&&(err>0))        //等待前面9ms的低电平过去                  
  74.                 {                        
  75.                         DelayMs(1);
  76.                         err--;
  77.                 }
  78.                 if(IRIN==1)                        //如果正确等到9ms低电平
  79.                 {
  80.                         err=500;
  81.                         while((IRIN==1)&&(err>0))                 //等待4.5ms的起始高电平过去
  82.                         {
  83.                                 DelayMs(1);
  84.                                 err--;
  85.                         }
  86.                         for(k=0;k<4;k++)                //共有4组数据
  87.                         {                                
  88.                                 for(j=0;j<8;j++)        //接收一组数据
  89.                                 {

  90.                                         err=60;               
  91.                                         while((IRIN==0)&&(err>0))//等待信号前面的560us低电平过去
  92. //                                        while (!IRIN)
  93.                                         {
  94.                                                 DelayMs(1);
  95.                                                 err--;
  96.                                         }
  97.                                         err=500;
  98.                                         while((IRIN==1)&&(err>0))         //计算高电平的时间长度。
  99.                                         {
  100.                                                 DelayMs(1);//0.14ms
  101.                                                 Time++;
  102.                                                 err--;
  103.                                                 if(Time>30)
  104.                                                 {
  105.                                                         EX0=1;
  106.                                                         return;
  107.                                                 }
  108.                                         }
  109.                                         IrValue[k]>>=1;         //k表示第几组数据
  110.                                         if(Time>=8)                        //如果高电平出现大于565us,那么是1
  111.                                         {
  112.                                                 IrValue[k]|=0x80;
  113.                                         }
  114.                                         Time=0;                //用完时间要重新赋值                                                        
  115.                                 }
  116.                         }
  117.                 }
  118.                 if(IrValue[2]!=~IrValue[3])
  119.                 {
  120.                         return;
  121.                 }
  122.         }                        
  123. }


  124.    

  125. void sjcl()
  126. {
  127.                 IrValue[4]=IrValue[0]>>4;                          //高位  用户码
  128.                 IrValue[5]=IrValue[0]&0x0f;                //低位        

  129.     IrValue[6]=IrValue[1]>>4;                          //高位   用户反码
  130.                 IrValue[7]=IrValue[1]&0x0f;                //低位        

  131.     IrValue[8]=IrValue[2]>>4;                          //高位   数据码
  132.                 IrValue[9]=IrValue[2]&0x0f;                //低位

  133.     IrValue[10]=IrValue[3]>>4;                          //高位  数据反码
  134.                 IrValue[11]=IrValue[3]&0x0f;                //低位        

  135. }

  136.         

  137. /*********************************************************************************************
  138. 函数名:延时函数
  139. 调  用:Delay30us();/Delay20ms();/Delay1s();
  140. 参  数:无
  141. 返回值:无
  142. 结  果:延时相应时间
  143. 备  注:振荡晶体为11.0592MHz(所有延时函数Copy自STC-ISP(v6.85).exe)
  144. **********************************************************************************************/
  145. void Delay30us()                //@11.0592MHz-40
  146. {
  147.         unsigned char i;

  148.         i = 11;
  149.         while (--i);
  150. }


  151. void Delay20ms()                //@11.0592MHz
  152. {
  153.         unsigned char i, j;

  154.         i = 36;
  155.         j = 217;
  156.         do
  157.         {
  158.                 while (--j);
  159.         } while (--i);
  160. }


  161. void Delay1s()                //@11.0592MHz
  162. {
  163.         unsigned char i, j, k;

  164.         _nop_();
  165.         i = 8;
  166.         j = 1;
  167.         k = 243;
  168.         do
  169.         {
  170.                 do
  171.                 {
  172.                         while (--k);
  173.                 } while (--j);
  174.         } while (--i);
  175. }





  176. /********************************************************************
  177. * 名称 : bit Busy(void)
  178. * 功能 : 这个是一个读状态函数,读出函数是否处在忙状态
  179. * 输入 : 输入的命令值
  180. * 输出 : 无
  181. ***********************************************************************/
  182. void Busy(void)
  183. {
  184. bit busy_flag = 1;
  185. IODATA  = 0xff;
  186. RS = 0;          //指令关
  187. Delay30us();
  188. RW = 1;          //读状态
  189. Delay30us();
  190. E = 1;        //使能高电平
  191. while(1)
  192. {
  193. busy_flag = (bit)(IODATA  & 0x80); //强制取最高位
  194. if(busy_flag == 0)
  195. {
  196. break;
  197. }
  198. }
  199. E = 0;           //E 1->0,rs=0,rw=1,
  200.            //将数据寄存器内的数据读到DB0~DB7,分别将状态标志BF和地址计数器(AC)内容读到DB7和DB6~DB0。

  201. }

  202. /********************************************************************
  203. * 名称 : lcd1602_write_cmd(uchar del)
  204. * 功能 : 1602命令函数
  205. * 输入 : 输入的命令值
  206. * 输出 : 无
  207. ***********************************************************************/
  208. void lcd1602_write_cmd(uchar del)
  209. {
  210. RS = 0;
  211. Delay30us();
  212. RW = 0;
  213. Delay30us();
  214. E = 0;
  215. Delay30us();
  216. IODATA  = del;
  217. Delay30us();
  218. E = 1;
  219. Delay30us();
  220. E = 0;                        //E 1->0,rs=0,rw=0, 将指令代码写入指令寄存器中
  221. }

  222. /********************************************************************
  223. * 名称 : lcd1602_write_data(uchar del)
  224. * 功能 : 1602写数据函数
  225. * 输入 : 需要写入1602的数据
  226. * 输出 : 无
  227. ***********************************************************************/

  228. void lcd1602_write_data(uchar del)
  229. {
  230. Busy();      
  231. Delay30us();
  232. RS = 1;
  233. Delay30us();
  234. RW = 0;
  235. Delay30us();
  236. E = 0;
  237. Delay30us();
  238. IODATA  = del;
  239. Delay30us();
  240. E = 1;
  241. Delay30us();
  242. E = 0;                          //E 1->0,rs=1,rw=0, 数据写入数据寄存器中
  243. }

  244. /********************************************************************
  245. * 名称 : lcd1602_init()
  246. * 功能 : 1602初始化,请参考1602的资料
  247. * 输入 : 无
  248. * 输出 : 无
  249. ***********************************************************************/
  250. void lcd1602_init(void)
  251. {
  252. Delay20ms();
  253. lcd1602_write_cmd(0x38);         //8位数据,双列,5*7字形
  254. Delay20ms();
  255. lcd1602_write_cmd(0x38);
  256. Delay20ms();
  257. lcd1602_write_cmd(0x38);
  258. Busy();
  259. lcd1602_write_cmd(0x08);// 只开显示
  260. Busy();   
  261. lcd1602_write_cmd(0x01);        //清屏
  262. Busy();
  263. lcd1602_write_cmd(0x06);//显示地址递增,即写一个数据后,显示位置右移一位
  264. Busy();
  265. lcd1602_write_cmd(0x0c);        //开启显示屏,关光标,光标不闪烁
  266. }

  267. /********************************************************************
  268. * 名称 : lcd1602_char(uchar hang,uchar lie,char sign)
  269. * 功能 : 改变液晶中某位的值,如果要让第一行,第五个字符显示"b" ,调用该函数如下
  270. L1602_char(1,5,'b')
  271. * 输入 : 行,列,需要输入1602的数据
  272. * 输出 : 无
  273. ***********************************************************************/
  274. void lcd1602_char(uchar hang,uchar lie,char sign)
  275. {
  276. uchar a;
  277. if(hang == 1)
  278. {
  279. a = 0x80;
  280. }
  281. if(hang == 2)
  282. {
  283. a = 0xc0;
  284. }
  285. a = a + lie - 1;
  286. Busy();
  287. lcd1602_write_cmd(a);  //置数据存贮器地址
  288. Busy();
  289. lcd1602_write_data(sign);
  290. }




  291. /********************************************************************
  292. * 名称 : lcd1602_string(uchar hang,uchar lie,uchar *p)
  293. * 功能 : 改变液晶中某位的值,如果要让第一行,第五个字符开始显示"ab cd ef" ,调用该函数如下
  294. L1602_string(1,5,"ab cd ef;")
  295. * 输入 : 行,列,需要输入1602的数据
  296. * 输出 : 无
  297. ***********************************************************************/

  298. void lcd1602_string(uchar hang,uchar lie,uchar *p)
  299. {
  300. uchar a;
  301. if(hang == 1)
  302. {
  303. a = 0x80;
  304. }
  305. if(hang == 2)
  306. {
  307. a = 0xc0;
  308. }
  309. a = a + lie - 1;
  310. while(1)
  311. {
  312. Busy();
  313. lcd1602_write_cmd(a);
  314. Busy();
  315. lcd1602_write_data(*p);
  316. a++;
  317. p++;   
  318. if((*p == '\0')||(a==0x90)||(a==0xd0)) //字符结束或每行最后一个字符
  319. {
  320. break;
  321. }
  322. }
  323. }




  324. /*********************************************************************************************
  325. 函数名:主函数
  326. 调  用:无
  327. 参  数:无
  328. 返回值:无
  329. 结  果:读DHT11数据并送到1602显示
  330. 备  注:
  331. **********************************************************************************************/
  332. void main (void)
  333. {
  334.     unsigned char k;
  335.        Delay20ms();
  336.         lcd1602_init();         //LCD初始化
  337.               IrInit();        
  338.         while(1)
  339.         {
  340.         sjcl();
  341.                         //LcdWriteData(0x37+IrValue[4]);        //将数值转换为该显示的ASCII码
  342.                         //LcdWriteData(IrValue[4]+0x30);        //将数值转换为该显示的ASCII码
  343.      
  344.       lcd1602_string(1,1,"HW:");
  345.         for(k=0;k<8;k++)
  346.                         {
  347.                 if(IrValue[k+4]>9)
  348.                                                                 {
  349.                 lcd1602_char(1,k+4,IrValue[k+4]+0x37);
  350.                 }
  351.                                  else
  352.                            {
  353.                                  lcd1602_char(1,k+4,IrValue[k+4]+0x30);
  354.                                   }



  355. }
  356. }
  357. }
  358.                                 
  359.                                 
复制代码

所有资料51hei提供下载:
红外解码.zip (56.82 KB, 下载次数: 50)

评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

回复

使用道具 举报

ID:499469 发表于 2019-6-4 20:46 | 显示全部楼层
按照 顺序显示 用户码 用户反码  数据码 和数据反码  ,空调遥控比较特殊,
回复

使用道具 举报

ID:277550 发表于 2019-6-4 23:02 | 显示全部楼层
还有一些是长度不同。


像格力的空调遥控器,是40位,各功能不同的拼合,每次都发送这40位。但起始码相似的


音响的也不同。
回复

使用道具 举报

ID:277550 发表于 2019-6-4 23:05 | 显示全部楼层
这有一篇空调编码的,可以参考 https://my.oschina.net/u/1242247/blog/3052004
回复

使用道具 举报

ID:398534 发表于 2019-6-5 12:40 | 显示全部楼层
本帖最后由 51_User 于 2019-6-5 14:20 编辑

我是新手,正好在学习红外解码,遇到困难,望指教。
我采用P32口中断和定时器方式接收解码红外信号,想法是中断为低电平跳边沿方式,红外遥控器按下一个按键后,发出的一帧数据里的每个脉冲都能引起中断响应,在中断响应程序里保存通过定时器计算出的前后两个低电平脉冲之间的时间,用于解码。
实际上运行程序,无法解码。
经过多次测试后,发现好像单片机的中断无法对红外遥控器发出的一帧数据里的每个脉冲都响应,实际上似乎只响应数次(原本以为33个脉冲,应该会有33次中断响应)。
请问我对红外信号的中断响应的理解是不是错了。
附上中断程序和定时器源码,请帮忙分析指教。

void Infraredrevive() interrupt 0
{
        if (timercounter>31 && timercounter<50)//判断是否为NEC编码引导码

        {
        //计算引导码接收次数13500x11.592/12/256=48,识别引导码的定时器计数为48次
       //识别引导码丢掉,不保存
                count_i=0;//数组下标清零,准备保存脉冲时间
        }

        
                if(newpulse_flag==1)          //下一个pulse来临,保存两个跳低电平pulse之间的计时器时间倍数
                {
                         timerArray[count_i]=timercounter;
                        count_i=count_i+1;
        
                }else
                {
                         newpulse_flag=1;//下一个pulse来临标志,也是为了识别第一个pulse的开始

                }        
        
                if (count_i==32)//33个脉冲信号,丢掉引导码,共32个信号
                {
                  decode_pulse();//33个脉冲信号接收结束,调用解码函数
                        count_i=0;

                }
                 timercounter=0;//timer时间倍数清零,用于下一个计数


}

void timer0() interrupt 1//定时器中断一次256x12/1105920秒
{
     timercounter++;

}


void IRinitial()//中断设定初始化
{               ///*中断设定开始
    EA=1;//中断总开关
//    IE0=0;//
    EX0=1;//外部中断0
    ET0=1;//定时器T0
    TR0=1;//启动T0定时器,后面需要放在中断程序里嵌套使用定时器 ,清零
//    TF0=0;//为timer溢出标志位,用于判断
    TMOD=0x02;//设置定时器t0为8位模式,自動裝入
    TH0=0;//196;//重装数据,为了精确时间,每次跳12/11059200秒
    TL0=0;//196;//初始值
    IT0=1;//下降沿触发
//中断设定结束*/
}

回复

使用道具 举报

ID:398534 发表于 2019-6-5 12:49 | 显示全部楼层
我是新手,正好在学习红外信号解码,遇到困难,望高手指教。
我想通过中断响应一次红外遥控器发出的一帧数据(有33个脉冲)的每个脉冲信号来启动定时器计算并保存脉冲信号的宽度来解码红外信号。
但是实际上一直不成功。
通过多次检查中间变量的值,发现一帧红外数据虽然有33个脉冲,但是实际上只会引起几次中断,并不是33此。
请问我原本希望一帧数据的33个脉冲都能引起中断响应的想法是错误的吗?实际上一帧数据会引起几次中断响应?
回复

使用道具 举报

ID:499469 发表于 2019-6-5 12:53 | 显示全部楼层
devcang 发表于 2019-6-4 23:02
还有一些是长度不同。

受教了 我是业余玩玩,40位还真闹不明白了 哈哈
回复

使用道具 举报

ID:73182 发表于 2019-6-5 14:01 | 显示全部楼层
感谢分享
回复

使用道具 举报

ID:425846 发表于 2019-6-7 16:40 来自手机 | 显示全部楼层
肯定会有33个下降沿啊
回复

使用道具 举报

ID:398534 发表于 2019-6-20 20:13 | 显示全部楼层
感谢回复。
那我可能抓错了。
自己再研究一下。
回复

使用道具 举报

ID:479579 发表于 2020-3-15 17:37 | 显示全部楼层
51_User 发表于 2019-6-20 20:13
感谢回复。
那我可能抓错了。
自己再研究一下。

中断程序如果执行周期过长,同时你进入中断后应该是关闭中断的。所以肯定会漏掉部分下降沿的。所以你那个方法在51单片机中无法实现。除非是有时钟频率更高的单片机来实现你的方法。
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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