紅外遙控是我們現實生活中必不可少的一員。遙控,顧名思義即在遙遠的地方進行控制,其實也就是說遠程通信。紅外遙控就是以紅外線為傳輸介質的遙控。
紅外遙控器是怎麼工作的?其實它内部就是有一個單片機,單片機的I/O口控制一個紅外LED燈(當然一般會加外部放大電路以增大傳輸距離),在内部編碼後發送出去,然後電視或者空調之類的可以用紅外遙控控制的電器在接收到紅外信号後就會進行解碼,并根據制定好的協議執行特定動作,比如電視可能換台、調節音量等。現在有的手機就帶有紅外遙控功能,其實也是用單片機的功能實現的。
紅外遙控解碼
有的時候,我們希望模拟一些遙控器的功能,這時候就要對這些遙控器的數據進行解碼。
在進行解碼前,我們先來了解一下紅外編碼。紅外編碼有很多種,最常用的編碼是NEC編碼。NEC格式的編碼如下:
NEC格式的紅外編碼是連續的32位二進制碼組。32位二進制碼組之前的引導碼,用于區分每次的傳輸;在起始碼之後的才是32位二進制碼組,其中8位用戶識别碼和8位反碼(反碼主要用于)校驗,用戶識别碼的作用主要是區分不同品牌的遙控器;接着就是8位操作碼和反碼,用于表示哪個按鍵被按下。
NEC格式的編碼除引導碼和起始碼外,用于交互數據的信息采用脈寬調制的串行碼,在38kHz的載波下,以脈寬為0.565ms、間隔0.56ms、周期為1.125ms的組合表示二進制的“0”;以脈寬為0.565ms、間隔1.685ms、周期為2.25ms的組合表示二進制的“1”,其波形如下所示:
了解了紅外編碼,就可以進行紅外解碼了。
實現方法
一般,進行紅外解碼我們需要先将接受到的有效信号與外界環境的幹擾信号分離開,這就是載波的作用。現在市面上有很多一體化紅外接收頭,内部是紅外接收二極管 放大電路 解調器。當接收到紅外信号後,先将其放大,然後把38kHz的信号保留下來。需要注意的是一般一體化紅外接收頭接收到信号時輸出是低電平,沒有接收信号時是高電平。
上面是常用紅外接收頭的引腳,“OUT”腳接單片機的I/O口,“-”腳接電源負極,“ ”腳接電源正極。
接好電路,就要進行解碼工作了.其實解碼工作主要是識别引導碼,還有數據“0”“1”。我們看波形圖,隻要識别紅外接收頭“OUT”引腳高低電平持續的時間就可以了。這個可以用定時器中斷完成,也可以用延時來判斷。由于隻完成解碼工作,我們就用延時的方法更方便。
首先,紅外接收頭沒有接到信号時是高電平,等待其變低。之後,每隔900us測量一次,若在10次内電平變高,說明這不是引導碼,退出重新開始;若10次内并沒有變高,說明這就是引導碼,繼續下一步。接着可以不判斷起始碼,就是直接等待引腳電平變高再變低,就可以接收數據了。
接收數據,就是進行判斷“0”和“1”。從位的定義我們可以發現“0”、“1”均以0.56ms的低電平開始,不同的是高電平的寬度不同,“0”為0.56ms,“1”為1.68ms,所以必須根據高電平的寬度區别“0”和“1”。
每一位“0”或者“1”在接收頭接收信号即低電平的時間都是一樣的,直接等待低電平過去,在高電平時延時600us,如果600us時還是高電平,說明是“1”;如果是低電平即高電平過去了,說明是“0”。将這些數據結合到一起,就是接收到的編碼了。
将單片機的串口與電腦接好,就可以在串口顯示我們讀到的紅外遙控數據了。
實現代碼:
#include <reg52.h>
// --- 紅外接收一體化輸出口 ----------------------------------
sbit IR_Out = P3^2;
bit START_Flag = 0;
bit BOOT_REPEATING_CODE_Flag = 0;
unsigned char DATA[4] = {0};
bdata unsigned char TEMP_BIT;
sbit B0 = TEMP_BIT^0;
sbit B1 = TEMP_BIT^1;
sbit B2 = TEMP_BIT^2;
sbit B3 = TEMP_BIT^3;
sbit B4 = TEMP_BIT^4;
sbit B5 = TEMP_BIT^5;
sbit B6 = TEMP_BIT^6;
sbit B7 = TEMP_BIT^7;
// --- 有無遙控信号判斷函數 ----------------------------------
bit START_Judge();
// --- 連發碼判斷函數 ----------------------------------------
bit BOOT_REPEATING_CODE_Judge();
// --- "0"和"1"識别 ------------------------------------------
bit H_L_LEVEL_Judge();
bit START_Judge()
{
bit TEMP_Flag = 1;
unsigned char i = 0;
//在正常無遙控信号時,一體化紅外接收頭輸出是高電平,程序一直在循環。
while ( IR_Out == 1);
//重複10次,目的是檢測在6876~8352微秒内如果出現高電平就退出解碼程序
for(i =0;i <9; i )
{
DELAY_Us(800); // 測試實際延時約為764~928us
if ( IR_Out == 1 )
{
TEMP_Flag = 0;
break;
}
}
return TEMP_Flag;
}
bit BOOT_REPEATING_CODE_Judge()
{
bit TEMP_Flag = 1;
while( IR_Out == 0 ) ; // 等待高電平避開9毫秒低電平引導脈沖
DELAY_Ms(1); // 測試實際延時約為1.007ms
DELAY_Ms(1); // 測試實際延時約為1.007ms
DELAY_Us(200); // 0.086ms
DELAY_Us(200); // 0.086ms
DELAY_Us(200); // 0.086ms
// 共計2.272ms
if( IR_Out == 0 )
{
TEMP_Flag = 1; // 是連發碼
}
else
{
TEMP_Flag = 0; // 不是連發碼,而是引導碼
}
return TEMP_Flag;
}
bit H_L_LEVEL_Judge()
{
while( IR_Out == 0 ); // 等待地址碼第一位的高電平信号
DELAY_Us(800); // 測試實際延時約為764~928us
if ( IR_Out == 1)
{
DELAY_Ms(1); // 測試實際延時約為1.007ms
return 1;
}
else
{
return 0;
}
// --- 串口初始化 --------------------------------------------
void UART_Initial();
void DELAY_Us(unsigned int Us)
{
unsigned int x;
for(x = 0; x <= (Us/200-1); x );
}
void DELAY_Ms(unsigned int Ms)
{
unsigned int x,y;
for(x = 0; x <= (Ms-1); x )
{
for(y = 0; y <= 120; y );
}
}
void main()
{
unsigned char i;
UART_Initial();
IR_Out = 1;
while(1)
{
START_Flag = START_Judge();
BOOT_REPEATING_CODE_Flag = BOOT_REPEATING_CODE_Judge();
if ( START_Flag && !BOOT_REPEATING_CODE_Flag )
{
for(i =0;i <4; i )
{
B0 = H_L_LEVEL_Judge();
B1 = H_L_LEVEL_Judge();
B2 = H_L_LEVEL_Judge();
B3 = H_L_LEVEL_Judge();
B4 = H_L_LEVEL_Judge();
B5 = H_L_LEVEL_Judge();
B6 = H_L_LEVEL_Judge();
B7 = H_L_LEVEL_Judge();
DATA[i] = TEMP_BIT;
}
for(i =0;i <4; i )
{
SBUF = DATA[i];
while( TI == 0 );
TI = 0;
}
}
}
}
void UART_Initial()
{
SCON = 0x50; // SCON: 模式 1, 8-bit UART, 使能接收
TMOD |= 0x20; // TMOD: timer 1, mode 2, 8-bit reload
TH1 = 0xFD; // TH1: reload value for 9600 baud @
// 11.0592MHz
TR1 = 1; // TR1: timer 1 run
EA = 0; // 關閉總中斷
ES = 0; // 關閉串口中斷
}
}
注
1、紅外遙控的編碼不隻NEC,還有很多也有廣泛應用如RC-5,RC-6等。
2、上面的代碼所用指令是STC 89C52單片機,如需用其它芯片請另行更改。
以上所有信息僅作為學習交流使用,不作為任何學習和商業标準。若您對文中任何信息有異議,歡迎随時提出,謝謝!
,更多精彩资讯请关注tft每日頭條,我们将持续为您更新最新资讯!