MADE: New MIDI player

This rewrites the MADE music player to remove the use of the MidiPlayer. This
fixes the following issues in Return to Zork:
- User music volume control did not work for AdLib.
- MIDI channels were remapped; sometimes channels were missing (f.e. at the
  lighthouse exterior).
- Some music tracks did not play because the current track had to be stopped
  before playing a new track (f.e. the music over the credits after the intro).
- MIDI played at half volume and volume control would reset when a new track
  started playing.
- "Native MT-32" flag was not checked, so the Miles driver was not used for
  hardware MT-32.
- MT-32 to GM instrument mapping did not work.
- Roland GS mode now works.
This commit is contained in:
Coen Rampen 2021-09-25 22:07:02 +02:00
parent d8491ecea4
commit 2d5e8dbee6
3 changed files with 123 additions and 115 deletions

View File

@ -109,11 +109,7 @@ MadeEngine::~MadeEngine() {
void MadeEngine::syncSoundSettings() {
Engine::syncSoundSettings();
bool mute = false;
if (ConfMan.hasKey("mute"))
mute = ConfMan.getBool("mute");
_music->setVolume(mute ? 0 : ConfMan.getInt("music_volume"));
_music->syncSoundSettings();
}
int16 MadeEngine::getTicks() {

View File

@ -20,31 +20,32 @@
*
*/
// FIXME: This code is taken from SAGA and needs more work (e.g. setVolume).
// MIDI and digital music class
// MIDI music class
#include "made/music.h"
#include "made/redreader.h"
#include "made/resource.h"
#include "audio/adlib_ms.h"
#include "audio/midiparser.h"
#include "audio/miles.h"
#include "common/config-manager.h"
#include "common/file.h"
#include "common/stream.h"
namespace Made {
MusicPlayer::MusicPlayer(bool milesAudio) : _isGM(false),_milesAudioMode(false) {
MusicType musicType = MT_INVALID;
if (milesAudio) {
MidiDriver::DeviceHandle dev = MidiDriver::detectDevice(MDT_MIDI | MDT_ADLIB | MDT_PREFER_MT32);
musicType = MidiDriver::getMusicType(dev);
Common::SeekableReadStream *adLibInstrumentStream = nullptr;
switch (musicType) {
case MT_ADLIB:
_milesAudioMode = true;
MusicPlayer::MusicPlayer(bool milesAudio) : _parser(0) {
MidiDriver::DeviceHandle dev = MidiDriver::detectDevice(MDT_MIDI | MDT_ADLIB | MDT_PREFER_MT32);
_driverType = MidiDriver::getMusicType(dev);
if (_driverType == MT_GM && ConfMan.getBool("native_mt32"))
_driverType = MT_MT32;
Common::SeekableReadStream *adLibInstrumentStream = nullptr;
switch (_driverType) {
case MT_ADLIB:
if (milesAudio) {
if (Common::File::exists("rtzcd.red")) {
// Installing Return to Zork produces both a SAMPLE.AD and
// a SAMPLE.OPL file, but they are identical. The resource
@ -53,115 +54,115 @@ MusicPlayer::MusicPlayer(bool milesAudio) : _isGM(false),_milesAudioMode(false)
}
_driver = Audio::MidiDriver_Miles_AdLib_create("SAMPLE.AD", "SAMPLE.OPL", adLibInstrumentStream);
delete adLibInstrumentStream;
break;
case MT_MT32:
_milesAudioMode = true;
_driver = Audio::MidiDriver_Miles_MT32_create("");
break;
default:
_milesAudioMode = false;
MidiPlayer::createDriver();
break;
} else {
_driver = new MidiDriver_ADLIB_MADE(OPL::Config::kOpl2);
}
} else {
MidiPlayer::createDriver();
break;
case MT_GM:
case MT_MT32:
if (milesAudio) {
_driver = Audio::MidiDriver_Miles_MIDI_create(MT_MT32, "");
} else {
_driver = new MidiDriver_MT32GM(MT_MT32);
}
break;
default:
_driver = new MidiDriver_NULL_Multisource();
break;
}
int ret = _driver->open();
if (ret == 0) {
if (musicType != MT_ADLIB) {
if (_nativeMT32)
_driver->sendMT32Reset();
else
_driver->sendGMReset();
}
if (_driver) {
_driver->property(MidiDriver::PROP_USER_VOLUME_SCALING, true);
if (_driver->open() != 0)
error("Failed to open MIDI driver.");
_driver->setTimerCallback(this, &timerCallback);
}
syncSoundSettings();
}
void MusicPlayer::send(uint32 b) {
if (_milesAudioMode) {
_driver->send(b);
return;
MusicPlayer::~MusicPlayer() {
if (_parser) {
_parser->stopPlaying();
delete _parser;
}
if ((b & 0xF0) == 0xC0 && !_isGM && !_nativeMT32) {
b = (b & 0xFFFF00FF) | MidiDriver::_mt32ToGm[(b >> 8) & 0xFF] << 8;
if (_driver) {
_driver->setTimerCallback(0, 0);
_driver->close();
delete _driver;
}
Audio::MidiPlayer::send(b);
}
void MusicPlayer::playXMIDI(GenericResource *midiResource, MusicFlags flags) {
Common::StackLock lock(_mutex);
void MusicPlayer::playXMIDI(GenericResource *midiResource) {
if (_parser) {
_parser->unloadMusic();
} else {
_parser = MidiParser::createParser_XMIDI(0, 0, 0);
if (_isPlaying)
return;
stop();
_parser->setMidiDriver(_driver);
_parser->setTimerRate(_driver->getBaseTempo());
_parser->property(MidiParser::mpSendSustainOffOnNotesOff, 1);
}
// Load XMID resource data
_isGM = true;
MidiParser *parser = MidiParser::createParser_XMIDI();
if (parser->loadMusic(midiResource->getData(), midiResource->getSize())) {
parser->setTrack(0);
parser->setMidiDriver(this);
parser->setTimerRate(_driver->getBaseTempo());
parser->property(MidiParser::mpCenterPitchWheelOnUnload, 1);
parser->property(MidiParser::mpSendSustainOffOnNotesOff, 1);
_parser = parser;
setVolume(127);
_isLooping = flags & MUSIC_LOOP;
_isPlaying = true;
} else {
delete parser;
}
_parser->loadMusic(midiResource->getData(), midiResource->getSize());
}
void MusicPlayer::playSMF(GenericResource *midiResource, MusicFlags flags) {
Common::StackLock lock(_mutex);
void MusicPlayer::playSMF(GenericResource *midiResource) {
if (_parser) {
_parser->unloadMusic();
} else {
_parser = MidiParser::createParser_SMF(0);
if (_isPlaying)
return;
stop();
_parser->setMidiDriver(_driver);
_parser->setTimerRate(_driver->getBaseTempo());
_parser->property(MidiParser::mpCenterPitchWheelOnUnload, 1);
}
// Load MIDI resource data
_isGM = true;
MidiParser *parser = MidiParser::createParser_SMF();
if (parser->loadMusic(midiResource->getData(), midiResource->getSize())) {
parser->setTrack(0);
parser->setMidiDriver(this);
parser->setTimerRate(_driver->getBaseTempo());
parser->property(MidiParser::mpCenterPitchWheelOnUnload, 1);
_parser = parser;
setVolume(127);
_isLooping = flags & MUSIC_LOOP;
_isPlaying = true;
} else {
delete parser;
}
_parser->loadMusic(midiResource->getData(), midiResource->getSize());
}
void MusicPlayer::pause() {
setVolume(-1);
_isPlaying = false;
if (_parser)
_parser->pausePlaying();
}
void MusicPlayer::resume() {
setVolume(127);
_isPlaying = true;
if (_parser)
_parser->resumePlaying();
}
void MusicPlayer::stop() {
if (_parser)
_parser->stopPlaying();
}
bool MusicPlayer::isPlaying() {
return _parser ? _parser->isPlaying() : false;
}
void MusicPlayer::syncSoundSettings() {
if (_driver)
_driver->syncSoundSettings();
}
void MusicPlayer::onTimer() {
if (_parser)
_parser->onTimer();
}
void MusicPlayer::timerCallback(void *data) {
((MusicPlayer *)data)->onTimer();
}
MidiDriver_ADLIB_MADE::MidiDriver_ADLIB_MADE(OPL::Config::OplType oplType) : MidiDriver_ADLIB_Multisource(oplType) {
_modulationDepth = MODULATION_DEPTH_LOW;
_vibratoDepth = VIBRATO_DEPTH_LOW;
_defaultChannelVolume = 0x7F;
}
} // End of namespace Made

View File

@ -25,33 +25,44 @@
#ifndef MADE_MUSIC_H
#define MADE_MUSIC_H
#include "audio/midiplayer.h"
#include "audio/adlib_ms.h"
#include "audio/mididrv.h"
#include "audio/mididrv_ms.h"
#include "audio/midiparser.h"
namespace Made {
class GenericResource;
enum MusicFlags {
MUSIC_NORMAL = 0,
MUSIC_LOOP = 1
};
class MusicPlayer : public Audio::MidiPlayer {
class MusicPlayer {
public:
MusicPlayer(bool milesAudio);
~MusicPlayer();
void playXMIDI(GenericResource *midiResource, MusicFlags flags = MUSIC_NORMAL);
void playSMF(GenericResource *midiResource, MusicFlags flags = MUSIC_NORMAL);
// void stop();
void pause() override;
void resume() override;
void playXMIDI(GenericResource *midiResource);
void playSMF(GenericResource *midiResource);
void stop();
void pause();
void resume();
// MidiDriver_BASE interface implementation
void send(uint32 b) override;
bool isPlaying();
void syncSoundSettings();
protected:
bool _isGM;
bool _milesAudioMode;
private:
MidiParser *_parser;
MidiDriver_Multisource *_driver;
MusicType _driverType;
static void timerCallback(void *refCon);
void onTimer();
};
class MidiDriver_ADLIB_MADE : public MidiDriver_ADLIB_Multisource {
public:
MidiDriver_ADLIB_MADE(OPL::Config::OplType oplType);
// TODO Implement AdLib driver logic for Manhole / LGoP2
};
} // End of namespace Made