求红外解码c程序带详细解释

求红外解码c程序带详细解释,第1张

#include<reg52.h> //包含头文件,一般情况不需要改动,头文件包含特殊功能寄存器的定义

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)

}


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

原文地址: http://outofmemory.cn/yw/8110473.html

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

发表评论

登录后才能评论

评论列表(0条)

保存