ATtiny10 登場
前に $2 足らずで売ってた 6ボタン学習リモコン、何個か買ったものの使い道が思い当たらず放置してたんですが、考えに考えてついに使い道を思いついたのです。
部屋の照明の ON/OFF(1)、エアコンの冷房の強(2)、弱(3)、暖房の強(4)、弱(5)、停止(6)、これで 6ボタンぴったり。個別のリモコンを使わなくても、これ一つをベッドに持ち込めば自分の寝床が部屋のコントロールセンターです。この学習リモコンは複数買ってあるので、家族それぞれに持たせることもできます。近年まれに見る良いアイディアです。
もちろんエアコンのリモコンはちょっと特殊なのは知ってます。
テレビのリモコンなんかは、音量を上げるボタンを押すと「音量を上げるボタン」のコードが送られ、それを受信したテレビが音量をひとつ上げるのです。
でもエアコンのリモコンには、「温度を上げるボタン」のコードはありません。リモコン自身が冷暖房等の運転状態や温度などを管理していて、温度を上げるボタンが押されたら自分が管理している温度の値をひとつ上げて、「冷房 25℃ 風量自動 にせよ」と運転状態一切合切を送るのです。
とどのつまり、学習させようとしている「冷房 強」とか「冷房 弱」みたいなのは温度の上下ボタンの話ではなく、「冷房 24℃」と「冷房 28℃」みたいな、特定の温度を決め打ちで学習させるということです。これなら 6ボタンでやれるわけです。
で結論から言うと、このアイディアは失敗に終わりました。例の学習リモコンではエアコンのリモコンを学習できなかったのです。前述のとおりエアコンのリモコンは一度の送信で運転状態を全て送るので、コマンドがとても長いのです。この学習リモコンは、この長いコマンドを覚えきれなかったわけです。
放置していた学習リモコンをなんとかして使う方法を思いついただけでしたが、このアイディアが素晴らしすぎて、それがいざ実現できないとるとなおさらコントロールセンターの夢が捨てられません。
そんな折、ATtiny10 がここにありました。
このワンチップマイコンで、学習とまではいかなくても特定ボタンリモコンを作ってしまえいいのです。見た目とは裏腹のスペックを持つこいつなら、赤外線リモコンの信号なんてたやすく再現してくれるはずです。というわけで早速開発に着手し、かのリモコンはまた放置と相成りました。
もっとピンを!
さて ATtiny10 ですが、乾電池 2本で動かせるので赤外線リモコンとしては丁度いいです。
処理速度も、必要ならば乾電池 2本で 8MHz 出せそうですから多分問題無いでしょう。
パワーダウンによって何もしていないときはほぼ電気を食わずに眠ることができ、この点もリモコンにぴったりです。
GPIO は 4本ですから、1本を赤外線LEDに割り当て、ボタンには残りの 3本を割り当てられそうです。欲しいボタンは 6個なので、ATtiny10 を 2つ使えばカバーできる気がします。安いマイコンですし、デュアルCPU とかカッコイイですし。
…
なんてダサい話がありますか。
タイニーなマイコンを使ってピン(GPIO)が足りないからマイコンおかわりとか、本末転倒の極みです。
だったら最初からピンがたくさんあるマイコンを使えばいいという話です。
こういうのとか。
しかしそういうマイコンはとても高性能で、たかだか赤外線リモコンにするにはもったいないです。実装されたハードウェアコンポーネントの大部分が使われずにシリコンの肥やしになるとか、作った人に申し訳が立ちません。
そもそも ATtiny10 にすら使わず余らせてしまうコンポーネントがあるというのにです。
例えばこれとか。
そう、ADC。
アナログ電圧をデジタル化して読み取るコンポーネントで、赤外線リモコンには関係の無いものです。こんなもの内蔵されていても、電源供給を止めて死んでいてもらうだけです。
しかしそもそも、こんな小さなマイコンで作れるアプリケーションで ADC とか何に使うんでしょうね?
たとえばこんな感じ?
おや?
ADC でボタンを読む
ボタンごとに異なるアナログ電圧を割り当てれば、 ADC を使って 1ポートで複数のボタンを読み取れそうなことは思いつきました。ただしリモコンとして実用化するには、省電力である必要があります。
先ほどの回路
これは Vcc から GND まで抵抗を通じて常にアイドル電流が流れてしまい、省電力ではありません。
また、パワーダウンした ATtiny10 を目覚めさせるためのピンレベル割込み(INT0)をかけるには、どのボタンが押されても入力ピンを Low レベルに引き下げられる必要がありますから、最も高いボタン電圧でも Vcc の 1/3(Low保証電圧、データシートより)以下にする必要があります。これはすなわち VREF を Vcc とする ADC のレンジも 1/3 しか使えないということで、なんかもったいないです。
※なおお気付きだと思いますが、ボタンの同時押しは判別できません。しかしリモコンなどの用途ではこれを許容できると思います
これらを踏まえて改良すると、こうなります。
先ほど Vcc に直結だった抵抗をマイコン自身の出力ピンにつないで、出力を制御することで常時アイドル電流が流れるのを抑制するのです。
制御手順はこうです。
マイコンはパワーダウン状態です。
INT0ピンはプルアップしてデジタル入力とし、INT0割込みを有効にしておきます。
BTN-OUT(PBx) は Low を出力します。
これで、どのボタンが押されても INT0 を Low に引き下げることができますし、抵抗にはボタンの押下まで電流は流れません。
ボタン押下によって INT0ピンが Low に引き下げられると、INT0割込みによってマイコンは目覚めて動き出します。
BTN-OUT(PBx) に High を出力します。これで Vcc(High) ~ GND が抵抗群で分圧され、ボタン毎に異なるアナログ電圧が現れます。
INT0ピンのプルアップはやめてアナログ入力とし、ADC によってピン電圧を読み取ります。
※プルアップをやめないと、それによってボタン電圧が狂ってしまいます
ボタンの読み取りが終了したら、BTN-OUT(PBx) は Low にします。
そして INT0ピンをプルアップしてデジタル入力とし、ピンの状態を読みます。
Low ならボタンはまだ離されておらず、High ならボタンが離されたということです。
必要に応じてポーリングするなどして、ボタンが離されたらパワーダウンして STEP1 に戻ります。ポーリングはパワーダウンと WDT を組み合わせてタイミングを作ると省電力にできると思います。
INT0割込みは Low になったことを検出できますが、ここでは High になったことを検出したいのでここでは使えません。
ボタン押下の瞬間はチャタリングなどによってボタン電圧が不安定なことがありますので、読み取りは数回連続で同じボタンが読めるまで繰り返すとよいと思います。
ボタン数の限界
ATtiny10 の ADC は 8bit なので、アナログ電圧を 0~255 の 256段階に読み分けることができます。
ではボタンもそのくらいの数まで増やせるでしょうか?
答えは No です。
抵抗 R には誤差があるため、上図のような回路でも電圧が全く均等に分圧されることは期待できないからです。
そこで、R として一般庶民にもお求めやすい ±5% くらいの精度の抵抗を念頭に置いて、以下を探ります。
この答えを求めるには、方程式をごりごり解いたり全組み合わせを手計算してみるなどの達成感の得られる方法もあるかと思いますが、Excel に計算を押し付けるという手もあります。
GND側のボタンを 0 としてボタンに番号を振ったとき、
あるボタンN(上図ではボタン3) の下側には N個(3個)の抵抗があり、上側には ボタン数 – N個(6 – 3 = 3個)の抵抗があります。
ボタンN の電圧の範囲の下限は、下側の抵抗全てが -5% で上側の抵抗全てが +5% のケースです。上限は逆に下側の抵抗全てが +5% で上側の抵抗全てが -5% のケースです。
これを式にするとこんな感じになるので、
ボタンN電圧下限 = Vcc * (N * 0.95) / (N * 0.95 + (ボタン数 – N) * 1.05)
ボタンN電圧上限 = Vcc * (N * 1.05) / (N * 1.05 + (ボタン数 – N) * 0.95)
これを Excel で表計算させてグラフにしたのがこちらです(偶然見つけた株価っていうグラフを調整したものです)。
グラフは Vcc を 1 としてます。ボタン0 は一番下の軸線上にあって見えないのですが、あります(GND電位なので、ここは動きようがありません)。横軸のボタン番号がズレてるように見えますが、見たいものは見られてるので問題ありません。
各ボタンの電圧範囲が隣のボタンのそれと重ならなければ分離して識別することが可能ということですが、21ボタンはダメで 20ボタンはギリギリ OK といったところですね。
※この計算では最悪値を求めていますが現実的にはこんな偏り方をすることは考えづらくもっとゆとりがあるはずで、あと何個か増やしても大丈夫かもしれませんね
ともかく、探った結果からこう言えます。
ちなみにボタン0 は GND 直結なのに一番上のボタンはなぜ信号線に直結しないかというと、ボタンの同時押し対策です。一番上のボタンが信号線直結だと、そのボタンとボタン0 を同時に押した場合に信号線が GND とショートします。その操作を排除できるのであれば、一番上の抵抗の上にボタンを付けることができ、最大ボタン数も 1増えることになります。
抵抗値の選び方
さて、R の値はどう決めたらよいでしょうか。ボタンの読み取り時には BTN-OUT(PBx) に High を出力して抵抗を通じて GND へ電流が流れますから、抵抗が大きい方が省電力になります。でもいくらでも大きくできるかというとそうでもなく、待機状態にボタン押下によって INT0割込みを発生させられる必要があります。
待機状態では INT0ピンは内蔵プルアップされていますので、どのボタンを押しても R によってプルアップに打ち勝ってピン電圧を Vcc の 1/3 の Lowレベルまで引き下げられることが条件となります。
ですので、このプルアップ抵抗を基準に R が決まるわけです。
ではこのプルアップ抵抗はどの程度なのかと言えば、それはデータシートに書いてあります。
最小値の 20kΩ で R群合成抵抗(の最大値)について不等式を解くとこんな感じです。
R群合成抵抗 / (R群合成抵抗 + プルアップ抵抗値) < 0.3
R群合成抵抗 < プルアップ抵抗値 * 3 / 7
R群合成抵抗 < 8.57kΩ
そして R群の合成抵抗が一番大きい時というのは真ん中のボタンを押した時です。
真ん中のボタンが押されているので、そのボタンから GNDレベルまでの抵抗の本数は抵抗の数の半数になります。そしてそれがGND直結と BTN-OUT(PBx) への接続で GNDレベルへ並列して接続されますから、抵抗値はさらに半分になります。つまり全部の抵抗を直列につないだ場合の 1/4 の抵抗値になります。
抵抗の本数はボタンの数に等しいので、
R群合成抵抗 = R * ボタンの数 / 4
先ほどの解と合わせると
R < 34.28kΩ / ボタンの数
ということになります。
入手しやすい抵抗値から選ぶとして、ボタンが 6個なら R は 5.6kΩ、20個なら 1.5kΩ あたりになります。もちろんこの値は上限値ですからそれ以下であればよく、ボタンが何個であろうと 1kΩ でいくのだ!という方針でも構わないと思います。抵抗に電流が流れるのはボタン電圧読み取り時の一瞬だけですしね。
あり余るボタンの未来
たった 6pin のマイコンに能動的な素子を追加することなく 20個ものボタンを繋げるとか、すごくないですか?こんなにボタンがあっても、ボタンの動作を書くための ROM の方が足りないなんてことになりかねない事態です。リモコン制作に端を発してますが、他にも応用の効くテクニックであることに間違いありません。GPIO 4本のうち気軽に使えるものは 3本、そのうちの 2本を 20ボタンに割いてもまだ 1本残ります。その 1本には、LED やらスピーカーやらシリアル通信やらであれこれできるポテンシャルがあります。その上気軽に使えないピンではありますがまだ 1本残っていますから、ピンチになった時に本気を出すこともできます。なにか面白いこと、やれそうですよね。
といったところで、ハードウェア面についての考察を終えたところでこの記事は一旦ここまで。
ボタン制御のソフトウェア制作と動作確認、そしてリモコン制作は別の記事にします。