aboutsummaryrefslogtreecommitdiff
path: root/engines/scumm/midiparser_eup.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'engines/scumm/midiparser_eup.cpp')
-rw-r--r--engines/scumm/midiparser_eup.cpp219
1 files changed, 219 insertions, 0 deletions
diff --git a/engines/scumm/midiparser_eup.cpp b/engines/scumm/midiparser_eup.cpp
new file mode 100644
index 0000000000..a9c40f99c5
--- /dev/null
+++ b/engines/scumm/midiparser_eup.cpp
@@ -0,0 +1,219 @@
+/* ScummVM - Scumm Interpreter
+ * Copyright (C) 2001-2006 The ScummVM project
+ *
+ * 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.
+ *
+ * $URL$
+ * $Id$
+ *
+ */
+
+#include "common/stdafx.h"
+#include "sound/midiparser.h"
+#include "sound/mididrv.h"
+#include "common/util.h"
+
+namespace Scumm {
+
+/**
+ * The FM-TOWNS Euphony version of MidiParser.
+ */
+class MidiParser_EUP : public MidiParser {
+protected:
+ byte _instruments[6][50]; // Two extra bytes for SysEx ID and channel #
+ byte *_instr_to_channel;
+ struct {
+ byte *enable;
+ int8 *channel;
+ int8 *volume;
+ int8 *transpose;
+ } _presets;
+ bool _loop;
+ byte _presend; // Tracks which startup implied events have been sent.
+ uint32 _base_tick; // Events times are relative to this base.
+
+protected:
+ void parseNextEvent (EventInfo &info);
+ void resetTracking();
+
+public:
+ bool loadMusic (byte *data, uint32 size);
+};
+
+
+
+//////////////////////////////////////////////////
+//
+// MidiParser_EUP implementation
+//
+//////////////////////////////////////////////////
+
+void MidiParser_EUP::parseNextEvent (EventInfo &info) {
+ byte *pos = _position._play_pos;
+
+ // FIXME: The presend is for sending init events
+ // that aren't actually in the stream. This would
+ // be for, e.g., instrument setup. Right now, we
+ // don't actually use the instruments specified
+ // in the music header. We're sending fixed GM
+ // program changes to get a reasonable "one-size-
+ // fits-all" sound until we actually support the
+ // FM synthesis capabilities of FM-TOWNS.
+ for (; _presend < 12; ++_presend) {
+ if (_instr_to_channel[_presend >> 1] >= 16)
+ continue;
+ info.start = pos;
+ info.delta = 0;
+ if (_presend & 1) {
+ byte *data = &_instruments[_presend >> 1][0];
+ data[1] = _instr_to_channel[_presend >> 1];
+ info.event = 0xF0;
+ info.ext.data = data;
+ info.length = 48;
+ } else {
+ info.event = 0xB0 | (_presend >> 1);
+ info.basic.param1 = 121;
+ info.basic.param2 = 0;
+ }
+ ++_presend;
+ return;
+ }
+
+ while (true) {
+ byte cmd = *pos;
+ if ((cmd & 0xF0) == 0x90) {
+ byte preset = pos[1];
+ byte channel = _presets.channel[preset];
+ if (channel >= 16)
+ channel = cmd & 0x0F;
+ uint16 tick = (pos[2] | ((uint16) pos[3] << 7)) + _base_tick;
+ int note = (int) pos[4] + _presets.transpose[preset];
+ int volume = (int) pos[5];
+ // HACK: Loom-Towns distaff tracks seem to
+ // contain zero-volume note events, so change
+ // those to full volume.
+ if (!volume)
+ volume = 127;
+ volume += _presets.volume[preset];
+ if (volume > 127)
+ volume = 127;
+ else if (volume < 0)
+ volume = 0;
+ pos += 6;
+ if (_presets.enable[preset]) {
+ uint16 duration = pos[1] | (pos[2] << 4);
+ info.start = pos;
+ uint32 last = _position._last_event_tick;
+ info.delta = (tick < last) ? 0 : (tick - last);
+ info.event = 0x90 | channel;
+ info.length = duration;
+ info.basic.param1 = note;
+ info.basic.param2 = volume;
+ pos += 6;
+ break;
+ }
+ pos += 6;
+ } else if (cmd == 0xF2) {
+ // This is a "measure marker" of sorts.
+ // It advances the "base time", to which
+ // all event times are relative.
+ _base_tick += (pos[3] << 7) | pos[2];
+ pos += 6;
+ } else if (cmd == 0xF8) {
+ // TODO: Implement this.
+ pos += 6;
+ } else if (cmd == 0xFD || cmd == 0xFE) {
+ // End of track.
+ if (_loop && false) {
+ // TODO: Implement this.
+ } else {
+ info.start = pos;
+ uint32 last = _position._last_event_tick;
+ info.delta = (_base_tick < last) ? 0 : (_base_tick - last);
+ info.event = 0xFF;
+ info.length = 0;
+ info.ext.type = 0x2F;
+ info.ext.data = pos;
+ break;
+ }
+ } else {
+ error("Unknown Euphony music event 0x%02X", (int) cmd);
+ memset(&info, 0, sizeof(info));
+ pos = 0;
+ break;
+ }
+ }
+ _position._play_pos = pos;
+}
+
+bool MidiParser_EUP::loadMusic (byte *data, uint32 size) {
+ unloadMusic();
+ byte *pos = data;
+ int i;
+
+ if (memcmp(pos, "SO", 2)) {
+ error("'SO' header expected but found '%c%c' instead.", pos[0], pos[1]);
+ return false;
+ }
+
+ byte numInstruments = pos[16];
+ pos += 16 + 2;
+ for (i = 0; i < numInstruments; ++i) {
+ _instruments[i][0] = 0x7C;
+ memcpy (&_instruments[i][2], pos, 48);
+ pos += 48;
+ }
+
+ // Load the prest pointers
+ _presets.enable = pos;
+ pos += 32;
+ _presets.channel = (int8 *) pos;
+ pos += 32;
+ _presets.volume = (int8 *) pos;
+ pos += 32;
+ _presets.transpose = (int8 *) pos;
+ pos += 32;
+
+ pos += 8; // Unknown bytes
+ _instr_to_channel = pos; // Instrument-to-channel mapping
+ pos += 6;
+ pos += 4; // Skip the music size for now.
+ pos++; // Unknown byte
+ byte tempo = *pos++;
+ _loop = (*pos++ != 1);
+ pos++; // Unknown byte
+
+ _num_tracks = 1;
+ _ppqn = 120;
+ _tracks[0] = pos;
+
+ // Note that we assume the original data passed in
+ // will persist beyond this call, i.e. we do NOT
+ // copy the data to our own buffer. Take warning....
+ resetTracking();
+ setTempo (1000000 * 60 / tempo);
+ setTrack (0);
+ return true;
+}
+
+void MidiParser_EUP::resetTracking() {
+ MidiParser::resetTracking();
+ _presend = 0;
+ _base_tick = 0;
+}
+
+MidiParser *MidiParser_createEUP() { return new MidiParser_EUP; }
+
+} // End of namespace Scumm