/* 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 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, see . * */ #ifndef AUDIO_MIDI_H #define AUDIO_MIDI_H #include "audio/mt32gm.h" #include "common/config-manager.h" #include "common/debug.h" // The initialization of the static const integral data members is done in the class definition, // but we still need to provide a definition if they are odr-used. const uint8 MidiDriver_MT32GM::MT32_DEFAULT_CHANNEL_VOLUME; const uint8 MidiDriver_MT32GM::GM_DEFAULT_CHANNEL_VOLUME; const uint8 MidiDriver_MT32GM::MAXIMUM_MT32_ACTIVE_NOTES; const uint8 MidiDriver_MT32GM::MAXIMUM_GM_ACTIVE_NOTES; // These are the power-on default instruments of the Roland MT-32 family. const byte MidiDriver_MT32GM::MT32_DEFAULT_INSTRUMENTS[8] = { 0x44, 0x30, 0x5F, 0x4E, 0x29, 0x03, 0x6E, 0x7A }; // These are the power-on default panning settings for channels 2-9 of the Roland MT-32 family. // Internally, the MT-32 has 15 panning positions (0-E with 7 being center). // This has been translated to the equivalent MIDI panning values (0-127). // These are used for setting default panning on GM devices when using them with MT-32 data. // Note that MT-32 panning is reversed compared to the MIDI specification. This is not reflected // here; the driver is expected to flip these values based on the _midiDeviceReversePanning // variable. const byte MidiDriver_MT32GM::MT32_DEFAULT_PANNING[8] = { // 7, 8, 7, 8, 4, A, 0, E 0x40, 0x49, 0x40, 0x49, 0x25, 0x5B, 0x00, 0x7F }; // This is the drum map for the Roland Sound Canvas SC-55 v1.xx. It had a fallback mechanism // to correct invalid drumkit selections. Some games rely on this mechanism to select the // correct Roland GS drumkit. Use this map to emulate this mechanism. // E.g. correct invalid drumkit 50: GS_DRUMKIT_FALLBACK_MAP[50] == 48 const uint8 MidiDriver_MT32GM::GS_DRUMKIT_FALLBACK_MAP[128] = { 0, 0, 0, 0, 0, 0, 0, 0, // STANDARD 8, 8, 8, 8, 8, 8, 8, 8, // ROOM 16, 16, 16, 16, 16, 16, 16, 16, // POWER 24, 25, 24, 24, 24, 24, 24, 24, // ELECTRONIC; TR-808 (25) 32, 32, 32, 32, 32, 32, 32, 32, // JAZZ 40, 40, 40, 40, 40, 40, 40, 40, // BRUSH 48, 48, 48, 48, 48, 48, 48, 48, // ORCHESTRA 56, 56, 56, 56, 56, 56, 56, 56, // SFX 0, 0, 0, 0, 0, 0, 0, 0, // No drumkit defined (fall back to STANDARD) 0, 0, 0, 0, 0, 0, 0, 0, // No drumkit defined 0, 0, 0, 0, 0, 0, 0, 0, // No drumkit defined 0, 0, 0, 0, 0, 0, 0, 0, // No drumkit defined 0, 0, 0, 0, 0, 0, 0, 0, // No drumkit defined 0, 0, 0, 0, 0, 0, 0, 0, // No drumkit defined 0, 0, 0, 0, 0, 0, 0, 0, // No drumkit defined 0, 0, 0, 0, 0, 0, 0, 127 // No drumkit defined; CM-64/32L (127) }; const uint8 MidiDriver_MT32GM::MT32_DISPLAY_NUM_CHARS; const uint32 MidiDriver_MT32GM::MT32_DISPLAY_MEMORY_ADDRESS; // Callback hooked up to the driver wrapped by the MIDI driver // object. Executes onTimer and the external callback set by // the setTimerCallback function. void MidiDriver_MT32GM::timerCallback(void *data) { MidiDriver_MT32GM *driver = (MidiDriver_MT32GM *)data; driver->onTimer(); } MidiDriver_MT32GM::MidiDriver_MT32GM(MusicType midiType) : _driver(nullptr), _nativeMT32(false), _enableGS(false), _midiDataReversePanning(false), _midiDeviceReversePanning(false), _scaleGSPercussionVolumeToMT32(false), _isOpen(false), _outputChannelMask(65535), // Channels 1-16 _baseFreq(250), _sysExDelay(0) { memset(_controlData, 0, sizeof(_controlData)); switch (midiType) { case MT_MT32: _midiType = MT_MT32; break; case MT_GM: case MT_GS: // Treat GS same as GM _midiType = MT_GM; break; default: error("MidiDriver_MT32GM: Unsupported music type %i", midiType); break; } for (int i = 0; i < MAXIMUM_SOURCES; ++i) { _availableChannels[i] = 0; // Default MIDI channel mapping: data channel == output channel for (int j = 0; j < MIDI_CHANNEL_COUNT; ++j) { _channelMap[i][j] = j; } } // Default MT-32 <> GM instrument mappings. _mt32ToGMInstrumentMap = _mt32ToGm; _gmToMT32InstrumentMap = _gmToMt32; _maximumActiveNotes = _midiType == MT_MT32 ? MAXIMUM_MT32_ACTIVE_NOTES : MAXIMUM_GM_ACTIVE_NOTES; _activeNotes = new ActiveNote[_maximumActiveNotes]; assert(_activeNotes); } MidiDriver_MT32GM::~MidiDriver_MT32GM() { if (_driver) { _driver->setTimerCallback(nullptr, nullptr); _driver->close(); delete _driver; } _driver = nullptr; for (int i = 0; i < MIDI_CHANNEL_COUNT; ++i) { delete _controlData[i]; } if (_activeNotes) delete[] _activeNotes; } int MidiDriver_MT32GM::open() { assert(!_driver); // Setup midi driver MidiDriver::DeviceHandle dev = MidiDriver::detectDevice(MDT_MIDI | (_midiType == MT_MT32 ? MDT_PREFER_MT32 : MDT_PREFER_GM)); MusicType deviceMusicType = MidiDriver::getMusicType(dev); if (!(deviceMusicType == MT_MT32 || deviceMusicType == MT_GM || deviceMusicType == MT_GS)) error("MidiDriver_MT32GM: detected music device uses unsupported music type %i", deviceMusicType); MidiDriver *driver = MidiDriver::createMidi(dev); bool nativeMT32 = deviceMusicType == MT_MT32 || ConfMan.getBool("native_mt32"); return open(driver, nativeMT32); } int MidiDriver_MT32GM::open(MidiDriver *driver, bool nativeMT32) { assert(!_driver); _driver = driver; _nativeMT32 = nativeMT32; _enableGS = ConfMan.getBool("enable_gs"); if (!_driver) return 255; if (_nativeMT32) _outputChannelMask = _midiType == MT_MT32 ? 1022 : 767; // Channels 2-10 / 1-8 and 10 _driver->property(MidiDriver::PROP_CHANNEL_MASK, _outputChannelMask); int ret = _driver->open(); if (ret != MidiDriver::MERR_ALREADY_OPEN && ret != 0) return ret; _timerRate = _driver->getBaseTempo(); _driver->setTimerCallback(this, timerCallback); initControlData(); initMidiDevice(); syncSoundSettings(); _isOpen = true; return 0; } void MidiDriver_MT32GM::initControlData() { for (int i = 0; i < MIDI_CHANNEL_COUNT; ++i) { _controlData[i] = new MidiChannelControlData(); _controlData[i]->volume = _controlData[i]->scaledVolume = (_nativeMT32 ? MT32_DEFAULT_CHANNEL_VOLUME : GM_DEFAULT_CHANNEL_VOLUME); _controlData[i]->pitchBendSensitivity = (_nativeMT32 ? MT32_PITCH_BEND_SENSITIVITY_DEFAULT : GM_PITCH_BEND_SENSITIVITY_DEFAULT); if (_nativeMT32 && i >= 1 && i <= 8) { _controlData[i]->program = MT32_DEFAULT_INSTRUMENTS[i - 1]; _controlData[i]->panPosition = MT32_DEFAULT_PANNING[i - 1]; } } } void MidiDriver_MT32GM::initMidiDevice() { if (_nativeMT32) { initMT32(_midiType != MT_MT32); } else { initGM(_midiType == MT_MT32, _enableGS); } } void MidiDriver_MT32GM::initMT32(bool initForGM) { sendMT32Reset(); if (initForGM) { // Set up MT-32 for GM data. // This is based on Roland's GM settings for MT-32. debug("Initializing MT-32 for General MIDI data"); byte buffer[17]; // Roland MT-32 SysEx for system area memcpy(&buffer[0], "\x41\x10\x16\x12\x10\x00", 6); // Set reverb parameters: // - Mode 2 (Plate) // - Time 3 // - Level 4 memcpy(&buffer[6], "\x01\x02\x03\x04\x66", 5); sysEx(buffer, 11); // Set partial reserve to match SC-55 memcpy(&buffer[6], "\x04\x08\x04\x04\x03\x03\x03\x03\x02\x02\x4C", 11); sysEx(buffer, 17); // Use MIDI instrument channels 1-8 instead of 2-9 memcpy(&buffer[6], "\x0D\x00\x01\x02\x03\x04\x05\x06\x07\x09\x3E", 11); sysEx(buffer, 17); // The MT-32 has reversed stereo panning compared to the MIDI spec. // GM does use panning as specified by the MIDI spec. _midiDeviceReversePanning = true; int i; // Set default GM panning (center on all channels) for (i = 0; i < 8; ++i) { send((0x40 << 16) | (MIDI_CONTROLLER_PANNING << 8) | (MIDI_COMMAND_CONTROL_CHANGE | i)); } // Set default GM instruments (0 on all channels). // This is expected to be mapped to the MT-32 equivalent by the driver. for (i = 0; i < 8; ++i) { send((0 << 8) | (MIDI_COMMAND_PROGRAM_CHANGE | i)); } // Set Pitch Bend Sensitivity to 2 semitones. for (i = 0; i < 8; ++i) { setPitchBendRange(i, 2); } setPitchBendRange(MIDI_RHYTHM_CHANNEL, 2); } } void MidiDriver_MT32GM::initGM(bool initForMT32, bool enableGS) { sendGMReset(); if (initForMT32) { // Set up the GM device for MT-32 MIDI data. // Based on iMuse implementation (which is based on Roland's MT-32 settings for GS) debug("Initializing GM device for MT-32 MIDI data"); // The MT-32 has reversed stereo panning compared to the MIDI spec. // GM does use panning as specified by the MIDI spec. _midiDeviceReversePanning = true; int i; // Set the default panning for the MT-32 instrument channels. for (i = 1; i < 9; ++i) { send((MT32_DEFAULT_PANNING[i - 1] << 16) | (MIDI_CONTROLLER_PANNING << 8) | (MIDI_COMMAND_CONTROL_CHANGE | i)); } // Set Channels 1-16 Reverb to 64, which is the // equivalent of MT-32 default Reverb Level 5 for (i = 0; i < 16; ++i) send((64 << 16) | (MIDI_CONTROLLER_REVERB << 8) | (MIDI_COMMAND_CONTROL_CHANGE | i)); // Set Channels 1-16 Chorus to 0. The MT-32 has no chorus capability. // (This is probably the default for many GM devices with chorus anyway.) for (i = 0; i < 16; ++i) send((0 << 16) | (MIDI_CONTROLLER_CHORUS << 8) | (MIDI_COMMAND_CONTROL_CHANGE | i)); // Set Channels 1-16 Pitch Bend Sensitivity to 12 semitones. for (i = 0; i < 16; ++i) { setPitchBendRange(i, 12); } if (enableGS) { // GS specific settings for MT-32 instrument mapping. debug("Additional initialization of GS device for MT-32 MIDI data"); // Note: All Roland GS devices support CM-64/32L maps if (getPercussionChannel() != nullptr) { // Set Percussion Channel to SC-55 Map (CC#32, 01H), then // Switch Drum Map to CM-64/32L (MT-32 Compatible Drums) // Bank select MSB: bank 0 getPercussionChannel()->controlChange(MIDI_CONTROLLER_BANK_SELECT_MSB, 0); // Bank select LSB: map 1 (SC-55) getPercussionChannel()->controlChange(MIDI_CONTROLLER_BANK_SELECT_LSB, 1); } // Patch change: 127 (CM-64/32L) send((127 << 8) | MIDI_COMMAND_PROGRAM_CHANGE | 9); // Set Channels 1-16 to SC-55 Map, then CM-64/32L Variation for (i = 0; i < 16; ++i) { if (getPercussionChannel() != nullptr && i == getPercussionChannel()->getNumber()) continue; // Bank select MSB: bank 127 (CM-64/32L) send((127 << 16) | (MIDI_CONTROLLER_BANK_SELECT_MSB << 8) | (MIDI_COMMAND_CONTROL_CHANGE | i)); // Bank select LSB: map 1 (SC-55) send((1 << 16) | (MIDI_CONTROLLER_BANK_SELECT_LSB << 8) | (MIDI_COMMAND_CONTROL_CHANGE | i)); // Patch change: 0 (causes bank select to take effect) send((0 << 16) | (0 << 8) | (MIDI_COMMAND_PROGRAM_CHANGE | i)); } byte buffer[12]; // Roland GS SysEx ID memcpy(&buffer[0], "\x41\x10\x42\x12", 4); // Set channels 1-16 Mod. LFO1 Pitch Depth to 4 memcpy(&buffer[4], "\x40\x20\x04\x04\x18", 5); for (i = 0; i < 16; ++i) { buffer[5] = 0x20 + i; buffer[8] = 0x18 - i; sysEx(buffer, 9); } // In Roland's GS MT-32 emulation settings, percussion channel expression // is locked at 80. This corrects a difference in volume of the SC-55 MT-32 // drum kit vs the drums of the MT-32. However, this approach has a problem: // the MT-32 supports expression on the percussion channel, so MIDI data // which uses this will play incorrectly. So instead, percussion channel // volume will be scaled by the driver by a factor 80/127. // Strangely, the regular GM drum kit does have a volume that matches the // MT-32 drums, so scaling is only necessary when using GS MT-32 emulation. _scaleGSPercussionVolumeToMT32 = true; // Change Reverb settings (as used by Roland): // - Character: 0 // - Pre-LPF: 4 // - Level: 35h // - Time: 6Ah memcpy(&buffer[4], "\x40\x01\x31\x00\x04\x35\x6A\x6B", 8); sysEx(buffer, 12); } // Set the default MT-32 patches. For non-GS devices these are expected to be // mapped to the GM equivalents by the driver. for (i = 1; i < 9; ++i) { send((MT32_DEFAULT_INSTRUMENTS[i - 1] << 8) | (MIDI_COMMAND_PROGRAM_CHANGE | i)); } // Regarding Master Tune: 442 kHz was intended for the MT-32 family, but // apparently due to a firmware bug the master tune was actually 440 kHz for // all models (see MUNT source code for more details). So master tune is left // at 440 kHz for GM devices playing MT-32 MIDI data. } } void MidiDriver_MT32GM::close() { if (_driver) { _driver->close(); } } bool MidiDriver_MT32GM::isReady(int8 source) { Common::StackLock lock(_sysExQueueMutex); // For an unspecified source, just return if the queue is empty or not. if (source < 0) return _sysExQueue.empty(); // For a specific source, check if there is a SysEx for that source in the // queue. for (Common::ListInternal::Iterator it = _sysExQueue.begin(); it != _sysExQueue.end(); it++) { if (it->source == source) return false; } return true; } uint32 MidiDriver_MT32GM::property(int prop, uint32 param) { switch (prop) { case PROP_MIDI_DATA_REVERSE_PANNING: if (param == 0xFFFF) return _midiDataReversePanning ? 1 : 0; _midiDataReversePanning = param > 0; break; default: return MidiDriver_Multisource::property(prop, param); } return 0; } void MidiDriver_MT32GM::send(uint32 b) { send(-1, b); } void MidiDriver_MT32GM::send(int8 source, uint32 b) { byte dataChannel = b & 0xf; int8 outputChannel = source < 0 ? dataChannel : mapSourceChannel(source, dataChannel); MidiChannelControlData &controlData = *_controlData[outputChannel]; processEvent(source, b, outputChannel, controlData); } // MIDI messages can be found at https://web.archive.org/web/20120128110425/http://www.midi.org/techspecs/midimessages.php void MidiDriver_MT32GM::processEvent(int8 source, uint32 b, uint8 outputChannel, MidiChannelControlData &controlData, bool channelLockedByOtherSource) { assert(source < MAXIMUM_SOURCES); byte command = b & 0xf0; byte op1 = (b >> 8) & 0xff; byte op2 = (b >> 16) & 0xff; if (command != MIDI_COMMAND_SYSTEM && controlData.source != source) { // A new source has sent an event on this channel. controlData.sourceVolumeApplied = false; controlData.source = source; if (source >= 0) // If the new source is a real MIDI source, apply controller // default values. applyControllerDefaults(source, controlData, outputChannel, channelLockedByOtherSource); } switch (command) { case MIDI_COMMAND_NOTE_OFF: case MIDI_COMMAND_NOTE_ON: if (!channelLockedByOtherSource) noteOnOff(outputChannel, command, op1, op2, source, controlData); break; case MIDI_COMMAND_PITCH_BEND: pitchBend(outputChannel, op1, op2, source, controlData, channelLockedByOtherSource); break; case MIDI_COMMAND_POLYPHONIC_AFTERTOUCH: // Not supported by MT-32 or GM polyAftertouch(outputChannel, op1, op2, source, controlData, channelLockedByOtherSource); break; case MIDI_COMMAND_CHANNEL_AFTERTOUCH: // Not supported by MT-32 channelAftertouch(outputChannel, op1, source, controlData, channelLockedByOtherSource); break; case MIDI_COMMAND_CONTROL_CHANGE: controlChange(outputChannel, op1, op2, source, controlData, channelLockedByOtherSource); break; case MIDI_COMMAND_PROGRAM_CHANGE: programChange(outputChannel, op1, source, controlData, channelLockedByOtherSource); break; case MIDI_COMMAND_SYSTEM: // The only supported system event is SysEx and that should be sent using the sysEx functions. warning("MidiDriver_MT32GM: send received system event (not processed): %x", b); break; default: warning("MidiDriver_MT32GM: Received unknown event %02x", command); break; } } void MidiDriver_MT32GM::applyControllerDefaults(uint8 source, MidiChannelControlData &controlData, uint8 outputChannel, bool channelLockedByOtherSource) { if (outputChannel != MIDI_RHYTHM_CHANNEL) { // Apply default bank and program only to melodic channels. if (_controllerDefaults.instrumentBank >= 0 && controlData.instrumentBank != _controllerDefaults.instrumentBank) { controlChange(outputChannel, MIDI_CONTROLLER_BANK_SELECT_MSB, _controllerDefaults.instrumentBank, source, controlData); controlChange(outputChannel, MIDI_CONTROLLER_BANK_SELECT_LSB, 0, source, controlData); } if (_controllerDefaults.program[outputChannel] >= 0 && controlData.program != _controllerDefaults.program[outputChannel]) { programChange(outputChannel, _controllerDefaults.program[outputChannel], source, controlData); } } else { // Apply default drumkit only to the rhythm channel. if (_controllerDefaults.drumkit >= 0 && controlData.program != _controllerDefaults.drumkit) { programChange(outputChannel, _controllerDefaults.drumkit, source, controlData); } } if (_controllerDefaults.channelPressure >= 0 && controlData.channelPressure != _controllerDefaults.channelPressure) { channelAftertouch(outputChannel, _controllerDefaults.channelPressure, source, controlData); } if (_controllerDefaults.pitchBend >= 0 && controlData.pitchWheel != _controllerDefaults.pitchBend) { pitchBend(outputChannel, _controllerDefaults.pitchBend & 0x7F, _controllerDefaults.pitchBend >> 7, source, controlData); } if (_controllerDefaults.modulation >= 0 && controlData.modulation != _controllerDefaults.modulation) { controlChange(outputChannel, MIDI_CONTROLLER_MODULATION, _controllerDefaults.modulation, source, controlData); } if (_controllerDefaults.volume >= 0 && controlData.volume != _controllerDefaults.volume) { controlChange(outputChannel, MIDI_CONTROLLER_VOLUME, _controllerDefaults.volume, source, controlData); } if (_controllerDefaults.panning >= 0 && controlData.panPosition != _controllerDefaults.panning) { controlChange(outputChannel, MIDI_CONTROLLER_PANNING, _controllerDefaults.panning, source, controlData); } if (_controllerDefaults.expression >= 0 && controlData.expression != _controllerDefaults.expression) { controlChange(outputChannel, MIDI_CONTROLLER_EXPRESSION, _controllerDefaults.expression, source, controlData); } // RPN will be changed by setting pitch bend sensitivity, so store the // current value. uint16 rpn = controlData.rpn; bool setRpn = false; if (_controllerDefaults.pitchBendSensitivity >= 0 && controlData.pitchBendSensitivity != _controllerDefaults.pitchBendSensitivity) { controlChange(outputChannel, MIDI_CONTROLLER_RPN_MSB, MIDI_RPN_PITCH_BEND_SENSITIVITY >> 8, source, controlData); controlChange(outputChannel, MIDI_CONTROLLER_RPN_LSB, MIDI_RPN_PITCH_BEND_SENSITIVITY & 0xFF, source, controlData); controlChange(outputChannel, MIDI_CONTROLLER_DATA_ENTRY_MSB, _controllerDefaults.pitchBendSensitivity, source, controlData); controlChange(outputChannel, MIDI_CONTROLLER_DATA_ENTRY_LSB, 0, source, controlData); if (rpn != controlData.rpn) // Active RPN was changed; reset it to previous value (or default). setRpn = true; } if (_controllerDefaults.rpn >= 0 && rpn != _controllerDefaults.rpn) { // Set RPN to the specified default value. rpn = _controllerDefaults.rpn; setRpn = true; } if (setRpn) { controlChange(outputChannel, MIDI_CONTROLLER_RPN_MSB, rpn >> 8, source, controlData); controlChange(outputChannel, MIDI_CONTROLLER_RPN_LSB, rpn & 0xFF, source, controlData); } } void MidiDriver_MT32GM::noteOnOff(byte outputChannel, byte command, byte note, byte velocity, int8 source, MidiChannelControlData &controlData) { if (!isOutputChannelUsed(outputChannel)) return; // Note On with velocity 0 is treated as Note Off bool addNote = command == MIDI_COMMAND_NOTE_ON && velocity != 0; if (addNote) { if (source >= 0 && !controlData.sourceVolumeApplied) // Source volume hasn't been applied yet. Do so now. controlChange(outputChannel, MIDI_CONTROLLER_VOLUME, controlData.volume, source, controlData); // Add the new note to the active note registration addActiveNote(outputChannel, note, source); } else { // Remove the note from the active note registration removeActiveNote(outputChannel, note, source); } _driver->send(command | outputChannel, note, velocity); } void MidiDriver_MT32GM::polyAftertouch(byte outputChannel, byte note, byte pressure, int8 source, MidiChannelControlData &controlData, bool channelLockedByOtherSource) { if (!channelLockedByOtherSource) _driver->send(MIDI_COMMAND_CHANNEL_AFTERTOUCH | outputChannel, note, pressure); } void MidiDriver_MT32GM::controlChange(byte outputChannel, byte controllerNumber, byte controllerValue, int8 source, MidiChannelControlData &controlData, bool channelLockedByOtherSource) { assert(source < MAXIMUM_SOURCES); // Standard MIDI controllers switch (controllerNumber) { case MIDI_CONTROLLER_BANK_SELECT_MSB: // Keep track of the current bank for each channel controlData.instrumentBank = controllerValue; break; case MIDI_CONTROLLER_MODULATION: controlData.modulation = controllerValue; break; case MIDI_CONTROLLER_DATA_ENTRY_MSB: if (controlData.rpn == MIDI_RPN_PITCH_BEND_SENSITIVITY) controlData.pitchBendSensitivity = controllerValue; break; case MIDI_CONTROLLER_VOLUME: controlData.volume = controllerValue; controlData.sourceVolumeApplied = true; if (source >= 0) { // Scale to source volume controllerValue = (controllerValue * _sources[source].volume) / _sources[source].neutralVolume; } if (_userVolumeScaling) { if (_userMute) { controllerValue = 0; } else { // Scale to user volume uint16 userVolume = _sources[source].type == SOURCE_TYPE_SFX ? _userSfxVolume : _userMusicVolume; // Treat SOURCE_TYPE_UNDEFINED as music controllerValue = (controllerValue * userVolume) >> 8; } } if (_scaleGSPercussionVolumeToMT32 && outputChannel == MIDI_RHYTHM_CHANNEL) { // Scale GS percussion channel volume to MT-32 level (80/127) controllerValue = (80 * controllerValue) >> 7; } // Source volume scaling might clip volume, so reduce to maximum controllerValue = MIN(controllerValue, (byte)0x7F); if (controlData.scaledVolume == controllerValue) { // Volume is already at this value, so no need to send it out // to the MIDI device. return; } controlData.scaledVolume = controllerValue; break; case MIDI_CONTROLLER_PANNING: if (_midiDeviceReversePanning != _midiDataReversePanning) { // Center panning is 0x40 controllerValue = 0x80 - controllerValue; if (controllerValue > 0x7F) controllerValue = 0x7F; } break; case MIDI_CONTROLLER_EXPRESSION: controlData.expression = controllerValue; break; case MIDI_CONTROLLER_SUSTAIN: controlData.sustain = controllerValue >= 0x40; if (!channelLockedByOtherSource && !controlData.sustain) { removeActiveNotes(outputChannel, true); } break; case MIDI_CONTROLLER_RPN_LSB: controlData.rpn &= 0xFF00; controlData.rpn |= controllerValue; break; case MIDI_CONTROLLER_RPN_MSB: controlData.rpn &= 0x00FF; controlData.rpn |= controllerValue << 8; break; case MIDI_CONTROLLER_RESET_ALL_CONTROLLERS: controlData.channelPressure = 0; controlData.pitchWheel = MIDI_PITCH_BEND_DEFAULT; controlData.modulation = 0; controlData.expression = MIDI_EXPRESSION_DEFAULT; controlData.sustain = false; if (!channelLockedByOtherSource) { removeActiveNotes(outputChannel, true); } controlData.rpn = MIDI_RPN_NULL; break; case MIDI_CONTROLLER_OMNI_ON: case MIDI_CONTROLLER_OMNI_OFF: case MIDI_CONTROLLER_MONO_ON: case MIDI_CONTROLLER_POLY_ON: // These act as an All Notes Off on MT-32, but also turn sustain off. // They are not part of GM, so should not be used in GM data. if (_midiType != MT_MT32) { warning("MidiDriver_MT32GM: unsupported GM controller %x", controllerNumber); return; } controlData.sustain = false; if (!channelLockedByOtherSource) removeActiveNotes(outputChannel, true); if (!_nativeMT32) { // MT-32 data on GM device. // These controllers might not be supported or have side effects // (changing omni or mono/poly mode). Send All Notes Off and // Sustain Off instead. if (!channelLockedByOtherSource) { controllerNumber = MIDI_CONTROLLER_ALL_NOTES_OFF; _driver->send(MIDI_COMMAND_CONTROL_CHANGE | outputChannel | (MIDI_CONTROLLER_SUSTAIN << 8) | (0 << 16)); } } // fall through case MIDI_CONTROLLER_ALL_NOTES_OFF: if (!channelLockedByOtherSource) removeActiveNotes(outputChannel, false); break; default: break; } if (!channelLockedByOtherSource) _driver->send(MIDI_COMMAND_CONTROL_CHANGE | outputChannel | (controllerNumber << 8) | (controllerValue << 16)); } bool MidiDriver_MT32GM::addActiveNote(uint8 outputChannel, uint8 note, int8 source) { Common::StackLock lock(_activeNotesMutex); for (int i = 0; i < _maximumActiveNotes; ++i) { ActiveNote &activeNote = _activeNotes[i]; if (activeNote.channel == 0xFF) { // Add the new note. activeNote.source = source; activeNote.channel = outputChannel; activeNote.note = note; activeNote.sustain = false; return true; } } warning("MidiDriver_MT32GM: Could not add active note %x on channel %i", note, outputChannel); return false; } bool MidiDriver_MT32GM::removeActiveNote(uint8 outputChannel, uint8 note, int8 source) { Common::StackLock lock(_activeNotesMutex); for (int i = 0; i < _maximumActiveNotes; ++i) { ActiveNote &activeNote = _activeNotes[i]; if (activeNote.channel == outputChannel && activeNote.source == source && activeNote.note == note) { if (_controlData[outputChannel]->sustain) { // Sustain is on, so the note should be turned off // when sustain is turned off. activeNote.sustain = true; return false; } else { // Turn off the existing note. activeNote.clear(); return true; } } } //warning("MidiDriver_MT32GM: Could not find active note %x on channel %i when removing", note, outputChannel); return false; } void MidiDriver_MT32GM::removeActiveNotes(uint8 outputChannel, bool sustainedNotes) { Common::StackLock lock(_activeNotesMutex); // Remove sustained or non-sustained notes from the active notes registration for (int i = 0; i < _maximumActiveNotes; ++i) { if (_activeNotes[i].channel == outputChannel && _activeNotes[i].sustain == sustainedNotes) { _activeNotes[i].clear(); } } } void MidiDriver_MT32GM::programChange(byte outputChannel, byte patchId, int8 source, MidiChannelControlData &controlData, bool channelLockedByOtherSource) { // remember patch id for the current MIDI-channel controlData.program = patchId; if (channelLockedByOtherSource) return; if (_instrumentRemapping && outputChannel != MIDI_RHYTHM_CHANNEL) // Apply instrument remapping (if specified) to instrument channels. patchId = _instrumentRemapping[patchId]; if (_midiType == MT_MT32) { if (outputChannel == MIDI_RHYTHM_CHANNEL && !(!_nativeMT32 && _enableGS && patchId == 0x7F)) { // Patch changes on the rhythm channel do nothing on an MT-32. // On GM/GS devices they might unintentionally change the drumkit. // Exception: changing the drumkit to the MT-32 drumkit on a GS // device. return; } if (!_nativeMT32 && !_enableGS) { // GM device: map the patch to GM equivalent patchId = mapMT32InstrumentToGM(patchId); } } else { // GM/GS MIDI if (outputChannel == MIDI_RHYTHM_CHANNEL) { // Correct possible wrong GS drumkit number patchId = GS_DRUMKIT_FALLBACK_MAP[patchId]; } else if (!_nativeMT32) { // Correct possible wrong bank / instrument variation byte correctedBank = correctInstrumentBank(controlData.instrumentBank, patchId); if (correctedBank != 0xFF) { // Send out a bank select for the corrected bank number controlChange(outputChannel, MIDI_CONTROLLER_BANK_SELECT_MSB, correctedBank, source, controlData); controlChange(outputChannel, MIDI_CONTROLLER_BANK_SELECT_LSB, 0, source, controlData); } } else { // GM on an MT-32: map the patch to the MT-32 equivalent patchId = mapGMInstrumentToMT32(patchId); } } // Finally send program change to MIDI device _driver->send(MIDI_COMMAND_PROGRAM_CHANGE | outputChannel | (patchId << 8)); } byte MidiDriver_MT32GM::mapMT32InstrumentToGM(byte mt32Instrument) { return _mt32ToGMInstrumentMap[mt32Instrument]; } byte MidiDriver_MT32GM::mapGMInstrumentToMT32(byte gmInstrument) { return _gmToMT32InstrumentMap[gmInstrument]; } byte MidiDriver_MT32GM::correctInstrumentBank(byte instrumentBank, byte patchId) { if (instrumentBank == 0 || patchId >= 120 || instrumentBank >= 64) // Usually, no bank select has been sent and no correction is // necessary. // No correction is performed on banks 64-127 or on the SFX // instruments (120-127). return 0xFF; // Determine the intended bank. This emulates the behavior of the // Roland SC-55 v1.2x. Instruments have 2, 1 or 0 sub-capital tones. // Depending on the selected bank and the selected instrument, the // bank will "fall back" to a sub-capital tone or to the capital // tone (bank 0). byte correctedBank = 0xFF; switch (patchId) { case 25: // Steel-String Guitar / 12-string Guitar / Mandolin // This instrument has 2 sub-capital tones. Bank selects 17-63 // are corrected to the second sub-capital tone at 16. if (instrumentBank >= 16) { correctedBank = 16; break; } // Corrections for values below 16 are handled below. // fall through case 4: // Electric Piano 1 / Detuned Electric Piano 1 case 5: // Electric Piano 2 / Detuned Electric Piano 2 case 6: // Harpsichord / Coupled Harpsichord case 14: // Tubular-bell / Church Bell case 16: // Organ 1 / Detuned Organ 1 case 17: // Organ 2 / Detuned Organ 2 case 19: // Church Organ 1 / Church Organ 2 case 21: // Accordion Fr / Accordion It case 24: // Nylon-string Guitar / Ukelele case 26: // Jazz Guitar / Hawaiian Guitar case 27: // Clean Guitar / Chorus Guitar case 28: // Muted Guitar / Funk Guitar case 30: // Distortion Guitar / Feedback Guitar case 31: // Guitar Harmonics / Guitar Feedback case 38: // Synth Bass 1 / Synth Bass 3 case 39: // Synth Bass 2 / Synth Bass 4 case 48: // Strings / Orchestra case 50: // Synth Strings 1 / Synth Strings 3 case 61: // Brass 1 / Brass 2 case 62: // Synth Brass 1 / Synth Brass 3 case 63: // Synth Brass 2 / Synth Brass 4 case 80: // Square Wave / Sine Wave case 107: // Koto / Taisho Koto case 115: // Woodblock / Castanets case 116: // Taiko / Concert BD case 117: // Melodic Tom 1 / Melodic Tom 2 case 118: // Synth Drum / 808 Tom // These instruments have one sub-capital tone. Bank selects 9-63 // are corrected to the sub-capital tone at 8. if (instrumentBank >= 8) { correctedBank = 8; break; } // Corrections for values below 8 are handled below. // fall through default: // The other instruments only have a capital tone. Bank selects // 1-63 are corrected to the capital tone. correctedBank = 0; break; } // Return the corrected bank, or 0xFF if no correction is needed. return instrumentBank != correctedBank ? correctedBank : 0xFF; } void MidiDriver_MT32GM::channelAftertouch(byte outputChannel, byte pressure, int8 source, MidiChannelControlData &controlData, bool channelLockedByOtherSource) { controlData.channelPressure = pressure; if (!channelLockedByOtherSource) _driver->send(MIDI_COMMAND_CHANNEL_AFTERTOUCH | outputChannel, pressure, 0); } void MidiDriver_MT32GM::pitchBend(byte outputChannel, uint8 pitchBendLsb, uint8 pitchBendMsb, int8 source, MidiChannelControlData &controlData, bool channelLockedByOtherSource) { controlData.pitchWheel = ((uint16)pitchBendMsb << 7) | (uint16)pitchBendLsb; if (!channelLockedByOtherSource) _driver->send(MIDI_COMMAND_PITCH_BEND | outputChannel, pitchBendLsb, pitchBendMsb); } void MidiDriver_MT32GM::sysEx(const byte *msg, uint16 length) { uint16 delay = sysExNoDelay(msg, length); if (delay > 0) g_system->delayMillis(delay); } uint16 MidiDriver_MT32GM::sysExNoDelay(const byte *msg, uint16 length) { if (!_nativeMT32 && length >= 3 && msg[0] == 0x41 && msg[2] == 0x16) // MT-32 SysExes have no effect on GM devices. return 0; // Send SysEx _driver->sysEx(msg, length); // Wait the time it takes to send the SysEx data uint16 delay = (length + 2) * 1000 / 3125; // Plus an additional delay for the MT-32 rev00 if (_nativeMT32) delay += 40; return delay; } void MidiDriver_MT32GM::sysExQueue(const byte *msg, uint16 length, int8 source) { SysExData sysEx; memcpy(sysEx.data, msg, length); sysEx.length = length; sysEx.source = source; _sysExQueueMutex.lock(); _sysExQueue.push_back(sysEx); _sysExQueueMutex.unlock(); } uint16 MidiDriver_MT32GM::sysExMT32(const byte *msg, uint16 length, const uint32 targetAddress, bool queue, bool delay, int8 source) { if (!_nativeMT32) // MT-32 SysExes have no effect on GM devices. return 0; byte sysExMessage[270]; uint16 sysExPos = 0; byte sysExByte; uint16 sysExChecksum = 0; memset(&sysExMessage, 0, sizeof(sysExMessage)); sysExMessage[0] = 0x41; // Roland sysExMessage[1] = 0x10; sysExMessage[2] = 0x16; // Model MT32 sysExMessage[3] = 0x12; // Command DT1 sysExChecksum = 0; sysExMessage[4] = (targetAddress >> 14) & 0x7F; sysExMessage[5] = (targetAddress >> 7) & 0x7F; sysExMessage[6] = targetAddress & 0x7F; for (byte targetAddressByte = 4; targetAddressByte < 7; targetAddressByte++) { assert(sysExMessage[targetAddressByte] < 0x80); // security check sysExChecksum -= sysExMessage[targetAddressByte]; } sysExPos = 7; for (int i = 0; i < length; ++i) { sysExByte = *msg++; assert(sysExPos < sizeof(sysExMessage)); assert(sysExByte < 0x80); // security check sysExMessage[sysExPos++] = sysExByte; sysExChecksum -= sysExByte; } // Calculate checksum assert(sysExPos < sizeof(sysExMessage)); sysExMessage[sysExPos++] = sysExChecksum & 0x7F; if (queue) { sysExQueue(sysExMessage, sysExPos, source); } else if (!delay) { return sysExNoDelay(sysExMessage, sysExPos); } else { sysEx(sysExMessage, sysExPos); } return 0; } void MidiDriver_MT32GM::metaEvent(int8 source, byte type, byte *data, uint16 length) { assert(source < MAXIMUM_SOURCES); if (type == 0x2F && source >= 0) // End of Track deinitSource(source); _driver->metaEvent(type, data, length); } void MidiDriver_MT32GM::stopAllNotes(bool stopSustainedNotes) { for (int i = 0; i < MIDI_CHANNEL_COUNT; ++i) { if (!isOutputChannelUsed(i)) continue; if (stopSustainedNotes) { _driver->send(MIDI_COMMAND_CONTROL_CHANGE | i, MIDI_CONTROLLER_SUSTAIN, 0); _controlData[i]->sustain = false; } _driver->send(MIDI_COMMAND_CONTROL_CHANGE | i, MIDI_CONTROLLER_ALL_NOTES_OFF, 0); } _activeNotesMutex.lock(); for (int i = 0; i < _maximumActiveNotes; ++i) { _activeNotes[i].clear(); } _activeNotesMutex.unlock(); } void MidiDriver_MT32GM::clearSysExQueue() { Common::StackLock lock(_sysExQueueMutex); _sysExQueue.clear(); } MidiChannel *MidiDriver_MT32GM::allocateChannel() { if (_driver) return _driver->allocateChannel(); return nullptr; } MidiChannel *MidiDriver_MT32GM::getPercussionChannel() { if (_driver) return _driver->getPercussionChannel(); return nullptr; } bool MidiDriver_MT32GM::isOutputChannelUsed(int8 outputChannel) { return outputChannel >= 0 && outputChannel < MIDI_CHANNEL_COUNT && _outputChannelMask & (1 << outputChannel); } uint32 MidiDriver_MT32GM::getBaseTempo() { if (_driver) { return _driver->getBaseTempo(); } return 1000000 / _baseFreq; } bool MidiDriver_MT32GM::allocateSourceChannels(uint8 source, uint8 numChannels) { assert(source < MAXIMUM_SOURCES); deinitSource(source); _allocationMutex.lock(); uint16 claimedChannels = 0; if (numChannels > 0) { for (int i = 0; i < MIDI_CHANNEL_COUNT; ++i) { if (!isOutputChannelUsed(i) || i == MIDI_RHYTHM_CHANNEL) continue; if (_controlData[i]->source == -1) { claimedChannels |= (1 << i); numChannels--; } if (numChannels == 0) break; } } if (numChannels > 0) { // Not enough channels available. _allocationMutex.unlock(); return false; } // Allocate the channels. for (int i = 0; i < MIDI_CHANNEL_COUNT; ++i) { if ((claimedChannels >> i) & 1) { _controlData[i]->source = source; } // Clear the source channel mapping. if (i != MIDI_RHYTHM_CHANNEL) _channelMap[source][i] = -1; } _allocationMutex.unlock(); _availableChannels[source] = claimedChannels; return true; } int8 MidiDriver_MT32GM::mapSourceChannel(uint8 source, uint8 dataChannel) { int8 outputChannel = _channelMap[source][dataChannel]; if (outputChannel == -1) { for (int i = 0; i < MIDI_CHANNEL_COUNT; ++i) { if ((_availableChannels[source] >> i) & 1) { _availableChannels[source] &= ~(1 << i); _channelMap[source][dataChannel] = i; outputChannel = i; break; } } if (outputChannel == -1) { warning("MidiDriver_MT32GM: Insufficient available channels for source %i", source); } } return outputChannel; } void MidiDriver_MT32GM::deinitSource(uint8 source) { assert(source < MAXIMUM_SOURCES); _sysExQueueMutex.lock(); // Remove any pending SysExes for this source from the queue. Common::ListInternal::Iterator it = _sysExQueue.begin(); while (it != _sysExQueue.end()) { if (it->source == source) { it = _sysExQueue.erase(it); } else { it++; } } _sysExQueueMutex.unlock(); MidiDriver_Multisource::deinitSource(source); // Free channels which were used by this source. for (int i = 0; i < MIDI_CHANNEL_COUNT; ++i) { if (!isOutputChannelUsed(i)) continue; if (_controlData[i]->source == source) { // Set the sustain default value if it is specified (typically // sustain would be turned off). if (_controllerDefaults.sustain >= 0 && _controlData[i]->sustain != (_controllerDefaults.sustain >= 0x40)) { controlChange(i, MIDI_CONTROLLER_SUSTAIN, _controllerDefaults.sustain, _controlData[i]->source, *_controlData[i]); } _controlData[i]->source = -1; } } _availableChannels[source] = 0xFFFF; // Reset the data to output channel mapping for (int i = 0; i < MIDI_CHANNEL_COUNT; ++i) { _channelMap[source][i] = i; } } void MidiDriver_MT32GM::applySourceVolume(uint8 source) { for (int i = 0; i < MIDI_CHANNEL_COUNT; ++i) { if (!isOutputChannelUsed(i)) continue; if (source == 0xFF || _controlData[i]->source == source) controlChange(i, MIDI_CONTROLLER_VOLUME, _controlData[i]->volume, _controlData[i]->source, *_controlData[i]); } } void MidiDriver_MT32GM::stopAllNotes(uint8 source, uint8 channel) { _activeNotesMutex.lock(); for (int i = 0; i < _maximumActiveNotes; ++i) { if ((source == 0xFF || _activeNotes[i].source == source) && (channel == 0xFF || _activeNotes[i].channel == channel)) { if (_activeNotes[i].sustain) { // Turn off sustain controlChange(_activeNotes[i].channel, MIDI_CONTROLLER_SUSTAIN, 0x00, _activeNotes[i].source, *_controlData[i]); } else { // Send note off noteOnOff(_activeNotes[i].channel, MIDI_COMMAND_NOTE_OFF, _activeNotes[i].note, 0x00, _activeNotes[i].source, *_controlData[i]); } } } _activeNotesMutex.unlock(); } void MidiDriver_MT32GM::onTimer() { MidiDriver_Multisource::onTimer(); _sysExQueueMutex.lock(); _sysExDelay -= (_sysExDelay > _timerRate) ? _timerRate : _sysExDelay; if (!_sysExQueue.empty() && _sysExDelay == 0) { // Ready to send next SysEx message to the MIDI device SysExData sysEx = _sysExQueue.front(); _sysExDelay = sysExNoDelay(sysEx.data, sysEx.length) * 1000; _sysExQueue.pop_front(); } _sysExQueueMutex.unlock(); } #endif