你家裡是否有一個電視機遙控器或者空調機遙控器呢?你是否也想讓它遙控其他的電器甚至讓它遙控您的電腦呢?那好,跟我一起做這個「紅外遙控解碼器」。
該小製作所需要的元件很少:單片機TA89C2051一隻,RS232接口電平與TTL電平轉換心片MAX232CPE一隻,紅外接收管一隻,晶振11.0592MHz,電解電容10uF4隻,10uF一隻,電阻1K1個,300歐姆左右1個,瓷片電容30P2個。發光二極體8個。價錢不足20元。
電路原理介紹:
主控制單元是單片機AT89C2051,中斷口INT0跟紅外接受管U1相連,接收紅外信號的脈衝,8個發光二極體作為顯示解碼輸出(也可以用來擴展接其他控制電路),U3是跟電腦串行口RS232相連時的電平轉換心片,9、10腳分別與單片機的1、2腳相連,(1腳為串行接收,2腳為串行發送),MAX232CPE的7、8腳分別接電腦串行口的2(接收)腳、3(發送腳)。晶振採用11.0592MHz,這樣才能使得通訊的波特率達到9600b/s,電腦一般默認值是9600b/s、8位數據位、1位停止位、無校驗位。電路就這麼簡單了,現在分析具體的編程過程吧。
開始位是以3.6ms低電平然後是3.6ms高電平,然後數據表示形式是0.9ms低電平0.9ms高電平周期為1.8ms表示「0」,0.9ms低電平2.4ms高電平周期為3.3ms表示「1」,編寫程序時,以大於3.4ms小於3.8ms高電平為起始位,以大於2.2ms小於2.7ms高電平表示「1」,大於0.84ms小於1.11ms高電平表示「0」。因此,我們主要用單片機測量高電平的長短來確定是「1」還是「0」即可。定時器0的工作方式設置為方式1:movtmod,#09h,這樣設置定時器0即是把GATE置1,16位計數器,最大計數值為2的16次方個機器周期,此方式由外中斷INT0控制,即INT0為高時才允許計數器計數。比如:
jnbp3.2,$
jbp3.2,$
clrtr0
這3條指令就可以測量一個高電平,接下來讀取計數值TH0,TL0就可以分辨是起始位還是「1」或「0」。在確定碼錶之前,您可以使用P0口的8個發光二極體來顯示編碼,16位編碼分兩次顯示:
movp0,keydata
acalldelay_1s;//1ms延時子程序
movp0,keydata+1
ljmpmain
根據P0相繼的兩次顯示的編碼,記錄每個按鍵的編碼,形成編碼表,即遙控器編碼的解碼完畢。碼錶確定之後,以後接收到遙控器的編碼之後,就與碼錶比較,找到匹配的碼項,並把該碼項對應的順序號輸出到P0口,同時也把順序號向串行口輸出到電腦,電腦接收該數據後由串口軟體決定如何處理。
程序不長,下面是完整的程序和注釋:(先看流程圖)
keydataequ30h;//該地址和31H地址用來存放遙控器按鍵編碼。
org00h
main:
movkeydata,#0;//清零
movtmod,#09h;//設置定時0方式1,GATE=1
movr7,#0;//計數器,用來計數是否滿8位
movr6,#0;//計數器,用來計數是否滿2位元組(解16位編碼)
jbp3.2,$;//是否為低電平
again:;//如果為低,繼續往下面執行
movtl0,#0;//清零TL0
movth0,#0;//清零TH0
setbtr0;//開啟定時器0
jnbp3.2,$;//等待高電平到來
jbp3.2,$;//高電平到來,此時開始計數
clrtr0;//高電平結束,停止計數
mova,th0;//讀取th0值,TL0忽略不計clrc;//
subba,#12;//
jcagain;//th0《12則轉,即小於3.4ms,你可以算一下這個時間
mova,#14;//
clrc;//
subba,th0;//和14比較,如果TH0》14則大於3.8ms
jcagain;//大於3.8ms,從新再檢測
nextbit:;//起始位找到了,然後下一位
movtl0,#0;//
movth0,#0;//
setbtr0;//啟動定時器
jnbp3.2,$;//等待高電平
jbp3.2,$;//高電平到來,此時開始計數
clrtr0;//高電平結束,停止計數
mova,th0;//讀取計數值,TL0忽略不計
clrc;//
subba,#8;//th0和8比較
jcnext;;;;//若《2.2ms則轉,再判斷是否大於0.84ms
mova,#10;//再跟10比較
clrc;//
subba,th0;//
jcagain;;;;;;;//若》2.7ms,則放棄,從新檢測
mova,keydata;//符合大於2.2ms小於2.7ms,即為「1」
setbc;//C=1
rrca;//把1移位進A
movkeydata,a;//保存
incr7;//計數器加1
cjner7,#8,nextbit;//是否滿8位
incr6;//計數加1
cjner6,#2,last8;//是否滿兩字節
sjmpseach;//不滿兩字節,再新採集
last8:;//滿1位元組,再接下來第二字節
movkeydata+1,a;//把第一字節編碼數據保存到31h裡
movr7,#0;//計數器R7清零
sjmpnextbit;//繼續採集數據
next:;//小於2.2ms時轉到這裡
mova,th0;//讀取計數值TH0
swapa;//高4位與低4位對換
movr1,a;//保存到R1
anltl0,#0f0h;//取TL0高4位,低4位忽略不計
mova,tl0;//
clrc;//
rrca;//
rrca;//
rrca;//
rrca;//
adda,r1;//
movr1,a;//
subba,#30;//以上幾行是把TH0的低4位和TL0的高4位合併為1位元組作為計數值
jcnextbit;//判斷是否《0.84ms,是則放棄,繼續採集
mova,r1;//否
clrc;//
cjnea,#64,continue;//跟64比較
continue:;//
jncnextbit;//a》64表示採樣值》1.11ms放棄
mova,keydata;//否則,符合位「0」
clrc;//C=0
rrca;//把零右移進A
movkeydata,a;//保存
incr7;//計數器加1
cjner7,#8,nextbit;//是否滿8位
incr6;//計數器加1
cjner6,#2,last_8;//是第一字節已經滿
sjmpseach;//
last_8:;//如果為第二字節
movkeydata+1,a;//則保存第一字節到31h
movr7,#0;//清零R7
sjmpnextbit;//
seach:;//匹配按鍵編碼
movr0,#-2;//按鍵編碼字節個數計數器
movr1,#-1;//按鍵順序計數器
seach1:;//
incr0;//
seach2:;//
incr0;//
incr1;//
cjner1,#29,compare;//是否R1=29
sjmpexit0;//
compare:;//開始匹配
mova,r0;//
movdptr,#keycode;//地址指針指向碼錶首址
movca,@a+dptr;//取碼
cjnea,keydata,seach1;//比較
incr0;//R0+1,再比較下一字節(每個按鍵編碼為2位元組)
mova,r0;//
;movdptr,#keycode;//
movca,@a+dptr;//比較
cjnea,keydata+1,seach2;//是否匹配,不匹配則繼續跟下一字節比較
movp1,r1;//如果匹配,把按鍵順序號輸出到p1
send:;//
movtmod,#20h;//設置timer1,mode2
movtl1,#0fdh;//設置定時器初值
movth1,#0fdh;//
movscon,#01010000b;//以上設置,即設置串口波特率係數為:9600,8,1,0
setbtr1;//啟動定時器1
loop_s:;//
movsbuf,r1;//把R1(按鍵順序號)輸出到串口
jnbti,$;//等待是否發送完畢
clrti;//發送完畢,清零TI
exit0:;//
ljmpmain;//循環
keycode:;//每兩字節代表一個按鍵的編碼
db11111000b,00000000b,11111100b,00000000b,11111001b,11000000b
db11111100b,11000000b,11111010b,00000000b,11111010b,00100000b
db11111010b,01000000b,11111010b,01100000b,11111010b,10000000b
db11111010b,10100000b,11111010b,11000000b,11111010b,11100000b
db11111011b,00000000b,11111011b,00100000b,11111011b,01000000b
db11111011b,01100000b,11111111b,01100000b,11111111b,10100000b
db10001100b,10001110b,10001101b,11101110b,10001100b,10101110b
db10001101b,11001110b,11111000b,11100000b,11111100b,10000000b
db11111100b,01000000b,11111001b,10100000b,11111100b,10100000b
db11111100b,01100000b
end
各種遙控器編碼不同,如果你採用的是其他遙控器,修改幾個參數即可(當然按鍵的編碼表肯定不同了),即計數器的值不同,不過有的遙控器有機器碼(機器碼每個按鍵都是一樣的),此時可以跳過機器碼的採集。