/* ScummVM - Graphic Adventure Engine * * ScummVM is the legal property of its developers, whose names * are too numerous to list here. Please refer to the COPYRIGHT * file distributed with this source distribution. * * 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 2 * 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 AUDIO_MILES_MIDIDRIVER_H #define AUDIO_MILES_MIDIDRIVER_H #include "audio/mididrv.h" #include "audio/mt32gm.h" #include "common/error.h" #include "common/mutex.h" #include "common/queue.h" #include "common/stream.h" namespace Audio { /** * @defgroup audio_miles Miles and XMIDI * @ingroup audio * * @brief API for managing XMIDI files used by Miles Sound System. * @{ */ // Miles Audio supported controllers for control change messages #define MILES_CONTROLLER_SELECT_PATCH_BANK 114 #define MILES_CONTROLLER_PROTECT_VOICE 112 #define MILES_CONTROLLER_PROTECT_TIMBRE 113 #define MILES_CONTROLLER_LOCK_CHANNEL 110 #define MILES_CONTROLLER_PROTECT_CHANNEL 111 #define MILES_CONTROLLER_PITCH_RANGE 6 #define MILES_CONTROLLER_PATCH_REVERB 59 #define MILES_CONTROLLER_PATCH_BENDER 60 #define MILES_CONTROLLER_REVERB_MODE 61 #define MILES_CONTROLLER_REVERB_TIME 62 #define MILES_CONTROLLER_REVERB_LEVEL 63 #define MILES_CONTROLLER_RHYTHM_KEY_TIMBRE 58 // 3 SysEx controllers, each range 5 // 32-36 for 1st queue // 37-41 for 2nd queue // 42-46 for 3rd queue #define MILES_CONTROLLER_SYSEX_RANGE_BEGIN 32 #define MILES_CONTROLLER_SYSEX_RANGE_END 46 #define MILES_CONTROLLER_SYSEX_QUEUE_COUNT 3 #define MILES_CONTROLLER_SYSEX_QUEUE_SIZE 32 #define MILES_CONTROLLER_SYSEX_COMMAND_ADDRESS1 0 #define MILES_CONTROLLER_SYSEX_COMMAND_ADDRESS2 1 #define MILES_CONTROLLER_SYSEX_COMMAND_ADDRESS3 2 #define MILES_CONTROLLER_SYSEX_COMMAND_DATA 3 #define MILES_CONTROLLER_SYSEX_COMMAND_FINAL_DATA 4 #define MILES_CONTROLLER_XMIDI_RANGE_BEGIN 110 #define MILES_CONTROLLER_XMIDI_RANGE_END 120 #define MILES_MT32_PATCHES_COUNT 128 #define MILES_MT32_CUSTOMTIMBRE_COUNT 64 #define MILES_MT32_PATCHDATA_COMMONPARAMETER_SIZE 14 #define MILES_MT32_PATCHDATA_PARTIALPARAMETER_SIZE 58 #define MILES_MT32_PATCHDATA_PARTIALPARAMETERS_COUNT 4 #define MILES_MT32_PATCHDATA_TOTAL_SIZE (MILES_MT32_PATCHDATA_COMMONPARAMETER_SIZE + (MILES_MT32_PATCHDATA_PARTIALPARAMETER_SIZE * MILES_MT32_PATCHDATA_PARTIALPARAMETERS_COUNT)) // Some engines using Miles assume a source neutral // volume of 256, so use this by default. #define MILES_DEFAULT_SOURCE_NEUTRAL_VOLUME 256 struct MilesMT32InstrumentEntry { byte bankId; byte patchId; byte commonParameter[MILES_MT32_PATCHDATA_COMMONPARAMETER_SIZE]; byte partialParameters[MILES_MT32_PATCHDATA_PARTIALPARAMETERS_COUNT][MILES_MT32_PATCHDATA_PARTIALPARAMETER_SIZE]; }; /** * Abstract class containing the interface for loading * the XMIDI timbres specified in the timbre chunks of * an XMIDI file. */ class MidiDriver_Miles_Xmidi_Timbres { public: virtual ~MidiDriver_Miles_Xmidi_Timbres() { } /** * Processes the timbre chunk specified for a track * in an XMIDI file. This will load the necessary * timbres into the MIDI device using SysEx messages. * * This function will likely return before all SysEx * messages have been sent. Use the isReady method to * check if the driver has finished preparing for * playback. Playback should not be started before * this process has finished. */ virtual void processXMIDITimbreChunk(const byte *timbreListPtr, uint32 timbreListSize) = 0; }; class MidiDriver_Miles_Midi : public MidiDriver_MT32GM, public MidiDriver_Miles_Xmidi_Timbres { public: MidiDriver_Miles_Midi(MusicType midiType, MilesMT32InstrumentEntry *instrumentTablePtr, uint16 instrumentTableCount); ~MidiDriver_Miles_Midi(); public: using MidiDriver_MT32GM::send; void send(int8 source, uint32 b) override; /** * De-initialize a source. Call this after playing a track or sound effect using this source. * This will unlock and unprotect channels used by this source and stop any active notes * from this source. * Automatically executed when an End Of Track meta event is received. */ void deinitSource(uint8 source) override; /** * Set the volume for this source. This will be used to scale the volume values in the MIDI * data from this source. Expected volume values are 0 - 256. * Note that source volume remains set for the source number even after deinitializing the * source. If the same source numbers are consistently used for music and SFX sources, the * source volume will only need to be set once. */ void setSourceVolume(uint8 source, uint16 volume) override; void stopAllNotes(bool stopSustainedNotes = false) override; void processXMIDITimbreChunk(const byte *timbreListPtr, uint32 timbreListSize) override; protected: void initControlData() override; void initMidiDevice() override; private: void writeRhythmSetup(byte note, byte customTimbreId); void writePatchTimbre(byte patchId, byte timbreGroup, byte timbreId, bool useSysExQueue = false); void writePatchByte(byte patchId, byte index, byte patchValue); void writeToSystemArea(byte index, byte value); const MilesMT32InstrumentEntry *searchCustomInstrument(byte patchBank, byte patchId); int16 searchCustomTimbre(byte patchBank, byte patchId); void setupPatch(byte patchBank, byte patchId, bool useSysExQueue = false); int16 installCustomTimbre(byte patchBank, byte patchId); private: /** * This stores the values of the MIDI controllers for * a MIDI channel. It is used to keep track of controller * values while a channel is locked, so they can be * restored when the channel is unlocked. */ struct MilesMidiChannelControlData : MidiChannelControlData { // Custom timbre data byte currentPatchBank; bool usingCustomTimbre; byte currentCustomTimbreId; MilesMidiChannelControlData() : currentPatchBank(0), usingCustomTimbre(false), currentCustomTimbreId(0) { } }; struct MidiChannelEntry { // True if this channel is locked. A locked channel will // only accept MIDI messages from the source that locked it. bool locked; // The channel in the MIDI data of the lock source that // is assigned to this locked output channel. This is a // reverse lookup for MidiSource::channelMap. // -1 if the channel is not locked. int8 lockDataChannel; // True if this channel is protected from locking. // The channel can still be locked, but unprotected // channels will be prioritized. bool lockProtected; // The source that protected this channel from locking. // -1 if the channel is not protected. int8 protectedSource; // The number of notes currently active on the channel. uint8 activeNotes; // The MIDI controller values currently used by the channel. MilesMidiChannelControlData *currentData; // The MIDI controller values set by the sources which are // not currently using the channel because it is locked. // These values will be set on the channel when the channel // is unlocked. MilesMidiChannelControlData *unlockData; MidiChannelEntry() : locked(false), lockDataChannel(-1), lockProtected(false), protectedSource(-1), activeNotes(0), currentData(0), unlockData(0) { } }; /** * Send out a control change MIDI message using the specified data. * @param controlData The new MIDI controller value will be set on this MidiChannelControlData * @param sendMessage True if the message should be sent out to the device */ void controlChange(byte outputChannel, byte controllerNumber, byte controllerValue, int8 source, MidiChannelControlData &controlData, bool channelLockedByOtherSource = false) override; bool addActiveNote(uint8 outputChannel, uint8 note, int8 source) override; bool removeActiveNote(uint8 outputChannel, uint8 note, int8 source) override; /** * Removes active notes from the active notes registration on the specified channel. * @param sustainedNotes True if only sustained notes should be removed; otherwise only regular active notes will be removed */ void removeActiveNotes(uint8 outputChannel, bool sustainedNotes) override; /** * Find and lock an output channel and reserve it for the specified * source. The output channel will be mapped to the specified data * channel. */ void lockChannel(uint8 source, uint8 dataChannel); /** * Find an output channel to lock. This will be based on the number * of active notes on the channels and whether the channel is * protected or not. * @param useProtectedChannels When true, protected channels are considered for locking * @returns The output channel to lock, or -1 if no channel is available */ int8 findLockChannel(bool useProtectedChannels = false); /** * Unlock an output channel. This will stop all notes on the channel, * restore the controller values and make it available to other sources. */ void unlockChannel(uint8 outputChannel); /** * Send a program change MIDI message using the specified data. * @param controlData The new program value will be set on this MidiChannelControlData * @param sendMessage True if the message should be sent out to the device */ void programChange(byte outputChannel, byte patchId, int8 source, MidiChannelControlData &controlData, bool channelLockedByOtherSource = false) override; void stopNotesOnChannel(uint8 outputChannelNumber); struct MidiCustomTimbreEntry { bool used; bool protectionEnabled; byte currentPatchBank; byte currentPatchId; uint32 lastUsedNoteCounter; MidiCustomTimbreEntry() : used(false), protectionEnabled(false), currentPatchBank(0), currentPatchId(0), lastUsedNoteCounter(0) {} }; struct MilesMT32SysExQueueEntry { uint32 targetAddress; byte dataPos; byte data[MILES_CONTROLLER_SYSEX_QUEUE_SIZE]; MilesMT32SysExQueueEntry() : targetAddress(0), dataPos(0) { memset(data, 0, sizeof(data)); } }; // stores information about all MIDI channels MidiChannelEntry _midiChannels[MIDI_CHANNEL_COUNT]; // stores information about all custom timbres MidiCustomTimbreEntry _customTimbres[MILES_MT32_CUSTOMTIMBRE_COUNT]; byte _patchesBank[MILES_MT32_PATCHES_COUNT]; // holds all instruments MilesMT32InstrumentEntry *_instrumentTablePtr; uint16 _instrumentTableCount; uint32 _noteCounter; // used to figure out, which timbres are outdated // Queues for Miles SysEx controllers MilesMT32SysExQueueEntry _milesSysExQueues[MILES_CONTROLLER_SYSEX_QUEUE_COUNT]; }; extern MidiDriver *MidiDriver_Miles_AdLib_create(const Common::String &filenameAdLib, const Common::String &filenameOPL3, Common::SeekableReadStream *streamAdLib = nullptr, Common::SeekableReadStream *streamOPL3 = nullptr); extern MidiDriver_Miles_Midi *MidiDriver_Miles_MT32_create(const Common::String &instrumentDataFilename); extern MidiDriver_Miles_Midi *MidiDriver_Miles_MIDI_create(MusicType midiType, const Common::String &instrumentDataFilename); /** @} */ } // End of namespace Audio #endif // AUDIO_MILES_MIDIDRIVER_H