sbit IR=P3^2 //红外接口标志
#define DataPort P0 //定义数据端口 程序中遇到DataPort 则用P0 替换
sbit LATCH1=P2^6//定义锁存使能端口 段锁存
sbit LATCH2=P2^7// 位锁存
/*------------------------------------------------
全局变量声明
------------------------------------------------*/
//unsigned char code dofly_DuanMa[10]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f}// 显示段码值0~9
unsigned char irtime//红外用全局变量
bit irpro_ok,irok
unsigned char IRcord[4]
unsigned char irdata[33]
/*------------------------------------------------
函数声明
------------------------------------------------*/
void Ir_work(void)
void Ircordpro(void)
/*------------------------------------------------
定时器0中断处理
------------------------------------------------*/
void tim0_isr (void) interrupt 1 using 1
{
irtime++ //用于计数2个下降沿之间的时间
}
/*------------------------------------------------
外部中断0中断处理
------------------------------------------------*/
void EX0_ISR (void) interrupt 0 //外部中断0服务函数
{
static unsigned char i//接收红外信号处理
static bit startflag //是否开始处理标志位
if(startflag)
{
if(irtime<63&&irtime>=33)//引导码 TC9012的头码,9ms+4.5ms
i=0
irdata[i]=irtime//存储每个电平的持续时间,用于以后判断是0还是1
irtime=0
i++
if(i==33)
{
irok=1
i=0
}
}
else
{
irtime=0
startflag=1
}
}
/*------------------------------------------------
定时器0初始化
------------------------------------------------*/
void TIM0init(void)//定时器0初始化
{
TMOD=0x02//定时器0工作方式2,TH0是重装值,TL0是初值
TH0=0x00//重载值
TL0=0x00//初始化值
ET0=1 //开中断
TR0=1
}
/*------------------------------------------------
外部中断0初始化
------------------------------------------------*/
void EX0init(void)
{
IT0 = 1 //指定外部中断0下降沿触发,INT0 (P3.2)
EX0 = 1 //使能外部中断
EA = 1 //开总中断
}
/*------------------------------------------------
键值处理
------------------------------------------------*/
void Ir_work(void)//红外键值散转程序
{
/* switch(IRcord[2])//判断第三个数码值
{
case 0x0c:DataPort=dofly_DuanMa[1]break//1 显示相应的按键值
case 0x18:DataPort=dofly_DuanMa[2]break//2
case 0x5e:DataPort=dofly_DuanMa[3]break//3
case 0x08:DataPort=dofly_DuanMa[4]break//4
case 0x1c:DataPort=dofly_DuanMa[5]break//5
case 0x5a:DataPort=dofly_DuanMa[6]break//6
case 0x42:DataPort=dofly_DuanMa[7]break//7
case 0x52:DataPort=dofly_DuanMa[8]break//8
case 0x4a:DataPort=dofly_DuanMa[9]break//9
default:break
} */
P1=IRcord[2]
irpro_ok=0//处理完成标志
}
/*------------------------------------------------
红外码值处理
------------------------------------------------*/
void Ircordpro(void)//红外码值处理函数
{
unsigned char i, j, k
unsigned char cord,value
k=1
for(i=0i<4i++) //处理4个字节
{
for(j=1j<=8j++) //处理1个字节8位
{
cord=irdata[k]
if(cord>7)//大于某值为1,这个和晶振有绝对关系,这里使用12M计算,此值可以有一定误差
value|=0x80
if(j<8)
{
value>>=1
}
k++
}
IRcord[i]=value
value=0
}
irpro_ok=1//处理完毕标志位置1
}
/*------------------------------------------------
主函数
------------------------------------------------*/
void main(void)
{
EX0init()//初始化外部中断
TIM0init()//初始化定时器
DataPort=0xfe//取位码 第一位数码管选通,即二进制1111 1110
LATCH2=1 //位锁存
LATCH2=0
while(1)//主循环
{
if(irok)//如果接收好了进行红外处理
{
Ircordpro()
irok=0
}
if(irpro_ok) //如果处理好后进行工作处理,如按对应的按键后显示对应的数字等
{
Ir_work()
}
}
}
#include <reg52.h> //特殊寄存器头文件#define c(x) (x*110592/120000) //是晶振值,为计数器计一下所需要的微秒数,120000为12M,110592为11.0592M
sbit Ir_Pin=P3^3 //位声明,把P3.3/外部中断1的状态读到Ir_Pin中
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 //定时器中断零程序
{
TH0=(65536-1000)/256
TL0=(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 //为定时器1赋初值
TR1=1 //开启定时器1
while(!Ir_Pin &&(TH1&0x80)==0) //判断,如果P3.3口为低电平则执行TR1=0
TR1=0 //关闭定时器1
return TH1*256+TL1//返回TH1*256+TL1的值
}
//=============================================================
unsigned int Ir_Get_High()//脉冲高电平时间
{
TL1=0
TH1=0 //为定时器1赋初值
TR1=1 //开启定时器1
while(Ir_Pin &&(TH1&0x80)==0) //判断,如果P3.3口为低电平则执行TR1=0
TR1=0//关闭定时器1
return TH1*256+TL1 //返回TH1*256+TL1的值
}
//==============================================================
main()
{
unsigned int temp
char i,j
Led_Index=1
TMOD=0x11
TL0=(65536-1000)%256
TH0=(65536-1000)/256 //定时器0设定约1000us中断一次,用于数码管扫描
EA=1 //开总中断
ET0=1 //定时计数器0的开放控制位
TR0=1 //定时器0的运行控制位
Led_Buf[0]=0
Led_Buf[1]=0
Led_Buf[2]=0
Led_Buf[3]=0 //显示区设成0
do{
restart:
while(Ir_Pin) //判断P3.3口
temp=Ir_Get_Low() //取脉冲为低电平的时间
if(temp<c(8500) || temp>c(9500)) continue //引导脉冲低电平9000
temp=Ir_Get_High()//取脉冲高电平时间
if(temp<c(4000) || temp>c(5000)) continue //引导脉冲高电平4500
for(i=0i<4i++)//4个字节
for(j=0j<8j++)//每个字节8位
{
temp=Ir_Get_Low()
if(temp<c(200) || temp>c(800)) goto restart //根据编码格式,低电平小于0.2ms大于0.8ms视为无效电平,重新检测
temp=Ir_Get_High()
if(temp<c(200) || temp>c(2000)) goto restart//根据编码格式,低电平小于0.2ms大于2ms视为无效电平,重新检测
Ir_Buf[i]>>=1 //把Ir_Buf[i]右移一位,然后赋值给Ir_Buf[i]
if(temp>c(1120)) Ir_Buf[i]|=0x80//根据编码格式,如果电平大于1.12ms,则把0x80赋值给Ir_Buf[i]
}
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)
}
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)