不过你贴的程序看不到引导码的处理过程,或者说没法知道startflag怎么置1的。最好也贴出来。
如果你这个接收程序的中断处理部分完整的话,可以确定所用的单片机不具有修改触发边沿的功能。只能根据第一个下降沿(引导码)和第一个用户码之间的时间来处理(减去0电平的时间)。
从程序上看,
if(startflag)
只是判断可能是起始标记。
if(irtime>32&&irtime<63)
//引导码
8-16ms间
bitnum=0
这个判断是否为引导码。如果是,则将数组键置0,准备开始存储数据。即:
irdata[bitnum]=irtime
irtime=0//这个就不说了
bitnum++//这个本来也不该说的,不过结合前面的引导码判断,就有问题了:
1、如果遥控器发射过程中红外线被挡住,irtime会很长···还需要继续存储么?
2、如果接收被干扰,会插入高电平,irtime
不确定···
3·······
简单说,判断过于简单,没法实用的。
话说,你自己有能力写个好的,为嘛还参考别人的?
这是采用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\x0d\x0a#include\x0d\x0a#define uchar unsigned char\x0d\x0a#define uint unsigned int\x0d\x0asbit lcden = P2^7\x0d\x0asbit lcdrs = P2^6\x0d\x0asbit lcdwr = P2^5\x0d\x0asbit IR = P3^2\x0d\x0auchar IRCOM[6]//数组,用于存储红外编码\x0d\x0auchar code table1[] = "remote control"\x0d\x0auchar code table2[] = "CODE:"\x0d\x0avoid delayms(uchar x)// 延时x*0.14ms\x0d\x0a{\x0d\x0auchar i\x0d\x0awhile(x--)\x0d\x0afor(i=0i//延时xms\x0d\x0a{\x0d\x0auchar i,j\x0d\x0afor(i=xi>0i--)\x0d\x0afor(j=110j>0j--)\x0d\x0a}\x0d\x0a/****************************LCD部分***********************************************/\x0d\x0avoid write_com(uchar com)\x0d\x0a{\x0d\x0alcden = 0\x0d\x0alcdrs = 0\x0d\x0alcdwr = 0\x0d\x0aP0 = com\x0d\x0adelay(5)\x0d\x0alcden = 1\x0d\x0adelay(5)\x0d\x0alcden = 0//别忘了lcden拉低\x0d\x0a}\x0d\x0avoid write_date(uchar date)\x0d\x0a{\x0d\x0alcden = 0\x0d\x0alcdrs = 1\x0d\x0alcdwr = 0\x0d\x0aP0 = date\x0d\x0adelay(5)\x0d\x0alcden = 1\x0d\x0adelay(5)\x0d\x0alcden = 0\x0d\x0a}\x0d\x0avoid lcd_init(void)\x0d\x0a{\x0d\x0alcden = 0\x0d\x0alcdrs = 0\x0d\x0alcdwr = 0\x0d\x0adelay(5)\x0d\x0awrite_com(0x38)\x0d\x0awrite_com(0x0c)\x0d\x0awrite_com(0x06)\x0d\x0awrite_com(0x01)\x0d\x0a}\x0d\x0a/*****************main()************************/\x0d\x0avoid main(void)\x0d\x0a{\x0d\x0auchar count=0\x0d\x0aIR = 1\x0d\x0alcd_init()\x0d\x0awrite_com(0x80)\x0d\x0awhile(table1[count]!='\0')\x0d\x0a{\x0d\x0awrite_date(table1[count])\x0d\x0acount++\x0d\x0adelay(5)\x0d\x0a}\x0d\x0acount = 0\x0d\x0awrite_com(0x80+0x40)\x0d\x0awhile(table2[count]!='\0')\x0d\x0a{\x0d\x0awrite_date(table2[count])\x0d\x0acount++\x0d\x0adelay(5)\x0d\x0a}\x0d\x0a\x0d\x0aIE = 0x81//开中断\x0d\x0aTCON = 0x01//脉冲负边沿触发\x0d\x0awhile(1)\x0d\x0a\x0d\x0a}\x0d\x0a/*********************红外中断**************************/\x0d\x0avoid IR_time() interrupt 0 \x0d\x0a{\x0d\x0auchar i,j,TimeNum=0//TimeNum用来计IR高电平次数 从而判断是0还是1\x0d\x0aEX0 = 0//关闭中断\x0d\x0adelayms(5)\x0d\x0aif(1 == IR)\x0d\x0a{\x0d\x0aEX0 = 1\x0d\x0areturn\x0d\x0a}\x0d\x0awhile(!IR) //跳过9ms前导低电平\x0d\x0adelayms(1)\x0d\x0afor(i=0i{\x0d\x0afor(j=0j{\x0d\x0awhile(IR) //跳过4.5ms的前导高电平\x0d\x0adelayms(1)\x0d\x0awhile(!IR) //跳过0.56ms的低电平\x0d\x0adelayms(1)\x0d\x0a\x0d\x0awhile(IR) \x0d\x0a{\x0d\x0aTimeNum++//计时高电平时间从而判断读取的是0还是1\x0d\x0adelayms(1)\x0d\x0a}\x0d\x0aif(TimeNum>=30)//按键按下时间过长 跳过\x0d\x0a{\x0d\x0aEX0 = 1\x0d\x0areturn\x0d\x0a}\x0d\x0aIRCOM[i] = IRCOM[i]>>1\x0d\x0aif(TimeNum >= 8) //8*0.14ms 这时读取的是1;\x0d\x0a{\x0d\x0aIRCOM[i] = IRCOM[i]|0x80\x0d\x0a}\x0d\x0aTimeNum = 0\x0d\x0a}\x0d\x0a}\x0d\x0aif(IRCOM[2]!=~IRCOM[3])//判断八位数据和八位数据反码是否相等\x0d\x0a{\x0d\x0aEX0 = 1\x0d\x0areturn\x0d\x0a}\x0d\x0aIRCOM[4] = IRCOM[2]&0x0f//取低四位\x0d\x0aIRCOM[5] = IRCOM[2]>>4//IRCOM[5]取IRCOM[2]高四位\x0d\x0aif(IRCOM[4] >9) //转换成字符\x0d\x0a{\x0d\x0aIRCOM[4] = IRCOM[4] + 0x37\x0d\x0a}\x0d\x0aelse\x0d\x0aIRCOM[4] = IRCOM[4] + 0x30\x0d\x0aif(IRCOM[5] >9)\x0d\x0a{\x0d\x0aIRCOM[5] = IRCOM[5] + 0x37\x0d\x0a}\x0d\x0aelse\x0d\x0aIRCOM[5] = IRCOM[5] + 0x30\x0d\x0adelay(5)\x0d\x0awrite_com(0x80 + 0x40 + 5)\x0d\x0awrite_date(IRCOM[5])\x0d\x0awrite_date(IRCOM[4])\x0d\x0aEX0 = 1//重新开启外部中断\x0d\x0a}欢迎分享,转载请注明来源:内存溢出
评论列表(0条)