单片机红外解码程序问题,完全不能理解

单片机红外解码程序问题,完全不能理解,第1张

“我并非一点基础也没有”

很遗憾,从你提的这三个问题来看,你的基础比0只多出一点点。

(1)Tc是个16位整型。这是16位整型的基本算法。Tc保存的是抓出的脉宽。

(2)引导码就是指定宽度的脉冲数据码是短一些的脉冲——所以才会那么大费周章地抓脉宽判断是引导码还是用户码。接收是否反相要查你所局银用接收头的手册,有可能跟你的程序是相桐祥宴反的。

(3)m不是字宴码节序号而是位序号。注意前面的/8。

加油啊

这是采用STC12C5A60S2单片机的红外解码程序及其说明。

采用脉宽调制的串行码,以脉宽为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位代码--------------------------

MOVR1,#IRUSERL

MOVR2,#04H

EXINT101:

MOVR6,#08H 每组数据位8位

EXINT102:

JNB IRIN,$ 等待地址码第一组数据的高电平信号

LCALL DELAY09MS 高电平开始后延时判断信号此时的高/低状态

MOVC,IRIN 将P3.2引脚此时启春的电平状态0或1存入C中

JNCINT1OUT 如果为0跳出

LCALL DELAY1MS

INT1OUT:

MOVA,@R1

RRCA 将C中的数据0/1移入A中最低位

MOV@R1,A 将A中的数据暂存在R1

DJNZ R6,EXINT102 接受完8位代码

INCR1

DJNZ R2,EXINT101 接受完4组32位代码

--------------数据码比较-------------------------------

MOVA,IRDATAL

LCALL SENDRXDAT

MOVA,IRDATAL

CPLA

CJNE A,IRDATAH,INTOUT1 判断数码正误,不等退出

MOVIR_DAT,IRDATAL 相等则保存正确数据

MOV A,IR_DAT

LCALL SENDRXDAT

SETB IRBIT

INTOUT1:

LCALL DELAY45MS

SETB EA 允许中断

POP 6

POP 2

POP 1

POPPSW

POPACC

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

***************************************************

#include<reg52.h>

#include<intrins.h>

#define uchar unsigned char

#define uint unsigned int

sbit lcden = P2^7

sbit lcdrs = P2^6

sbit lcdwr = P2^5

sbit IR = P3^2

uchar IRCOM[6]//数组,用于存储红外编码

uchar code table1[] = "remote control"

uchar code table2[] = "CODE:"

void delayms(uchar x)// 延时x*0.14ms

{

uchar i

while(x--)

for(i=0i<13i++){}

}

void delay(uchar x) //延时xms

{

uchar i,j

for(i=xi>0i--)

for(j=110j>0j--)

}

/****************************LCD部分***********************************************/

void write_com(uchar com)

{

lcden = 0

lcdrs = 0

lcdwr = 0

P0 = com

delay(5)

lcden = 1

delay(5)

lcden = 0 //别忘了lcden拉低

}

void write_date(uchar date)

{

lcden = 0

lcdrs = 1

lcdwr = 0

P0 = date

delay(5)

lcden = 1

delay(5)

lcden = 0

}

void lcd_init(void)

{

lcden = 0

lcdrs = 0

lcdwr = 0

delay(5)

write_com(0x38)

write_com(0x0c)

write_com(0x06)

write_com(0x01)

}

/*****************main()************************/

void main(void)

{

uchar count=0

IR = 1

lcd_init()

write_com(0x80)

while(table1[count]!='\0')

{

write_date(table1[count])

count++

delay(5)

}

count = 0

write_com(0x80+0x40)

while(table2[count]!='\0')

{

write_date(table2[count])

count++

delay(5)

}

IE = 0x81//开中断

TCON = 0x01//脉冲负边沿触发

while(1)

}

/*********************红外中断**************************/

void IR_time() interrupt 0

{

uchar i,j,TimeNum=0//TimeNum用来计IR高电平次数 从而判断是0还是1

EX0 = 0//关闭中断

delayms(5)

if(1 == IR)

{

EX0 = 1

return

}

while(!IR) //跳祥余孙谨链过9ms前导低电平

delayms(1)

for(i=0i<4i++)

{

for(j=0j<8j++)

{

while(IR) //跳过4.5ms的前导高电平

delayms(1)

while(!IR) //跳毁敬过0.56ms的低电平

delayms(1)

while(IR)

{

TimeNum++//计时高电平时间从而判断读取的是0还是1

delayms(1)

}

if(TimeNum>=30)//按键按下时间过长 跳过

{

EX0 = 1

return

}

IRCOM[i] = IRCOM[i]>>1

if(TimeNum >= 8) //8*0.14ms 这时读取的是1;

{

IRCOM[i] = IRCOM[i]|0x80

}

TimeNum = 0

}

}

if(IRCOM[2]!=~IRCOM[3])//判断八位数据和八位数据反码是否相等

{

EX0 = 1

return

}

IRCOM[4] = IRCOM[2]&0x0f//取低四位

IRCOM[5] = IRCOM[2]>>4 //IRCOM[5]取IRCOM[2]高四位

if(IRCOM[4] >9) //转换成字符

{

IRCOM[4] = IRCOM[4] + 0x37

}

else

IRCOM[4] = IRCOM[4] + 0x30

if(IRCOM[5] >9)

{

IRCOM[5] = IRCOM[5] + 0x37

}

else

IRCOM[5] = IRCOM[5] + 0x30

delay(5)

write_com(0x80 + 0x40 + 5)

write_date(IRCOM[5])

write_date(IRCOM[4])

EX0 = 1//重新开启外部中断

}


欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/yw/8226707.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-04-14
下一篇2023-04-14

发表评论

登录后才能评论

评论列表(0条)

    保存