mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-15 06:18:33 +00:00
134 lines
3.3 KiB
C++
134 lines
3.3 KiB
C++
/* 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.
|
|
*
|
|
*/
|
|
|
|
#include "audio/midiparser.h"
|
|
#include "common/textconsole.h"
|
|
|
|
#include "toltecs/toltecs.h"
|
|
#include "toltecs/music.h"
|
|
#include "toltecs/resource.h"
|
|
|
|
namespace Toltecs {
|
|
|
|
MusicPlayer::MusicPlayer(bool isGM) : _isGM(isGM), _buffer(NULL) {
|
|
MidiPlayer::createDriver();
|
|
|
|
int ret = _driver->open();
|
|
if (ret == 0) {
|
|
if (_nativeMT32)
|
|
_driver->sendMT32Reset();
|
|
else
|
|
_driver->sendGMReset();
|
|
|
|
_driver->setTimerCallback(this, &timerCallback);
|
|
}
|
|
}
|
|
|
|
void MusicPlayer::send(uint32 b) {
|
|
if ((b & 0xF0) == 0xC0 && !_isGM && !_nativeMT32) {
|
|
b = (b & 0xFFFF00FF) | MidiDriver::_mt32ToGm[(b >> 8) & 0xFF] << 8;
|
|
}
|
|
|
|
Audio::MidiPlayer::send(b);
|
|
}
|
|
|
|
void MusicPlayer::playMIDI(const byte *data, uint32 size, bool loop) {
|
|
Common::StackLock lock(_mutex);
|
|
|
|
stopAndClear();
|
|
|
|
_buffer = new byte[size];
|
|
memcpy(_buffer, data, size);
|
|
|
|
MidiParser *parser;
|
|
|
|
if (!memcmp(data, "FORM", 4))
|
|
parser = MidiParser::createParser_XMIDI(NULL);
|
|
else
|
|
parser = MidiParser::createParser_SMF();
|
|
|
|
if (parser->loadMusic(_buffer, size)) {
|
|
parser->setTrack(0);
|
|
parser->setMidiDriver(this);
|
|
parser->setTimerRate(_driver->getBaseTempo());
|
|
parser->property(MidiParser::mpCenterPitchWheelOnUnload, 1);
|
|
parser->property(MidiParser::mpSendSustainOffOnNotesOff, 1);
|
|
|
|
_parser = parser;
|
|
|
|
syncVolume();
|
|
|
|
_isLooping = loop;
|
|
_isPlaying = true;
|
|
} else {
|
|
delete parser;
|
|
}
|
|
}
|
|
|
|
void MusicPlayer::stopAndClear() {
|
|
Common::StackLock lock(_mutex);
|
|
stop();
|
|
|
|
delete[] _buffer;
|
|
_buffer = NULL;
|
|
}
|
|
|
|
Music::Music(ArchiveReader *arc) : MusicPlayer(true), _arc(arc) {
|
|
_sequenceResIndex = -1;
|
|
}
|
|
|
|
void Music::playSequence(int16 sequenceResIndex) {
|
|
_sequenceResIndex = sequenceResIndex;
|
|
|
|
int32 resourceSize = _arc->getResourceSize(sequenceResIndex);
|
|
byte *data = new byte[resourceSize];
|
|
_arc->openResource(sequenceResIndex);
|
|
_arc->read(data, resourceSize);
|
|
_arc->closeResource();
|
|
|
|
if (!memcmp(data, "FORM", 4))
|
|
playMIDI(data, resourceSize, true); // music tracks are always looping
|
|
else
|
|
// Sanity check: this should never occur
|
|
error("playSequence: resource %d isn't XMIDI", sequenceResIndex);
|
|
|
|
delete[] data;
|
|
}
|
|
|
|
void Music::stopSequence() {
|
|
_sequenceResIndex = -1;
|
|
stopAndClear();
|
|
}
|
|
|
|
void Music::saveState(Common::WriteStream *out) {
|
|
out->writeSint16LE(_sequenceResIndex);
|
|
}
|
|
|
|
void Music::loadState(Common::ReadStream *in) {
|
|
_sequenceResIndex = in->readSint16LE();
|
|
|
|
if (_sequenceResIndex >= 0)
|
|
playSequence(_sequenceResIndex);
|
|
}
|
|
|
|
} // End of namespace Made
|