STC8A8K64S4A12最小系统板06 红外接收

  • new world
  • LV5工程师
  • |      2017-07-11 12:11:20
  • 浏览量 1457
  • 回复:7

1、红外遥控系统

通用红外遥控系统由发射和接收两大部分组成,应用编/解码专用集成电路芯片来进行控制操作,如图1所示。 发射部分包括键盘矩阵、编码调制、LED红外发送器; 接收部分包括光、电转换放大器、解调、解码电路。

下面,我们将使用下面两种设备:

另外,使用51单片机进行解码。 2、原理图

从原理图看出,IR的data脚与51的PD2(P3.2)相连。 2、红外发射原理 要对红外遥控器所发的信号进行解码,必须先理解这些信号。 a) 波形 首先来看看,当我们按下遥控器时,红外发射器是发送了一个什么样的信号波形,如下图:

由上图所示,当一个键按下超过22ms,振荡器使芯片激活,将发射一组108ms的编码脉冲(由位置1所示)。如果键按下超过108ms仍未松开,接下来发射的代码(连发代码由位置3所示)将仅由起始码(9ms)和结束码(2.5ms)组成。下面把位置1的波形放大:

由位置1的波形得知,这108ms发射代码由一个起始码(9ms),一个结果码(4.5ms),低8位地址码(用户编码)(9ms~18ms),高8位地址码(用户编码)(9ms~18ms),8位数据码(键值数据码)(9ms~18ms)和这8位数据的反码(键值数据码反码)(9ms~18ms)组成。 b) 编码格式 遥控器发射的信号由一串0和1的二进制代码组成.不同的芯片对0和1的编码有所不同。通常有曼彻斯特编码和脉冲宽度编码。XS-091遥控板的0和1采用PWM方法编码,即脉冲宽度调制。下图为一个发射波形对应的编码方法: 放大0和1的波形如下图: 这种编码具有以下特征:以脉宽为0.565ms、间隔0.56ms、周期为1.125ms的组合表示二进制的“0”;以脉宽为0.565ms、间隔1.685ms、周期为2.25ms的组合表示二进制的“1”。 3、红外接收原理 a) 波形 红外接收头将38K载波信号过虑,接收到的波形刚好与发射波形相反:

放大,位定义0和位定义1波形如下:

4、解码原理及算法

注:代码宽度算法

16位地址码的最短宽度:1.12×16=18ms 16位地址码的最长宽度:2.24ms×16=36ms

可以得知8位数据代码及其8位反代码的宽度和不变:(1.12ms+2.24ms)×8=27ms

所有32位代码的宽度为(18ms+27ms)~(36ms+27ms) 对于红外线遥控对于很多电子爱好者来讲,都感觉到非常神奇,看不到,摸不着,但能实现无线遥控,其实控制的关键就是我们要用单片机芯片来识别红外线遥控器发出红外光信号,即我们通常所说的解码。单片机得知发过来的是什么信号,然后再做出相应的判断与控制,如我们按电视机遥控器的频道按钮,则单片机会控制更换电视频道,如按的是遥控器音量键,则单片机会控制增减音量。
解码的关键是如何识别“0”和“1” !! 从位的定义我们可以发现“0”、“1”均以 0.56ms的低电平开始,不同的是高电平的宽度不同!,“0”为0.56ms,“1”为1.68ms,所以必须根据高电平的宽度区别“0”和“1”。 如果从0.56ms低电平过后,开始延时,0.56ms以后,若读到的电平为低,说明该位为“0”,反之则为“1”,为了可靠起见,延时必须比0.56ms长些,但又不能超过1.12ms,否则如果该位为“0”,读到的已是下一位的高电平,因此取(1.12ms+0.56ms)/2=0.84ms最为可靠,一般取0.84ms左右均可。根据码的格式,应该等待9ms的起始码和4.5ms的结果码完成后才能读码。 5、实例代码:
#include "include.h"







	

sbit LED1 = P3^0;

sbit LED2 = P3^1;

sbit IRIN = P3^2;

sbit LED3 = P1^0;





unsigned char KeyValue;     //机器码  

unsigned char MaValue;      //键值码;  

unsigned char num;    //所记录IR数值  

unsigned char flag;



/***********************************************

*延时程序

*ms:延时时间

**********************************************/

void delay_ms(uint ms)

{

	for(;ms>0;ms--)

	{

		



		unsigned char i, j;



		_nop_();

		_nop_();

		_nop_();

		i = 11;

		j = 190;

		do

		{

			while (--j);

		} while (--i);



	}

}





/*********************************************

*延时1 us

**********************************************/

void delay_us(uint us)

{

	for(;us>0;us--)

	{

		_nop_();

		_nop_();

		_nop_();

	}

}











/** 

 * 读码  

 */  

unsigned char GetCode()  

{   

    unsigned char n;  

  

    static temp = 0;   

  

    for( n = 0; n < 8; n++ )   

    {  

        while(!IRIN);  // 等待高电平,开始解码  

  

        delay_us(840); // 延时0.84ms  

  

        if(IRIN) // 若仍然为高电平,则为1,否则为0  

        {  

           temp = (0x80|(temp>>1));  // 1    

           while(IRIN); //等待跳变成低电平  

        }  

        else {   

            temp=(0x00|(temp>>1));  // 0  

        }  

    }   

  

    return temp;  

}  





void main()

{

	IT0 = 1;	//设置为下降沿触发

	EX0 = 1;	//打开外部触发中断1

	

	UART_Init();

	EA = 1;		//打开总中断

	

	

	while(1)

	{

		//LED1 = !LED1;

		  

		    if(flag == 1)

			{

			 

			EX0 =0;

			USART1_send(num);

			

			USART1_sendchar('\r');

			USART1_sendchar('\n');

				

			flag = 0;

			EX0 =1;

			}

		

			



		

	}

}





void IR_read() interrupt 0 using 1

{

	

    /** 

     * 用户码和机器码  

     */  

    unsigned char addrl,addrh,num1,num2;   

  

    EA = 0;  //先关闭外部中断0  

  

    delay_ms(9); // 检测9ms开始码  

	

    if (IRIN) {     // 检测是否为干扰信号  

        EA = 1;     // 重新开启外部中断0  

        return ;    // 退出解码  

    }  

           

    while(!IRIN);   // 等待跳为高电平  

  

    //delay_ms(4); // 检测4.5ms结果码  

	delay_us(4500);

   

	 

	

    if (IRIN) {     // 检测是否为干扰信号  

        EA = 1;     // 重新开启外部中断0  

        return ;    // 退出解码  

    }  

  

	

    // 读码  

    addrl=GetCode(); // 用户编码高位  

    addrh=GetCode(); // 用户编码低位  

    num1=GetCode();  // 机器码  

    num2=GetCode();  // 机器码反码  

	

	

	LED3 = !LED3;

   

    //校验是否为错码  

	/*

    if(num1!=~num2)  

    {   

        KeyValue=14;   

        EA=1;   

        return;  

    }  

    */

	

	

   // KeyValue=num2;  

    //MaValue=addrh;

	

	flag = 1;

   

	num= addrl;

	num = addrh;

	num = num1;

	num = num2;

  

    EA=1;  

}
6、程序的实验结果
我们使用遥控器对着红外接头,依次按动,我们从串口接收到的数据为。后边的红框中的是“\r\n”的十六制显示。 7、完整程序
  • 0
  • 收藏
  • 举报
  • 分享
我来回复

登录后可评论,请 登录注册

所有回答 数量:5
xhuaihe 2017-08-01
瞅瞅,学习学习
0   回复
举报
发布
new world 回复 2017-08-01
共同学习:handshake:handshake
0   回复
举报
ld@digiic.com 回复 2017-08-04
这是嘛呢呢
0   回复
举报
qq1328512480 2017-07-31
点赞world哥!
0   回复
举报
发布
952632352@qq.co 2017-07-30
不错,继续加油!
0   回复
举报
发布
new world 2017-07-11
其实非常简单的,你可以试一下
0   回复
举报
发布
瞎折腾 2017-07-11
学习了,我还真没弄过遥控
0   回复
举报
发布
x
收藏成功!点击 我的收藏 查看收藏的全部帖子