⑴ 紅外線遙控器解碼C語言程序代碼 SM0038接受頭
//////////////////////////////////////
//晶振頻率為6MHz 一個機器周期2us //
//實現按鍵地址碼、指令碼的數碼管顯示//
//2010-06-01 //
//////////////////////////////////////
#include<reg52.h>
#define uchar unsigned char
#define uint unsigned int
sbit la=P2^6;
sbit wela=P2^7;
bit overflag,dataover;//開始接收數據,數據處理完畢
uchar timedata;//定時器0溢出次數
uchar chubus[33];//33Byte數據 timedata組成的數組
uchar jieguos[4];//地址碼,地址反碼,數據碼,數據反碼;
void initial()//初始化
{
IT0=1;EX0=1;//負邊沿觸發
TMOD=0x01;ET0=1;TR0=1;//模式1,十六位定時器
TH0=0xff;TL0=0x06;//0.5ms溢出
EA=1; //開總中斷
}
void time0() interrupt 1//定時器0中斷
{TH0=0xff;TL0=0x06;
timedata++;
}
void ex0() interrupt 0//外部中斷0,接收數據
{
static bit startflag;//開始接收
static uchar i;
if(startflag)
{
if(timedata<32&&timedata>=16) i=0;
chubus[i]=timedata;
timedata=0;
i++;
if(i==33){overflag=1;i=0;}
}
else
{
startflag=1;
timedata=0;
}
}
void chulidata()
{
uchar chubu;//初步數據
uchar jieguo;//結果數據
uchar x,y,z=1;
for(x=0;x<4;x++)//處理四組數據
{
for(y=1;y<=8;y++)//處理一組數據8Byte
{
chubu=chubus[z];
jieguo=jieguo>>1;
if(chubu>3) jieguo=jieguo|0x80;//大於1.5mS為1
z++;
}
jieguos[x]=jieguo;
jieguo=0;
}
dataover=1;
}
void delay(uint z)
{
uint x ,y;
for(x=z;x>0;x--)
for(y=110;y>0;y--);
}
void display()
{
uchar gao,gao1;
uchar di,di1;
uchar code table[]={
0x3f,0x06,0x5b,0x4f,
0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,
0x39,0x5e,0x79,0x71};
gao1=table[jieguos[0]/16];
di1=table[jieguos[0]%16];
gao=table[jieguos[2]/16];
di=table[jieguos[2]%16];
la=1;
P0=gao1;
la=0;
P0=0xff;
wela=1;
P0=0x7e;
wela=0;
delay(2);
la=1;
P0=di1;
la=0;
P0=0xff;
wela=1;
P0=0xfd;
wela=0;
delay(2);//地址碼
la=1;
P0=gao;
la=0;
P0=0xff;
wela=1;
P0=0x77;
wela=0;
delay(2);
la=1;
P0=di;
la=0;
P0=0xff;
wela=1;
P0=0x6f;
wela=0;
delay(2);//指令碼
}
void main()
{
initial();//初始化
while(1)
{
if(overflag)//數據接收完畢
{
chulidata();//處理數據,完成標志dataover
overflag=0;
}
if(dataover)
{
display();//數碼管顯示
}
}
}
⑵ 基於51單片機紅外遙控代碼(C語言)
以下文件是51單片機實現遙控解碼,通過數碼管顯示鍵碼的程序,P0口驅動數碼管段選,p2.6和p2.7為數碼管位選,接收頭連到P3.2口。此程序以通過驗證,可以直接編譯使用,另外還有一個繼電器和蜂鳴器的控制,不用可以屏蔽掉。
;********************************************************************************
;* 描述: *
;* 遙控鍵值讀取器 *
;* 數碼管顯示, P0口為數碼管的數據口 *
;* *
;********************************************************************************
;遙控鍵值解碼-數碼管顯示 *
;********************************************************************************/
#include <reg51.h>
#include <intrins.h>
void IR_SHOW();
void delay(unsigned char x);//x*0.14MS
void delay1(unsigned char ms);
void beep();
sbit IRIN = P3^2;
sbit BEEP = P3^7;
sbit RELAY= P1^3;
sbit GEWEI= P2^7;
sbit SHIWEI= P2^6;
unsigned char IRCOM[8];
unsigned char code table[16] =
{0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90,0x88,0x83,0xc6,0xa1,0x86,0x8e};
main()
{
IE = 0x81; //允許總中斷中斷,使能 INT0 外部中斷
TCON = 0x1; //觸發方式為脈沖負邊沿觸發
delay(1);
IRIN=1;
BEEP=1;
RELAY=1;
for(;;)
{
IR_SHOW();
}
} //end main
void IR_IN() interrupt 0 using 0
{
unsigned char i,j,k,N=0;
EA = 0;
I1:
for (i=0;i<4;i++)
{
if (IRIN==0) break;
if (i==3) {EA =1;return;}
}
delay(20);
if (IRIN==1) goto I1; //確認IR信號出現
while (!IRIN) //等 IR 變為高電平
{delay(1);}
for (j=0;j<4;j++)
{
for (k=0;k<8;k++)
{
while (IRIN) //等 IR 變為低電平
{delay(1);}
while (!IRIN) //等 IR 變為高電平
{delay(1);}
while (IRIN) //計算IR高電平時長
{
delay(1);
N++;
if (N>=30) {EA=1;return;}
}
IRCOM[j]=IRCOM[j] >> 1;
if (N>=8) {IRCOM[j] = IRCOM[j] | 0x80;}
N=0;
}//end for k
}//end for j
if (IRCOM[2]!=~IRCOM[3]) {EA=1;return;}
IRCOM[5]=IRCOM[2] & 0x0F;
IRCOM[6]=IRCOM[2] & 0xF0;
IRCOM[6]=IRCOM[6] >> 4;
beep();
EA = 1;
}
void IR_SHOW()
{
P0 = table[IRCOM[5]];
GEWEI = 0;
SHIWEI = 1;
delay1(4);
P0 = table[IRCOM[6]];
SHIWEI = 0;
GEWEI = 1;
delay1(4);
}
void beep()
{
unsigned char i;
for (i=0;i<100;i++)
{
delay(5);
BEEP=!BEEP;
}
BEEP=1;
}
void delay(unsigned char x)//x*0.14MS
{
unsigned char i;
while(x--)
{
for (i = 0; i<13; i++) {}
}
}
void delay1(unsigned char ms)
{
unsigned char i;
while(ms--)
{
for(i = 0; i<120; i++)
{
_nop_();
_nop_();
_nop_();
_nop_();
}
}
}
⑶ 高分請教高人,紅外遙控解碼程序C語言,要求每一行都解釋為什麼什麼作用。非常感謝!!先給100分,好再加
#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=0;i<4;i++) //4個位元組
for(j=0;j<8;j++) //每個位元組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);
}
⑷ 紅外遙控器同一個鍵控制一個I/O口,第一次開,第二次關,比如P0^1開啟關閉C語言代碼高手請進。
第一,收到你要的按鍵後,是取反某個位,而不是賦值。如P10=~P10;(建議取個名字)
第二,為了避免一次按鍵,多次觸發,對按鍵值操作後,需要進行清除處理;(即賦值成無效的按鍵值),如 IrValue[2]=0xff;(如果沒有按鍵是 0xff的話)
第三,最正確的做法是,收到遙控後,額外提供一個標志位(如 IR_UpdateFlag),程序識別標志位後,進行按鍵選擇,處理完按鍵後,清除標志位。這樣可以避免程序重復、多餘執行按鍵選擇(雖然是無效的按鍵,但那也是選擇),可以提高系統執行效率!