uc/OS-II下ARM7定時器捕獲實現(xiàn)紅外解碼
LPC2103芯片帶有定時器捕獲,可以設置為下降沿,上升沿或雙邊沿捕獲中斷。因為結合到此款紅外編碼方式,所以采用下降沿進行捕獲。定時器0因為用作了系統(tǒng)節(jié)拍定時器,所以我選擇了定時器1的捕獲。
本文引用地址:http://www.ex-cimer.com/article/201611/319111.htm定時器1進行如下的初始化。
[plain]view plaincopyprint?
- voidSetTimer(void)
- {
- T1TCR=0x02;//關閉復位定時器1
- T1PR=10;//11分頻,約1us計時一次(外設時鐘11.0592MHZ)
- T1CCR=0x06;//下降沿捕獲并產生中斷
- T1IR=0x10;//清除定時器1捕獲0中斷
- T1TCR=0x01;//開啟定時器1
- VICVectAddr1=(uint32)Timer1_Handler;//中斷向量相關設置
- VICVectCntl1=(0x20|0x05);
- VICIntEnable|=(1<<5);
- }
然后在定時器1中斷服務函數(shù)里,就算出相鄰兩次下降沿之間的差值。然后通過消息郵箱把消息發(fā)送到脈寬檢測任務進行處理。
[plain]view plaincopyprint?
- voidTimer1_Exception()
- {
- staticuint32tOld;//保存舊的下降沿捕獲值
- uint32tNew;//保存新的下降沿捕獲值
- statici;
- OS_ENTER_CRITICAL();
- T1IR=0x10;//清除定時器1捕獲0中斷
- tNew=T1CR0;
- tValue=tNew-tOld;//得到兩次下降沿之間的差值
- tOld=tNew;//以便下次中斷處理
- OSMboxPost(Msg,(void*)tValue);//發(fā)送消息郵箱,行為同步
- OS_EXIT_CRITICAL();
- VICVectAddr=0x00;
- }
在檢測脈寬任務里,我只需要根據(jù)測定脈寬與本來編碼原有的脈寬進行比較判斷,然后進行相關移位數(shù)據(jù)操作,得到數(shù)據(jù)碼值。
因為我的遙控器有點不同,地址碼與地址反碼不互反,所以不能進行地址的判斷,所以濾除掉了引導碼與地址碼,直接進行了數(shù)據(jù)碼的處理。
只要數(shù)據(jù)碼與數(shù)據(jù)反碼取反相同,則調試LED閃爍一下。
當然我還沒有具體知道遙控器按鍵對應的具體碼制是多少,而且還沒有檢測到連發(fā)碼,留待下次把碼制通過串口發(fā)送到上位機進行顯示。
[plain]view plaincopyprint?
[plain]view plaincopyprint?
這是檢測脈寬任務的核心代碼:
while(1) - {
- OSMboxPend(Msg,0,&err);//等待脈寬檢測消息
- if(tValue>2145&&tValue<2345)//進行脈寬檢測
- {
- ucTemp=1;//邏輯1
- }
- elseif(tValue>1025&&tValue<1225)
- {
- ucTemp=0;//邏輯0
- }
- elseif(tValue>13400&&tValue<13600)
- {
- ucCounter=0;//引導碼
- usData0=0;
- usData1=0;
- flag=1;
- }
- else
- {
- continue;
- }
- if(flag)//數(shù)據(jù)處理過程
- {
- ucCounter++;
- if(ucCounter<16)
- {
- usData0|=(uint16)ucTemp;
- usData0<<=1;
- }
- elseif(ucCounter==16)
- {
- usData0|=(uint16)ucTemp;
- }
- elseif(ucCounter<32)
- {
- usData1|=(uint16)ucTemp;
- usData1<<=1;
- }
- elseif(ucCounter==32)
- {
- usData1|=(uint16)ucTemp;
- flag=0;
- OSSemPost(Sem);//發(fā)送信號量,進行碼制轉換任務,我的任務只是簡單的實現(xiàn)了判斷解碼是否成功。
- }
- }
- }
評論