正點原子的ALIENTEK遙控器
用戶碼00ff
16進制鍵值碼表
45 46 47
44 40 43
07 15 09
16 19 0d
0c 18 5e
08 1c 5a
42? 4a
紅外接收管 通用型即可
---------------------------------------------------------------------------
下面是類似的波形圖,這個是網(wǎng)上找的,圖片較大,縮小后看上去有些模糊,不過沒關(guān)系,數(shù)據(jù)手冊上一般都有
遙控器在發(fā)射紅外信號之前,我們的mcu已經(jīng)開啟了定時器
在定時器中斷函數(shù)中的全局變量irTimeCounts++ 一直在自加
irTimeCounts多長時間加一次? 或者說多長時間進入一次定時器中斷函數(shù)呢?
在方式2時,t=256*12/11059200 約為277.78us
============================================================
1.對于1.125ms的時間,timer0會中斷1.125/t=4.05次 也就是要5次,算上各種誤差,(頂多5次,至少4次)
2.對于2.25ms 的時間,timer0會中斷 2.25/t=8.1次, 也就是要9次,算上各種誤差,(頂多9次,至少8次)
對于這里的次數(shù),比N次小一點,就是N次,比N次大一點,就是N+1次,因為不會有半次,
誤差給它正負0.4次足夠了,給了誤差后再算次數(shù),直接舍入就行
那么我們就檢測irTimeCounts的值,
如果小于6或7,那么接收到的數(shù)據(jù)為0
如果大于6或7,那么接收到的數(shù)據(jù)為1
從計算結(jié)果來看,我覺得0.56,1.125,2.25這些值的出現(xiàn)是比較合理的,
能有效避免由于器件誤差和環(huán)境造成的數(shù)據(jù)誤判,而且計算也方便
C代碼
#include"my51.h"
#include"ir.h"
#include"smg.h"
voidmain()
{
timer0Init();//定時器0初始化
int0Init();//外部中斷0初始化
while(1)
{
if(irTimeCountsArrProcess())//如果成功接收并解析完成一幀數(shù)據(jù)
{//就讓蜂鳴器響一下
beep=0;//蜂鳴器開啟
led4=~led4;//4號燈反轉(zhuǎn)一下
}
displaySMG(irCodeByteDataProcessForSmg());//顯示
beep=1;//蜂鳴器關(guān)閉
//由于displaySMG()函數(shù)執(zhí)行時間較短,故蜂鳴器響聲時間也較短,聽到滴了一下
}
}
C代碼
#ifndef_51IR_H_
#define_51IR_H_
#include"my51.h"
externu8datasmgWela[7];//數(shù)碼管顯示的數(shù)據(jù)
externvoidint0Init();//外部中斷0初始化
externvoidtimer0Init();//定時器0初始化
externboolirTimeCountsArrProcess();//成功解析一幀中斷數(shù)據(jù)返回TRUE
externu8*irCodeByteDataProcessForSmg();//將遙控器碼值處理成數(shù)碼管可顯示數(shù)據(jù)
#endif
C代碼
#include"ir.h"
u8irTimeCounts=0;//定時器0在方式2下8位自動重裝時的中斷計數(shù)值
u8irTimeCountsArr[32]={0};//存放紅外接收數(shù)據(jù)時的中斷次數(shù)記錄值,
u8bitNum=0;//標志當前接收的是第幾個比特位
u8irReceFlag=0;//紅外接收一幀數(shù)據(jù)未完成標志,為1時完成
u8irCodeByteData[4]={0};//保存接收到的4個字節(jié)的有效數(shù)據(jù)
u8irTimeCountsArrProcessOk=0;//對接收到的33位數(shù)據(jù)處理未完成標志,1完成
voidint0Init();//外部中斷0初始化
voidtimer0Init();//定時器0初始化
boolirTimeCountsArrProcess();//解析中斷次數(shù),即解碼
u8*irCodeByteDataProcessForSmg();//將遙控器碼值處理成數(shù)碼管可顯示數(shù)據(jù)
u8*irCodeByteDataProcessForSmg()//將解碼的4字節(jié)數(shù)據(jù)處理成數(shù)碼管可顯示的數(shù)據(jù)
{
if(irTimeCountsArrProcessOk)//檢測一幀數(shù)據(jù)是否解析完成
{
//這里的用戶碼只顯示低八位,因為高八位反正都是00(手上2個遙控器都是00)
//然后還顯示遙控鍵值及其反碼,我們的數(shù)碼管只有6位,只顯示3字節(jié)數(shù)據(jù)
if(irCodeByteData[2]+irCodeByteData[3]!=0xff)//校驗數(shù)據(jù)的完整性
{//最后一個字節(jié)是鍵碼的反碼
led6=0;//調(diào)試代碼
}
else
{
smgWela[0]=irCodeByteData[1]>>4;//取高4位
smgWela[1]=irCodeByteData[1]&0x0f;//取低4位
smgWela[2]=irCodeByteData[2]>>4;
smgWela[3]=irCodeByteData[2]&0x0f;
smgWela[4]=irCodeByteData[3]>>4;
smgWela[5]=irCodeByteData[3]&0x0f;
smgWela[6]=0xff;//小數(shù)點全不顯
}
irTimeCountsArrProcessOk=0;//標志清零,下一次有未解析的數(shù)據(jù)時才會再解析
}
returnsmgWela;
}
boolirTimeCountsArrProcess()//對接收到的32位的中斷次數(shù)數(shù)據(jù)進行解析
{
u8i,j,k,value=0;
if(irReceFlag)//檢測是否已經(jīng)接收到新的4字節(jié)的紅外通信數(shù)據(jù)
{
for(j=0;j<4;j++)//有4個字節(jié)
{
for(i=0;i<8;i++) //對每個字節(jié)的8位數(shù)據(jù)處理