2002.11.27- 2016.7.19 ブレッドボードを使った Arduino流 ものづくり (ATtiny4313版)[pdf] (ATmega328版はこちら・tinyBasic版はこちら) 小山智史(弘前大学教育学部) 0. 準備 1. デジタル出力(1): 光るものを作る(LEDの点滅) 2. デジタル入力: スイッチ操作できるものを作る(LEDのオンオフ) 3. アナログ出力: LEDの明るさを変える 4. デジタル出力(2): 数字を表示する 5. アナログ入力: アナログ値をデジタル表示する 6. 感じるものを作る 7. 音の出るものを作る 8. 動くものを作る 9. 応用例 (付録1) Arduinoプログラムの仕様の抜粋(ATtiny4313版) (付録2) センサーモジュール (付録3) 使用する主な部品 私達の身近にある電子機器の多くにコンピュータが内蔵されており、このようなコンピュータは「マイコン」と呼ばれま す。マイクロコンピュータあるいはマイクロコントローラの略です。マイコンにはメーカーが開発したプログラムが書き込ま れています。リモコンの中に入っているマイコンには、押したボタンに応じて決まったパターンで赤外線をオンオフする プログラムが書き込まれています。このプログラムはマイコン製造時に書き込まれ、後から書き換えることはできません が、フラッシュメモリ(電気的に書き換え可能なプログラムメモリ)を搭載したマイコンを使えば、私達もさまざまな機器を 作ることができます。 マイコンを使った機器を設計・製作するには、電子回路、論理回路、コンピュータの仕組み、プログラミングなど広範 囲の知識が要求されます。これは大変なことではありますが、一方ではこれらのことを学ぶ格好の教材であるということ を意味しています。 このテキストでは、AVRマイコン(ATtiny4313)[1]を使った実習を行いながら、「マイコンを使ったものづくり」について 学習します。また、最近注目されているArduino[2]の開発環境(Arduino IDE)を使います(Arduino IDEは使わずに行う こともできます)。例題の多くは、ArduinoIDEに用意されている「スケッチの例」をそのまま、あるいは多少アレンジして 取り上げています。 ☆ コンピュータの仕組みを学びたい方は「AVRマイコンで学ぶコンピュータの仕組み」[3]を参考にしてください。 ☆ AVRライターはArduinoISPまたはHIDaspx(ヒダピオ)を使います。「AVRライター」[4]をご覧ください。 ☆ ここでは複雑な電子回路は登場していませんが、電気回路の最低限の知識は前提としています。 ☆ 本テキストは実習時に適宜解説や補足を行うことを前提にしています。自学自習には適していないかもしれません。 0. 準備 0.1 Arduino開発環境の準備 1. Arduino 1.6.9(Windows ZIP file)をダウンロードし、マウスの右クリックで[すべて展開]します(時間がかかります)。 2. 「c:/arduino」フォルダを作成し、この中に上で展開したArduino-1.6.9フォルダの中のArduino-1.6.9フォルダを移動します。ま た、c:/arduinoの下に「src」フォルダを作成し、ここに作成したプログラムを保存することにします。 3. c:/arduino/Arduino-1.6.9/arduino.exeのショートカットを作り、デスクトップまたはスタートメニューに置きます。 4. hardware-1.6.9.zipを解凍し、展開したhardwareフォルダをArduino-1.6.9/hardwareに上書きします。これにより以下のことが行 われます。 ◦ hardware/arduino/avrフォルダのboards.txtとprogrammers.txtとplatform.txtを差し替え。 ◦ hardware/arduino/avr/variantsフォルダにattiny4313を追加。 ◦ hardware/arduino/avr/cores/arduinoフォルダにservo4313.hを追加。tone.cppとwiring_analog.cを差し替え。 ◦ hardware/arduino/avr/libraries/Wire/srcフォルダを差し替え。 ◦ hardware/arduino/avr/bootloaders/dummy/blink4313.hexを追加。 ◦ hardware/tools/avr/binフォルダにhidspx-2012-0326[4]のhidspx.exeとhidspx.iniとfuse_j.txtを追加。 ◦ hardware/tools/avr/binフォルダにcat.exeとrm.exeとtest.exeを追加(Arduino-1.0.6より) ◦ hardware/tools/avr/binフォルダにArduinoISP4313.bat, HIDapio4313.bat, HIDaspx4313.bat, arduino.batを追加。 5. ArduinoIDEを利用する場合 ◦ Arduinoを起動し、[ファイル][環境設定]で、スケッチブックの保存場所を「c:/arduino/src」にします。また、エディタの文 字の大きさを「20」程度にすると見やすいです。 ◦ AVRライタにArduinoISPを使う場合は、[ツール][ボード]で「ArduinoISP - ATtiny4313」を選択し、ライタを接続した後、 [ツール][シリアルポート]でポートを選択します。また、[ツール][書込装置]を「Arduino as ISP」にします。 ◦ AVRライタにHIDapioを使う場合は、 [ツール][ボード]で「HIDapio - ATtiny4313」を選択し、[ツール][書込装置]で 「HIDapio」を選択します。 6. ArduinoIDEを利用しない場合は、AVRライタに合わせてarduino-1.6.9/hardware/tools/avr/binフォルダのArduinoISP4313.bat またはHIDapio4313.batのショートカットを作り、名前を「Arduino」としてデスクトップにおきます。 1 0.2 ブレッドボードの使い方 ここでは下図左のブレッドボードを使います。ボードの内部は下図右のように接続されています。 (概観) (内部の接続) 電子工作の第一歩として、下図左の回路図を組み立ててみましょう。実際の配線は下図右のようになります。これを実体配線図と 言います。なお、結線はジャンパーワイヤを使って行います。配線を行う時は、必ず電池のスイッチを切るようにします。 (回路図) (実体配線図) 電池のスイッチを入れてLEDは点灯しましたか? 下の写真は実際の様子です。 (12cm×5cm 程の板を用意し、ブレッドボードと電池ケースを貼りつけています。) ここで、ブレッドボード内部の接続を念頭に、下図のような電流の流れをイメージすることが重要です。LEDは電流が流れるから 点灯するのです。 2 下図のように押しボタンスイッチを追加し、スイッチを押すとLEDが点灯するようにしてみましょう。スイッチを押すと電流が流れる わけですが、ここでも「電流が流れる経路」を(ブレッドボードの内部結線を含めて)イメージしてください。 (回路図) (実体配線図) 次のステップへの準備として、AVRマイコン(ATtiny4313)を接続しましょう。電池のスイッチは切り、AVRマイコンの向きに気を付 けて差し込んでください。 (回路図) (実体配線図) ここで、AVRマイコンにラベルを貼っておきます(テプラファイル)。 3 1. デジタル出力(1): 光るものを作る(LEDの点滅) 1.1 LEDを点灯・消灯する はじめの一歩、「Lチカ」です。以下の回路を組み立ててください。 回路図 実体配線図 以下のプログラムは、Arduino IDE の [スケッチの例][01.Basics][Blink] の正味の部分を抜粋したものです。「//」以降行末までは コメントですので、入力しなくても構いません。 Blink.ino: ArduinoIDE [スケッチの例][01.Basics][Blink] int led=13; void setup(){ pinMode(led, OUTPUT); } void loop(){ digitalWrite(led, HIGH); delay(1000); digitalWrite(led, LOW); delay(1000); } // 変数ledに13を代入する // { }内をはじめに1度だけ実行する // ledピン(13ピン)を出力にする // // // // // { }内を繰り返し実行する ledピンにHIGHを出力する 1秒(1000ms)待つ ledピンにLOWを出力する 1秒(1000ms)待つ ArduinoIDEを利用する場合は、Arduinoを起動し、下図のようにプログラムを入力し、c:/arduinoフォルダにBlinkという名前で保存 してください。そのフォルダにBlinkフォルダが作られ、中にBlink.inoというファイルが作られますので、確認してください。そして、電 池のスイッチを入れて ボタンの操作でマイコンに書き込み、LEDが点滅することを確かめてください。プログラムに誤りがあった り、うまく書き込みができない場合は、エラーメッセージが表示されます。 ArduinoIDEを利用する場合の表示画面 ArduinoIDEを利用しない場合は、下図のようにメモ帳等でc:/arduino/srcフォルダにBlink.inoを作成し、そのファイルアイコンを 「Arduino」にドラッグ&ドロップすると、プログラムが書き込まれます。 4 ArduinoIDEを利用しない場合のドラッグ&ドロップ操作 以下、プログラムの内容について簡単に説明します。 • 1行目で整数型の変数「led」に13を代入します。 • 2~4行目の setup(){ } の中は電源投入後に1度だけ実行され、ここでは「pinMode(led,OUTPUT)」でledピン(13ピン)を出力 に設定しています。 • 5~10行目のloop(){ } の中は繰り返し実行されます。 • 6行目の「digitalWrite(led,HIGH)」はledピンにHIGH(高い電圧)を出力します。 • 7行目の「delay(1000)」は1秒(1000ms)待ちます。 • 8行目の「digitalWrite(led,LOW)」はledピンにLOW(低い電圧)を出力します。 • 9行目の「delay(1000)」は1秒(1000ms)待ちます。 • 以下、6~9行目が繰り返されます。 下図のようにマイコン内部のスイッチがプログラムで切り替えられ、LEDが点滅すると考えればわかりやすいと思います。スイッチ がHIGHになるとLEDに電流が流れ、LOWになると電流が流れません。 この例のように、Arduinoのプログラムは以下のように作ります。なお、Arduinoではプログラムのことをスケッチと呼びますが、この テキストでは一般的な「プログラム」という言葉を使います。 1. プログラム全体を通して使う変数を冒頭に定義します(この例では1行目)。変数はデータ(値)を入れる入れ物のことです。中 に入れるデータによって入れ物の大きさは異なります。データの型については付録1を参照してください。 2. setup(){ }の中に、電源投入後に1度だけ実行したいプログラムを記載します(この例では2~4行目)。 3. loop(){ }の中に、繰り返して実行したいプログラムを記載します(この例では5~10行目)。 4. 各行の「//」以降はコメントとなります。また、この例にはありませんが、「/* */」で囲まれた箇所は改行も含めてコメントとなりま す。 5. HIGHやLOWやOUTPUTは「組み込み定数」で、あらかじめ値が定義されています(付録1)。 6. pinMode()やdigitalWrite()はあらかじめ用意された「組み込み関数」です(付録1)。関数を独自に定義することもできます。 (練習) LEDを0.5秒点灯、0.5秒消灯を繰り返すようにプログラムを変更しなさい。また、0.2秒点灯、1.8秒消灯を繰り返すように変更 しなさい。 1.2 LEDを高速に点滅させる LEDの点灯時間と消灯時間を変更してみます。 7行目と9行目を「delay(1000)」→「delay(100)」→「delay(20)」→「delay(10)」→「delay(1)」と変えて、点滅の様子を確認してくださ い。「delay(10)」の時は10ms点灯、10ms消灯を繰り返し、50Hzの点滅になります。「delay(20)」(25Hz)の時は点滅しているのがわか りますが、「delay(10)」(50Hz)では点滅を感じなくなります。人間の目が感じる「点滅のちらつき」はフリッカと呼ばれ、高速の点滅(50 ~60Hz)ではちらつきを感じなくなります。 通常の蛍光灯は100Hz(50Hzの2倍)で点滅しますが、蛍光管の状態によっては発光にムラが生じ50Hzで点滅し、その場合はちら つきが気になることがあります。インバーター式の蛍光灯はちらつきを抑えるために超高速(数10kHz)で点滅させています。また、 ブラウン管テレビが毎秒30フレーム(実際にはインターレースで60Hzのフリッカ)、映画のフィルムが毎秒24フレーム(実際には同じ フレームを2度映すことで48Hzのフリッカ)になっているのは、このような目の特性を考慮してのことです。 5 (練習) LEDの点滅速度を30Hz, 40Hz, 50Hz, 60Hzと変化させ、ちらつきが感じられるかどうかを、ブレッドボードを静止した場合と、 左右に振った場合の両方で調べなさい。 (練習) 緑色LEDと1kΩの抵抗をD12ピンに追加接続するよう、下の回路図に書き加えなさい。また、blink.inoを参考に(1) 実際ブ レッドボードに部品を追加し、追加した緑色LEDを1秒毎に点滅させなさい(blink1.ino)。(2) 赤色LEDと緑色LEDが1秒毎に交互に 点灯するようにしなさい(blink2.ino)。 6 2. デジタル入力: スイッチ操作できるものを作る(LEDのオンオフ) 2.1 スイッチが押されている時にLEDを点灯する スイッチと抵抗を追加し、下図の回路にします。 以下のプログラムは、buttonピンがHIGHならば(スイッチが押されていれば)LEDを点灯し、LOWならばLEDを消灯します。 8行目で、「digitalRead(button)」でbuttonピン(2ピン)がHIGH(高い電圧)かLOW(低い電圧)かを読み取り、HIGHであれば9行目が 実行され、そうでなければ11行目が実行されます。 Button.ino: ArduinoIDE [スケッチの例][02.Digital][Button] int button=2; // 変数buttonに2を代入する int led=13; // 変数ledに13を代入する void setup(){ pinMode(button, INPUT); // buttonピンを入力にする(この行はなくても良い) pinMode(led, OUTPUT); // ledピン(13ピン)を出力にする } void loop(){ if(digitalRead(button)==HIGH){//buttonピンがHIGHならば digitalWrite(led, HIGH); // ledピンにHIGHを出力する }else{ // そうでなければ digitalWrite(led, LOW); // ledピンにLOWを出力する } } 2.2 スイッチ操作で交互に点灯・消灯する 以下のプログラムは、スイッチを押すと交互にLEDが点灯・消灯します。考え方は、「スイッチが押された瞬間に(つまりさっき押さ れていなくて今押されたら)LED表示を反転する」というものです。変数buttonStateには「今のスイッチの状態(LOWまたはHIGH)」、 変数lastStateには「さっきのスイッチの状態(LOWまたはHIGH)」、変数ledStateには「今のLEDの点灯状態(LOWまたはHIGH)」を 保存します。 11行目は、現在のbuttonピンの状態(HIGHまたはLOW)を読み取り、buttonStateに代入します。12行目で、それがlastState(直前 のbuttonの状態)と違っていて、かつそれがHIGHなら、ledState(ledの状態)を反転します。13行目でledStateをledピンに出力しま す。 Debounce.ino: ArduinoIDE [スケッチの例][02.Digital][Debounce] の簡易版 int button=2; // 変数buttonに2を代入する int led=13; // 変数ledに13を代入する int ledState=HIGH; // 変数ledStateをHIGHにする(現在の点灯状態) int buttonState; // 現在のbuttonの状態 int lastState; // 直前のbuttonの状態 void setup(){ pinMode(button, INPUT); // buttonピンを入力にする pinMode(led, OUTPUT); // ledピンを出力にする } void loop(){ // 以下を繰り返す buttonState=digitalRead(button); // buttonピンを読みbuttonStateに代入 if(buttonState!=lastState && buttonState==HIGH) ledState=!ledState; // それが前の状態と違いかつHIGHならledStateを反転 digitalWrite(led, ledState);// ledピンにledStateの値を出力 lastState=buttonState; // lastStateを更新する delay(10); // チャタリングを考慮し少し待つ } 15行目の「delay(10)」がないと、誤動作することがあります。それは、スイッチを押した時に、ごく短い時間にその接点がオンに なったりオフになったりするからです。スイッチを離した時も同様です。スイッチによって異なりますが、10ms程で落ち着きます。この 現象は「チャタリング」と呼ばれます。 (練習) 15行目のdelay(10)を極端に長くし、例えば待ち時間が2秒になるようにして、どのような動作になるか調べなさい。 (練習) 緑色LEDを増設し、スイッチを押すと赤色LEDと緑色LEDが交互に点灯するようにしなさい。 7 2.3 スイッチを4回押すと点灯する 以下のプログラムは、スイッチ操作4回に一度LEDが点灯します。 上の例ではledStateを記憶し、スイッチが押されるとその値を反転させていましたが、ここではCounterにスイッチが押された回数を 記憶します。12行目で、buttonState(現在のbuttonの状態)がlastState(直前のbuttonの状態)と違っていて、かつそれがHIGHなら、 Counterの値に1を足しこみます。14行目で、「Counter%4==0」がtrueつまりCounterの値を4で割った余りが0であればledにHIGHを 出力し、そうでなければLOWを出力します。 StateChangeDetection.ino: ArduinoIDE [スケッチの例][02.Digital][StateChangeDetection] (シリアル通信に関係する行を除き、 loop() の最後に「delay(10)」を挿入) int button=2; // 変数buttonに2を代入する int led=13; // 変数ledに13を代入する int Counter=0; // 変数Counter(スイッチ操作回数)を0にする int buttonState; // 現在のbuttonの状態 int lastState=0; // 直前のbuttonの状態 void setup(){ pinMode(button, INPUT); // buttonピンを入力にする pinMode(led, OUTPUT); // ledピンを出力にする } void loop() { // 以下を繰り返す buttonState=digitalRead(button); // buttonピンを読みbuttonSteteに代入 if(buttonState!=lastState && buttonState==HIGH) Counter++; // それが前の状態と違いかつHIGHならCounterの値に+1 lastState=buttonState; // lastStateを更新する if(Counter%4==0) digitalWrite(led, HIGH); // Counterの値が4の倍数ならledピンにHIGHを出力し else digitalWrite(led, LOW); // そうでなければledピンにLOWを出力 delay(10); // チャタリングを考慮し少し待つ } (練習) 14行目の「Counter%4==0」の箇所を「Counter%7==0」や「Counter%4!=0」などに変えて、どのような動作になるか調べなさ い。 2.4 プルアップ抵抗 話を2.1のButton.inoに戻しましょう。以下にプログラムと回路図を再掲します。 Button.ino: ArduinoIDE [スケッチの例][02.Digital][Button] int button=2; // 変数buttonに2を代入する int led=13; // 変数ledに13を代入する void setup(){ pinMode(button, INPUT); // buttonピンを入力にする(この行はなくても良い) pinMode(led, OUTPUT); // ledピンを出力にする } void loop(){ if(digitalRead(button)==HIGH){//buttonピンがHIGHならば digitalWrite(led, HIGH); // ledピンにHIGHを出力する }else{ // そうでなければ digitalWrite(led, LOW); // ledピンにLOWを出力する } } このスイッチ入力の箇所をピックアップした図が下図(1)です。マイコンの入力信号はスイッチがオンの時にHIGH(高い電圧)、オ フの時にLOW(低い電圧)となります。 ここで、抵抗とスイッチを入れ替えると(2)のようになります。こうすると、マイコンの入力信号はスイッチがオンの時にLOW、オフの 時にHIGHとなります。このように接続した時の抵抗を「プルアップ抵抗」といいます。「高い電圧に引っ張り上げるための抵抗」とい う意味です。プログラムは変えていませんから、スイッチを押せば消灯、離せば点灯します。押した時に点灯させるためには以下の 箇所を書き換えます。 if(digitalRead(button)==HIGH){ // buttonピンがHIGHならば(スイッチが押されていれば) ↓ if(digitalRead(button)==LOW){ // buttonピンがLOWならば(スイッチが押されていれば) 8 こうすると、スイッチを押せば点灯し、離せば消灯します。 実は、プルアップ抵抗はマイコンの中に用意されていて、 pinMode(button, INPUT_PULLUP); とするとそれが有効となり、外部に抵抗をつなぐ必要がなくなります(以下のプログラムの4行目)。図の(3)はこの様子を示したもので す。 (1)前出の入力回路 (2)スイッチと抵抗を逆に接続 (3)マイコン内蔵の抵抗を利用 抵抗をひとつ省略できただけですが、回路は下図のようにシンプルになります。 Button.ino: ArduinoIDE [スケッチの例][02.Digital][DigitalInputPullup] int button=2; // 変数buttonに2を代入する int led=13; // 変数ledに13を代入する void setup(){ pinMode(button, INPUT_PULLUP); // buttonピンをプルアップ入力にする pinMode(led, OUTPUT); // ledピンを出力にする } void loop(){ if(digitalRead(button)==LOW){// buttonピンがLOWならば digitalWrite(led, HIGH); // ledピンにHIGHを出力する }else{ // そうでなければ digitalWrite(led, LOW); // ledピンにLOWを出力する } } (練習) 上記のようにスイッチを接続した時、2.2の「スイッチ操作で交互に点灯・消灯する」ようにDebounce.inoを修正しなさい。 (練習) 上記のようにスイッチを接続した時、2.3の「スイッチを4回押すと点灯する」ようにStateChangeDetection.inoを修正しなさい。 (練習) 2個の押しボタンスイッチを使い、一方のスイッチを押すとLEDが点灯し、もう一方のスイッチを押すと消灯するようにしなさ い。 2.5 スイッチが押されるとLEDを3秒点灯する 以下のプログラムは、スイッチが押されるとLEDを3秒点灯します。 3sec.ino: int button=2; // int led=13; // void setup(){ pinMode(button, INPUT_PULLUP);// pinMode(led, OUTPUT); // } void loop(){ if(digitalRead(button)==LOW){ // digitalWrite(led, HIGH); // delay(3000); // }else{ // digitalWrite(led, LOW); // } } 変数buttonに2を代入する 変数ledに13を代入する buttonピンをプルアップ入力にする ledピンを出力にする buttonピンがLOWならば ledピンにHIGHを出力する 3秒待つ そうでなければ ledピンにLOWを出力する 9 3. アナログ出力: LEDの明るさを変える これまでは、LEDが点灯するか消灯するかでした。しかし、点灯時の明るさを変化させたいこともあります。 3.1 LEDの明るさを変える 1.2ではLEDの点灯時間と消灯時間を変えてどのように見えるかを確認しました。ここでは7行目と9行目を「delay(1)」にしてみま す。 Blink.ino: ArduinoIDE [スケッチの例][01.Basics][Blink] (7行目と9行目を1000→1に変更) int led=13; void setup(){ pinMode(led, OUTPUT); } void loop(){ digitalWrite(led, HIGH); delay(1); digitalWrite(led, LOW); delay(1); } // 変数ledに13を代入する // { }内をはじめに1度だけ実行する // ledピン(13ピン)を出力にする // // // // // { }内を繰り返し実行する ledピンにHIGHを出力する 1ms待つ ledピンにLOWを出力する 1ms待つ 既に経験したように、点灯と消灯を高速に切り替えると、私達の目は点滅を認識できなくなります。代わりに、常時点灯している場 合よりも少しだけ暗く感じます。「平均的に」半分の明るさになるからです。下図左は、この時のledピンの出力の波形を示したもので す。8行目のLOWをHIGHにすれば常時点灯となりますから、LEDの明るさを比較してみてください。私達の目は、明るさが半分に なっても、「ちょっと暗くなった」ぐらいにしか感じません。 次に、7行目はそのままにして9行目を「delay(9)」としてみてください。こうすることにより、1ms点灯、9ms消灯を繰り返すようにな り、明るさはより暗く感じられます。この時の波形が上図中央です。 HIGHとLOWの時間が等しい時「デューティ比50%」、常時HIGHの時「デューティ比100%」、HIGHが1msでLOWが9msなら 「デューティ比10%」といいます。この技術は「パルス幅変調(PWM)」と呼ばれ、モータのスピードコントロールや、デジタルオー ディオアンプなどに利用されています。 さて、Arduinoでは analogWrite( ) という関数が用意されていて、 analogWrite(led, val) とすると、ledピンにアナログ値val(0~255)が出力されます。正確に言えば、0/255~255/255のデューティ比で上記のようなPWM信 号が出力されます。ただし、analogWrite()が利用できるピンは7, 11, 12, 13(tone()と併用する場合は7, 11のみ)に限られています。 pwm.ino: int led=13; void setup(){ analogWrite(led, 10); } void loop(){ } // 変数ledに13を代入する // { }内をはじめに1度だけ実行する // ledピンにアナログ値10を出力する // { }内は空 (練習) 3行目の「10」を0~255の間でいくつか変えて、LEDの明るさを観察しなさい。また、その時の波形をオシロスコープで観察 して記録しなさい。 (練習) ボタンを押す度に「消灯→暗い→明るい→消灯→...」の動作をするプログラムを作りなさい。 10 3.2 LEDの明るさを連続的に変える デューティ比を連続的に変化させれば、下図のように任意の波形を表すことができます。 以下は、デューティ比を0%から100%まで少しずつ変化させるプログラムです。暗い状態から次第に明るくなり、その後次第に暗 くなり、これを繰り返します。プログラムの6行目でbrightの値をアナログ出力し、7行目でbrightの値をfadeだけプラスします。 Fade.ino: ArduinoIDE [スケッチの例][01.Basics][Fade] (led 9→13に変更) int led=13; // 変数ledに13を代入する int bright=5; // 変数brightに5を代入する int fade=5; // 変数fadeに5を代入する void setup(){ } void loop(){ // 以下を繰り返す analogWrite(led, bright); // ledピンにbrightの値をアナログ出力する bright=bright+fade; // brightの値にfadeの値を足し込む if(bright==0 || bright==255) // brightの値が0または255だったら fade=-fade; // fadeの符号を逆にする delay(30); // 30ms待つ } (練習) スイッチを押すとLEDがだんだん明るくなり、スイッチを離すだんだん暗くなるプログラムを作りなさい。 11 4. デジタル出力(2): 数字を表示する 4.1 7セグメントLEDに数字を表示する 数字表示LED(7セグメントLED)に数字を表示してみます。下図の回路を組み立ててください。 以下のプログラムは、数字表示LEDに「5」を表示するものです。プログラムの冒頭でchar型の配列を定義し、LED[0]には0x3f, LED[1]には0x06, ..., LED[6]には0x67が入ります。setupの中の2行目でその値を参照し、PORTDに出力しています。 このプログラムはArduino風ではありません。Arduinoのピン番号は、マイコン本来のポート毎の番号 PB0~PB7, PD0~PD6, PA0,PA1 に通し番号を独自に割り振ったものです。この例では、LEDに接続された7本の信号線に一度に値を出力できるので、 Arduino風ではないこの方法をとりました。 Count.ino: char LED[]={ // gfedcba 0b00111111, 0b00000110, 0b01011011, 0b01001111, 0b01100110, 0b01101101, 0b01111101, 0b00100111, 0b01111111, 0b01100111 }; void setup(){ DDRD=0b01111111; PORTD=LED[5]; } void loop(){ } // 配列LEDに値を入れる // // // // // // // // // // 0x3f, 0x06, 0x5b, 0x4f, 0x66, 0x6d, 0x7d, 0x27, 0x7f, 0x67, LED[0] LED[1] LED[2] LED[3] LED[4] LED[5] LED[6] LED[7] LED[8] LED[9] // 0x7f, PD0~PD6を出力にする PD0~PD6はLEDのa~gのセグメントに接続されていて、それぞれ 1 を出力した時に点灯する回路となっています。下表は、各 数字を表示する際にPD0~PD6の各々に何を出力すればよいかと、PORTDに出力すべき2進数値と16進数値を示したものです。 数字 PD6(g) PD5(f) PD4(e) PD3(d) PD2(c) PD1(b) PD0(a) 16進 0 0 1 1 1 1 1 1 0x3F 1 0 0 0 0 1 1 0 0x06 2 1 0 1 1 0 1 1 0x5B 3 1 0 0 1 1 1 1 0x4F 4 1 1 0 0 1 1 0 0x66 5 1 1 0 1 1 0 1 0x6D 6 1 1 1 1 1 0 1 0x7D 7 0 1 0 0 1 1 1 0x27 8 1 1 1 1 1 1 1 0x7F 9 1 1 0 0 1 1 1 0x67 (練習) 表示する数字を他の数字に変更してみなさい。 12 4.2 数字表示をカウントアップする 表示される数字を1秒毎にカウントアップするには次のようにします。ポイントは配列を参照する9行目のLED[i]です。また、プログ ラム冒頭の配列LEDに値を入れる箇所は、簡潔にするために16進表記にしています。 Countup.ino: char LED[]={ // 配列LEDに値を入れる 0x3f, 0x06, 0x5b, 0x4f, 0x66, 0x6d, 0x7d, 0x27, 0x7f, 0x67 }; void setup(){ DDRD=0x7f; // PD0~PD6を出力にする } void loop(){ for(int i=0;i<10;i++){ PORTD=LED[i]; delay(1000); } } 以下は、スイッチを押すと数字表示がカウントアップするプログラムです。 ButtonCountup.ino: int button=2; // 変数buttonに2を代入する int count=0; // 変数countに0を代入する char LED[]={ // 配列LEDに値を入れる 0x3f, 0x06, 0x5b, 0x4f, 0x66, 0x6d, 0x7d, 0x27, 0x7f, 0x67 }; void setup(){ pinMode(button, INPUT_PULLUP);// buttonピンをプルアップ入力にする DDRD=0x7f; // PD0~PD6を出力にする } void loop(){ if(count<9&&digitalRead(button)==LOW){//countが9未満でbuttonがLOWなら count++; // countの値をプラス1する delay(300); // チャタリングを考慮して少し待つ } PORTD=LED[count]; } (練習) 数字表示が9~0にカウントダウンするプログラム Countdown.ino を作りなさい。 (練習) Countup.ino を、1秒毎ではなく50ms毎に高速でカウントアップするようにしなさい。また、スイッチが押されるとカウントを停 止するようにしなさい。 (練習) 1秒毎に「H」「E」「L」「L」「O」を表示し、その後3秒間消え、再びこれを繰り返すプログラムを作りなさい。また、「E」「r」「r」「o」 「r」の表示にしてみなさい。また、自分の名前を表示できるか工夫してみなさい。 13 4.3 数字の2桁表示 2桁以上の数字を表示するにはどうしたらいいでしょうか。マイコンから各LEDのa~gに別々につないでそれぞれ表示させたい値 を出力すればいいのですが(スタティック点灯)、それではマイコンのピンが不足してしまいます。このような場合はダイナミック点灯 と呼ばれる方法が用いられます。この方法は、身近にあるさまざまな表示装置に使われています。 下の回路図(上)はこれまで用いた7セグメントLEDを2個用いたもので、回路図(下)は2桁LED(SN440502)を用いたものです。2桁 LEDは、ひとつのパッケージの中に2個の7セグメントLEDが入っていて、中でa~gが接続されています。いずれの場合も、a~gに 表示したい数字のパターンを出力し、D11ピンを「0(低い電圧)」にすると「10の桁」に、D12ピンを0にすると「1の桁」に表示されま す。 いずれかの回路を組み立て、動作を確認してください。 2digit.ino: int digit1=11; // int digit2=12; // char LED[]={ // 0x3f, 0x06, 0x5b, 0x4f, 0x66, }; void setup(){ DDRD=0x7f; // pinMode(digit1, OUTPUT); // pinMode(digit2, OUTPUT); // } void loop(){ digitalWrite(digit1, HIGH);// digitalWrite(digit2, HIGH);// PORTD=LED[2]; // digitalWrite(digit1,LOW); // delay(1000); // digitalWrite(digit1,HIGH); // PORTD=LED[3]; // digitalWrite(digit2,LOW); // delay(1000); // digitalWrite(digit2,HIGH); // } 変数digit1に11を代入する 変数digit2に12を代入する 配列LEDに値を入れる 0x6d, 0x7d, 0x27, 0x7f, 0x67 PD0~PD6を出力にする digit1(11ピン)を出力にする digit2(12ピン)を出力にする 10の桁を消灯 1の桁を消灯 「2」を出力(まだ表示されない) 10の桁に表示 1秒待つ 10の桁を消灯 「3」を出力(まだ表示されない) 1の桁に表示 1秒待つ 1の桁を消灯 ここで、2箇所の「delay(1000)」を「delay(100)」→「delay(20)」→「delay(10)」→「delay(1)」と変えてみてください。1.2の時と同様にフ リッカが私たちの目にはわからなくなります。 次に、0~99の数値を表示するにはどうしたらいいでしょうか。今、以下のように表示させたい数「23」を変数Tに入れておくことにし ます。10の位は「T/10」、1の位は「T%10(Tを10で割った余り)」で求め、出力しています。他は前と同じです。 int T=23; // 表示させたい数 char LED[]={ // 配列LEDに値を入れる 0x3f, 0x06, 0x5b, 0x4f, 0x66, 0x6d, 0x7d, 0x27, 0x7f, 0x67 }; void setup(){ DDRD=0x7f; // PD0~PD6を出力にする } void loop(){ digitalWrite(11,HIGH); // 10の桁を消灯 digitalWrite(12,HIGH); // 1の桁を消灯 PORTD=LED[T/10]; // 10の位を出力(まだ表示されない) digitalWrite(11,LOW); // 10の桁に表示 delay(1); // 1ms待つ digitalWrite(11,HIGH); // 10の桁を消灯 PORTD=LED[T%10]; // 1の位を出力(まだ表示されない) digitalWrite(12,LOW); // 1の桁に表示 delay(1); // 1ms待つ digitalWrite(12,HIGH); // 1の桁を消灯 } 14 5. アナログ入力: アナログ値をデジタル表示する 5.1 アナログ値の入力 analogRead(A0)は、A0ピンの電圧(0~3V)に応じて0~1023の値をとります。ただし、ここで用いている方法では正確な電圧値を 読み取ることはできず、用途は限られます。 以下のプログラムは、ボリュームの値に応じてLEDの点滅時間が変化します。 AnalogInput.ino: ArduinoIDE [スケッチの例][03.Analog][AnalogInput] int led=13; void setup(){ pinMode(led, OUTPUT); } void loop(){ int val=analogRead(A0); digitalWrite(led, HIGH); delay(val); digitalWrite(led, LOW); delay(val); } // 変数ledに13を代入する // ledピン(13ピン)を出力にする // // // // // // 以下を繰り返す valにA0ピンのアナログ値(0~1023)を代入する ledピンにHIGHを出力する valの値だけ待つ ledピンにLOWを出力する valの値だけ待つ 5.2 アナログ値に応じてLEDを点灯・消灯する 以下のプログラムは、ボリュームのアナログ値(0~1023)が600未満の時にLEDを点灯します。 AutoLight.ino: int led=13; // void setup(){ pinMode(led, OUTPUT); // } void loop(){ // int val=analogRead(A0); // if(val<600) digitalWrite(led, else digitalWrite(led, } 変数ledに13を代入する ledピン(13ピン)を出力にする 以下を繰り返す valにA0ピンのアナログ値(0~1023)を代入する HIGH); // valが600未満ならledピンをHIGH LOW); // そうでなければLOW 5.3 アナログ値に応じて明るさを変える(調光) 以下のプログラムは、ボリュームのアナログ値(0~1023)に応じてLEDの明るさを連続的に変えます。 AnalogLight.ino: int led=13; void setup(){ pinMode(led, OUTPUT); } void loop(){ int val=analogRead(A0); analogWrite(led, val/4); } // 変数ledに13を代入する // ledピン(13ピン)を出力にする // 以下を繰り返す // valにA0ピンのアナログ値(0~1023)を代入する // ledにアナログ値(0~255)を出力 15 5.4 アナログ値に応じて色を変える フルカラーLEDは中にRGB三色のLEDが入っています。以下のプログラムは、フルカラーLEDの色相(HUE)をボリュームのアナ ログ値(0~1023)に応じて変化させます。アナログ値を一旦0~360°に変換してHに代入し、その値に応じたRGB各LEDの明るさを 0~255で求め、AnalogWrite()で出力しています。プログラム中のmap(H, 0,120, 255,0)はHの値0~120に対応する255~0の値 を返します。明るさが不足する場合は、電池を1本追加し、3V→4.5Vにしてください。 Hue.ino: int Rled=11, Gled=13, Bled=12; void setup(){ pinMode(Rled, OUTPUT); pinMode(Gled, OUTPUT); pinMode(Bled, OUTPUT); } void loop(){ int R, G, B; int H=(long)360*analogRead(A0)/1024; // H: 0-360 if(H<120){ R=map(H, 0,120, 255,0);G=255-R;B=0;} else if(H<240){ R=0; G=map(H, 120,240, 255,0); B=255-G;} else{ R=map(H, 240,360, 0,255); G=0; B=255-R;} analogWrite(Rled, R); analogWrite(Gled, G); analogWrite(Bled, B); } (練習) ボリュームのアナログ値(0~1023)に応じて、LEDの明るさを変えなさい。 5.5 アナログ値を数字で表示する 以下のプログラムは、ボリュームのアナログ値(0~1023)に応じて、0~9の数字を表示させます。 AnalogDigital.ino char LED[]={ 0x3f, 0x06, 0x5b, 0x4f, }; void setup(){ DDRD=0x7f; } void loop(){ int val=analogRead(A0); val=val/103; PORTD=LED[val]; } // 配列LEDに値を入れる 0x66, 0x6d, 0x7d, 0x27, 0x7f, 0x67 // PD0~PD6を出力にする // valにA0ピンのアナログ値(0~1023)を代入 // valの値を103で割る(0~9の値になる) // val(0~9)に応じたPD0~PD6をPORTDに出力 16 5.6 アナログ値を数字2桁で表示する 以下のプログラムは、ボリュームのアナログ値(0~1023)に応じて、0~99の数字2桁を表示させます。 AnalogDigital2.ino int digit1=11; // 変数digit1に11を代入する int digit2=12; // 変数digit2に12を代入する char LED[]={ // 配列LEDに値を入れる 0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x27,0x7f,0x67 }; void setup(){ DDRD=0x7f; // PD0~PD6を出力にする pinMode(digit1, OUTPUT);// digit1(11ピン)を出力にする pinMode(digit2, OUTPUT);// digit2(12ピン)を出力にする } void loop(){ int val=analogRead(A0); // valにA0ピンのアナログ値を代入 val=map(val, 0,1023, 0,99);//0~1023の値を0~99の値にする digitalWrite(digit1, HIGH);// 10の桁を消灯 digitalWrite(digit2, HIGH);// 1の桁を消灯 PORTD=LED[val/10]; // 10の桁を出力(まだ表示されない) digitalWrite(digit1,LOW); // 10の桁に表示 delay(1); // 1ms待つ digitalWrite(digit1,HIGH); // 10の桁を消灯 PORTD=LED[val%10]; // 1の桁を出力(まだ表示されない) digitalWrite(digit2,LOW); // 1の桁に表示 delay(1); // 1ms待つ digitalWrite(digit2,HIGH); // 1の桁を消灯 } 17 6. 感じるものを作る: センサー 6.1 スイッチをセンサーとして使う 押しボタンスイッチやスライドスイッチは、操作用のスイッチですが、センサーとしても活躍しています。例えば、CD/DVDデッキや コピー機やプリンタなど動きを伴う機器には、所定の位置に来たがどうかを検知するために、マイクロスイッチが必ずといっていい ほど使われています。このような使われ方をするスイッチを「リミットスイッチ」と呼ぶこともあります。 9.7のモーターカーでは車の先端にマイクロスイッチを使いました。ふたつの金属片を接触させるだけでスイッチ(センサー)となる ので、オリジナルのスイッチを作ることも難しくはありません。所定の水位になったかどうかを検知するピンポン球を使った浮力スイッ チ、傾けると金属ボールが移動する傾斜スイッチなど、用途に応じて工夫するのも楽しいでしょう。 6.2 明るさセンサー(CDS)を使う 5.2のボリュームの代わりに明るさセンサー(CDS)をつないでみましょう。明るくなるほどCDSの抵抗値が低くなり、A0ピンの電圧は 高くなります。一定以下に暗くなるとLEDが点灯します。プログラムは5.2と同様です。ただし、7行目の「600」の値は適切な値に変 更する必要があります。 AutoLight.ino: int led=13; // void setup(){ pinMode(led, OUTPUT); // } void loop(){ // int val=analogRead(A0); // if(val<600) digitalWrite(led, else digitalWrite(led, } 変数ledに13を代入する ledピン(13ピン)を出力にする 以下を繰り返す valにA0ピンのアナログ値(0~1023)を代入する HIGH); // valが600未満ならledピンをHIGH LOW); // そうでなければLOW 街路灯などでは、境界値付近でledが点灯したり消灯したり不安定になるのを避けるために、下図のような「ヒステリシス特性」を持 たせます。 プログラムの8行目はledが消灯(ledStateがOFF)している時に600未満の明るさになったらledを点灯(ledStateをON)し、9行目はled が点灯(ledStateがON)している時に700以上の明るさになったらledを消灯(ledStateをOFF)します。 Hysteresis.ino: int led=13; // 変数ledに13を代入する int ledState=LOW; // 変数ledStateにLOWを代入する void setup(){ pinMode(led, OUTPUT); // ledピン(13ピン)を出力にする } void loop(){ // 以下を繰り返す int val=analogRead(A0); // valにA0ピンのアナログ値(0~1023)を代入する if(ledState==LOW && val<600) ledState=HIGH; else if(ledState==HIGH && val>=700) ledState=LOW; if(ledState==LOW) digitalWrite(led, LOW); // ledStateがLOWならledピンをLOW else digitalWrite(led, HIGH);// そうでなければHIGH } (練習) 8行目の600と9行目の700の値を変えて、適切な値をみつけなさい。 18 6.3 温度センサー(サーミスタ)を使う CDSの代わりに温度センサー(サーミスタ)をつないでみましょう。温度が高くなるほどサーミスタの抵抗値が低くなり、A0ピンの電 圧は高くなります。温度がある値以上になるとLEDが点灯します。プログラムは上と同様です。ただし、7行目の「250」の値は適切な 値に変更する必要があります。 AutoLight.ino: int led=13; // void setup(){ pinMode(led, OUTPUT); // } void loop(){ // int val=analogRead(A0); // if(val<250) digitalWrite(led, else digitalWrite(led, } 変数ledに13を代入する ledピン(13ピン)を出力にする 以下を繰り返す valにA0ピンのアナログ値(0~1023)を代入する HIGH); // valが250未満ならledピンをHIGH LOW); // そうでなければLOW (練習) LEDをエアコンに置き換えて考えてみましょう。街路灯の場合と同様、境界値付近で点灯したり消灯したり不安定になるの を避けるために、下図のような「ヒステリシス特性」を持たせます。 ヒステリシス特性を持たせた Hysteresis.ino でその動作を確かめなさい。また、600と700の値を変えて、適切な値をみつけなさい。 (練習) 5.4を参考に、温度が高ければ赤、温度が低ければ青というようにLEDの色が変わるようにしてみなさい。 6.4 明るさを計測する(数字表示) 以下のプログラムは、明るさを0~9の数字で表示させます。プログラムは5.5と同様です。 AnalogDigital.ino char LED[]={ 0x3f, 0x06, 0x5b, 0x4f, }; void setup(){ DDRD=0x7f; } void loop(){ int val=analogRead(A0); val=val/103; PORTD=LED[val]; } // 配列LEDに値を入れる 0x66, 0x6d, 0x7d, 0x27, 0x7f, 0x67 // PD0~PD6を出力にする // valにA0ピンのアナログ値(0~1023)を代入 // valの値を103で割る(0~9の値になる) // val(0~9)に応じたPD0~PD6をPORTDに出力 19 6.5 温度を計測する(数字表示) 以下のプログラムは、温度を0~9の数字で表示させます。プログラムは5.5と同様です。ただし、A0ピンの電圧は温度が変化して もあまり大きく変化しません。このような場合は、9行目を以下のように変更し、220~280の値valを0~9に変換します。 val=val/30; ↓ val=map(val, 220,280, 0,9); AnalogDigital.ino char LED[]={ // 配列LEDに値を入れる 0x3f, 0x06, 0x5b, 0x4f, 0x66, 0x6d, 0x7d, 0x27, 0x7f, 0x67 }; void setup(){ DDRD=0x7f; // PD0~PD6を出力にする } void loop(){ int val=analogRead(A0); // valにA0ピンのアナログ値(0~1023)を代入 val=map(val, 220,280, 0,9); // 220~280の値を0~9の値にする PORTD=LED[val]; // val(0~9)に応じたPD0~PD6をPORTDに出力 } 6.6 音を検知する: 音量センサー 以下のプログラムは、音を検知したらLEDを3秒点灯するものです。 3sec.ino: int TH=150; int led=13; void setup(){ pinMode(led, OUTPUT); } void loop(){ if(analogRead(A0)>=TH){ digitalWrite(led, HIGH); delay(3000); }else{ digitalWrite(led, LOW); } } // 変数THに音量の境界値を代入する // 変数ledに13を代入する // ledピンを出力にする // // // // // 音量がTHよりも大きければ ledピンにHIGHを出力する 3秒待つ そうでなければ ledピンにLOWを出力する (練習) 4.2を参考に、手をたたくとカウントアップするプログラムSoundCountup.inoを作りなさい。 6.7 赤外線反射センサーを使う 赤外線の反射光で数mm程度の距離に反射物を検知するセンサーです。9.7のモーターカーと組み合わせると、ライントレースや 衝突回避などが可能になります。以下のプログラムは、数mm程度の距離に反射物を検知したらLEDを点灯するものです。ただし、 7行目の「250」の値は適切な値に変更する必要があります。 AutoLight.ino: int led=13; // void setup(){ pinMode(led, OUTPUT); // } void loop(){ // int val=analogRead(A0); // if(val<250) digitalWrite(led, else digitalWrite(led, } 変数ledに13を代入する ledピン(13ピン)を出力にする 以下を繰り返す valにA0ピンのアナログ値(0~1023)を代入する HIGH); // valが250未満ならledピンをHIGH LOW); // そうでなければLOW 20 7. 音の出るものを作る ここではブザー音で電子オルゴールを作ってみましょう。仕組みに興味のある方は「電子オルゴールの仕組み」[6]、和音など音 にこだわる方は「私だけの電子オルゴール」[7]を参考にしてください。 7.1 音を出す Blink.inoで、LEDと同じ箇所に圧電ブザー(Bzz)を接続し、「1秒ごとにLEDを点滅させるプログラム」Blink.inoを実行してくださ い。するとブザーからカチカチと音が聞こえます。ここで、点灯時間と消灯時間を以下のように変えてみましょう。変更箇所は7行目 と9行目です。 Blink.ino: ArduinoIDE [スケッチの例][01.Basics][Blink] (7行目と9行目を1000→1に変更) // 変数ledに13を代入する // { }内をはじめに1度だけ実行する // ledピン(13ピン)を出力にする int led=13; void setup(){ pinMode(led, OUTPUT); } void loop(){ digitalWrite(led, HIGH); delay(1); digitalWrite(led, LOW); delay(1); } // // // // // { }内を繰り返し実行する ledピンにHIGHを出力する 1ms待つ ledピンにLOWを出力する 1ms待つ この時の波形は下図のように500Hzの矩形波となり、聞こえるのは500Hzの音です(といっても倍音成分が含まれていますが)。 (練習) オシロスコープでトーン信号の波形を観測しなさい。また、WaveGeneなどの信号発生ソフトを用い、パソコンで500Hzの矩 形波を出し、音を比べてみなさい。マイコンの内蔵クロックはそれほど正確ではありませんので多少の音程の違いがあるかもしれま せん。 7.2 音程を変える Arduinoでは、指定した周波数の音を、指定した時間、出力するtone()という関数が利用できます。以下は時報の音を出すプログ ラムです。 BzzTone.ino: int bzz=13; void setup(){ tone(bzz, 784, 200); delay(600); tone(bzz, 784, 200); delay(600); tone(bzz, 784, 200); delay(600); tone(bzz, 1046, 1200); delay(1200); noTone(bzz); } void loop(){ } // 変数bzzに13を代入する // // // // // // // // // bzzピンに784Hzの音を200ms出力する(ポッ) 600ms待つ bzzピンに784Hzの音を200ms出力する(ポッ) 600ms待つ bzzピンに784Hzの音を200ms出力する(ポッ) 600ms待つ bzzピンに1046Hzの音を1200ms出力する(ポーン) 1200ms待つ toneを終了 (練習) 440Hzのラの音(A4)に対し1オクターブ高いラ(A5)の音は880Hzで、その間は隣同士(半音)の周波数の比が一定(21/12)とな るように音階が作られます。表計算ソフトを用い、A4~A6の各音の周波数を計算しなさい。上のプログラム中の784Hzが「ソ」、 1046Hzが「ド」であることが確認できます。 (練習) 上のプログラムを参考に、簡単な曲を演奏させるプログラムを作りなさい。 21 7.3 楽譜を演奏する 以下のプログラムは、配列として記述した楽譜情報を参照しながら演奏します。NOTE_C4などの周波数値はpitches.hファイルで 定義されています。 toneMelody.ino: ArduinoIDE [スケッチの例][02.Digital][toneMelody] (bzzPin 8→13に変更) #include "pitches.h" // pitches.hファイルを読み込む int bzz=13; // 変数bzzに13を代入する int melody[]={ // 楽譜データ(音階) NOTE_C4, NOTE_G3,NOTE_G3, NOTE_A3, NOTE_G3,0, NOTE_B3, NOTE_C4 }; int noteDurations[]={ 4,8,8,4,4,4,4,4 }; // 楽譜データ(音符) void setup(){ for(int thisNote=0; thisNote<8; thisNote++){ int noteDuration = 1000/noteDurations[thisNote]; tone(bzz, melody[thisNote],noteDuration); int pauseBetweenNotes = noteDuration*1.30; delay(pauseBetweenNotes); noTone(bzz); // toneを終了 } } void loop(){ } 楽譜はさまざまな方法で表現することができます。以下は、少し長い曲を演奏するプログラムです。 playNote.ino: const PROGMEM int16_t notes[]={ // 楽譜データ #define C 262 #define D 294 #define E 330 #define F 349 #define G 392 #define A 440 #define B 494 #define c 523 #define d 587 #define e 659 #define f 698 #define g 784 #define R 1 // Rest R,D,D,B,A,G,D,R, R,D,D,B,A,G,E,R, R,E,E,c,B,A,F,R, R,d,d,d,c,A,B,R, R,D,D,B,A,G,D,R, R,D,D,B,A,G,E,R, R,E,E,c,B,A,d,d, d,d,e,d,c,A,G,R, d,R, B,B,B,R, B,B,B,R, B,d,G,A,B,R,R,R, c,c,c,c,c,B,B,B, B,A,A,B,A,R, d,R, B,B,B,R, B,B,B,R, B,d,G,A,B,R,R,R, c,c,c,c,c,B,B,B, d,d,c,A,G,R, g,R,0 }; int Tempo=300; // 変数Tempoに300を代入する int bzz=13; // 変数bzzに13を代入する void setup(){ pinMode(bzz, OUTPUT); // bzzピンを出力にする } void loop(){ // 以下を繰り返す for(int i=0;;i++){ // 楽譜データを順に int note=pgm_read_word(¬es[i]); // 変数noteにひとつ読み if(note==0) break; // noteが0なら1曲演奏終了 else if(note!=R) tone(bzz, note, Tempo); // noteがR(休符)でもなければ // bzzピンにnote[Hz]の周波数の音をTempo[ms]出力する delay(Tempo*1.1); // Tempoだけ待つ } delay(Tempo*8); // しばらくしてから演奏を繰り返す } (練習) メロディー(notes[]の部分)を変え、必要な場合は音階を拡張し、好きな曲の電子オルゴールを作りなさい。 22 7.4 楽器を作る 以下のプログラムは、ボタンを押すと、ボリュームで決まる音程で、ドレミ...の音を出します。0~1023のアナログ値をmap関数を 使って0~11の値に変換し、notes[ ]配列の周波数を参照しています。 Instrument.ino: int bzz=13; // 変数bzzに13を代入する int button=2; // 変数buttonに2を代入する // ド レ ミ ファ ソ ラ シ ド レ ミ ファ ソ int notes[]={262, 294, 330, 349, 392, 440, 494, 523, 587, 659, 698, 784}; void setup(){ pinMode(bzz,OUTPUT); // bzzピンを出力にする pinMode(button, INPUT_PULLUP); // buttonピンをプルアップ入力にする } void loop(){ // 以下を繰り返す if(digitalRead(button)==LOW){ // buttonがLOWなら int val=analogRead(A0); // valにA0ピンのアナログ値(0~1023)を代入する val=map(val, 0,1023, 0,11); // valを0~11の値に変換する int note=notes[val]; // noteにvalに対応する周波数を代入する tone(bzz,note,200); // noteを200ms delay(200); // 200ms待つ } } (練習) ボリュームの代わりに明るさセンサーで音程が変わるようにしなさい。また、必要ならば音階を拡張しなさい。 7.5 しゃべるものを作る 音声合成LSI(ATP3011F4)を使ってスピーカから合成音声を出します。 Talk.ino: void setup(){ Serial.begin(9600); // 9600bpsで通信を開始 while(1){ Serial.write('?'); // '?'を送信する delay(300); // 300ms待つ if(Serial.available()>0&&Serial.read()=='>')break; } // '>'がきたらwhile終了 } void loop(){ // 以下を繰り返す Serial.println("ju'nbi/o'-kei");//「準備OK!」 delay(2000); // 2秒待つ } (練習) 10行目を変えて、違う言葉をしゃべらせてみなさい。 7.6 音で遊ぶ 音量センサーと楽譜の演奏を組み合わせ、「合いの手」楽器を作ってみます。 MicTone.ino: #define #define #define #define #define #define #define #define #define #define #define #define #define #define SOuL 207 RAL 220 RAuL 233 SIL 247 DO 262 DOu 277 RE 294 REu 311 MI 330 FA 349 FAu 370 SO 392 SOu 415 RA 440 23 #define RAu 466 #define SI 494 #define DOH 523 #define DOuH 554 #define REH 587 #define REuH 622 #define MIH 659 #define FAH 698 #define FAuH 740 #define SOH 784 #define R 0 // Rest #define Tempo 100 #define L1 32*Tempo #define L2_ 24*Tempo #define L2 16*Tempo #define L4_ 12*Tempo #define L4 8*Tempo #define L8_ 6*Tempo #define L8 4*Tempo #define L16 2*Tempo #define L32 1*Tempo int TH=150; // 変数THに音量の境界値を代入する(値は要調整) int bzz=13; int b(int note, int tempo){ if(note>0){ tone(bzz, note, tempo); delay(tempo*1.1); return 0; }else{ unsigned long lastmillis=millis(); int s=0; while(millis()<lastmillis+tempo) if(analogRead(A0)<TH) s=1; return s; } } void setup(){ pinMode(bzz, OUTPUT); } void loop(){ for(;;){ b(R,1000); // 1秒待つ while(analogRead(A0)>=TH); // チョ(音を待つ) b(R,L8); if(!b(R,L4)) continue; // チョイ if(b(R,L8)) continue; // の if(!b(R,L4)) continue; // チョイ if(b(R,L8)) continue; b(SIL,L16);b(RAL,L16);b(SOuL,L16);b(RAL,L32);// シラソ#ラ if(!b(R,L4)) continue; // チョン(ド) b(0,L32); b(RE,L16);b(DO,L16);b(SIL,L16);b(DO,L32); // レドシド if(!b(R,L4)) continue; // チョン(ミ) b(0,L32); b(FA,L16);b(MI,L16);b(REu,L16);b(MI,L16); // ファミレ#ミ b(SI,L16);b(RA,L16);b(SOu,L16);b(RA,L16); // シラソ#ラ b(SI,L16);b(RA,L16);b(SOu,L16);b(RA,L32); // シラソ#ラ if(!b(R,L4)) continue; // チョン(ド) b(0,L32); b(RA,L8);b(DOH,L8); // ラド b(SI,L8);b(RA,L8);b(SO,L8);b(RA,L8); // シラソラ b(SI,L8);b(RA,L8);b(SO,L8);b(RA,L8); // シラソラ b(SI,L8);b(RA,L8);b(SO,L8);b(FAu,L16); // シラソファ# if(!b(R,L4)) continue; // チョン(ミ) b(DO,L16);b(MI,L16);b(SO,L16); b(SO,L16);b(R,L16);b(MI,L16);b(SO,L4); } } (練習) 違うメロディーやリズムにしてみなさい。 24 8. 動くものを作る これまでの例では、マイコンの出力ピンにLEDや圧電ブザーを接続しました。AVRマイコンは最大20mAの電流を流すことができ るので、この範囲であれば直接接続できます。この章では、モータなど大きな電流が流れる素子を駆動する例を紹介します。大き な電流が流れる回路を制御する場合は、「電流が流れる経路」を意識することが回路の誤動作を防ぐことにつながります。 8.1 リレーでさまざまなものをオンオフする さまざまな機器をオンオフ制御したい場合は、「リレー」が最も汎用性に富んでいます。ただし、ここでは交流100Vの機器ではな く、電池で動作する機器のみを考えます。おもちゃやラジオなど、制御対象の電圧や電流や極性などを気にせずに接続することが できます。その際、「BDアダプタ」と組み合わせると、簡単に接続できるかもしれません(BDアダプタは製作も容易です)。 下図に2種類のリレーSS1A05(5V 10mA 接点容量0.5A), Y14H-1C-3DS(3V 50mA 接点容量1A)を使った回路例を示します。マ イコンに逆電圧がかからないよう、リレーと並列にダイオードを入れます。 Blink.ino: ArduinoIDE [スケッチの例][01.Basics][Blink] (7行目と9行目を1000→1に変更) int led=13; void setup(){ pinMode(led, OUTPUT); } void loop(){ digitalWrite(led, HIGH); delay(1); digitalWrite(led, LOW); delay(1); } // 変数ledに13を代入する // { }内をはじめに1度だけ実行する // ledピン(13ピン)を出力にする // // // // // { }内を繰り返し実行する ledピンにHIGHを出力する 1ms待つ ledピンにLOWを出力する 1ms待つ (練習) 上の回路を組み立て、電池で動くオモチャをつないで、Blink.inoの動作を確かめなさい。また、2.2を参考に、スイッチを押 すと交互に動作・停止するようにしてみなさい。 8.2 モータを動かす DCモータ(小型の直流モータ)は大きな電流が流れるので、マイコンに直接接続することはできず、トランジスタやFETを使って駆 動します。「ミニドラムマシン」では、バチを動かすのにDCモータを使い、2SC2120で駆動しました。赤外線リモコン自動車ではDC モータを2SK3142で駆動しました。 モータのオンオフだけでなく、正転逆転させたい場合は、Hブリッジ回路の専用IC(例えばTA7291P)を使います。ザリガニロボット や自律走行台車ではTA7257Pを使いました。IC内部での電圧降下が少なくないので注意が必要です。 Blink.ino: ArduinoIDE [スケッチの例][01.Basics][Blink] (7行目と9行目を1000→1に変更) int led=13; void setup(){ pinMode(led, OUTPUT); } void loop(){ digitalWrite(led, HIGH); delay(1); digitalWrite(led, LOW); delay(1); } // 変数ledに13を代入する // { }内をはじめに1度だけ実行する // ledピン(13ピン)を出力にする // // // // // { }内を繰り返し実行する ledピンにHIGHを出力する 1ms待つ ledピンにLOWを出力する 1ms待つ 回転数を制御したい場合は、analogWrite()を用います。以下は3.2のLEDの代わりにモーターを接続した例です。 Fade.ino: ArduinoIDE [スケッチの例][01.Basics][Fade] int led=13; // 変数ledに13を代入する 25 int bright=5; // 変数brightに5を代入する int fade=5; // 変数fadeに5を代入する void setup(){ } void loop(){ // 以下を繰り返す analogWrite(led, bright); // ledピンにbrightの値をアナログ出力する bright=bright+fade; // brightの値にfadeの値を足す if(bright==0 || bright==255)//brightの値が0または255だったら fade=-fade; // fadeの符号を逆にする delay(30); // 30ms待つ } 8.3 振動モータを動かす Midiコントローラ「ジングルベル」では、振動モータFM34F(100mA以下)を使い、「鈴」を鳴らました。2SC2120(最大800mA)や 2SC735など、少し多目に電流を流せるトランジスタを使っています。 Blink.ino: ArduinoIDE [スケッチの例][01.Basics][Blink] (7行目と9行目を1000→1に変更) int led=13; void setup(){ pinMode(led, OUTPUT); } void loop(){ digitalWrite(led, HIGH); delay(1); digitalWrite(led, LOW); delay(1); } // 変数ledに13を代入する // { }内をはじめに1度だけ実行する // ledピン(13ピン)を出力にする // // // // // { }内を繰り返し実行する ledピンにHIGHを出力する 1ms待つ ledピンにLOWを出力する 1ms待つ 8.4 サーボモータを動かす サーボモータは5Vで動作しますが、4.5Vでも支障なく動作しますので、ここではそのようにします。 Sweep4313.ino: #include <servo4313.h> int pos=0; int d=1; void setup(){ servoAttach(13); } void loop(){ servoWrite(pos); pos+=d; if(pos==179) d=-1; else if(pos==0) d=1; delay(15); } // servo4313.hを読み込む(12, 13ピンのみ対応) // 変数posに0を代入する(サーボモータの位置: 0~180) // 変数dに1を代入する(変化は1度ずつ) // 13ピンにサーボモータを接続する // // // // // // 以下を繰り返す サーボモータをposの位置にする posにdを足し込む posが179になったらdを-1にする posが0になったらdを1にする 少し待つ Knob4313.ino #include <servo4313.h> // void setup(){ servoAttach(13); // } void loop(){ // int val=analogRead(A0);// val=val*179/1023; // servoWrite(val); // delay(15); // } servo4313.hを読み込む(12, 13ピンのみ対応) 13ピンにサーボモータを接続する 以下を繰り返す valにA0ピンのアナログ値(0~1023)を代入する valの値を0~179にする サーボモータをvalの位置にする 少し待つ (練習) ボリュームの代わりに明るさセンサーを使い、明るさで位置が変わるようにしなさい。 26 9. 応用例 9.1 I2Cデバイスを使う 最近は、SDAとSCLの2本の信号線(GndとV+を加えれば4本)で接続するさまざまなI2Cデバイスが入手できます。例えば液晶ディ スプレイやモータードライバーや温湿度センサーや加速度センサーなどがあります。 ここでは、液晶ディスプレイに表示する温湿度計を紹介します。液晶ディスプレイのライブラリ(ST7032)は公開されているものを入 手し、librariesフォルダに入れておきます。 Thermometer.ino: #include <Wire.h> #include <ST7032.h> #define HDC 0x40 ST7032 lcd; void setup(){ lcd.begin(16, 2); // 16文字×2行の液晶 lcd.setContrast(45); // 0-63...10(5V), 30(3.3V), 45(3V) } void loop(){ Wire.beginTransmission(HDC); Wire.write(0x00); Wire.endTransmission(); delay(20); Wire.requestFrom(HDC, 4); uint16_t t=Wire.read(); t=(t<<8)|Wire.read(); // 温度を取得 uint16_t h=Wire.read(); h=(h<<8)|Wire.read(); // 湿度を取得 t=(((t>>8)*165)>>8)-40; // 温度の値を変換 h=(((h>>8)*100)>>8); // 湿度の値を変換 lcd.clear(); // 表示クリヤー lcd.setCursor(0,0); // 位置を指定し lcd.print(t); lcd.write(0xdf);lcd.write('C'); // 温度を表示 lcd.setCursor(8,0); // 位置を指定し lcd.print(h); lcd.print("%RH"); // 湿度を表示 delay(1000); // 1秒待つ } 9.2 赤外線リモコンを作る テレビなどのリモコンは、ボタンを押すと赤外線LEDが「決められたタイミングで点滅」するように作られています。どのように点滅し ているかは、リモコン受光器に向けてリモコンのボタンを押し、受光器の信号をオシロスコープで観測するとわかります。 実際に観測してみると、各社テレビのリモコン信号は以下のようになっています。図の着色した部分が信号が「オン」の箇所で、 「オン」とは「赤外線LEDを13μs点灯し、13μs消灯し、これを一定時間繰り返す」ことです。また、「オフ」とは「一定時間消灯する」こと です。これを参考に、赤外線LEDを点滅させるプログラムを作ればよいわけです。 27 電源 ON/OFF 40 チャンネルUP 40 チャンネルDOWN40 ボリュームUP 40 ボリュームDOWN 40 bf bf bf bf bf 12 1b 1f 1a 1e ed e4 e0 e5 e1 東芝製テレビのリモコン信号 電源 ON/OFF 02 チャンネルUP 02 チャンネルDOWN02 ボリュームUP 02 ボリュームDOWN 02 20 20 20 20 20 80 80 80 80 80 00 00 00 00 00 3d 34 35 20 21 bd b4 b5 a0 a1 パナソニック製テレビのリモコン信号 電源 ON/OFF 8f チャンネルUP 8f チャンネルDOWN8f ボリュームUP 8f ボリュームDOWN 8f 12 12 12 12 12 16 11 12 14 15 d1 a1 91 f1 e1 シャープ製テレビのリモコン信号 電源 ON/OFF 95 チャンネルUP 90 チャンネルDOWN91 ボリュームUP 92 ボリュームDOWN 93 0 0 0 0 0 SONY製テレビのリモコン信号 以下の回路とプログラムは、Upボタンを押すと各社リモコンの「チャンネルアップ」信号が次々出力され、Downボタンを押すと各 社リモコンの「チャンネルダウン」信号が次々出力されます。実際には各社の信号を出す必要はありません。 Remocon.ino: int led=13, UpButton=1, DownButton=5; void f38kHz(int n){ // 26μs × n周期 volatile char i; int j; for(j=0;j<n;j++){ digitalWrite(led, HIGH); i=0;i++; // 13μs点灯 digitalWrite(led, LOW); i=0; // 13μs消灯 } } void tx1(){f38kHz(300/26); delayMicroseconds(1800);}// 「1」の赤外線信号 void tx0(){f38kHz(300/26); delayMicroseconds(700);} // 「0」の赤外線信号 void sx1(){delayMicroseconds(600);f38kHz(1200/26); }// 「1」の赤外線信号(SONY) void sx0(){delayMicroseconds(600);f38kHz(600/26); } // 「0」の赤外線信号(SONY) void txbyte(uint8_t data){ // 1バイトのdataの赤外線信号 if(data&0x01) tx1(); else tx0(); // ビット0 if(data&0x02) tx1(); else tx0(); // ビット1 28 if(data&0x04) if(data&0x08) if(data&0x10) if(data&0x20) if(data&0x40) if(data&0x80) tx1(); tx1(); tx1(); tx1(); tx1(); tx1(); else else else else else else } void sxbyte(uint8_t data){ if(data&0x01) sx1(); else if(data&0x02) sx1(); else if(data&0x04) sx1(); else if(data&0x08) sx1(); else if(data&0x10) sx1(); else if(data&0x20) sx1(); else if(data&0x40) sx1(); else if(data&0x80) sx1(); else } tx0(); tx0(); tx0(); tx0(); tx0(); tx0(); // // // // // // ビット2 ビット3 ビット4 ビット5 ビット6 ビット7 sx0(); sx0(); sx0(); sx0(); sx0(); sx0(); sx0(); sx0(); // // // // // // // // // 1バイトのdataの赤外線信号(SONY) ビット0 ビット1 ビット2 ビット3 ビット4 ビット5 ビット6 ビット7 void setup(){ pinMode(led, OUTPUT); pinMode(UpButton, INPUT_PULLUP); pinMode(DownButton, INPUT_PULLUP); } void loop(){ if(digitalRead(UpButton)==LOW){ // UpButtonが押されると f38kHz(9000/26); delayMicroseconds(4500); // 東芝 CH UP txbyte(0x40); txbyte(0xbf); txbyte(0x1b); txbyte(0xe4); tx0(); delay(100); // ******************************* 少し時間をおいて f38kHz(3600/26); delayMicroseconds(1600); // パナソニック CH UP txbyte(0x02); txbyte(0x20); txbyte(0x80); txbyte(0x00); txbyte(0x34); txbyte(0xb4); tx0(); delay(100); // ******************************* 少し時間をおいて f38kHz(3600/26); delayMicroseconds(1600); // パナソニックBD CH UP txbyte(0x02); txbyte(0x20); txbyte(0xb0); txbyte(0x00); txbyte(0x34); txbyte(0x84); tx0(); delay(100); // ******************************* 少し時間をおいて txbyte(0x8f); txbyte(0x12); delay(42); // シャープ CH UP txbyte(0x11); txbyte(0xa1); tx0(); delay(100); // ******************************* 少し時間をおいて f38kHz(2500/26);sxbyte(0x90);sx0();sx0();sx0();sx0();delay(27);//SONY CH UP f38kHz(2500/26); sxbyte(0x90); sx0();sx0();sx0();sx0(); delay(27); f38kHz(2500/26); sxbyte(0x90); sx0();sx0();sx0();sx0(); delay(1000); }else if(digitalRead(DownButton)==LOW){ // DownButtonが押されると f38kHz(9000/26); delayMicroseconds(4500); // 東芝 CH DOWN txbyte(0x40); txbyte(0xbf); txbyte(0x1f); txbyte(0xe0); tx0(); delay(100); // ******************************* 少し時間をおいて f38kHz(3600/26); delayMicroseconds(1600); // パナソニック CH DOWN txbyte(0x02); txbyte(0x20); txbyte(0x80); txbyte(0x00); txbyte(0x35); txbyte(0xb5); tx0(); delay(100); // ******************************* 少し時間をおいて f38kHz(3600/26); delayMicroseconds(1600); // パナソニックBD CH DOWN txbyte(0x02); txbyte(0x20); txbyte(0xb0); txbyte(0x00); txbyte(0x35); txbyte(0x85); tx0(); delay(100); // ******************************* 少し時間をおいて txbyte(0x8f); txbyte(0x12); delay(42); // シャープTV CH DOWN txbyte(0x12); txbyte(0x91); tx0(); delay(100); // ******************************* 少し時間をおいて f38kHz(2500/26);sxbyte(0x91);sx0();sx0();sx0();sx0();delay(27);//SONY CH DN f38kHz(2500/26); sxbyte(0x91); sx0();sx0();sx0();sx0();delay(27); f38kHz(2500/26); sxbyte(0x91); sx0();sx0();sx0();sx0(); delay(1000); } } 29 9.3 押しボタン信号機を作る 押しボタン信号機の制御プログラムです。 Signal.ino: int buttonPin=2; int RPin=16, YPin=15, GPin=14, RRPin=10, GGPin=9, buttonMemo; void carsig(int r, int y, int g){ digitalWrite(RPin,r); digitalWrite(YPin,y); digitalWrite(GPin,g); } void mansig(int r,int g){digitalWrite(RRPin,r); digitalWrite(GGPin,g);} void setup(){ pinMode(RPin, OUTPUT); pinMode(YPin, OUTPUT); pinMode(GPin, OUTPUT); pinMode(RRPin,OUTPUT); pinMode(GGPin,OUTPUT); pinMode(buttonPin,INPUT_PULLUP); carsig(LOW,LOW,HIGH); mansig(HIGH,LOW); // 車青・歩赤 } void loop(){ int button=digitalRead(buttonPin); if(button!=buttonMemo && button==LOW){ delay(4000); carsig(LOW,HIGH,LOW); delay(4000); // 車黄4秒 carsig(HIGH,LOW,LOW); mansig(LOW,HIGH); // 車赤・歩青16秒 delay(16000); for(int i=0;i<8;i++){ // 歩青点滅8秒 mansig(LOW,HIGH); delay(500); mansig(LOW,LOW); delay(500); } mansig(HIGH,LOW); delay(4000); // 歩赤4秒 carsig(LOW,LOW,HIGH); // 車青 } buttonMemo=button; } 以下は歩行者用の「カッコー」の音を付け加えた例です。 Signal.ino: int bzz=11; // 変数bzzに11を代入する int button=2; // 変数buttonに2を代入する int RPin=16; // 変数RPinに16を代入する int YPin=15; // 変数YPinに15を代入する int GPin=14; // 変数GPinに14を代入する int RRPin=10; // 変数RRPinに10を代入する int GGPin=9; // 変数GGPinに9を代入する int lastState=0; // 直前のbuttonの状態 void carsig(int r, int y, int g){ // 関数定義 digitalWrite(RPin, r); // RPinピンにrを出力する digitalWrite(YPin, y); // YPinピンにyを出力する digitalWrite(GPin, g); // GPinピンにgを出力する } void mansig(int r, int g){ // 関数定義 digitalWrite(RRPin, r); // RRPinピンにrを出力する digitalWrite(GGPin, g); // GGPinピンにgを出力する } void setup(){ pinMode(bzz, OUTPUT); pinMode(RPin, OUTPUT); pinMode(YPin, OUTPUT); pinMode(GPin, OUTPUT); pinMode(RRPin,OUTPUT); pinMode(GGPin,OUTPUT); pinMode(button,INPUT_PULLUP); carsig(LOW, LOW, HIGH); mansig(HIGH,LOW); // 車青・歩赤 } void loop(){ int i, Tempo=200, button=digitalRead(buttonPin), e=659, c=523; 30 if(button!=buttonMemo && button==LOW){ delay(4000); carsig(LOW,HIGH,LOW); delay(4000); // 車黄4秒 carsig(HIGH,LOW,LOW); mansig(LOW,HIGH); // 車赤・歩青 for(i=0;i<5;i++){ // カッコー16秒 tone(bzzPin, e, Tempo/2); delay(Tempo*2.5); tone(bzzPin, c, Tempo); delay(Tempo*5.5); tone(bzzPin, e, Tempo/2); delay(Tempo); tone(bzzPin, e, Tempo/2); delay(Tempo); tone(bzzPin, c, Tempo); delay(Tempo*6); } for(i=0;i<8;i++){ // 歩青点滅8秒 mansig(LOW,HIGH); delay(500); mansig(LOW,LOW); delay(500); } mansig(HIGH,LOW); delay(4000); // 歩赤4秒 carsig(LOW,LOW,HIGH); // 車青 } buttonMemo=button; } また、以下はこれを発展させた本格的な「交通安全教室用信号機」です。高輝度LED(12V 60mA)を駆動するために、専用のドラ イバIC TD62083(最大500mA)を使っています。 9.4 Midi楽器を作って演奏させる こちらの例では、Midi楽器を制御して演奏させています。振動モーターで鈴をシャンシャン鳴らしながら「ジングルベル」を演奏し ます。 31 9.5 ドラムの演奏 こちらの例では、ドラムを演奏しながらブザー音のメロディーが流れます。ドラムの位置にバチの位置決めにサーボモータ (PICO/STD/F)、バチを打ち下ろすのにDCモータを使いました。DCモータは大きな電流が流れるので、2SC2120(最大800mA)を 使いました。 9.6 鉄琴を演奏させる こちらの例では、鉄琴を演奏しながらブザー音のメロディーが流れます。バチ(マレット)の位置決めにサーボモータ (PICO/STD/F)、バチを打ち下ろすのにDCモータを使いました。DCモータは大きな電流が流れるので、2SC2120(最大800mA)を 使いました。 32 9.7 モーターカーを動かす DCモータを2個使ったモーターカーです。車の先端のリミットスイッチがオンになると、少し後退し、右回転して、再び前進します。 DCモータの駆動には専用のHブリッジIC(TA7291P)を使いました。 Motorcar.ino: int motorR1=14, motorR2=13; // 右モーター 00:停止 01:後退 10:前進 int motorL1=16, motorL2=15; // 左モーター 00:停止 01:後退 10:前進 int sw=8; void lfwd(){ digitalWrite(motorL1, HIGH); digitalWrite(motorL2, LOW);} void lbwd(){ digitalWrite(motorL1, LOW); digitalWrite(motorL2, HIGH);} void lstop(){digitalWrite(motorL1, LOW); digitalWrite(motorL2, LOW);} void rfwd(){ digitalWrite(motorR1, HIGH); digitalWrite(motorR2, LOW);} void rbwd(){ digitalWrite(motorR1, LOW); digitalWrite(motorR2, HIGH);} void rstop(){digitalWrite(motorR1, LOW); digitalWrite(motorR2, LOW);} void Stop(){ lstop(); rstop(); delay(1000);} // 停止 void Fwd(){ Stop(); lfwd(); rfwd();} // 前進 void Bwd(){ Stop(); lbwd(); rbwd();} // 後退 void turnRight(){ Stop(); lfwd(); delay(200); Stop();} // 右旋回 void turnLeft(){ Stop(); rfwd(); delay(200); Stop();} // 左旋回 void setup(){ pinMode(motorL1, OUTPUT); pinMode(motorL2, OUTPUT); pinMode(motorR1, OUTPUT); pinMode(motorR2, OUTPUT); pinMOde(sw, INPUT_PULLUP); delay(1000); } void loop(){ fwd(); // 前進し while(digitalRead(sw)==HIGH); // ぶつかったら Bwd(); delay(300); // 少し後退し Right(); // 右旋回 } 以下はヒダピオ制御カー(Topman)のcpuをATtiny4313に交換し、上と同様の動作をさせたものです。 hcc.ino: int motorR1=4, motorR2=5; int motorL1=6, motorL2=7; int swR=10, swL=11; void void void void void void void void void void void // 右モーター 01:後退 10:前進 11:停止 // 左モーター 01:後退 10:前進 11:停止 // 左右のタッチスイッチ 0:オン 1:オフ lfwd(){ digitalWrite(motorL1, HIGH); digitalWrite(motorL2, LOW);} lbwd(){ digitalWrite(motorL1, LOW); digitalWrite(motorL2, HIGH);} lstop(){digitalWrite(motorL1, HIGH); digitalWrite(motorL2, HIGH);} rfwd(){ digitalWrite(motorR1, HIGH); digitalWrite(motorR2, LOW); } rbwd(){ digitalWrite(motorR1, LOW); digitalWrite(motorR2, HIGH);} rstop(){digitalWrite(motorR1, HIGH); digitalWrite(motorR2, HIGH);} Stop(){ lstop(); rstop();} // 停止 Fwd(){ Stop(); lfwd(); rfwd();} // 前進 Bwd(){ Stop(); lbwd(); rbwd();} // 後退 turnRight(){Stop(); lfwd(); delay(300); Stop();} // 右旋回 turnLeft(){ Stop(); rfwd(); delay(300); Stop();} // 左旋回 void setup(){ pinMode(motorL1, OUTPUT); pinMode(motorL2, OUTPUT); 33 pinMode(motorR1, OUTPUT); pinMode(motorR2, OUTPUT); pinMode(swR, INPUT_PULLUP); pinMode(swL, INPUT_PULLUP); Stop(); } void loop(){ Stop();delay(500); Fwd(); // 前進し while(digitalRead(swR)==HIGH && digitalRead(swL)==HIGH);// ぶつかったら Stop();delay(500); Bwd();delay(500); // 少し後退し Stop();delay(500); turnRight(); // 右旋回 } ヒダピオ制御カーのピン(参考) 機能 ピン番号 pinMode 値 DIPスイッチ1 0 INPUT_PULLUP 0:オン 1:オフ DIPスイッチ2 1 INPUT_PULLUP 0:オン 1:オフ ブザー 2,3 OUTPUT 右モーター 4,5 OUTPUT 00:- 01:後退 10:前進 11:停止 左モーター 6,7 OUTPUT 00:- 01:後退 10:前進 11:停止 LED 8 OUTPUT 0:消灯 1:点灯 CDS 9 INPUT 0:暗い 1:明るい 右タッチスイッチ 10 INPUT_PULLUP 0:オン 1:オフ 左タッチスイッチ 11 INPUT_PULLUP 0:オン 1:オフ リモコン STARTボタン 12 INPUT_PULLUP 0:オン 1:オフ リモコン LEFTボタン 13 INPUT_PULLUP 0:オン 1:オフ リモコン RIGHTボタン 14 INPUT_PULLUP 0:オン 1:オフ リモコン BACKボタン 15 INPUT_PULLUP 0:オン 1:オフ リモコン FORWARDボタン 16 INPUT_PULLUP 0:オン 1:オフ 下の写真は下山大氏が考案した「ザリガニロボット」で、メカ部分が単純で簡単に製作できるのが特長です。ブレッドボードを使っ た製作例ではありませんので悪しからず。走り方を工夫したり、センサーと組み合わせると面白いでしょう。 (参考資料) [1] ATtiny4313 データシート, http://www.atmel.com/Images/doc8246.pdf [2] Arduino, http://www.arduino.cc [3] AVRマイコンで学ぶコンピュータの仕組み, http://siva.cc.hirosaki-u.ac.jp/usr/koyama/avr/avrtext.html [4] AVRライター, http://siva.cc.hirosaki-u.ac.jp/usr/koyama/avr/avrwriter.html [5] グラフで考える直流回路, http://siva.cc.hirosaki-u.ac.jp/usr/koyama/lecture/it/elec.html [6] 電子オルゴールの仕組み, http://siva.cc.hirosaki-u.ac.jp/usr/koyama/avr/hoso3.html [7] 私だけの電子オルゴール, http://siva.cc.hirosaki-u.ac.jp/usr/koyama/mymelo/ 34 (付録1) Arduinoプログラムの仕様の抜粋(ATtiny4313版) (1) 入出力に利用できるピン(ATtiny4313の場合) 種類 ピン番号 デジタル出力ピン 0~16 デジタル入力ピン アナログ出力ピン* 7, 11, 12, 13 アナログ入力ピン A0 * ブログラム中でtone()を使っている場合は 7, 11を使う (2) データの型 型 サイズ 値 char 1バイト -128~127 int8_t unsigned char uint8_t 1バイト 0~255 byte int 2バイト -32768~32767 int16_t unsigned int 2バイト 0~65535 uint16_t long 4バイト -21474836480~2147483647 int32_t unsigned long 4バイト 0~4294967295 uint32_t float 4バイト 3.4028235E+38~-3.4028235E+38 (3) 組み込み定数 定数名 内容 HIGH, LOW デジタル出力するまたはデジタル入力されたピンの状態 OUTPUT, INPUT, INPUT_PULLUP pinMode()で設定するデジタルピンの入出力 A0 アナログpin(ATtiny4313ではA0だけが利用できる) DDRB, PORTB, PB0~PB7 ATtiny4313固有のポートおよびビット指定 DDRD, PORTD, PD0~PD7 ATtiny4313固有のポートおよびビット指定 (4) 組み込み関数 関数名 内容 pinMode(pin, OUTPUT) pinをOUTPUTにする(OUTPUT, INPUT, INPUT_PULLUP) digitalRead(pin) pinの値を読んでHIGHまたはLOWの値を返す digitalWrite(pin, HIGH) pinにHIGHを出力する(値はHIGHまたはLOW) analogRead(A0) A0ピンのアナログ値を読み、0~1023の値を返す analogWrite(pin, value) pin(7,11,12,13のいずれか)にvalue(0~255)を出力する map(val, fromL,fromH, toL,toH) fromL~fromHの値val(整数)をtoL~toHの値に変換する delay(t) t[ms]待つ delayMicroseconds(t) t[μs]待つ tone(pin, freq, t) pinにfreq[Hz]の矩形波信号をt[ms]出力する noTone() tone()を終了する servoAttach(pin) pinをサーボ制御ピンにする(pinは12, 13のいずれか) servoWrite(pos) サーボの角度をpos(0~180)にする Serial.begin(speed) speed[bps]でシリアル通信を開始する Serial.available() 受信した文字数を返す Serial.read() 受信した1文字を返す Serial.write(c) cを送信する Serial.print(s) 文字列sを送信する Serial.println(s) 文字列sと改行を送信する 35 (付録2) センサーモジュール ブレッドボードで利用できるセンサーモジュールの製作例です。 モジュール 回路図 可変抵抗モジュール 明るさセンサーモジュール 赤外線反射モジュール 音量センサーモジュール 36 製作例 (付録3) 使用する主な部品 名称 部品表 概観 備考 ブレッドボード EIC-301 秋月 190円 ブレッドボード ジャンパーワイヤ 15cm 秋月 10本 300円 ブレッドボード ジャンパーワイヤ EIC-J-S 秋月 250円 LED 秋月 10個 120円 フルカラーLED OSTA5131A 秋月 50円 赤外線LED 秋月 10個 100円 数字表示LED C-551SR 秋月 40円 圧電スピーカー 秋月 2個 100円 プッシュスイッチ DS-660R-C 千石 84円 抵抗 1/4W 100Ω, 220Ω, 1kΩ, 10kΩ 秋月 100本 100円 フィルムコンデンサ 0.01μF 秋月 10円 CDS(光センサー) 秋月 30円 サーミスタ(温度センサー) 秋月 50円 フォトリフレクタ(赤外線反射センサー) 秋月 50円 AVRマイコン(ATtiny4313) 妙楽堂 200円 電池ケース(単3×2本 スイッチ付) 秋月 60円 サーボモーター SG-90 秋月 400円 振動モーター 秋月 2個 100円 2SC2120 秋月 20個 200円 リレー SS1A05 秋月 5個 280円 リレー Y14H-1C-3DS 秋月 80円 ベニア板(5.5cm×12cm) ブレッドボードと電池ケースを貼り付けます。 [email protected] 37
© Copyright 2024 ExpyDoc