一、紅外遙控解碼部分從昨天開始整,一直到現在才解碼成功!中途遇到了不少問題,結果出來后還是覺得有必要總結一下,唉!
1、首先我又是懷疑我硬件電平不兼容德問題,后來給接上3.3V的電壓,還是不行,好吧,算失敗了,在網上查閱了比較多的帖子,也找了比較多的資料,最終還是決定用原本那個生了銹的遙控來解碼!
2、然后準備參照著原來51的思想來移植代碼,也確實找到類似的代碼貌似使用的2.0的庫寫的,單步調試了半天,總感覺在延時部分出了點問題,所以比較郁悶,好吧,分析來分析去的,結果真的是沒有半點現象啊!果斷網上求助,游蕩了一會,壓根沒人理,高手不屑一顧呀!!偶然間讓我遇到了原子哥的那段紅外的代碼,拖出來分析,所以就有了今晚解碼成功的結果!
3、我照著原子的移植,我用的是自己的延時,也就是系統定時器,MTD,單步調試的時候,發現居然死在了systick那里,進不了中斷,一步步觀察,好像導致進不了中斷的原因就是:我已經進了外部中斷,心想,沒道理啊,系統定時器的優先級不應該是高于外部中斷的么,因為他是核決定的呀(至少我是這么想的),然后又查了相關資料,據說系統定時器的中斷優先級是最低的,這時候我才恍然大悟!
現在開始分析代碼,雖然說原子的代碼風格不怎么樣,但是個人覺得他真的好牛逼,庫函數是人家ST公司搞出來的,我想,原子的這套代碼,應該基本上是他自己一個人整出來的吧!
二、所謂紅外遙控!(針對我手上的紅外遙控)
1、紅外解碼一直是單片機中應用較多的,需要設備加裝專用解碼芯片,這就大大減輕了單片機的負擔。需要單片機樣例使用延時做紅外解碼,比較容易理解,
下面通過TC9012和uPD6121芯片為例大致了解解碼原理:
先看一接收頭產生的波形圖,這是原子的一張圖
% U, K“ ?3 K2 _( j‘ a! e: K: o
從上圖可以看出 9.0ms高電平+4.5ms低電平稱為頭碼,用于識別是否遙控碼開始,這是一張連續發射碼的波形圖(就是一直按下某一遙控器按鍵)。; n5 [
+ z; ^4 d( T# L) h” Y6 B5 j3 T
頭碼過后會出現4個8位的數據,我們最終目的就是要把這個 32位(4x8)從一體化紅外接收頭提取出來,并轉換成16進制數,用于區分不同按鍵按下得出的不同數值。
在遙控器發射波形中,可以知道,8位數中的0或者1不是用高低電平表示,而是用不同的低電平的寬度表示,0.565ms表示0,1.69ms表示1,2個位中間還會有一個0.56ms的高電平
看到如上圖波形,表示單片機引腳可以接收到的波形,我們只要通過單片機讀取波形并分析波形的寬度,然后分辨出是頭碼,還是0或者1,最后整理出這組碼的16進制組合。正確的解碼結果是按同一個按鍵得出的16進制數值是不變化的。通過這個原理,我們可以分辨出每個按鍵的鍵值。
! z7 B/ `2 Q: z
基本原理分析如下,如接收到頭碼是4.5ms低電平+4.5ms高電平,我們分析 第一個下降沿到第二個下降沿的寬度是 9ms,我們判斷這個頭碼可以給定一個范圍,只要數據在這個范圍內則認為頭碼是正確的,檢測頭碼正確后接著檢測剩下的32位數值。
2、用自己的話概括就是:平常是高電平---》按鍵按下---》產生引導碼(9+4.5)ms---》然后判斷是不是連續發送---》1還是0---》存儲碼值---》轉換碼值!簡單就是這樣!
3、首先是我的主程序,代碼注釋都非常詳細,不解釋了!注意碼值需要依據自己的遙控而定,我就是單步測試出來的!
#include“stm32f10x.h”
#include“Usart.h”
#include“stdio.h”
#include“Remote_Control.h”
#include“Delay.h”
/**************************PA1接紅外接收端************************************/
/************由于沒有做外設測試的程序是:按鍵PA0僅一個LED燈*******************/
/*******由于沒有做外設測試的程序是:串口采用的是PA9-》(T《-》T),PA9-》(R《-》R)*****/
intmain(void)
{
u8key;
USART1_Config();
delay_init(72);//延時初始化
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//設置NVIC中斷分組2:2位搶占優先級,2位響應優先級
printf(“rn(”__DATE__“-”__TIME__“)rn”);
Remote_Init();
while(1)
{
if(Remote_Rdy)
{
key=Remote_Process();
switch(key)
{
case0x68:printf(“0n”);break;//0
case0x30:printf(“1n”);break;//1
case0x18:printf(“2n”);break;//2
case0x7a:printf(“3n”);break;//3
case0x10:printf(“4n”);break;//4
case0x38:printf(“5n”);break;//5
case0x5a:printf(“6n”);break;//6
case0x42:printf(“7n”);break;//7
case0x4a:printf(“8n”);break;//8
case0x52:printf(“9n”);break;//9
default:break;
}
}
}
}
4、然后是驅動程序
/*-------------------------協議--------------------------
開始拉低9ms,接著是一個4.5ms的高脈沖,通知器件開始傳送數據了
接著是發送4個8位二進制碼,第一二個是遙控識別碼(REMOTE_ID),第一個為
正碼(0),第二個為反碼(255),接著兩個數據是鍵值,第一個為正碼
第二個為反碼。發送完后40ms,遙控再發送一個9ms低,2ms高的脈沖,
表示按鍵的次數,出現一次則證明只按下了一次,如果出現多次,則可
以認為是持續按下該鍵。
---------------------------------------------------------*/
#include“Remote_Control.h”
#include“Delay.h”
u32Remote_Odr=0;//命令暫存處
u8Remote_Cnt=0;//按鍵次數,此次按下鍵的次數
u8Remote_Rdy=0;//紅外接收到數據
/************************初始化紅外接收引腳的設置**********************************/
/******************選擇PA1腳作為外部中斷,用于紅外輸入*****************************/
voidRemote_Init(void)
{
GPIO_InitTypeDefGPIO_InitStructure;//GPIO
NVIC_InitTypeDefNVIC_InitStructure;//中斷
EXTI_InitTypeDefEXTI_InitStructure;//外部中斷線
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_AFIO,ENABLE);
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU ;?
責任編輯;zl
評論
查看更多