51单片机红外接收端的程序怎么写,我想知道写的方法和原理,最好有一个具体的模版,好让我参考. 我在弄一个51单片机接收红外信号控制LED灯亮度的程序,写了...
;采用脉宽调制的串行码,以脉宽为0.565ms、间隔0.56ms、周期为1.125ms的组合表示二进制的"0";
;以脉宽为0.565ms、间隔1.685ms、周期为2.25ms的组合表示二进制的"1
;上述"0"和"1"组成的32位二进制码经38kHz的载频进行二次调制以提高发射效率,
;达到降低电源功耗的目的。然后再通过红外发射二极管产生红外线向空间发射
;遥控编码是连续的32位二进制码组,其中前16位为用户识别码,能区别不同的电器设备,
;防止不同机种遥控码互相干扰。该芯片的用户识别码固定为十六进制01H
;后16位为8位操作码(功能码)及其反码。
;当一个键按下超过36ms,振荡器使芯片激活,将发射一组108ms的编码脉冲,这108ms发射代码由一个起始码(9ms),
;一个结果码(4.5ms),低8位地址码(9ms~18ms),高8位地址码(9ms~18ms),8位数据码(9ms~18ms)
;和这8位数据的反码(9ms~18ms)组成。如果键按下超过108ms仍未松开,
;接下来发射的代码(连发代码)将仅由起始码(9ms)和结束码(2.5ms)组成。
;
;解码的关键是如何识别"0"和"1",接收端而言,"0"是0.56ms的高+0.56ms的低。"1"是1.68ms的高+0.56ms的低。
;所以可以根据高电平的宽度区别"0"和"1"。当高电平出现时开始延时,0.56ms以后,若读到的电平为低,
;说明该位为"0",反之则为"1",为了可靠起见,延时必须比0.56ms长些,但又不能超过1.12ms,否则如果该位为"0",
;读到的已是下一位的高电平,因此取(1.12ms+0.56ms)/2=0.84ms最为可靠,一般取0.84ms左右均可。
;为了共用引导部分延时程序,这里用0.9ms延时。
;-------------红外解码程序---------------------------
EXINT0:
PUSH ACC
PUSH PSW
PUSH 1
PUSH 2
PUSH 6
CLR EA ;暂时关闭中断请求
MOV R6,#10
EXINT10:
LCALL DELAY09MS ;调用900us延时子程序
JB IRIN,INTOUT1 ;判断P3.2是否有高电平,如果有就退出解码程序
DJNZ R6,EXINT10 ;循环10次,检测在900微妙中是否存在高电平。以上完成对遥控信号的9000微秒的初始低电平信号的识别。
JNB IRIN,$ ;等待高电平避开9毫秒低电平引导脉冲
LCALL DELAY45MS ;延时4.5毫秒
;-------------接受32位代码--------------------------
MOV R1,#IRUSERL
MOV R2,#04H
EXINT101:
MOV R6,#08H ;每组数据位8位
EXINT102:
JNB IRIN,$ ;等待地址码第一组数据的高电平信号
LCALL DELAY09MS ;高电平开始后延时判断信号此时的高/低状态
MOV C,IRIN ;将P3.2引脚此时的电平状态0或1存入C中
JNC INT1OUT ;如果为0跳出
LCALL DELAY1MS
INT1OUT:
MOV A,@R1
RRC A ;将C中的数据0/1移入A中最低位
MOV @R1,A ;将A中的数据暂存在R1
DJNZ R6,EXINT102 ;接受完8位代码
INC R1
DJNZ R2,EXINT101 ;接受完4组32位代码
;--------------数据码比较-------------------------------
MOV A,IRDATAL
; LCALL SENDRXDAT
MOV A,IRDATAL
CPL A
CJNE A,IRDATAH,INTOUT1 ;判断数码正误,不等退出
MOV IR_DAT,IRDATAL ;相等则保存正确数据
MOV A,IR_DAT
; LCALL SENDRXDAT
SETB IRBIT
INTOUT1:
LCALL DELAY45MS
SETB EA ;允许中断
POP 6
POP 2
POP 1
POP PSW
POP ACC
RETI
;;*****************11.0592*900=9953******************
DELAY09MS: ;6
PUSH 4 ;4
PUSH 3 ;4
MOV R4,#20 ;2
DLY900:
MOV R3,#122 ;2
DJNZ R3,$ ;4
DJNZ R4,DLY900 ;4
MOV R4,#11 ;2
DJNZ R4,$ ;4
POP 3 ;3
POP 4 ;3
RET ;4
;TOTAL=9952
;;*****************11.0592*560=6193******************
DELAY056: ;6
PUSH 4 ;4
PUSH 3 ;4
MOV R4,#12 ;2
DLY5600:
MOV R3,#122 ;2
DJNZ R3,$ ;4
DJNZ R4,DLY5600 ;4
MOV R4,#71 ;2
DJNZ R4,$ ;4
POP 3 ;3
POP 4 ;3
RET ;4
;TOTAL=6194
;;*****************11.0592*4500=49766****************
DELAY45MS: ;6
PUSH 4 ;4
PUSH 3 ;4
MOV R4,#52 ;2
DLY45:
MOV R3,#236 ;2
DJNZ R3,$ ;4
DJNZ R4,DLY45 ;4
MOV R4,#85 ;2
DJNZ R4,$ ;4
POP 3 ;3
POP 4 ;3
RET ;4
;;TOTAL=49768
;;*****************11.0592*1000=11059****************
DELAY1MS: ;6
PUSH 4 ;4
PUSH 3 ;4
MOV R4,#20 ;2
DLY1MS:
MOV R3,#136 ;2
DJNZ R3,$ ;4
DJNZ R4,DLY1MS ;4
MOV R4,#8 ;2
DJNZ R4,$ ;4
POP 3 ;3
POP 4 ;3
RET ;4
;TOTAL=11060
;;***************************************************
DELAY100US: ;6
PUSH 4 ;4
MOV R4,#140 ;2
DJNZ R4,$ ;4
MOV R4,#131 ;2
DJNZ R4,$ ;4
POP 4 ;3
RET ;4
;TOTAL=1105
;;***************************************************
遥控器解码非常简单,我给你个参考程序,你直接下载我上传的附近程序,直接用你也可以研究研究,看看我的设计程序思路!
/***************************************************************************
******* NEC型遥控器协议解码
***************************************************************************/
#include "STC89C5X.H"
#include "NEC_IR.H"
#include "TYPEDEF.H"
/*
* 宏定义红外端口
*/
#define NEC_IRinput P3_2 //红外数据输入端口
/*
* 宏定义蜂鸣器端口
*/
#define NEC_BEEP P2_3 //蜂鸣器数据输入端口
/*
* 变量
*/
INT8U CAIy_Byte; //计算接收码变量
INT16U CAIy_Data; //接收码暂存变量
INT8U NEC_IRBUF[4]; //接收码保存数组
/*
* 初始化T0定时器
*/
void NEC_SystemInit(void)
{
// T0/INT0中断配置
TMOD = 0x01;
//
EX0 = 1;
//
// ET0 = 1;
IT0 = 1;
// IE0 = 1;
// 开启总中断
EA = 1;
}
/*
* NEC遥控器按键音子函数
*/
void NEC_BeepByte(void)
{
unsigned char i;
unsigned int t = 2000;
for(i = 200; i > 0; i--)
{
NEC_BEEP = 0;
}
while(t > 0)t--;
NEC_BEEP = 1;
}
/*
* 采样红外触发外部中断低电平数据
*/
unsigned int INT0_INTERRUPT_LSB()
{
TL0 = 0;
TH0 = 0;
TR0 = 1;
while(~NEC_IRinput && (TH0 & 0x80) == 0);
TR0 = 0;
return (TH0 * 256 + TL0);
}
/*
* 采样红外没有触发外部中断高电平数据
*/
unsigned int INT0_INTERRUPT_MSB()
{
TL0 = 0;
TH0 = 0;
TR0 = 1;
while(NEC_IRinput && (TH0 & 0x80) == 0);
TR0 = 0;
return (TH0 * 256 + TL0);
}
/*
* INT0外部中断0服务程序
*/
void INT0_IRQ(void) interrupt 0
{
unsigned char i;
unsigned char k = 3;
// unsigned char NEC_CHAN;
/* 拉高红外数据端口准备采样有效数据 */
NEC_IRinput = 1;
/* 接收红外低电平数据采样 */
CAIy_Data = INT0_INTERRUPT_LSB();
/* 判断接收到的红外采样数据是否在9ms引导码范围内 */
if((CAIy_Data < NEC_8500) || (CAIy_Data > NEC_9500))
{
/* 如果超出或者低于引导码低电平9ms程序直接跳出 */
return;
}
/* 清除采样变量 */
CAIy_Data &= 0x00;
/* 接收红外高电平数据采样 */
CAIy_Data = INT0_INTERRUPT_MSB();
/* 判断接收到的红外采样数据是否在4.5ms引导码范围内 */
if((CAIy_Data < NEC_4300) || (CAIy_Data > NEC_4600))
{
/* 如果超出或者低于引导码高电平4.5ms程序直接跳出 */
return;
}
/*
* NEC引导接收正确后,进入8用户码和8按键码接收
*/
for(i = 0; i < 32; i++)
{
CAIy_Byte >>= 1; /* 接收数据位左移一位 */
/* 接收红外低电平数据采样 */
CAIy_Data = INT0_INTERRUPT_LSB();
/* 判断低电平是否在NEC协议0.56ms范围以内 */
if((CAIy_Data < NEC_200) || (CAIy_Data > NEC_800))
{
/* 如果超出NEC协议0.56ms低电平直接跳出接收 */
return;
}
/* 接收红外高电平数据采样 */
CAIy_Data = INT0_INTERRUPT_MSB();
/* 判断高电平是否在NEC协议1.68ms范围以内 */
if((CAIy_Data < NEC_200) || (CAIy_Data > NEC_2000))
{
/* 如果超出NEC协议1.68ms低电平直接跳出接收 */
return;
}
/* 判断高电平是否在NEC协议1.68范围内 */
if(CAIy_Data >= NEC_1120)
{
/* 如果符合NEC协议1.68ms为高电平1变量高位制1 */
CAIy_Byte |= 0x80;
}
/* 用来区分8位用户正码、8位用户反码、8位按键正码、8位按键反码 */
if(i == 7 || i == 15 || i == 23 || i == 31)
{
k++;
if(k == 4)
{
k = 0;
}
NEC_IRBUF[k] = CAIy_Byte; // 把接收到的数据存放到数组
CAIy_Byte = 0x00; // 清除接收数据暂存变量
}
}
/* 判断NEC协议用户码是否正确 */
if((NEC_IRBUF[0] != 0x00) || (NEC_IRBUF[0] != ~NEC_IRBUF[1]))
{
/* 如果NEC协议用户码不对程序直接跳出 */
return;
}
/* 把接收到的十进制数据转换成十六进制数据 */
/* NEC_CHAN = NEC_IRBUF[2] / 16;
NEC_IRBUF[2] = NEC_IRBUF[2] % 16;
NEC_IRBUF[2] = NEC_IRBUF[2] + NEC_CHAN * 10; */
/* NEC遥控器按键音 */
NEC_BeepByte();
}
额,这个程序我这倒是有,你要不?
一个按键想通过51单片机红外控制数码管倒计时,请问红外收发程序怎么写啊~
首先你要理解红外发射的原理和硬件才好写程序。
另,这样 的程序在百度文库里面是有的,只要你理解它的原理,下载个程序改进下就可以用了。
哥们,你中断这么写可以?别用ET0控制,用TR0控制才对,在调用中断的位置跳入中断后,关闭TR0,出去在打开,从头看到尾,编程逻辑应该没啥问题
51单片机红外解码C程序
答:// 解码值在Im[2]中,当IrOK=1时解码有效。 /* 51单片机红外遥控解码程序 */ //用遥控器对准红外接收头,按下遥控器按键,在数码管前两位上就会显示对应按键的编码 include <reg52.h> define uchar unsigned char sbit dula=P2^6;sbit wela=P2^7;uchar code table[]={0x3f,0x...
51单片机红外线遥控LED程序,疑惑中!!!
答:红外收发中,IRDATA[2]与IRDATA[3]是取反的关系。也就是说两个数对应各位前者为1后者就为0 其余的问题,都是根据红外接收时序来编的程序,以下总结以下红外收发时序供参考,你读懂就可以理解了。采用脉宽调制的串行码,以脉宽为0.565ms、间隔0.56ms、周期为1.125ms的组合表示二进制的"0";;以...
单片机红外控制程序,上电后数码管显示8个0,遥控按每反应,这是什么原因...
答:include <reg51.h> sbit IRIN = P3^2; //红外接收器数据线 sbit RELAY= P1^4; //继电器驱动线 sbit BEEP = P1^5; //蜂鸣器驱动线 void IRdelay(char x); //x*0.14MS void beep();unsigned char IRCOM[7];extern unsigned char Y0;/***/ void IRInit(){ IE |= ...
用红外接受发射管和单片机来测速度,但是单片机测速度的程序不会写
答:由于我不清楚你的电路检测到物体时是高电平还是低电平。现在假设是:高电平表示有物体通过,低电平表示没有物体通过。再假设你的P0.0接入红外检测脉冲,用T0定时计数器计数,用51单片机的程序如下(至于初始化,结果处理之类的,你就自己写吧,我也不太清楚你最后是要怎么处理的):/// TR0 = 0;wh...
单片机做的红外接收,请问怎么理解这个for()语句呢,in是连接红外接收管的...
答:你好,那个for是一次接受8个bit的循环。下图是红外接收头的输出信号。所以每个二进制来的前面都会先拉低一下,如果是1的话就是一个比较宽的脉冲,如果是零的话,1的时长跟0差不多。那个while(!in)是等开头的低电平,然后等待一段时间,之后判断是不是还是1,如意是1的话,这个就是输出1,否者...
关于51单片机红外解码程序,哪位大侠帮我看下
答:我以前做的一个项目,红外遥控开关,解码部分的code,供参考 6121码,外部中断0,at89s52 void int0() interrupt 0 //外部中断1服务函数,红外解码程序 { static uchar wei; //定义静态变量 static uchar pp; //定义静态变量 if(tt<56&&tt>50) {d2=0; tt=0;pp=0;wei=0;}//...
用51单片机制作学习型红外遥控器的原理
答:用51单片机制作学习型红外遥控器的原理 主要是C程序方面很难写出来??发射模块的程序思路怎么写???最好有C程序addby2004的是一个接受时的解码的过程吧??假如我要的遥控器是自己做的而且是用51单片机根据按键的不同发射... 主要是C程序方面很难写出来??发射模块的程序思路怎么写???最好有C程序addby2004 的...
51单片机红外线解码的程序
答:ir_code[2]&0x0f有意义 是 将高4位清0只取低4位 估计你的表格里只有16个七段码 因此查表时只能对高4位和低4位(都不大于15)分别处理和显示 /16和数学的除法相似,不过只取整数部分,不理会余数 如 15/16=0 17/16=1 18/16=1 32/16=2 35/16=2 0x0f化成二进制是 0000...
基于单片机的红外通信装置
答:红外发射器:可以用单片机产生38K的载波,也可以用555振荡产生,也可以用红外遥控器 红外接收头:HS0038,SM0038,T4148,都是一体化红外接收头电路十分简单 设计过程:发射红外信号,单片机接收,识别红外信号,9ms低电平,4.5ms高电平,通信的话最好用遥控器,最好有遥控器的编码方式,遥控器有两种编码...
求单片机红外遥控解码识别长按和短按键的C语言例子
答:以NEC格式为例:按键一次: 依次发送 引导码 + 地址 + 地址取反 + 数据 + 数据取反 长按键: 隔110ms左右发一次引导码(重复),并不带任何数据(全部为高电平)所以根据这个特点可以识别长按键:程序接收一帧数据后,提取出地址和数据,然后判断 ...//有地址和数据的,为一次按键 ...//地址和数...