290 lines
11 KiB
C++
290 lines
11 KiB
C++
/*!
|
|
* @file MIDI.h
|
|
* Project Arduino MIDI Library
|
|
* @brief MIDI Library for the Arduino
|
|
* @author Francois Best
|
|
* @date 24/02/11
|
|
* @license MIT - Copyright (c) 2015 Francois Best
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
* THE SOFTWARE.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include "midi_Defs.h"
|
|
#include "midi_Platform.h"
|
|
#include "midi_Settings.h"
|
|
#include "midi_Message.h"
|
|
|
|
#include "serialMIDI.h"
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
BEGIN_MIDI_NAMESPACE
|
|
|
|
/*! \brief The main class for MIDI handling.
|
|
It is templated over the type of serial port to provide abstraction from
|
|
the hardware interface, meaning you can use HardwareSerial, SoftwareSerial
|
|
or ak47's Uart classes. The only requirement is that the class implements
|
|
the begin, read, write and available methods.
|
|
*/
|
|
template<class Transport, class _Settings = DefaultSettings, class _Platform = DefaultPlatform>
|
|
class MidiInterface
|
|
{
|
|
public:
|
|
typedef _Settings Settings;
|
|
typedef _Platform Platform;
|
|
typedef Message<Settings::SysExMaxSize> MidiMessage;
|
|
|
|
public:
|
|
inline MidiInterface(Transport&);
|
|
inline ~MidiInterface();
|
|
|
|
public:
|
|
void begin(Channel inChannel = 1);
|
|
|
|
// -------------------------------------------------------------------------
|
|
// MIDI Output
|
|
|
|
public:
|
|
inline void sendNoteOn(DataByte inNoteNumber,
|
|
DataByte inVelocity,
|
|
Channel inChannel);
|
|
|
|
inline void sendNoteOff(DataByte inNoteNumber,
|
|
DataByte inVelocity,
|
|
Channel inChannel);
|
|
|
|
inline void sendProgramChange(DataByte inProgramNumber,
|
|
Channel inChannel);
|
|
|
|
inline void sendControlChange(DataByte inControlNumber,
|
|
DataByte inControlValue,
|
|
Channel inChannel);
|
|
|
|
inline void sendPitchBend(int inPitchValue, Channel inChannel);
|
|
inline void sendPitchBend(double inPitchValue, Channel inChannel);
|
|
|
|
inline void sendPolyPressure(DataByte inNoteNumber,
|
|
DataByte inPressure,
|
|
Channel inChannel) __attribute__ ((deprecated));
|
|
|
|
inline void sendAfterTouch(DataByte inPressure,
|
|
Channel inChannel);
|
|
inline void sendAfterTouch(DataByte inNoteNumber,
|
|
DataByte inPressure,
|
|
Channel inChannel);
|
|
|
|
inline void sendSysEx(unsigned inLength,
|
|
const byte* inArray,
|
|
bool inArrayContainsBoundaries = false);
|
|
|
|
inline void sendTimeCodeQuarterFrame(DataByte inTypeNibble,
|
|
DataByte inValuesNibble);
|
|
inline void sendTimeCodeQuarterFrame(DataByte inData);
|
|
|
|
inline void sendSongPosition(unsigned inBeats);
|
|
inline void sendSongSelect(DataByte inSongNumber);
|
|
inline void sendTuneRequest();
|
|
|
|
inline void sendRealTime(MidiType inType);
|
|
|
|
inline void beginRpn(unsigned inNumber,
|
|
Channel inChannel);
|
|
inline void sendRpnValue(unsigned inValue,
|
|
Channel inChannel);
|
|
inline void sendRpnValue(byte inMsb,
|
|
byte inLsb,
|
|
Channel inChannel);
|
|
inline void sendRpnIncrement(byte inAmount,
|
|
Channel inChannel);
|
|
inline void sendRpnDecrement(byte inAmount,
|
|
Channel inChannel);
|
|
inline void endRpn(Channel inChannel);
|
|
|
|
inline void beginNrpn(unsigned inNumber,
|
|
Channel inChannel);
|
|
inline void sendNrpnValue(unsigned inValue,
|
|
Channel inChannel);
|
|
inline void sendNrpnValue(byte inMsb,
|
|
byte inLsb,
|
|
Channel inChannel);
|
|
inline void sendNrpnIncrement(byte inAmount,
|
|
Channel inChannel);
|
|
inline void sendNrpnDecrement(byte inAmount,
|
|
Channel inChannel);
|
|
inline void endNrpn(Channel inChannel);
|
|
|
|
inline void send(const MidiMessage&);
|
|
|
|
public:
|
|
void send(MidiType inType,
|
|
DataByte inData1,
|
|
DataByte inData2,
|
|
Channel inChannel);
|
|
|
|
// -------------------------------------------------------------------------
|
|
// MIDI Input
|
|
|
|
public:
|
|
inline bool read();
|
|
inline bool read(Channel inChannel);
|
|
|
|
public:
|
|
inline MidiType getType() const;
|
|
inline Channel getChannel() const;
|
|
inline DataByte getData1() const;
|
|
inline DataByte getData2() const;
|
|
inline const byte* getSysExArray() const;
|
|
inline unsigned getSysExArrayLength() const;
|
|
inline bool check() const;
|
|
|
|
public:
|
|
inline Channel getInputChannel() const;
|
|
inline void setInputChannel(Channel inChannel);
|
|
|
|
public:
|
|
static inline MidiType getTypeFromStatusByte(byte inStatus);
|
|
static inline Channel getChannelFromStatusByte(byte inStatus);
|
|
static inline bool isChannelMessage(MidiType inType);
|
|
|
|
// -------------------------------------------------------------------------
|
|
// Input Callbacks
|
|
|
|
public:
|
|
inline void setHandleMessage(void (*fptr)(const MidiMessage&)) { mMessageCallback = fptr; };
|
|
inline void setHandleError(ErrorCallback fptr) { mErrorCallback = fptr; }
|
|
inline void setHandleNoteOff(NoteOffCallback fptr) { mNoteOffCallback = fptr; }
|
|
inline void setHandleNoteOn(NoteOnCallback fptr) { mNoteOnCallback = fptr; }
|
|
inline void setHandleAfterTouchPoly(AfterTouchPolyCallback fptr) { mAfterTouchPolyCallback = fptr; }
|
|
inline void setHandleControlChange(ControlChangeCallback fptr) { mControlChangeCallback = fptr; }
|
|
inline void setHandleProgramChange(ProgramChangeCallback fptr) { mProgramChangeCallback = fptr; }
|
|
inline void setHandleAfterTouchChannel(AfterTouchChannelCallback fptr) { mAfterTouchChannelCallback = fptr; }
|
|
inline void setHandlePitchBend(PitchBendCallback fptr) { mPitchBendCallback = fptr; }
|
|
inline void setHandleSystemExclusive(SystemExclusiveCallback fptr) { mSystemExclusiveCallback = fptr; }
|
|
inline void setHandleTimeCodeQuarterFrame(TimeCodeQuarterFrameCallback fptr) { mTimeCodeQuarterFrameCallback = fptr; }
|
|
inline void setHandleSongPosition(SongPositionCallback fptr) { mSongPositionCallback = fptr; }
|
|
inline void setHandleSongSelect(SongSelectCallback fptr) { mSongSelectCallback = fptr; }
|
|
inline void setHandleTuneRequest(TuneRequestCallback fptr) { mTuneRequestCallback = fptr; }
|
|
inline void setHandleClock(ClockCallback fptr) { mClockCallback = fptr; }
|
|
inline void setHandleStart(StartCallback fptr) { mStartCallback = fptr; }
|
|
inline void setHandleContinue(ContinueCallback fptr) { mContinueCallback = fptr; }
|
|
inline void setHandleStop(StopCallback fptr) { mStopCallback = fptr; }
|
|
inline void setHandleActiveSensing(ActiveSensingCallback fptr) { mActiveSensingCallback = fptr; }
|
|
inline void setHandleSystemReset(SystemResetCallback fptr) { mSystemResetCallback = fptr; }
|
|
|
|
inline void disconnectCallbackFromType(MidiType inType);
|
|
|
|
private:
|
|
void launchCallback();
|
|
|
|
void (*mMessageCallback)(const MidiMessage& message) = nullptr;
|
|
ErrorCallback mErrorCallback = nullptr;
|
|
NoteOffCallback mNoteOffCallback = nullptr;
|
|
NoteOnCallback mNoteOnCallback = nullptr;
|
|
AfterTouchPolyCallback mAfterTouchPolyCallback = nullptr;
|
|
ControlChangeCallback mControlChangeCallback = nullptr;
|
|
ProgramChangeCallback mProgramChangeCallback = nullptr;
|
|
AfterTouchChannelCallback mAfterTouchChannelCallback = nullptr;
|
|
PitchBendCallback mPitchBendCallback = nullptr;
|
|
SystemExclusiveCallback mSystemExclusiveCallback = nullptr;
|
|
TimeCodeQuarterFrameCallback mTimeCodeQuarterFrameCallback = nullptr;
|
|
SongPositionCallback mSongPositionCallback = nullptr;
|
|
SongSelectCallback mSongSelectCallback = nullptr;
|
|
TuneRequestCallback mTuneRequestCallback = nullptr;
|
|
ClockCallback mClockCallback = nullptr;
|
|
StartCallback mStartCallback = nullptr;
|
|
ContinueCallback mContinueCallback = nullptr;
|
|
StopCallback mStopCallback = nullptr;
|
|
ActiveSensingCallback mActiveSensingCallback = nullptr;
|
|
SystemResetCallback mSystemResetCallback = nullptr;
|
|
|
|
// -------------------------------------------------------------------------
|
|
// MIDI Soft Thru
|
|
|
|
public:
|
|
inline Thru::Mode getFilterMode() const;
|
|
inline bool getThruState() const;
|
|
|
|
inline void turnThruOn(Thru::Mode inThruFilterMode = Thru::Full);
|
|
inline void turnThruOff();
|
|
inline void setThruFilterMode(Thru::Mode inThruFilterMode);
|
|
|
|
private:
|
|
void thruFilter(byte inChannel);
|
|
|
|
private:
|
|
bool parse();
|
|
inline void handleNullVelocityNoteOnAsNoteOff();
|
|
inline bool inputFilter(Channel inChannel);
|
|
inline void resetInput();
|
|
|
|
public:
|
|
Transport* getTransport() { return &mTransport; };
|
|
|
|
private:
|
|
Transport& mTransport;
|
|
|
|
private:
|
|
Channel mInputChannel;
|
|
StatusByte mRunningStatus_RX;
|
|
StatusByte mRunningStatus_TX;
|
|
byte mPendingMessage[3];
|
|
unsigned mPendingMessageExpectedLength;
|
|
unsigned mPendingMessageIndex;
|
|
unsigned mCurrentRpnNumber;
|
|
unsigned mCurrentNrpnNumber;
|
|
bool mThruActivated : 1;
|
|
Thru::Mode mThruFilterMode : 7;
|
|
MidiMessage mMessage;
|
|
|
|
int8_t mLastError;
|
|
|
|
unsigned long mLastMessageSentTime;
|
|
unsigned long mLastMessageReceivedTime;
|
|
unsigned long mSenderActiveSensingPeriodicity;
|
|
bool mReceiverActiveSensingActivated;
|
|
|
|
private:
|
|
inline StatusByte getStatus(MidiType inType,
|
|
Channel inChannel) const;
|
|
|
|
inline void UpdateLastSentTime()
|
|
{
|
|
if (mSenderActiveSensingPeriodicity)
|
|
mLastMessageSentTime = Platform::now();
|
|
};
|
|
|
|
};
|
|
|
|
// -----------------------------------------------------------------------------
|
|
|
|
unsigned encodeSysEx(const byte* inData,
|
|
byte* outSysEx,
|
|
unsigned inLenght,
|
|
bool inFlipHeaderBits = false);
|
|
unsigned decodeSysEx(const byte* inSysEx,
|
|
byte* outData,
|
|
unsigned inLenght,
|
|
bool inFlipHeaderBits = false);
|
|
|
|
END_MIDI_NAMESPACE
|
|
|
|
#include "MIDI.hpp"
|