Chinaunix首页 | 论坛 | 博客
  • 博客访问: 137416
  • 博文数量: 35
  • 博客积分: 1410
  • 博客等级: 上尉
  • 技术积分: 289
  • 用 户 组: 普通用户
  • 注册时间: 2010-06-21 20:39
文章分类

全部博文(35)

文章存档

2010年(35)

我的朋友

分类: 嵌入式

2010-06-22 22:20:53

红外遥控在生产和生活中应用越来越广泛,不同的红外遥控芯片有不同的发码协议,但一般都是由引导码,系统码,键码三部分组成.引导码是告诉接收机准备接收红外遥控码.系统码是识别码,不同的遥控芯片有不同的误别码,以免搞错.

  遥控器上不同的按键有不同的键码,系统码和键码都是16位码,8位正码,8位反码.SC6122的系统码是FF00,FF00互为反码,键码1EF10也是互为反码.
  
SC6122的引导码为低电平为9000微秒,高电平为4500微秒.当然高电平不可能精确为9000微秒,8000微秒到10000微秒都看作是正常范围,低电平在4000-5000之间都看作是正常范围.引导码后的32位编码(16位系统码和16位键码)不管高低电平,载波时间都是560微秒,但低电平持续时间是1125微秒,高电平持续时间是2250微秒,所以低电平除去载波时间大约是560微秒,高电平除去载波时间大约是1680微秒.低电平也有一个波动范围,400-700之间都看作是正常的,具体多少可以通过示波器测量出来.高电平也有一个波动范围,400-2000之间都看作是正常的,具体多少也是根据经验.当然范围越宽,捕捉红外线的范围也越宽,越精确.在捕捉到有高低电平之间,560-1680之间取一个中间值1120微秒,认为小于1120微秒是低电平,大于1120微秒是高电平.

广告插播信息
维库最新热卖芯片:

  下面有两个经过实践能在实验板上显示键码的程序,一个是汇编写的,一个是用C写的,与大家一起探讨遥控器.

  以下程序能在LCD上显示系统码和键码,按不同的按键,系统码不变,变的是键码.有不懂的地方可以在留言本上留言.

RS EQU P2.5                 ;这几个是LCD引脚.
RW EQU P2.6 
E EQU P2.7 

IRR EQU P3.3                ;
红外接收的输出接P3.3.

BUF EQU 30H ;30H-33H
保存解码结果


;=============================================
ORG 0000H
AJMP MAIN

;=============================================
ORG 0030H
MAIN:
MOV SP,#70H ;
堆栈指针设到70H的地方

ACALL INIT_LCD                                    ;
初始化LCD

MOV R7,#10
ACALL DELAY_MS

MOV DPTR,#MSG1
CALL DISPLAY_LINE1                        ;
在第一行显示 Test8: IR Reader
MOV DPTR,#MSG2
CALL DISPLAY_LINE2                        ;
在第二行显示www.mcuedu.com  


MAIN_LOOP:
JB IRR,$ ;
等待接收头信号为低
ACALL GET_LOW ;
测量引导脉冲低电平
CLR C
MOV A,R7
SUBB A,#(8000/50) ;SC6122
的引导脉冲低电平为9000US,我们只要测到低电平的值在8000-10000US范围内就认为合格的.
JC MAIN_LOOP ;
如果小于8000US,不对,重新等待接收

CLR C
MOV A,R7
SUBB A,#(10000/50)
JNC MAIN_LOOP

ACALL GET_HIGH ;
测量引导脉冲高电平
CLR C
MOV A,R7
SUBB A,#(4000/50)
JC MAIN_LOOP                    ;
如果小于4000US,不对,重新等待接收

CLR C
MOV A,R7
SUBB A,#(5000/50)
JNC MAIN_LOOP                        ;
如果大于5000US,不对,重新等待接收


MOV R0,#BUF ;

MOV R5,#8 ;SC6122
发的码有32,我们用4个字节来存放,每个字节有8

IR_NEXT:

CALL GET_LOW

CLR C
MOV A,R7
SUBB A,#(300/50) ;300US
JC MAIN_LOOP                                ;
低电平小于300微秒认为不对,重新接收

CLR C
MOV A,R7
SUBB A,#(800/50) ;800US
JNC MAIN_LOOP                                ;
低电平大于800微秒认为不对,重新接收


ACALL GET_HIGH
CLR C
MOV A,R7
SUBB A,#(300/50) ;300US
JC MAIN_LOOP                                ;
高电平小于300微秒认为不对,重新接收

CLR C
MOV A,R7
SUBB A,#(2000/50) ;2000US
JNC MAIN_LOOP                                     ;
高电平大于2000微秒认为不对,重新接收


CLR C
MOV A,R7
SUBB A,#(1120/50) ;                            ;
跟中间值1120进行比较


RRC A
MOV @R0,A                                    ;
通过CY移到间接地址R0中去
DJNZ R5,IR_NEXT                                    ;8
位移完了吗
MOV R5,#8
INC R0
MOV A,R0
XRL A,#(BUF+4)
JNZ IR_NEXT ;
如果不到4个字节,接收下一个


MOV DPTR,#MSG_6122
ACALL DISPLAY_LINE1 ;
显示格式名称

ACALL DISPLAY_IR_CODE ;
显示码


AJMP MAIN_LOOP
;============================================ 
MSG1: DB " Test8: IR Reader "
MSG2: DB " www.mcuedu.com "
MSG_6122: DB " Format: SC6122 "
;============================================

;转为ASCII码在LCD在显示
TO_ASCII:
CJNE A,#0AH,TO_ASCII_1
TO_ASCII_1:
JC TO_ASCII_2 ;
小于10
ADD A,#(A-10)
RET
TO_ASCII_2:
ADD A,#0
RET
;============================================
DISPLAY_IR_CODE:
MOV A,#0C0H ;
显示在第二行
ACALL SEND_COMMAND_BYTE ;
设置DDRAM地址

MOV R0,#BUF
DISPLAY_IR_CODE_NEXT:
MOV A,@R0
SWAP A
ANL A,#0FH                            ;
分离出高字节
ACALL TO_ASCII                                ;
转为ASCII
ACALL SEND_DATA_BYTE                             ;
显示 
MOV A,@R0
ANL A,#0FH                            ;
分离出低字节
ACALL TO_ASCII                                 ;
转为ASCII
ACALL SEND_DATA_BYTE                            ;
显示 
MOV A,#
ACALL SEND_DATA_BYTE                                ;
显示空格

INC R0
MOV A,R0
XRL A,#(BUF+4)
JNZ DISPLAY_IR_CODE_NEXT

MOV R0,#8 ;
2行共有20个字符,前面显示用了12,再用8个空格填满
DISPLAY_IR_CODE_B:
MOV A,#
ACALL SEND_DATA_BYTE
DJNZ R0,DISPLAY_IR_CODE_B

RET


;============================================
;
测量低电平时间,50US采样一次,R71一次,比如低电平时间为9000US,测得R7的结果为180(0B4H)
;OUTPUT: R7
GET_LOW:
MOV R7,#00H

GET_LOW_NEXT:
MOV R6,#20 ;
在晶振为11.0592M,50US需要46个机器周期,
DJNZ R6,$ ;
这条指令执行需要2个机器周期

JB IRR,GET_LOW_RTN ;
接收头为高电平,结束测量
INC R7
MOV A,R7
JNZ GET_LOW_NEXT ;
R7是否有溢出

GET_LOW_RTN:
RET
;============================================
;
测量高电平时间,50US采样一次,R71一次,比如高电平时间为4500US,测得R7的结果为90
;OUTPUT: R7
GET_HIGH:
MOV R7,#00H

GET_HIGH_NEXT:
MOV R6,#20 ;
在晶振为11.0592M,50US需要46个机器周期,
DJNZ R6,$ ;
这条指令执行需要2个机器周期

JNB IRR,GET_HIGH_RTN ;
接收头为低电平,结束测量
INC R7
MOV A,R7
JNZ GET_HIGH_NEXT ;
R7是否有溢出

GET_HIGH_RTN:
RET

;============================================

DELAY_MS:
MOV R6,#250
DELAY_MS_NEXT:
NOP
NOP
DJNZ R6,DELAY_MS_NEXT
DJNZ R7,DELAY_MS
RET
;============================================ 
;INPUT: R7
DELAY:
DJNZ R7,$
RET
;============================================
;
LCD写一个命令字节
;INPUT: ACC
SEND_COMMAND_BYTE:
CLR RS
CLR RW

MOV P0,A

SETB E
NOP
NOP
NOP
NOP
NOP
NOP
CLR E

MOV R7,#100
ACALL DELAY
RET
;===============================================
;
LCD写一个数据字节
;INPUT: ACC
SEND_DATA_BYTE:
SETB RS
CLR RW

MOV P0,A

SETB E
NOP
NOP
NOP
NOP
NOP
NOP
CLR E

MOV R7,#100
ACALL DELAY
RET
;======================================================
;
初始化LCD
INIT_LCD:
MOV A,#30H
ACALL SEND_COMMAND_BYTE
ACALL SEND_COMMAND_BYTE
ACALL SEND_COMMAND_BYTE
MOV A,#38H ;
设置工作方式
ACALL SEND_COMMAND_BYTE
MOV A,#0CH ;
显示状态设置
ACALL SEND_COMMAND_BYTE
MOV A,#01H ;
清屏
ACALL SEND_COMMAND_BYTE
MOV A,#06H ;
输入方式设置
ACALL SEND_COMMAND_BYTE
RET
;=======================================================
;
在第一行显示
;INPUT: DPTR
指向要显示的内容
DISPLAY_LINE1:
MOV A,#080H
DISPLAY_LINE1_A:
ACALL SEND_COMMAND_BYTE ;
设置DDRAM地址
MOV R6,#20
DISPLAY_LINE1_NEXT:
CLR A
MOVC A,@A+DPTR
ACALL SEND_DATA_BYTE
INC DPTR
DJNZ R6,DISPLAY_LINE1_NEXT
MOV R7,#100
ACALL DELAY
RET
;=======================================================
;
在第二行显示
;INPUT: DPTR
指向要显示的内容
DISPLAY_LINE2:
MOV A,#0C0H
AJMP DISPLAY_LINE1_A
;=======================================================

END

下面是一个用C写的遥控器程序.能在数码管上显示键码.

#include

#define c(x) (x*110592/120000) 

sbit Ir_Pin=P3^3;

unsigned char code Led_Tab[]={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,
0xf8,0x80,0x90,0x88,0x83,0xC6,0xA1,0x86,0x8E};                        //
共阳极数码显示码0-F.
unsigned char code Led_Sel[]={0xe,0xd,0xb,0x7};

unsigned char Led_Buf[4]; //
显示缓冲区
char Led_Index;                   //
位选         

unsigned char Ir_Buf[4]; //
用于保存解码结果

//==============================================================
//
数码管扫描
timer0() interrupt 1 using 1
{
TL0=65536-1000;
TH0=(65536-1000)/256; //
定时器0设定约1000us中断一次,用于数码管扫描
P0=0xff;
P2=Led_Sel[Led_Index];                            //
位选
P0=Led_Tab[Led_Buf[Led_Index]];                    //
段选

if(++Led_Index>3) Led_Index=0;                    //
四个扫描完了,到第一个数码管
}
//==============================================================
unsigned int Ir_Get_Low()
{
TL1=0;
TH1=0;
TR1=1;
while(!Ir_Pin && (TH1&0x80)==0);                
TR1=0;           
return TH1*256+TL1;
}
//=============================================================
unsigned int Ir_Get_High()
{
TL1=0;
TH1=0;
TR1=1;
while(Ir_Pin && (TH1&0x80)==0);
TR1=0;
return TH1*256+TL1;
}
//==============================================================
main()
{
unsigned int temp;
char i,j;
Led_Index=1;

TMOD=0x11;
TL0=65536-1000;
TH0=(65536-1000)/256; //
定时器0设定约1000us中断一次,用于数码管扫描
EA=1;
ET0=1;
TR0=1;

Led_Buf[0]=0;
Led_Buf[1]=0;
Led_Buf[2]=0;
Led_Buf[3]=0; //
显示区设成0

do{
restart:
while(Ir_Pin);
temp=Ir_Get_Low();
if(tempc(9500)) continue;//
引导脉冲低电平9000
temp=Ir_Get_High();
if(tempc(5000)) continue;//
引导脉冲高电平4500
for(i=0;i<4;i++) //4
个字节
for(j=0;j<8;j++) //
每个字节8
{
temp=Ir_Get_Low();
if(tempc(800)) goto restart;
temp=Ir_Get_High();
if(tempc(2000)) goto restart;
Ir_Buf[i]>>=1;
if(temp>c(1120)) Ir_Buf[i]|=0x80;
}
Led_Buf[0]=Ir_Buf[2]&0xf;
Led_Buf[1]=(Ir_Buf[2]/16)&0xf;
Led_Buf[2]=Ir_Buf[3]&0xf;
Led_Buf[3]=(Ir_Buf[3]/16)&0xf; //
显示结果
}while(1);
}
阅读(1594) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~