PT2221发送的红外编码最前面为13.5ms的引导码,而后为32位的数据,32位数据的前16位为用户码, ;用于区别各个厂家的遥控器的不同,而后16位为数据码,分前8位和后8位,前后8位互为反码,用于检验 ;这里我们只接收了16位的数据,别的数据则丢掉了,对16的数据在程序中进行了检验 ;解码的结果用两种方式输出:1.用串口以9600的波特率发送,2.四位一体数码管显示 ;一体化接P3.2,即外部中断0 ;数码管:P1口送段码,P0口低四位送位码 ;程序中采用11.0592M的晶振,若用12M晶振只需按程序注释的提示选择相应程序即可
b_bitequ20h ;数码管个位数存放内存位置
a_bitequ21h ;数码管十位数存放内存位置
ORG0000H
AJMPMAIN;转入主程序
ORG0003H ;外部中断P3.2脚INT0入口地址
AJMPINT;转入外部中断服务子程序(解码程序)
;=============以下为主程序进行CPU中断方式设置===================================
MAIN:SETBEA ;打开CPU总中断请求
SETBIT0 ;设定INT0的触发方式为脉冲负边沿触发
SETBEX0 ;打开INT0中断请求
AJMP$
;============以下为进入P3.2脚外部中断子程序,也就是解码程序======================
INT:CLREA ;暂时关闭CPU的所有中断请求,只处理当前中断,避免受外部信号干扰
MOVR6,#10
SB:ACALLYS1;调用882微秒延时子程序
JBP3.2,EXIT;延时882微秒后判断P3.2脚是否出现高电平如果有就退出解码程序,(882um就出现高电平说明是干扰信号)
DJNZR6,SB;重复10次,目的是检测在8820微秒内如果出现高电平就退出解码程序
;=======以上完成对遥控信号的9ms的初始低电平信号的识别======================
JNBP3.2, $ ;等待高电平避开9毫秒低电平引导脉冲
ACALLYS2;延时4.74毫秒避开4.5毫秒的结果码
MOVR7,#16 ;忽略前16位系统识别码,PT2221共发送32位数据,前16位为用户码,后16为数据码才是我们需要的
JJJJA:JNBP3.2,$ ;等待地址码第一位的高电平信号
LCALLYS1;高电平开始后用882微秒的时间尺去判断信号此时的高低电平状态
MOVC,P3.2 ;将P3.2引脚此时的电平状态0或1存入C中
JNCUUUA;如果为0就跳转到UUUA
LCALLYS3;检测到高电平1的话延时1毫秒等待脉冲高电平结束
UUUA:DJNZR7,JJJJA
;==============到此为止,已跳过13.5ms的引导码和前16位的用户码=====================
MOVR1,#1AH ;设定1AH为起始RAM区
MOVR2,#2 ;接收从1AH到1BH的2个内存,用于存放操作码和操作反码,
;16位数据分前8位和后8位,后8位是前8位的反码,用于检验
PP:MOVR3,#8 ;每组数据为8位
JJJJ:JNBP3.2,$ ;等待地址码第一位的高电平信号
LCALLYS1;高电平开始后用882微秒的时间尺去判断信号此时的高低电平状态
MOVC,P3.2 ;将P3.2引脚此时的电平状态0或1存入C中
JNCUUU;如果为0就跳转到UUU
LCALLYS3;检测到高电平1的话延时1毫秒等待脉冲高电平结束,然后把C中的1存入R1
UUU:MOVA,@R1;将R1中的数据给A
RRCA;将C中的值0或1移入A中的最低位
MOV@R1,A;将A中的数暂时存放在R1数值的内存中
DJNZR3,JJJJ;接收满8位换一个内存
INCR1;对R1中的值加1,换下一个RAM
DJNZR2,PP;接收完8位数据码和8位数据反码,存放在1AH/1BH中
;==================上面的程序已完成了16位数据码的接收工作,下面进行检验==================
MOVA,1AH
CPLA;对1AH取反后和1BH比较
CJNEA,1BH,EXIT;如果不等表示接收数据发生错误,放弃
;====================校验正确则继续执行下面程序,不正确则作为干扰丢弃===================
LCALLSEND;调用串口发送程序,把16位数据码用串口发送出去
LCALLDISPLAY ;把16位数据码送数码管显示
CLRP3.5 ;P3.5口上的LED闪烁一次,表示解码成功
LCALLYS2
LCALLYS2
LCALLYS2
SETBP3.5
;==================退出子程序,解码出错时退出解码子程序========================
EXIT:SETBEA ;允许中断
RETI;退出解码子程序
;===================串口发送子程序============================================
SEND:MOVa,1AH ;将1AH中的十六进制数转换成16进制
MOVb,#16 ;10进制/16=16进制,此时输出按键的16进制编码,00~1F
divab
SWAPA
ADDA,B
MOV30H,a;转换为16进制代码后存到30H
MOVTMOD,#20H
MOVTL1,#0FDH
MOVTH1,#0FDH
MOVSCON,#40H
MOVPCON,0
CLRTI
CLRRI
SETBTR1
MOVR0,#30H
MOVA,@R0
MOVSBUF,A
WAIT:JNBTI,WAIT
CLRTI
RET
;******************************************显示子程序********************************************
display:MOVa,1AH ;将1AH中的十六进制数转换成10进制或16进制
MOVb,#10 ;10进制/10=10进制,此时输出按键对应的键号,00~31
;MOV b,#16 ;10进制/16=16进制,此时输出按键的16进制编码,00~1F
divab
MOVa_bit,a;十位在a
MOVb_bit,b;个位在b
MOVdptr,#numtab;指定查表启始地址
MOVr0,#4
dpl1:MOVr1,#250 ;显示1000次
dplop:MOVa,b_bit ;取个位数
MOV CA,@A+DPTR;查个位数的7段代码
MOVp1,a;送出个位的7段代码
SETBp0.0 ;开个位显示
acalld1ms ;显示1ms
CLRP0.0
MOVa,a_bit ;取十位数
MOV CA,@A+DPTR;查十位数的7段代码
MOVp1,a;送出十位的7段代码
SETBP0.1 ;开十位显示
acalld1ms ;显示1ms
CLRP0.1
CLRP0.2
CLRP0.3
djnzr1,dplop;100次没完循环
djnzr0,dpl1;4个100次没完循环
ret
D1MS:MOVR4,#250 ;延时1ms
DJNZR4,$
RET
numtab:DB3FH,06H,5BH,4FH,66H,6DH,7DH,07H ;字形码表"0","1","2"~"7"
DB7FH,6FH,77H,7CH,39H,5EH,79H,71H ;字形码表"8","9","A"~"F"
;=============延时子程序1,精确延时882微秒=========================================
;YS1: MOV R4,#20 ;12M晶振请用此句延时
YS1:MOVR4,#17 ;11.0592M晶振请用此句延时
D1:MOVR5,#20
DJNZR5,$
DJNZR4,D1
RET
;============延时子程序2,精确延时4740微秒==========================================
YS2:MOVR4,#10
;D2: MOV R5,#235 ;12M晶振请用此句延时
D2:MOVR5,#217 ;11.0592M晶振请用此句延时
DJNZR5,$
DJNZR4,D2
RET
;=============延时程序3,精确延时1000微秒===========================================
YS3:MOVR4,#2
;D3: MOV R5,#248 ;12M晶振请用此句延时
D3:MOVR5,#230 ;11.0592M晶振请用此句延时
DJNZR5,$
DJNZR4,D3
RET
;=========================
END