2014年12月1日 星期一

[Arduino] 用 IRrepeater 抓取 kbro 凱擘遙控器的 IR raw code

[Arduino] Use IRrepeator to capture IR raw code of kbro TV box's remote control (English)

2016-03-15 update:
本文的方法只能抓到部分凱擘遙控碼, 完整抓法請參考下文:
[Arduino] 完整抓取 凱擘 kbro 遙控器紅外線原碼並發出重新發出以控制機上盒 (NEW)

由於 Sony Locationfree 已經停止支援, 內建的 IR code 找不到支援 kbro凱擘數位機上盒的 code set. 造成無法使用 Locationfree 選台, 而且 Loctionfree LF-BOX1 內建的 IR code 也是有限, 除了 kbro 機上盒不支援以外, 對於 Tivo 的支援也有很多按鍵無法使用.

所以我一直想用 Arduino 做一個 IR 轉譯器(IR translater)的功能, 把 Locationfree 發出來的 Tivo IR code 轉為 kbro IR code. 這樣一來靠一個 Arduino 轉換器就可以讓 Locationfree 支援各種新的機上盒. 同樣的功能也能用在Slingbox上, 不用苦惱為了得到 IR code 的 bin file 到處求爺爺告奶奶.

在製作 IR translater 之前, 抓取 Tivo 和 kbro 遙控器的 IR code 是重要的先遣工作. 本篇文章介紹如何抓取 IR code. 特別的是, 用傳統網路上找到的方法都是用Aurdino IR library 內建的 IRrecvDump 來作為抓取程式, 但是遇到 kbro 這種特別"高拐"的遙控器就抓不出來. 所以我這裡另外介紹一個自創的方法更方便去抓出正確的 IR raw code, 就是使用 IRrepeater 程式.

首先還是先用大家最常用的 IRrecvDump 來抓, 下圖是IR接收器的接線:, 我用了三pin的立體聲座來接IR接收器, 紅色線接到 Arduino nano 的 5V 輸出, 黑線接到地線, 白線接到 digital 11 pin 作為資料輸入:


更近一點: 紅線 5V, 黑線 GND, 白線 D11

回到 Arduino IDE, 在 Borad 選擇我用的 Arduino Nano w/ ATmega328:

詳細步驟就不說了, 網路上很多中英文都有, 安裝 IR library 記得下載後直接拷貝到 Sketchbook folder 下的 libraries 就可以被 Arduino IDE 找到.

選了IRrecvDump example, compiler/upload 之後打開 Serial Monitor 觀察抓到的 IR code, 印出來大致像這樣:

Decoded NEC: A10C7807 (32 bits)
Raw (68): -23088 8950 -4450 550 -1650 550 -550 550 -1650 550 -550 550 -550 550 -550 550 -550 550 -1650 550 -550 550 -550 550 -500 600 -500 550 -1650 550 -1650 550 -550 550 -550 550 -550 550 -1650 550 -1650 550 -1650 550 -1600 600 -500 550 -550 550 -550 550 -550 550 -550 550 -550 550 -550 550 -550 550 -1650 550 -1650 550 -1650 550
A10C7807

如果有順利 parser 出 IR code type, 就會像上面的 NEC 字樣印出來,  後面跟著一組IR code代號, 後面印出 Raw code. IR code 和 Raw code 都可以用, 如果有辨認出 type, 則直接用 IR code, 沒有的話就用 Raw code, 注意 Raw length 也需要記下來.

在Raw 的使用上, 第一個數字要被排除, 以上面抓出來的為例子, -23088 就應該排除, 不要貼到後續 IRsend 的程式中, 然後把所有 raw code 陣列中的負數都改為正數, 總長度 68 -1. 這樣就可以貼到 IRsend 中的 unsigned int array 去測試. 以上面的例子, 修改後變成這樣:

8950 4450 550 1650 550 550 550 1650 550 550 550 550 550 550 550 550 550 1650 550 550 550 550 550 500 600 500 550 1650 550 1650 550 550 550 550 550 550 550 1650 550 1650 550 1650 550 1600 600 500 550 550 550 550 550 550 550 550 550 550 550 550 550 550 550 1650 550 1650 550 1650 550

再貼到程式裡面, 弄成array的話記得數字間加上逗號, 程式類似長這樣:
https://github.com/smallbeetw/arduinosketch/blob/master/Panasonic2KbroIRtrans/Panasonic2KbroIRtrans.ino#L50

以下列出 Tivo 的 Channel up/down button, 由於抓到 NEC type, 就不列 Raw code了:

TIVO
    Channel up:        NEC: A10C7807 (32 bits)
    Channel down:    NEC: A10CF807 (32 bits)

Tivo的遙控器是這樣長相:


取得 IR code 之後, 就可以試試看將 IR code 透過 IR LED 發射出去讓設備(電視或機上盒)接收, 接收後設備有對應的動作就是code抓對了. 硬體線路很簡單, 直接將 IR LED 長腳接 D3 pin, 短腳接GND:
講到 IR LED 的接腳有個很蠢的故事, 就是我第一次試的時候沒有接電阻就直接把LED長腳接到 Arduino 5V 輸出的 pin, 瞬間 LED 就炸掉了, 這是第一次親身體驗LED炸開, 眼睛會比較危險.
上面我做的時候沒接電阻, 因為D3 pin程式寫的時候輸出電流不會持續輸出, 所以不至於像 5V pin 那樣炸掉. 不過還是串個電阻會比較安全.

軟體的部份比dump還簡單,  在 Arduino IDE 上選 IRsendDemo, 改掉 type 和 IR code 代號, 編譯完upload就可以了. 用 Serial Monitor 隨便打個字母驅動 IR code 發射, 看看機上盒有無反應就知道是否 IR code 抓錯.

 如果實驗的時候你懷疑 IR LED 有沒有亮, 由於紅外線肉眼無法看到, 可以拿一台數位像機打開用鏡頭和數位像機的螢幕去看看 IR LED 有沒有亮起來就可以了.

以上是大多數網路教學用 IRrecvDump 來抓 IR code 的方法. 但是這個方法並非百試百靈, 有部份遙控器的 IR raw code 很難用 IRrecvDump正確抓到. 原因不明, 講句廢話應該是和 Arduino IR library 的實作有關, 我不是 IR 專家, 也沒有示波器來驗證, 所以無從得知真正的問題.

剛好手上要抓的 KBRO 凱擘數位機上盒的遙控器就是屬於這種:

為了這個KBRO的 IR code, 我卡關卡了一整天, 弄到半夜兩點才找到方法, 我的做法是改之前我寫的 IR repeator 程式, 把它加上 print IR raw code 的程式:
https://github.com/smallbeetw/arduinosketch/blob/master/IRrepeator/IRrepeator.ino

這個的好處有點像把 IRrecvDump和 IRsend 兩支程式合併起來, 就是 Arduino 透過 IR 接收器抓到 raw code 之後立刻用 sendRaw() 把它從 IR LED 再重新打出去, 所以叫做 IRrepeator. 這種小設備在很多拍賣都有賣, 就是 IR 延長設備, 譬如你的電視 IR 接收窗被音箱遮到, 或者撥放機放在電視櫃裡面遙控不到, 就可以用這種把 IR code 重新打出去的程式.

硬體的接線如下, 就是把之前的 IR 接收器和 IR LED 都一併接上麵包板:

簡易電路架構圖如下:

用 IRrepeator 抓 raw code 的好處是, 如果 Arduino 抓到正確 raw code 的機率很低, 但不是完全沒有的時候, 也就是說我們把遙控器對著接收器一直按按鍵, repeator也會一直把IR code打出去給設備, 直到看到機上盒收到訊號, 我們就知道這組 raw code 是真正有效的, 可以把它紀錄下來.

簡單的說, IRrepeator 讓使用者可以立刻測試抓出來的 raw code 是否真正有效. 就不需要等到把 raw code 貼到 IRsend 程式之後測試才發現根本是抓錯 code, 又要用 IRrecvDump 抓一次.

用這個方法我先抓了 kbro 的 channel up/down 兩個按鍵的 raw code, 有空再繼續抓其他的(發現Arduino IR library 抓 kbro失敗率真的很高):

KBRO
  Channel UP:         250,850,250,1000,250,700,250,750,200,750,250,2450,250,750,200,750,250
  Channel DOWN:   250,850,250,850,250,750,200,750,250,700,250,2600,250,750,200,750,250

我用 sendRaw(sendRawCodes,17,38) , 後面加上 delay(40), 實測過沒有問題. 網路說一般使用 38 hz 作為 IR 傳遞的原因是因為自然界比較少用到這個頻率, 可以避免太多雜訊.

除了用 IRrepeator 對付這種有時候可以抓到 IR code 的遙控器以外. 還有一些狀況, 像是據說 Sony 有時候會用 40 hz, 或者像 Samsung 的 raw code 長度會大於 RAWBUF 的預設值 100 (這時候請修改 IR library 中的這個 define 從 100 改為 400). 還有 delay 可以設定長一點, 用 1000 來測試, 避免太快發下一個 IR code. 這些都是在無法順利用 IRrecvDump 抓出 IR code 時可以試的方向.

這次我順便抓了 LF-Box1 的 RM-Box1 遙控器的 IR code, 由於這款遙控器市場上很難得到, 所以特別紀錄在這裡:



RM-BOX1
  UP:                    Decoded SONY: 4EB31 (20 bits)
  DOWN:             Decoded SONY: CEB31 (20 bits)
  LEFT:                 Decoded SONY: AEB31 (20 bits)
  RIGHT:               Decoded SONY: 2EB31 (20 bits)
  決定:                  Decoded SONY: 6EB31 (20 bits)
  畫面xxx:             Decoded SONY: 14B31 (20 bits)
  右下:                   Decoded SONY: 34B31 (20 bits)
  左上:                   Decoded SONY: ACB31 (20 bits)
  戾xxx:                  Decoded SONY: 54B31 (20 bits)
  電源:                   Decoded SONY: A8B31 (20 bits)
  入力切換:           Decoded SONY: A4B31 (20 bits)
  畫面表示:           Decoded SONY: 44B31 (20 bits)
  設定:                   Decoded SONY: 4B31 (20 bits)
  XXX電源:           Decoded SONY: A90 (12 bits)
  入力切換(左上): Decoded SONY: A50 (12 bits)
  音量(+):               Decoded SONY: 490 (12 bits)
  音量(-):                Decoded SONY: C90 (12 bits)
  消音:                    Decoded SONY: 290 (12 bits)

more: [Arduino] 製作紅外線遙控轉換器 IR translate Panasonic 電漿電視轉 Kbro 凱擘
more: [Arduino] 完整抓取 凱擘 kbro 遙控器紅外線原碼並發出重新發出以控制機上盒