You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
964 lines
30 KiB
964 lines
30 KiB
/*
|
|
MicroDexed
|
|
|
|
MicroDexed is a port of the Dexed sound engine
|
|
(https://github.com/asb2m10/dexed) for the Teensy-3.5/3.6/4.x with audio shield.
|
|
Dexed ist heavily based on https://github.com/google/music-synthesizer-for-android
|
|
|
|
(c)2018-2022 H. Wirtz <wirtz@parasitstudio.de>
|
|
(c)2021-2022 H. Wirtz <wirtz@parasitstudio.de>, M. Koslowski <positionhigh@gmx.de>
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software Foundation,
|
|
Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*/
|
|
|
|
#ifndef MIDI_DEVICES_H
|
|
#define MIDI_DEVICES_H
|
|
#include "config.h"
|
|
extern config_t configuration;
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
#include <USBHost_t36.h>
|
|
#endif
|
|
// override default sysex size settings
|
|
struct MicroDexedSettings : public midi::DefaultSettings
|
|
{
|
|
static const unsigned SysExMaxSize = 4104; // Accept SysEx messages up to 1024 bytes long.
|
|
};
|
|
#ifdef MIDI_DEVICE_DIN
|
|
MIDI_CREATE_CUSTOM_INSTANCE(HardwareSerial, MIDI_DEVICE_DIN, midi_serial, MicroDexedSettings);
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
USBHost usb_host;
|
|
MIDIDevice midi_usb(usb_host);
|
|
#endif
|
|
/* #ifdef MIDI_DEVICE_USB
|
|
static const unsigned sUsbTransportBufferSize = 16;
|
|
typedef midi::UsbTransport<sUsbTransportBufferSize> UsbTransport;
|
|
UsbTransport sUsbTransport;
|
|
MIDI_CREATE_INSTANCE(UsbTransport, sUsbTransport, midi_onboard_usb);
|
|
#endif */
|
|
void handleNoteOn(byte inChannel, byte inNumber, byte inVelocity, byte device);
|
|
void handleNoteOff(byte inChannel, byte inNumber, byte inVelocity, byte device);
|
|
void handleControlChange(byte inChannel, byte inData1, byte inData2);
|
|
void handleAfterTouch(byte inChannel, byte inPressure);
|
|
void handlePitchBend(byte inChannel, int inPitch);
|
|
void handleProgramChange(byte inChannel, byte inProgram);
|
|
void handleAfterTouchPoly(byte inChannel, byte inNumber, byte inVelocity);
|
|
void handleSystemExclusive(byte *data, uint len);
|
|
//void handleSystemExclusiveChunk(const byte *data, uint len, bool last);
|
|
void handleTimeCodeQuarterFrame(byte data);
|
|
void handleSongSelect(byte inSong);
|
|
void handleTuneRequest(void);
|
|
void handleClock(void);
|
|
void handleStart(void);
|
|
void handleContinue(void);
|
|
void handleStop(void);
|
|
void handleActiveSensing(void);
|
|
void handleSystemReset(void);
|
|
//void handleRealTimeSystem(void);
|
|
void MD_sendControlChange(uint8_t channel, uint8_t cc, uint8_t value);
|
|
#define MIDI_BY_DIN "MIDI_DIN"
|
|
#define MIDI_BY_USB_HOST "MIDI_USB_HOST"
|
|
#define MIDI_BY_USB "USB_MIDI"
|
|
void handle_generic(byte inChannel, byte inData1, byte inData2, const char *midi_device, midi::MidiType event)
|
|
{
|
|
char text[10];
|
|
switch(event) {
|
|
case midi::NoteOn:
|
|
handleNoteOn(inChannel, inData1, inData2, 0);
|
|
strcpy(text, "NoteOn");
|
|
break;
|
|
case midi::NoteOff:
|
|
handleNoteOff(inChannel, inData1, inData2, 0);
|
|
strcpy(text, "NoteOff");
|
|
break;
|
|
case midi::ControlChange:
|
|
handleControlChange(inChannel, inData1, inData2);
|
|
strcpy(text, "CC");
|
|
break;
|
|
case midi::AfterTouchChannel:
|
|
handleAfterTouch(inChannel, inData1);
|
|
strcpy(text, "Mono AT");
|
|
break;
|
|
case midi::PitchBend:
|
|
handlePitchBend(inChannel, inData1);
|
|
strcpy(text, "PB");
|
|
break;
|
|
case midi::ProgramChange:
|
|
handleProgramChange(inChannel, inData1);
|
|
strcpy(text, "PC");
|
|
break;
|
|
case midi::AfterTouchPoly:
|
|
handleAfterTouchPoly(inChannel, inData1, inData2);
|
|
strcpy(text, "Poly AT");
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.printf("[%s] %s", midi_device, text);
|
|
#endif
|
|
// MIDI THRU
|
|
if (configuration.sys.soft_midi_thru == 1)
|
|
{
|
|
#ifdef MIDI_DEVICE_USB
|
|
if(strcmp(MIDI_BY_USB, midi_device)) {
|
|
switch(event) {
|
|
case midi::NoteOn:
|
|
usbMIDI.sendNoteOn(inData1, inData2, inChannel);
|
|
break;
|
|
case midi::NoteOff:
|
|
usbMIDI.sendNoteOff(inData1, inData2, inChannel);
|
|
break;
|
|
case midi::ControlChange:
|
|
usbMIDI.sendControlChange(inData1, inData2, inChannel);
|
|
break;
|
|
case midi::AfterTouchChannel:
|
|
usbMIDI.sendAfterTouch(inData1, inChannel);
|
|
break;
|
|
case midi::PitchBend:
|
|
usbMIDI.sendPitchBend(inData1, inChannel);
|
|
break;
|
|
case midi::ProgramChange:
|
|
usbMIDI.sendProgramChange(inData1, inChannel);
|
|
break;
|
|
case midi::AfterTouchPoly:
|
|
usbMIDI.sendAfterTouch(inData1, inData2, inChannel);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_USB"));
|
|
#endif
|
|
}
|
|
#endif
|
|
#ifdef MIDI_DEVICE_DIN
|
|
if(strcmp(MIDI_BY_DIN, midi_device)) {
|
|
switch(event) {
|
|
case midi::NoteOn:
|
|
midi_serial.sendNoteOn(inData1, inData2, inChannel);
|
|
break;
|
|
case midi::NoteOff:
|
|
midi_serial.sendNoteOff(inData1, inData2, inChannel);
|
|
break;
|
|
case midi::ControlChange:
|
|
midi_serial.sendControlChange(inData1, inData2, inChannel);
|
|
break;
|
|
case midi::AfterTouchChannel:
|
|
midi_serial.sendAfterTouch(inData1, inChannel);
|
|
break;
|
|
case midi::PitchBend:
|
|
midi_serial.sendPitchBend(inData1, inChannel);
|
|
break;
|
|
case midi::ProgramChange:
|
|
midi_serial.sendProgramChange(inData1, inChannel);
|
|
break;
|
|
case midi::AfterTouchPoly:
|
|
midi_serial.sendAfterTouch(inData1, inData2, inChannel);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_DIN"));
|
|
#endif
|
|
}
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
if(strcmp(MIDI_BY_USB_HOST, midi_device)) {
|
|
switch(event) {
|
|
case midi::NoteOn:
|
|
midi_usb.sendNoteOn(inData1, inData2, inChannel);
|
|
break;
|
|
case midi::NoteOff:
|
|
midi_usb.sendNoteOff(inData1, inData2, inChannel);
|
|
break;
|
|
case midi::ControlChange:
|
|
midi_usb.sendControlChange(inData1, inData2, inChannel);
|
|
break;
|
|
case midi::AfterTouchChannel:
|
|
midi_usb.sendAfterTouch(inData1, inChannel);
|
|
break;
|
|
case midi::PitchBend:
|
|
midi_usb.sendPitchBend(inData1, inChannel);
|
|
break;
|
|
case midi::ProgramChange:
|
|
midi_usb.sendProgramChange(inData1, inChannel);
|
|
break;
|
|
case midi::AfterTouchPoly:
|
|
midi_usb.sendAfterTouch(inData1, inData2, inChannel);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_USB_HOST"));
|
|
#endif
|
|
}
|
|
#endif
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.println();
|
|
#endif
|
|
}
|
|
void handleSystemExclusive_generic(byte *data, uint len, const char *midi_device) {
|
|
handleSystemExclusive(data, len);
|
|
#ifdef DEBUG
|
|
Serial.printf("[%s] SysEx", midi_device);
|
|
#endif
|
|
// MIDI THRU
|
|
if (configuration.sys.soft_midi_thru == 1)
|
|
{
|
|
#ifdef MIDI_DEVICE_USB
|
|
if(strcmp(MIDI_BY_USB, midi_device)) {
|
|
usbMIDI.sendSysEx(len, data);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_USB"));
|
|
#endif
|
|
}
|
|
#endif
|
|
#ifdef MIDI_DEVICE_DIN
|
|
if(strcmp(MIDI_BY_DIN, midi_device)) {
|
|
midi_serial.sendSysEx(len, data);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_DIN"));
|
|
#endif
|
|
}
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
if(strcmp(MIDI_BY_USB_HOST, midi_device)) {
|
|
midi_usb.sendSysEx(len, data);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_USB_HOST"));
|
|
#endif
|
|
}
|
|
#endif
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.println();
|
|
#endif
|
|
}
|
|
void handleSystemCommon_generic(byte inData1, const char *midi_device, midi::MidiType event)
|
|
{
|
|
char text[10];
|
|
switch(event) {
|
|
case midi::TimeCodeQuarterFrame:
|
|
handleTimeCodeQuarterFrame(inData1);
|
|
strcpy(text, "TimeCodeQuarterFrame");
|
|
break;
|
|
case midi::SongSelect:
|
|
handleSongSelect(inData1);
|
|
strcpy(text, "SongSelect");
|
|
break;
|
|
case midi::TuneRequest:
|
|
handleTuneRequest();
|
|
strcpy(text, "TuneRequest");
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.printf("[%s] %s", midi_device, text);
|
|
#endif
|
|
// MIDI THRU
|
|
if (configuration.sys.soft_midi_thru == 1)
|
|
{
|
|
#ifdef MIDI_DEVICE_USB
|
|
if(strcmp(MIDI_BY_USB, midi_device)) {
|
|
switch(event) {
|
|
case midi::TimeCodeQuarterFrame:
|
|
usbMIDI.sendTimeCodeQuarterFrame(0xF1, inData1);
|
|
break;
|
|
case midi::SongSelect:
|
|
usbMIDI.sendSongSelect(inData1);
|
|
break;
|
|
case midi::TuneRequest:
|
|
usbMIDI.sendTuneRequest();
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_USB"));
|
|
#endif
|
|
}
|
|
#endif
|
|
#ifdef MIDI_DEVICE_DIN
|
|
if(strcmp(MIDI_BY_DIN, midi_device)) {
|
|
switch(event) {
|
|
case midi::TimeCodeQuarterFrame:
|
|
midi_serial.sendTimeCodeQuarterFrame(inData1);
|
|
break;
|
|
case midi::SongSelect:
|
|
midi_serial.sendSongSelect(inData1);
|
|
break;
|
|
case midi::TuneRequest:
|
|
midi_serial.sendTuneRequest();
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_DIN"));
|
|
#endif
|
|
}
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
if(strcmp(MIDI_BY_USB_HOST, midi_device)) {
|
|
switch(event) {
|
|
case midi::TimeCodeQuarterFrame:
|
|
midi_usb.sendTimeCodeQuarterFrame(0xF1, inData1);
|
|
break;
|
|
case midi::SongSelect:
|
|
midi_usb.sendSongSelect(inData1);
|
|
break;
|
|
case midi::TuneRequest:
|
|
midi_usb.sendTuneRequest();
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_USB_HOST"));
|
|
#endif
|
|
}
|
|
#endif
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.println();
|
|
#endif
|
|
}
|
|
void handleRealtime_generic(const char *midi_device, midi::MidiType event)
|
|
{
|
|
char text[10];
|
|
switch(event) {
|
|
case midi::Clock:
|
|
handleClock();
|
|
strcpy(text, "Clock");
|
|
break;
|
|
case midi::Start:
|
|
handleStart();
|
|
strcpy(text, "Start");
|
|
break;
|
|
case midi::Continue:
|
|
handleContinue();
|
|
strcpy(text, "Continue");
|
|
break;
|
|
case midi::Stop:
|
|
handleStop();
|
|
strcpy(text, "Stop");
|
|
break;
|
|
case midi::ActiveSensing:
|
|
handleActiveSensing();
|
|
strcpy(text, "ActiveSensing");
|
|
break;
|
|
case midi::SystemReset:
|
|
handleSystemReset();
|
|
strcpy(text, "SystemReset");
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.printf("[%s] %s", midi_device, text);
|
|
#endif
|
|
// MIDI THRU
|
|
if (configuration.sys.soft_midi_thru == 1)
|
|
{
|
|
#ifdef MIDI_DEVICE_USB
|
|
if(strcmp(MIDI_BY_USB, midi_device)) {
|
|
usbMIDI.sendRealTime(event);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_USB"));
|
|
#endif
|
|
}
|
|
#endif
|
|
#ifdef MIDI_DEVICE_DIN
|
|
if(strcmp(MIDI_BY_DIN, midi_device)) {
|
|
midi_serial.sendRealTime(event);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_DIN"));
|
|
#endif
|
|
}
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
if(strcmp(MIDI_BY_USB_HOST, midi_device)) {
|
|
midi_usb.sendRealTime(event);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_USB_HOST"));
|
|
#endif
|
|
}
|
|
#endif
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.println();
|
|
#endif
|
|
}
|
|
///* void handleSystemExclusiveChunk_MIDI_DEVICE_DIN(byte *data, uint len, bool last)
|
|
// void handlRealTimeSystem_generic(byte inRealTime, byte midi_device) {
|
|
// handleRealTimeSystem();
|
|
// #ifdef DEBUG
|
|
// switch(midi_device) {
|
|
// case MIDI_DIN:
|
|
// Serial.print(F("[MIDI_DIN] RealTimeSystem"));
|
|
// break;
|
|
// case MIDI_USB_HOST:
|
|
// Serial.print(F("[MIDI_USB_HOST] RealTimeSystem"));
|
|
// break;
|
|
// case USB_MIDI:
|
|
// Serial.print(F("[USB_MIDI] RealTimeSystem"));
|
|
// break;
|
|
// }
|
|
// #endif
|
|
// if (configuration.sys.soft_midi_thru == 1)
|
|
// {
|
|
// #ifdef MIDI_DEVICE_USB
|
|
// if(midi_device != USB_MIDI) {
|
|
// usbMIDI.sendRealTime(inRealTime);
|
|
// #ifdef DEBUG
|
|
// Serial.print(F(" THRU->MIDI_USB"));
|
|
// #endif
|
|
// }
|
|
// #endif
|
|
// #ifdef MIDI_DEVICE_DIN
|
|
// if(midi_device != MIDI_DIN) {
|
|
// midi_serial.sendRealTime((midi::MidiType)inRealTime);
|
|
// #ifdef DEBUG
|
|
// Serial.print(F(" THRU->MIDI_DIN"));
|
|
// #endif
|
|
// }
|
|
// #endif
|
|
// #ifdef MIDI_DEVICE_USB_HOST
|
|
// if(midi_device != MIDI_USB_HOST) {
|
|
// midi_usb.sendRealTime(inRealTime);
|
|
// #ifdef DEBUG
|
|
// Serial.print(F(" THRU->MIDI_USB_HOST"));
|
|
// #endif
|
|
// }
|
|
// #endif
|
|
// }
|
|
// #ifdef DEBUG
|
|
// Serial.println();
|
|
// #endif
|
|
// }
|
|
/*****************************************
|
|
MIDI_DEVICE_DIN
|
|
*****************************************/
|
|
#ifdef MIDI_DEVICE_DIN
|
|
void handleNoteOn_MIDI_DEVICE_DIN(byte inChannel, byte inNoteNumber, byte inVelocity)
|
|
{
|
|
handle_generic(inChannel, inNoteNumber, inVelocity, MIDI_BY_DIN, midi::NoteOn);
|
|
}
|
|
void handleNoteOff_MIDI_DEVICE_DIN(byte inChannel, byte inNoteNumber, byte inVelocity)
|
|
{
|
|
handle_generic(inChannel, inNoteNumber, inVelocity, MIDI_BY_DIN, midi::NoteOff);
|
|
}
|
|
void handleControlChange_MIDI_DEVICE_DIN(byte inChannel, byte inData1, byte inData2)
|
|
{
|
|
handle_generic(inChannel, inData1, inData2, MIDI_BY_DIN, midi::ControlChange);
|
|
}
|
|
void handleAfterTouch_MIDI_DEVICE_DIN(byte inChannel, byte inPressure)
|
|
{
|
|
handle_generic(inChannel, inPressure, '\0', MIDI_BY_DIN, midi::AfterTouchChannel);
|
|
}
|
|
void handlePitchBend_MIDI_DEVICE_DIN(byte inChannel, int inPitch)
|
|
{
|
|
handle_generic(inChannel, inPitch, '\0', MIDI_BY_DIN, midi::PitchBend);
|
|
}
|
|
void handleProgramChange_MIDI_DEVICE_DIN(byte inChannel, byte inProgram)
|
|
{
|
|
handle_generic(inChannel, inProgram, '\0', MIDI_BY_DIN, midi::ProgramChange);
|
|
}
|
|
void handleAfterTouchPoly_MIDI_DEVICE_DIN(byte inChannel, byte inNoteNumber, byte inVelocity)
|
|
{
|
|
handle_generic(inChannel, inNoteNumber, inVelocity, MIDI_BY_DIN, midi::AfterTouchPoly);
|
|
}
|
|
void handleSystemExclusive_MIDI_DEVICE_DIN(byte * data, uint len)
|
|
{
|
|
handleSystemExclusive_generic(data, len, MIDI_BY_DIN);
|
|
}
|
|
/* void handleSystemExclusiveChunk_MIDI_DEVICE_DIN(byte *data, uint len, bool last)
|
|
{
|
|
handleSystemExclusiveChunk(data, len, last);
|
|
#ifdef DEBUG
|
|
Serial.print(F("[MIDI_DIN] SysExChunk"));
|
|
#endif
|
|
if (configuration.sys.soft_midi_thru == 1)
|
|
{
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
midi_usb.sendSysEx(len, data, last);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_USB_HOST"));
|
|
#endif
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB
|
|
usbMIDI.sendSysEx(len, data, last);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_USB"));
|
|
#endif
|
|
#endif
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.println();
|
|
#endif
|
|
} */
|
|
void handleTimeCodeQuarterFrame_MIDI_DEVICE_DIN(byte data)
|
|
{
|
|
handleSystemCommon_generic(data, MIDI_BY_DIN, midi::TimeCodeQuarterFrame);
|
|
}
|
|
void handleSongSelect_MIDI_DEVICE_DIN(byte inSong)
|
|
{
|
|
handleSystemCommon_generic(inSong, MIDI_BY_DIN, midi::SongSelect);
|
|
}
|
|
void handleTuneRequest_MIDI_DEVICE_DIN(void)
|
|
{
|
|
handleSystemCommon_generic('\0', MIDI_BY_DIN, midi::TuneRequest);
|
|
}
|
|
void handleClock_MIDI_DEVICE_DIN(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_DIN, midi::Clock);
|
|
}
|
|
void handleStart_MIDI_DEVICE_DIN(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_DIN, midi::Start);
|
|
}
|
|
void handleContinue_MIDI_DEVICE_DIN(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_DIN, midi::Continue);
|
|
}
|
|
void handleStop_MIDI_DEVICE_DIN(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_DIN, midi::Stop);
|
|
}
|
|
void handleActiveSensing_MIDI_DEVICE_DIN(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_DIN, midi::ActiveSensing);
|
|
}
|
|
void handleSystemReset_MIDI_DEVICE_DIN(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_DIN, midi::SystemReset);
|
|
}
|
|
/* void handlRealTimeSysteme_MIDI_DEVICE_DIN(byte inRealTime)
|
|
{
|
|
handleRealTimeSystem_generic(MIDI_DIN);
|
|
} */
|
|
#endif // MIDI_DEVICE_DIN
|
|
/*****************************************
|
|
MIDI_DEVICE_USB_HOST
|
|
*****************************************/
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
void handleNoteOn_MIDI_DEVICE_USB_HOST(byte inChannel, byte inNoteNumber, byte inVelocity)
|
|
{
|
|
handle_generic(inChannel, inNoteNumber, inVelocity, MIDI_BY_USB_HOST, midi::NoteOn);
|
|
}
|
|
void handleNoteOff_MIDI_DEVICE_USB_HOST(byte inChannel, byte inNoteNumber, byte inVelocity)
|
|
{
|
|
handle_generic(inChannel, inNoteNumber, inVelocity, MIDI_BY_USB_HOST, midi::NoteOff);
|
|
}
|
|
void handleControlChange_MIDI_DEVICE_USB_HOST(byte inChannel, byte inData1, byte inData2)
|
|
{
|
|
handle_generic(inChannel, inData1, inData2, MIDI_BY_USB_HOST, midi::ControlChange);
|
|
}
|
|
void handleAfterTouch_MIDI_DEVICE_USB_HOST(byte inChannel, byte inPressure)
|
|
{
|
|
handle_generic(inChannel, inPressure, '\0', MIDI_BY_USB_HOST, midi::AfterTouchChannel);
|
|
}
|
|
void handlePitchBend_MIDI_DEVICE_USB_HOST(byte inChannel, int inPitch)
|
|
{
|
|
handle_generic(inChannel, inPitch, '\0', MIDI_BY_USB_HOST, midi::PitchBend);
|
|
}
|
|
void handleProgramChange_MIDI_DEVICE_USB_HOST(byte inChannel, byte inPitch)
|
|
{
|
|
handle_generic(inChannel, inPitch, '\0', MIDI_BY_USB_HOST, midi::ProgramChange);
|
|
}
|
|
void handleAfterTouchPoly_MIDI_DEVICE_USB_HOST(byte inChannel, byte inNoteNumber, byte inVelocity)
|
|
{
|
|
handle_generic(inChannel, inNoteNumber, inVelocity, MIDI_BY_USB_HOST, midi::AfterTouchPoly);
|
|
}
|
|
void handleSystemExclusive_MIDI_DEVICE_USB_HOST(byte * data, uint len)
|
|
{
|
|
handleSystemExclusive_generic(data, len, MIDI_BY_USB_HOST);
|
|
}
|
|
/* void handleSystemExclusiveChunk_MIDI_DEVICE_USB_HOST(byte *data, uint len, bool last)
|
|
{
|
|
handleSystemExclusiveChunk(data, len, last);
|
|
#ifdef DEBUG
|
|
Serial.print(F("[MIDI_USB_HOST] SysExChunk"));
|
|
#endif
|
|
if (configuration.sys.soft_midi_thru == 1)
|
|
{
|
|
#ifdef MIDI_DEVICE_DIN
|
|
midi_serial.sendSysEx(len, data, last);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_DIN"));
|
|
#endif
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB
|
|
usbMIDI.sendSysEx(len, data, last);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_USB"));
|
|
#endif
|
|
#endif
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.println();
|
|
#endif
|
|
} */
|
|
void handleTimeCodeQuarterFrame_MIDI_DEVICE_USB_HOST(midi::DataByte data)
|
|
{
|
|
handleSystemCommon_generic(data, MIDI_BY_USB_HOST, midi::TimeCodeQuarterFrame);
|
|
}
|
|
void handleSongSelect_MIDI_DEVICE_USB_HOST(byte inSong)
|
|
{
|
|
handleSystemCommon_generic(inSong, MIDI_BY_USB_HOST, midi::SongSelect);
|
|
}
|
|
void handleTuneRequest_MIDI_DEVICE_USB_HOST(void)
|
|
{
|
|
handleSystemCommon_generic('\0', MIDI_BY_USB_HOST, midi::TuneRequest);
|
|
}
|
|
void handleClock_MIDI_DEVICE_USB_HOST(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_USB_HOST, midi::Clock);
|
|
}
|
|
void handleStart_MIDI_DEVICE_USB_HOST(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_USB_HOST, midi::Start);
|
|
}
|
|
void handleContinue_MIDI_DEVICE_USB_HOST(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_USB_HOST, midi::Continue);
|
|
}
|
|
void handleStop_MIDI_DEVICE_USB_HOST(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_USB_HOST, midi::Stop);
|
|
}
|
|
void handleActiveSensing_MIDI_DEVICE_USB_HOST(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_USB_HOST, midi::ActiveSensing);
|
|
}
|
|
void handleSystemReset_MIDI_DEVICE_USB_HOST(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_USB_HOST, midi::SystemReset);
|
|
}
|
|
/* void handlRealTimeSystem_MIDI_DEVICE_USB_HOST(midi::MidiType inRealTime)
|
|
{
|
|
handleRealTimeSystem_generic(inRealTime, MIDI_USB_HOST);
|
|
} */
|
|
#endif // MIDI_DEVICE_USB_HOST
|
|
/*****************************************
|
|
MIDI_DEVICE_USB
|
|
*****************************************/
|
|
#ifdef MIDI_DEVICE_USB
|
|
void handleNoteOn_MIDI_DEVICE_USB(byte inChannel, byte inNoteNumber, byte inVelocity)
|
|
{
|
|
handle_generic(inChannel, inNoteNumber, inVelocity, MIDI_BY_USB, midi::NoteOn);
|
|
}
|
|
void handleNoteOff_MIDI_DEVICE_USB(byte inChannel, byte inNoteNumber, byte inVelocity)
|
|
{
|
|
handle_generic(inChannel, inNoteNumber, inVelocity, MIDI_BY_USB, midi::NoteOff);
|
|
}
|
|
void handleControlChange_MIDI_DEVICE_USB(byte inChannel, byte inData1, byte inData2)
|
|
{
|
|
handle_generic(inChannel, inData1, inData2, MIDI_BY_USB, midi::ControlChange);
|
|
}
|
|
void handleAfterTouch_MIDI_DEVICE_USB(byte inChannel, byte inPressure)
|
|
{
|
|
handle_generic(inChannel, inPressure, '\0', MIDI_BY_USB, midi::AfterTouchChannel);
|
|
}
|
|
void handlePitchBend_MIDI_DEVICE_USB(byte inChannel, int inPitch)
|
|
{
|
|
handle_generic(inChannel, inPitch, '\0', MIDI_BY_USB, midi::PitchBend);
|
|
}
|
|
void handleProgramChange_MIDI_DEVICE_USB(byte inChannel, byte inProgram)
|
|
{
|
|
handle_generic(inChannel, inProgram, '\0', MIDI_BY_USB, midi::ProgramChange);
|
|
}
|
|
void handleAfterTouchPoly_MIDI_DEVICE_USB(byte inChannel, byte inNoteNumber, byte inVelocity)
|
|
{
|
|
handle_generic(inChannel, inNoteNumber, inVelocity, MIDI_BY_USB, midi::AfterTouchPoly);
|
|
}
|
|
void handleSystemExclusive_MIDI_DEVICE_USB(byte * data, uint len)
|
|
{
|
|
handleSystemExclusive_generic(data, len, MIDI_BY_USB);
|
|
}
|
|
/* FLASHMEM void handleSystemExclusiveChunk_MIDI_DEVICE_USB(byte *data, uint len, bool last)
|
|
{
|
|
handleSystemExclusiveChunk(data, len, last);
|
|
#ifdef DEBUG
|
|
Serial.print(F("[MIDI_USB] SysExChunk"));
|
|
#endif
|
|
if (configuration.sys.soft_midi_thru == 1)
|
|
{
|
|
#ifdef MIDI_DEVICE_DIN
|
|
midi_serial.sendSysEx(len, data, last);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_DIN"));
|
|
#endif
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
midi_usb.sendSysEx(len, data, last);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" THRU->MIDI_USB_HOST"));
|
|
#endif
|
|
#endif
|
|
}
|
|
#ifdef DEBUG
|
|
Serial.println();
|
|
#endif
|
|
} */
|
|
void handleTimeCodeQuarterFrame_MIDI_DEVICE_USB(midi::DataByte data)
|
|
{
|
|
handleSystemCommon_generic(data, MIDI_BY_USB, midi::TimeCodeQuarterFrame);
|
|
}
|
|
void handleSongSelect_MIDI_DEVICE_USB(byte inSong)
|
|
{
|
|
handleSystemCommon_generic(inSong, MIDI_BY_USB, midi::SongSelect);
|
|
}
|
|
void handleTuneRequest_MIDI_DEVICE_USB(void)
|
|
{
|
|
handleSystemCommon_generic('\0', MIDI_BY_USB, midi::TuneRequest);
|
|
}
|
|
void handleClock_MIDI_DEVICE_USB(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_USB, midi::Clock);
|
|
}
|
|
void handleStart_MIDI_DEVICE_USB(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_USB, midi::Start);
|
|
}
|
|
void handleContinue_MIDI_DEVICE_USB(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_USB, midi::Continue);
|
|
}
|
|
void handleStop_MIDI_DEVICE_USB(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_USB, midi::Stop);
|
|
}
|
|
void handleActiveSensing_MIDI_DEVICE_USB(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_USB, midi::ActiveSensing);
|
|
}
|
|
void handleSystemReset_MIDI_DEVICE_USB(void)
|
|
{
|
|
handleRealtime_generic(MIDI_BY_USB, midi::SystemReset);
|
|
}
|
|
/* FLASHMEM void handleRealTimeSystem_MIDI_DEVICE_USB(byte inRealTime)
|
|
{
|
|
handleRealTimeSystem_generic(inRealTime, USB_MIDI);
|
|
} */
|
|
#endif // MIDI_DEVICE_USB
|
|
FLASHMEM void MD_sendControlChange(uint8_t channel, uint8_t cc, uint8_t value)
|
|
{
|
|
#ifdef DEBUG
|
|
Serial.print(F("[MD] SendControlChange CH:"));
|
|
Serial.print(channel, DEC);
|
|
Serial.print(F(" CC:"));
|
|
Serial.print(cc);
|
|
Serial.print(F(" VAL:"));
|
|
Serial.print(value);
|
|
#endif
|
|
#ifdef MIDI_DEVICE_DIN
|
|
midi_serial.sendControlChange(cc, value, channel);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" MIDI-DIN"));
|
|
#endif
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
midi_usb.sendControlChange(cc, value, channel);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" MIDI-USB-HOST"));
|
|
#endif
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB
|
|
usbMIDI.sendControlChange(cc, value, channel);
|
|
#ifdef DEBUG
|
|
Serial.print(F(" MIDI-USB"));
|
|
#endif
|
|
#endif
|
|
#ifdef DEBUG
|
|
Serial.println();
|
|
#endif
|
|
}
|
|
/*****************************************
|
|
HELPER FUNCTIONS
|
|
*****************************************/
|
|
FLASHMEM void setup_midi_devices(void)
|
|
{
|
|
#ifdef MIDI_DEVICE_DIN
|
|
// Start serial MIDI
|
|
midi_serial.begin(DEFAULT_MIDI_CHANNEL);
|
|
midi_serial.setHandleNoteOn(handleNoteOn_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleNoteOff(handleNoteOff_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleControlChange(handleControlChange_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleAfterTouchChannel(handleAfterTouch_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandlePitchBend(handlePitchBend_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleProgramChange(handleProgramChange_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleSystemExclusive(handleSystemExclusive_MIDI_DEVICE_DIN);
|
|
//midi_serial.setHandleSystemExclusiveChunk(handleSystemExclusiveChunk_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleTimeCodeQuarterFrame(handleTimeCodeQuarterFrame_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleAfterTouchPoly(handleAfterTouchPoly_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleSongSelect(handleSongSelect_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleTuneRequest(handleTuneRequest_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleClock(handleClock_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleStart(handleStart_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleContinue(handleContinue_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleStop(handleStop_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleActiveSensing(handleActiveSensing_MIDI_DEVICE_DIN);
|
|
midi_serial.setHandleSystemReset(handleSystemReset_MIDI_DEVICE_DIN);
|
|
//midi_serial.setHandleRealTimeSystem(handleRealTimeSystem_MIDI_DEVICE_DIN);
|
|
#ifdef DEBUG
|
|
Serial.println(F("MIDI_DEVICE_DIN enabled"));
|
|
#endif
|
|
#endif
|
|
// start up USB host
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
usb_host.begin();
|
|
midi_usb.setHandleNoteOn(handleNoteOn_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleNoteOff(handleNoteOff_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleControlChange(handleControlChange_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleAfterTouchChannel(handleAfterTouch_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandlePitchChange(handlePitchBend_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleProgramChange(handleProgramChange_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleSystemExclusive(handleSystemExclusive_MIDI_DEVICE_USB_HOST);
|
|
//midi_usb.setHandleSystemExclusiveChunk(handleSystemExclusiveChunk_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleTimeCodeQuarterFrame(handleTimeCodeQuarterFrame_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleAfterTouchPoly(handleAfterTouchPoly_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleSongSelect(handleSongSelect_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleTuneRequest(handleTuneRequest_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleClock(handleClock_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleStart(handleStart_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleContinue(handleContinue_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleStop(handleStop_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleActiveSensing(handleActiveSensing_MIDI_DEVICE_USB_HOST);
|
|
midi_usb.setHandleSystemReset(handleSystemReset_MIDI_DEVICE_USB_HOST);
|
|
//midi_usb.setHandleRealTimeSystem(handleRealTimeSystem_MIDI_DEVICE_USB_HOST);
|
|
#ifdef DEBUG
|
|
Serial.println(F("MIDI_DEVICE_USB_HOST enabled."));
|
|
#endif
|
|
#endif
|
|
// check for onboard USB-MIDI
|
|
#ifdef MIDI_DEVICE_USB
|
|
usbMIDI.begin();
|
|
usbMIDI.setHandleNoteOn(handleNoteOn_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleNoteOff(handleNoteOff_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleControlChange(handleControlChange_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleAfterTouchChannel(handleAfterTouch_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandlePitchChange(handlePitchBend_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleProgramChange(handleProgramChange_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleSystemExclusive(handleSystemExclusive_MIDI_DEVICE_USB);
|
|
//usbMIDI.setHandleSystemExclusiveChunk(handleSystemExclusiveChunk_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleTimeCodeQuarterFrame(handleTimeCodeQuarterFrame_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleAfterTouchPoly(handleAfterTouchPoly_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleSongSelect(handleSongSelect_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleTuneRequest(handleTuneRequest_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleClock(handleClock_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleStart(handleStart_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleContinue(handleContinue_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleStop(handleStop_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleActiveSensing(handleActiveSensing_MIDI_DEVICE_USB);
|
|
usbMIDI.setHandleSystemReset(handleSystemReset_MIDI_DEVICE_USB);
|
|
//usbMIDI.setHandleRealTimeSystem(handleRealTimeSystem_MIDI_DEVICE_USB);
|
|
#ifdef DEBUG
|
|
Serial.println(F("MIDI_DEVICE_USB enabled."));
|
|
#endif
|
|
#endif
|
|
}
|
|
FLASHMEM void check_midi_devices(void)
|
|
{
|
|
#ifdef MIDI_DEVICE_DIN
|
|
midi_serial.read();
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB
|
|
usbMIDI.read();
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
usb_host.Task();
|
|
midi_usb.read();
|
|
#endif
|
|
}
|
|
FLASHMEM void send_sysex_voice(uint8_t midi_channel, uint8_t* data)
|
|
{
|
|
uint8_t checksum = 0;
|
|
uint8_t vd[161];
|
|
// Send SYSEX data also via MIDI
|
|
//vd[0] = 0xF0; // SysEx start
|
|
vd[0] = 0x43; // ID=Yamaha
|
|
vd[1] = midi_channel; // Sub-status and MIDI channel
|
|
vd[2] = 0x00; // Format number (0=1 voice)
|
|
vd[3] = 0x01; // Byte count MSB
|
|
vd[4] = 0x1B; // Byte count LSB
|
|
for (uint8_t n = 0; n < 155; n++)
|
|
{
|
|
checksum -= data[n];
|
|
vd[5 + n] = data[n];
|
|
}
|
|
vd[160] = checksum & 0x7f; // Checksum
|
|
//vd[162] = 0xF7; // SysEx end
|
|
#ifdef MIDI_DEVICE_DIN
|
|
midi_serial.sendSysEx(161, vd); // Send to DIN MIDI
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB
|
|
usbMIDI.sendSysEx(161, vd); // Send to USB MIDI
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
midi_usb.sendSysEx(161, vd); // Send to USB-HOST MIDI
|
|
#endif
|
|
}
|
|
FLASHMEM void send_sysex_bank(uint8_t midi_channel, uint8_t* bank_data)
|
|
{
|
|
#ifdef MIDI_DEVICE_DIN
|
|
midi_serial.sendSysEx(4104, bank_data); // Send to DIN MIDI
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB
|
|
// Sysex bank dump is splitted due to Windows USB driver limitations
|
|
usbMIDI.sendSysEx(2048, bank_data, true); // Send to USB MIDI
|
|
delay(50);
|
|
usbMIDI.sendSysEx(2048, bank_data+2048, true);
|
|
delay(50);
|
|
usbMIDI.sendSysEx(8, bank_data+4096, true);
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
midi_usb.sendSysEx(4104, bank_data); // Send to USB-HOST MIDI
|
|
#endif
|
|
}
|
|
FLASHMEM void send_sysex_param(uint8_t midi_channel, uint8_t var, uint8_t val, uint8_t param_group)
|
|
{
|
|
uint8_t s[5];
|
|
s[0] = 0x43; // ID=Yamaha
|
|
s[1] = midi_channel; // Sub-status and MIDI channel
|
|
s[2] = (param_group & 5) << 2; // Format number (0=1 voice)
|
|
if (param_group == 0)
|
|
{
|
|
s[2] |= 1;
|
|
s[3] = var & 0x7f;
|
|
}
|
|
else
|
|
{
|
|
s[3] = var & 0x7f;
|
|
}
|
|
s[4] = val & 0x7f;
|
|
#ifdef MIDI_DEVICE_DIN
|
|
midi_serial.sendSysEx(5, s); // Send to DIN MIDI
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB
|
|
usbMIDI.sendSysEx(5, s); // Send to USB MIDI
|
|
#endif
|
|
#ifdef MIDI_DEVICE_USB_HOST
|
|
midi_usb.sendSysEx(5, s); // Send to USB-HOST MIDI
|
|
#endif
|
|
}
|
|
|
|
#endif // MIDI_DEVICES_H
|
|
|