哥用了幾天時間終于完成了第一個程序,當然也狠狠地借參閱了別人的的程序。 這是第一種解碼方式
第一部分:(解碼部分)
#include<reg52.h>
#include "IRTXD.h"
uchar IRData[4]={0x01,0xFE,0xaa,0x55};
uchar IRData1[4]={0x01,0xFE,0x55,0xaa};
sbit K1=P2^0;
sbit K2=P2^1;
sbit IRD=P3^2;
uchar IRRData[4];
bit RX_Flag=0;
/*****************************************
延時程序
*******************************************/
void delayms(uint xms)
{
uint i,j;
for(i=xms;i>0;i--)
for(j=110;j>0;j--);
}
/******************************************
0.12ms延時
******************************************/
void delayus()
{
uchar xms=18;
while(xms--);
}
/*******************************************
外部中斷初始化
*******************************************/
void EX0_init()
{
EX0=1;
}
/*******************************************
主函數
*******************************************/
void main()
{
TimeInit();
EX0_init();
while(1)
{
if(K1==0) //K1=0發送IRData數據
{
delayms(10);
if(K1==0)
{
IRCodeSend(IRData);
delayms(100);
}
while(!K1);
}
if(K2==0) //K2=0發送IRDdata1數據
{
delayms(10);
if(K2==0)
{
IRCodeSend(IRData1);
delayms(100);
}
while(!K2);
}
while(RX_Flag) //紅外解碼結束處理數據
{
RX_Flag=0;
if(IRRData[2]==0xaa)
{
P0=0xf0;
}
if(IRRData[2]==0x55)
{
P0=0x0f;
}
}
}
}
/***********************************************************
外部中斷0(紅外信號解碼)
***********************************************************/
void IRReceive() interrupt 0 //P3.2口作為紅外線信號的接收端.
{
uchar i,j;
uint Num=0;
EX0=0;
delayms(10); //中斷口延時防IO電平波動原理和開關一樣(非常重要若是沒有此延時接收信號不穩定)
while(IRD) //若是高電平信號退出程序
{
EX0=1;
return ;
}
EX0=0;
while(!IRD) // 等待9ms低電平結束
delayus();
for(i=0;i<4;i++)
{
for(j=0;j<8;j++)
{
while(IRD) //等待4.5ms高電平結束
delayus();
while(!IRD) //等待0.56ms低電平結束
delayus();
while(IRD)
{
delayus();
Num++;
if(Num>30) //高電平
{
EX0=1;
return ;
}
}
IRRData[i]=IRRData[i]>>1;
if(Num>9)
IRRData[i]=IRRData[i]|0x80; //當高電平時間大于1.12ms的時候為高電平 ,接收信號先讀低位右移
Num=0;
}
}
if(IRRData[0]!=~IRRData[1])
{
EX0=1;
return ;
}
EX0=1;
RX_Flag=1; //紅外線信號接收解碼結束標志位
}
第二部分(信號發送部分)
#include<reg52.h>
#include "IRTXD.h"
sbit Ls138A=P2^2;
sbit Ls138B=P2^3;
sbit Ls138c=P2^4;
sbit IRLed=P1^0; //P1.0作為紅外線信號的發射端口,發射載波為38khz 的數據
/*****************************************************
9ms高電位+4.5ms低電平的引導碼
*****************************************************/
void ProIRStart()
{
TH0=0xdc; //9ms高電平初值 (注定時器0沒有啟動ET0作為延時計數)
TL0=0xd8;
TR0=1;
TR1=1;
while(!TF0);
TF0=0;
TR0=0;
TR1=0;
IRLed=1;
TH0=0xee; //4.5ms低電平初值
TL0=0X6C;
TR0=1;
TR1=0;
while(!TF0);
TF0=0;
TR0=0;
TR1=0;
IRLed=1;
}
/***************************************************
·0.5ms結束碼
****************************************************/
void SendOver() //若沒有結速碼當解碼部分采取下降沿觸發時無法解碼
{
TH0=0xfe;
TL0=0X0C;
TR0=1;
TR1=1;
while(!TF0);
TF0=0;
TR0=0;
TR1=0;
IRLed=1;
}
/***************************************************
數據0位發送函數
****************************************************/
void IRSendCode0()
{
TH0=0xfd;
TL0=0xd0;
TR0=1;
TR1=1;
while(!TF0); // 0.56ms高電平
TF0=0;
TR0=0;
TR1=0;
IRLed=1;
TH0=0xfd;
TL0=0xd0;
TR0=1;
TR1=0;
while(!TF0); //0.56ms低電平
TF0=0;
TR0=0;
TR1=0;
IRLed=1;
}
/***************************************************
數據1位發送函數
***************************************************/
void IRSendCode1()
{
TH0=0xfd;
TL0=0xd0;
TR0=1;
TR1=1;
while(!TF0); // 0.56ms¸高電平發送
TF0=0;
TR0=0;
TR1=0;
IRLed=1;
TH0=0xf9;
TL0=0x70;
TR0=1;
TR1=0;
while(!TF0); //1.68ms低電平發送
TF0=0;
TR0=0;
TR1=0;
IRLed=1;
}
/****************************************************/
void IRCodeSend(uchar *p)
{
uchar i,j;
uchar IRSendDat;
ProIRStart(); //發送引導碼
for(j=0;j<4;j++)
{
IRSendDat=*p;
p++;
for(i=0;i<8;i++)
{
if(IRSendDat&0x80) //數據發送時先發高位再發低位
IRSendCode1();
else
IRSendCode0();
IRSendDat=IRSendDat<<1;
}
}
SendOver();
}
/**************************************
定時器1初始化
**************************************/
void TimeInit()
{
TMOD|=0x21; //定時器工作方式賦13us的定時初值產生38khz信號
TH1=0xf3;
TL1=TH1;
EA=1;
ET1=1;
}
/*****************************************
定時器1中斷函數
*****************************************/
void time1() interrupt 3 //產生38KHZ載波 定時器1每隔13us中斷一次 剛好是38khz的信號
{
IRLed=~IRLed;
}
|