ATMEGA328を搭載した Arduino Duemilanove 互換機で音をPWM D/A変換出力するシンセサイザーライブラリです。
Nombre | Tamaño | Rev. | Tiempo | Autor | Log Message |
---|---|---|---|---|---|
examples | |||||
PWMDAC_Synth.h | 17.45 k | 3c6fdee | 2020-04-18 23:56:21 | Akiyoshi Kamide | Doc/keywords update, and macro refact... |
README.txt | 7.88 k | 3c6fdee | 2020-04-18 23:56:21 | Akiyoshi Kamide | Doc/keywords update, and macro refact... |
keywords.txt | 804 | 3c6fdee | 2020-04-18 23:56:21 | Akiyoshi Kamide | Doc/keywords update, and macro refact... |
[PWMDAC_Synth - PWM DAC synthesizer library for Arduino] ver.20200418 https://osdn.jp/users/kamide/pf/PWMDAC_Synth/wiki/FrontPage これは、CAmiDion http://www.yk.rim.or.jp/~kamide/music/chordhelper/hardware/ の2号機以降で実装した音源をライブラリとして独立させた、 Arduino互換機用のPWMシンセサイザライブラリです。 ●Arduinoで楽器を作ろうとして、こんな問題にぶち当たったことはありませんか? ・音を出すなら tone() が手っ取り早いけど、和音が出ない… ・analogWrite() で PWM 出力しても、パルス波の周波数が490Hzと 低すぎるから、1kHz前後の音なんて乗せられない、どうしよう… そんな問題も、AVRのプリスケーラ(分周器)の設定を変えて パルス波の周波数を最大にすれば一気に解決。 このライブラリは、規則正しい最速の周期で発生させたタイマー割り込みで、 その瞬間の波の高さに応じ、PWMのパルス幅を毎回リアルタイムに素早く更新する ことにより、矩形波だけでなく、色々な波形の音を出力します。 複数のボイスを使った多重加算によりPWMのパルス幅を計算しているので、 和音の出力も可能です。 外部音源チップなしで Arduino 本体そのものにシンセサイザの音源を 実装したいときに活用してください。 ●インストール 展開した PWMDAC_Synth フォルダを、Windows の マイドキュメント\Arduino\libraries\ に置くだけでインストールできます。 置いた後、Arduino IDE のメニュー [ファイル] → [スケッチの例] に PWMDAC_Synth が現れることを確認してください。 ●使い方 基本的な使い方を下記に示します。なお、ノートオン、ノートオフなど、音の出し方については Arduino IDE のメニューからたどれる「スケッチの例」を参考にしてください。 【サンプルコード】 // // 必要に応じ、PWMDAC_Synth.h をインクルードする前に #define で下記を指定できます。 // #define PWMDAC_OUTPUT_PIN 3 // PWM出力ピン番号(省略可:下記「●出力ピン(PWM)」参照) #define PWMDAC_POLYPHONY 6 // 同時発音数(省略可:下記「●同時発音数」を参照) #define PWMDAC_NOTE_A_FREQUENCY 440 // A音のチューニング周波数(省略可:デフォルト440Hz) #define PWMDAC_CHANNEL_PRIORITY_SUPPORT // 特定チャンネルの優先度を上げる機能を有効化(使わない場合は省略) #include <PWMDAC_Synth.h> // // 波形テーブルを、必要な分だけ定義します。 // PWMDAC_CREATE_WAVETABLE(squareWavetable, PWMDAC_SQUARE_WAVE); // 矩形波 PWMDAC_CREATE_WAVETABLE(triangleWavetable, PWMDAC_TRIANGLE_WAVE); // 三角波 PWMDAC_CREATE_WAVETABLE(sineWavetable, PWMDAC_SINE_WAVE); // 正弦波 PWMDAC_CREATE_WAVETABLE(shepardToneSineWavetable, PWMDAC_SHEPARD_TONE); // シェパードトーン(無限音階) PWMDAC_CREATE_WAVETABLE(sawtoothWavetable, PWMDAC_SAWTOOTH_WAVE); // のこぎり波 // // 波形とエンベロープパラメータ(ADSR)を Instrument 構造体に束ねて // プログラムメモリ領域に音色データを生成します。 // 内側の {} にはADSRの値を逆順で指定します(値の範囲については後述の「●音色変更」を参照)。 // PROGMEM const Instrument instrument = {sawtoothWavetable, {9, 0, 11, 4}}; // // 生成した音色を指定してPWMDAC_Synthの実体(インスタンス)を生成します。 // MIDIチャンネル1〜16がすべてこの音色に初期化されます。 // PWMDAC_CREATE_INSTANCE(&instrument); setup() { PWMDACSynth::setup(); // 初期化(必須) #if defined(PWMDAC_CHANNEL_PRIORITY_SUPPORT) // 必要に応じて、メロディパートが入っているMIDIチャンネルの優先度を高くします //(詳細については後述の「●チャンネル優先度指定」を参照)。 PWMDACSynth::getChannel(1)->setPriority(0xC0); #endif } loop() { PWMDACSynth::update(); // 状態更新(必須) // ▲ これは、減衰などのADSRエンベロープ形状の現在位置を一つ進めたり、 // モジュレーションによって変化する時間ごとの周波数を更新するための関数です。 // 呼び出し頻度は自分でカウンタを用意するなどの方法でコントロールしてください。 } ●同時発音数(ボイス数) デフォルトは6重和音です。 PWMDAC_POLYPHONY に数値を設定してコンパイルし直すことで増減可能です。 ※ 同時発音数を増やしていくと、その分 ISR() (Interrupt Service Routine: 割り込みサービスルーチン)における ループ回数が増え、多忙になってきます。あまりに多忙過ぎて loop() を実行する 暇がなくなると、動作不能に陥ります。そうなる寸前が、同時発音数の事実上の上限です。 ※ ISR() などで動作速度が要求されるため、PWMDAC_Synth.h には #pragma GCC optimize ("-O3") が指定されています。コンパイルの最適化基準を、このライブラリだけ速度優先にするためです。 特定のMIDIチャンネルに対しボイスアサインの優先度を上げることもできます。 後述の「●チャンネル優先度指定」を参照。 ●出力ピン(PWM) PWM音声出力用として指定できるピン番号は 3,9,10,11 のいずれかです。 デフォルトのピン番号は 3 です。 3,11 を選択した場合は TIMER2、9,10 を選択した場合は TIMER1 が使われます。 なお、ピン番号 5,6 も PWM 端子ですが、これを指定してもコンパイルエラーになり、 利用できません。これは TIMER0 用の ISR() が、Arduino の millis() 関数を 実装するためすでに使われていて、再定義できなかったためです。 PWM 出力ピンは、そのまま PC の LINE IN などにつないでも一応音は聞こえますが、 ローパスフィルタで可聴周波数だけを通すようにすると、より聞きやすくなります。 ●MIDIチャンネル操作 PWMDACSynth::getChannel() で MIDIチャンネル番号(1〜16) ⇔ MIDIチャンネルへのポインタ を相互変換できます。MIDIチャンネルへのポインタを介して、 チャンネル単位に持たせるパラメータを操作できます。 (操作方法については PWMDAC_Synth.h を参照) なお、ノートオン、ノートオフ、ピッチベンドなど、今出ている音 (アサインされているボイス)に即座に反映させるには、PWMDACSynth のほうで 用意しているメンバ関数を呼び出し、チャンネルとボイスを同時に更新する必要があります。 ●MIDI関数 PWMDACSynth で定義されたいくつかの static オブジェクトは、 MIDIライブラリの MIDI.setHandleXxxx() に直接指定できる形式になっています。 ただし、接続相手のMIDIデバイスによっては、NOTE OFF の代わりに velocity=0 の NOTE ON を送ってくることがあるので、その場合だけ NOTE OFF を呼び出すといった工夫が必要になります。 MIDIチャンネル番号は、MIDIライブラリに合わせて 1〜16 の範囲で 指定するように作ってあります。 ●音色変更 波形テーブルとADSRエンベロープパラメータを、それぞれ MidiChannel クラスの wavetable と envelope[] に指定することで、音色を変更できます。 また、Instrument 構造体定数をプログラムメモリ領域に作り、それを programChange() に指定するだけで、波形とエンベロープを一度に設定できます。 MIDIのプログラムチェンジを実装するときに便利です。 【波形】 wavetable 波形テーブルは、要素数256のbyte型PROGMEM配列として生成します。 それを wavetable に指定することで、波形を切り替えることができます。 波形関数のマクロを PWMDAC_CREATE_WAVETABLE() マクロに指定するか、 またはそれにならって自分で配列を定義することで、任意の波形を生成することが可能です。 ※ 波高は(255÷同時発音数:デフォルト6重和音のとき42)を超えないようにしてください。 同時発音数いっぱいに鳴らしたときに最大音量255を超えてオーバーフローし、ノイズが 発生する恐れがあります。 【ADSRエンベロープ】 envelope[] ADSR_ATTACK_VALUE - アタック時間(0〜15、大きいほどノートオン直後の立ち上がりがゆっくり) ADSR_DECAY_VALUE - ディケイ時間(0〜15、大きいほどノートオン後の減衰がゆっくり) ADSR_SUSTAIN_VALUE - サスティンレベル(0〜255、減衰が止まったあと維持する音量) ADSR_RELEASE_VALUE - リリース時間(0〜15、大きいほどノートオフ後の減衰がゆっくり) 実時間は loop() 内で update() を呼び出す頻度によって変わります。 ●チャンネル優先度指定 この機能は、特定のMIDIチャンネル(例:メロディパート)の音が他のMIDIチャンネルの音に かき消される現象が目立つ場合に、そのMIDIチャンネルに対するボイスアサインの優先度を 高くするための機能です。 この機能を使うには PWMDAC_CHANNEL_PRIORITY_SUPPORT を #define する必要があります。 #define PWMDAC_CHANNEL_PRIORITY_SUPPORT PWMDACSynth::getChannel(チャンネル番号)->setPriority(チャンネル優先度); チャンネル優先度には 0x00(0、最低:デフォルト)〜 0xFF(255、最高)を指定できます。 基本的な考え方として、ADSRエンベロープで変動した音量の最も小さいボイスから順に、 他のMIDIチャンネルに譲ることを原則としています。 そのうえで、チャンネル優先度のビット反転値(1の補数)を「音量閾値」とし、 それを超える音量のボイスを、そのチャンネルがより優先的に占有し続けるようにしています。 優先度をあまり高くしすぎると、十分小さい音量にもかかわらずボイスを占有し続けてしまい、 多重和音感が失われることがありますので、適宜調整してください。 作者:@きよし - Akiyoshi Kamide http://www.yk.rim.or.jp/~kamide/