2015-06-26 02:51:30 +02:00
|
|
|
/* 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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2015-06-27 22:43:32 +02:00
|
|
|
#ifndef AUDIO_MILES_MIDIDRIVER_H
|
|
|
|
#define AUDIO_MILES_MIDIDRIVER_H
|
2015-06-26 02:51:30 +02:00
|
|
|
|
|
|
|
#include "audio/mididrv.h"
|
2020-07-27 14:43:21 +02:00
|
|
|
#include "audio/mt32gm.h"
|
|
|
|
|
2015-06-26 02:51:30 +02:00
|
|
|
#include "common/error.h"
|
2020-05-29 17:50:14 +02:00
|
|
|
#include "common/mutex.h"
|
2020-06-26 16:27:33 +02:00
|
|
|
#include "common/queue.h"
|
2015-07-04 12:55:49 +02:00
|
|
|
#include "common/stream.h"
|
2015-06-26 02:51:30 +02:00
|
|
|
|
2015-06-27 22:43:32 +02:00
|
|
|
namespace Audio {
|
2015-06-26 02:51:30 +02:00
|
|
|
|
2020-11-30 02:54:01 +01:00
|
|
|
/**
|
|
|
|
* @defgroup audio_miles Miles and XMIDI
|
|
|
|
* @ingroup audio
|
|
|
|
*
|
|
|
|
* @brief API for managing XMIDI files used by Miles Sound System.
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
2015-06-26 02:51:30 +02:00
|
|
|
// Miles Audio supported controllers for control change messages
|
2015-06-27 13:56:48 +02:00
|
|
|
#define MILES_CONTROLLER_SELECT_PATCH_BANK 114
|
|
|
|
#define MILES_CONTROLLER_PROTECT_VOICE 112
|
|
|
|
#define MILES_CONTROLLER_PROTECT_TIMBRE 113
|
2020-05-29 17:50:14 +02:00
|
|
|
#define MILES_CONTROLLER_LOCK_CHANNEL 110
|
|
|
|
#define MILES_CONTROLLER_PROTECT_CHANNEL 111
|
2015-06-27 13:56:48 +02:00
|
|
|
#define MILES_CONTROLLER_PITCH_RANGE 6
|
2015-06-27 17:04:26 +02:00
|
|
|
#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
|
|
|
|
|
2015-06-28 23:52:50 +02:00
|
|
|
// 3 SysEx controllers, each range 5
|
|
|
|
// 32-36 for 1st queue
|
|
|
|
// 37-41 for 2nd queue
|
|
|
|
// 42-46 for 3rd queue
|
2015-06-27 17:04:26 +02:00
|
|
|
#define MILES_CONTROLLER_SYSEX_RANGE_BEGIN 32
|
2015-06-28 23:52:50 +02:00
|
|
|
#define MILES_CONTROLLER_SYSEX_RANGE_END 46
|
|
|
|
|
|
|
|
#define MILES_CONTROLLER_SYSEX_QUEUE_COUNT 3
|
|
|
|
#define MILES_CONTROLLER_SYSEX_QUEUE_SIZE 32
|
|
|
|
|
2020-06-04 15:04:05 +02:00
|
|
|
#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
|
2015-06-27 17:04:26 +02:00
|
|
|
|
|
|
|
#define MILES_CONTROLLER_XMIDI_RANGE_BEGIN 110
|
|
|
|
#define MILES_CONTROLLER_XMIDI_RANGE_END 120
|
2015-06-26 02:51:30 +02:00
|
|
|
|
2020-05-29 17:50:14 +02:00
|
|
|
#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))
|
|
|
|
|
2020-07-27 14:43:21 +02:00
|
|
|
// Some engines using Miles assume a source neutral
|
|
|
|
// volume of 256, so use this by default.
|
|
|
|
#define MILES_DEFAULT_SOURCE_NEUTRAL_VOLUME 256
|
|
|
|
|
2021-08-03 21:49:46 +02:00
|
|
|
enum MilesVersion {
|
|
|
|
MILES_VERSION_2 = 2,
|
|
|
|
MILES_VERSION_3
|
|
|
|
};
|
|
|
|
|
2020-05-29 17:50:14 +02:00
|
|
|
struct MilesMT32InstrumentEntry {
|
|
|
|
byte bankId;
|
|
|
|
byte patchId;
|
2020-07-27 14:43:21 +02:00
|
|
|
byte commonParameter[MILES_MT32_PATCHDATA_COMMONPARAMETER_SIZE];
|
|
|
|
byte partialParameters[MILES_MT32_PATCHDATA_PARTIALPARAMETERS_COUNT][MILES_MT32_PATCHDATA_PARTIALPARAMETER_SIZE];
|
2020-05-29 17:50:14 +02:00
|
|
|
};
|
|
|
|
|
2020-06-26 22:23:08 +02:00
|
|
|
/**
|
|
|
|
* 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:
|
2020-10-16 23:36:57 +02:00
|
|
|
virtual ~MidiDriver_Miles_Xmidi_Timbres() { }
|
|
|
|
|
2020-06-26 22:23:08 +02:00
|
|
|
/**
|
|
|
|
* 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;
|
|
|
|
};
|
|
|
|
|
2020-07-27 14:43:21 +02:00
|
|
|
class MidiDriver_Miles_Midi : public MidiDriver_MT32GM, public MidiDriver_Miles_Xmidi_Timbres {
|
2020-05-29 17:50:14 +02:00
|
|
|
public:
|
|
|
|
MidiDriver_Miles_Midi(MusicType midiType, MilesMT32InstrumentEntry *instrumentTablePtr, uint16 instrumentTableCount);
|
2020-07-27 14:43:21 +02:00
|
|
|
~MidiDriver_Miles_Midi();
|
2020-05-29 17:50:14 +02:00
|
|
|
|
|
|
|
public:
|
2020-07-27 14:43:21 +02:00
|
|
|
using MidiDriver_MT32GM::send;
|
2020-05-29 17:50:14 +02:00
|
|
|
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.
|
|
|
|
*/
|
2020-07-27 14:43:21 +02:00
|
|
|
void deinitSource(uint8 source) override;
|
2020-05-29 17:50:14 +02:00
|
|
|
|
2020-06-27 16:02:20 +02:00
|
|
|
void stopAllNotes(bool stopSustainedNotes = false) override;
|
2020-05-29 17:50:14 +02:00
|
|
|
|
2021-07-24 22:38:33 +02:00
|
|
|
uint32 property(int prop, uint32 param) override;
|
|
|
|
|
2020-07-27 14:43:21 +02:00
|
|
|
void processXMIDITimbreChunk(const byte *timbreListPtr, uint32 timbreListSize) override;
|
2020-05-29 17:50:14 +02:00
|
|
|
|
|
|
|
protected:
|
2020-07-27 14:43:21 +02:00
|
|
|
void initControlData() override;
|
|
|
|
void initMidiDevice() override;
|
2021-05-15 22:03:28 +02:00
|
|
|
void applySourceVolume(uint8 source) override;
|
2020-05-29 17:50:14 +02:00
|
|
|
|
|
|
|
private:
|
|
|
|
void writeRhythmSetup(byte note, byte customTimbreId);
|
2020-06-26 16:27:33 +02:00
|
|
|
void writePatchTimbre(byte patchId, byte timbreGroup, byte timbreId, bool useSysExQueue = false);
|
2020-05-29 17:50:14 +02:00
|
|
|
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);
|
|
|
|
|
2020-06-26 16:27:33 +02:00
|
|
|
void setupPatch(byte patchBank, byte patchId, bool useSysExQueue = false);
|
2020-05-29 17:50:14 +02:00
|
|
|
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.
|
|
|
|
*/
|
2020-07-27 14:43:21 +02:00
|
|
|
struct MilesMidiChannelControlData : MidiChannelControlData {
|
2020-05-29 17:50:14 +02:00
|
|
|
// Custom timbre data
|
|
|
|
byte currentPatchBank;
|
|
|
|
|
|
|
|
bool usingCustomTimbre;
|
|
|
|
byte currentCustomTimbreId;
|
|
|
|
|
2020-07-27 14:43:21 +02:00
|
|
|
MilesMidiChannelControlData() : currentPatchBank(0),
|
2020-05-29 17:50:14 +02:00
|
|
|
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.
|
2020-07-27 14:43:21 +02:00
|
|
|
MilesMidiChannelControlData *currentData;
|
2020-05-29 17:50:14 +02:00
|
|
|
// 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.
|
2020-07-27 14:43:21 +02:00
|
|
|
MilesMidiChannelControlData *unlockData;
|
2020-05-29 17:50:14 +02:00
|
|
|
|
|
|
|
MidiChannelEntry() : locked(false),
|
|
|
|
lockDataChannel(-1),
|
|
|
|
lockProtected(false),
|
|
|
|
protectedSource(-1),
|
2020-07-27 14:43:21 +02:00
|
|
|
activeNotes(0),
|
|
|
|
currentData(0),
|
|
|
|
unlockData(0) { }
|
2020-05-29 17:50:14 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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
|
|
|
|
*/
|
2020-07-27 14:43:21 +02:00
|
|
|
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;
|
2020-05-29 17:50:14 +02:00
|
|
|
/**
|
|
|
|
* 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
|
|
|
|
*/
|
2020-07-27 14:43:21 +02:00
|
|
|
void removeActiveNotes(uint8 outputChannel, bool sustainedNotes) override;
|
2020-05-29 17:50:14 +02:00
|
|
|
/**
|
|
|
|
* 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
|
|
|
|
*/
|
2020-07-27 14:43:21 +02:00
|
|
|
void programChange(byte outputChannel, byte patchId, int8 source, MidiChannelControlData &controlData, bool channelLockedByOtherSource = false) override;
|
2020-05-29 17:50:14 +02:00
|
|
|
|
|
|
|
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;
|
2020-07-27 14:43:21 +02:00
|
|
|
byte data[MILES_CONTROLLER_SYSEX_QUEUE_SIZE];
|
2020-05-29 17:50:14 +02:00
|
|
|
|
|
|
|
MilesMT32SysExQueueEntry() : targetAddress(0),
|
|
|
|
dataPos(0) {
|
|
|
|
memset(data, 0, sizeof(data));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-07-24 22:38:33 +02:00
|
|
|
// the version of Miles AIL/MSS to emulate
|
|
|
|
MilesVersion _milesVersion;
|
|
|
|
|
2020-05-29 17:50:14 +02:00
|
|
|
// stores information about all MIDI channels
|
2020-07-27 14:43:21 +02:00
|
|
|
MidiChannelEntry _midiChannels[MIDI_CHANNEL_COUNT];
|
2020-05-29 17:50:14 +02:00
|
|
|
|
|
|
|
// 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
|
|
|
|
|
2020-06-26 16:27:33 +02:00
|
|
|
// Queues for Miles SysEx controllers
|
|
|
|
MilesMT32SysExQueueEntry _milesSysExQueues[MILES_CONTROLLER_SYSEX_QUEUE_COUNT];
|
2020-05-29 17:50:14 +02:00
|
|
|
};
|
|
|
|
|
2021-07-13 21:26:23 +02:00
|
|
|
extern MidiDriver_Multisource *MidiDriver_Miles_AdLib_create(const Common::String &filenameAdLib, const Common::String &filenameOPL3, Common::SeekableReadStream *streamAdLib = nullptr, Common::SeekableReadStream *streamOPL3 = nullptr);
|
2015-06-26 02:51:30 +02:00
|
|
|
|
2020-05-29 17:50:14 +02:00
|
|
|
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);
|
2020-11-30 02:54:01 +01:00
|
|
|
/** @} */
|
2015-06-27 22:43:32 +02:00
|
|
|
} // End of namespace Audio
|
2015-06-26 02:51:30 +02:00
|
|
|
|
2015-06-27 22:43:32 +02:00
|
|
|
#endif // AUDIO_MILES_MIDIDRIVER_H
|