diff options
author | Jamieson Christian | 2003-05-18 23:36:38 +0000 |
---|---|---|
committer | Jamieson Christian | 2003-05-18 23:36:38 +0000 |
commit | febf19cb3f8fab19e280eda0392f0a3cc010a4a7 (patch) | |
tree | 71c30d9d1842a0a0c56ebb8aa79390760ce801e4 | |
parent | f6d1e711945718261a298904ba6012dd7c119346 (diff) | |
download | scummvm-rg350-febf19cb3f8fab19e280eda0392f0a3cc010a4a7.tar.gz scummvm-rg350-febf19cb3f8fab19e280eda0392f0a3cc010a4a7.tar.bz2 scummvm-rg350-febf19cb3f8fab19e280eda0392f0a3cc010a4a7.zip |
SMF/GMF implementation of MidiParser
svn-id: r7650
-rw-r--r-- | sound/midiparser_smf.cpp | 507 |
1 files changed, 507 insertions, 0 deletions
diff --git a/sound/midiparser_smf.cpp b/sound/midiparser_smf.cpp new file mode 100644 index 0000000000..b429f7484a --- /dev/null +++ b/sound/midiparser_smf.cpp @@ -0,0 +1,507 @@ +/* ScummVM - Scumm Interpreter + * Copyright (C) 2001-2003 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + * + * $Header$ + * + */ + +#include "midiparser.h" +#include "mididrv.h" +#include "common/util.h" + +#include <stdio.h> +#include <memory.h> + +////////////////////////////////////////////////// +// +// The Standard MIDI File version of MidiParser +// +////////////////////////////////////////////////// + +class MidiParser_SMF : public MidiParser { +protected: + byte *_data; + byte *_buffer; + uint16 _num_tracks; + byte *_tracks [16]; + + byte _active_track; + byte *_play_pos; + uint32 _play_time; + uint32 _last_event_time; + byte _running_status; // Cached MIDI command + + uint32 _ppqn; + uint32 _psec_per_tick; // Microseconds per delta tick + + int _lock; + +protected: + uint32 read4high (byte * &data) { + uint32 val = 0; + int i; + for (i = 0; i < 4; ++i) val = (val << 8) | *data++; + return val; + } + uint16 read2low (byte * &data) { + uint16 val = 0; + int i; + for (i = 0; i < 2; ++i) val |= (*data++) << (i * 8); + return val; + } + uint32 readVLQ (byte * &data); + + void compressToType0(); + void playToTime (uint32 psec, bool transmit); + void allNotesOff(); + +public: + bool loadMusic (byte *data, uint32 size); + void unloadMusic(); + + void setMidiDriver (MidiDriver *driver) { _driver = driver; } + void setTimerRate (uint32 rate) { _timer_rate = rate; } + void onTimer(); + + void setTrack (byte track); + void jumpToTick (uint32 tick); +}; + + + +////////////////////////////////////////////////// +// +// MidiParser_SMF implementation +// +// Much of this code is adapted from the XMIDI +// implementation from the exult project. +// +////////////////////////////////////////////////// + +// This is the conventional (i.e. SMF) variable length quantity +uint32 MidiParser_SMF::readVLQ (byte * &data) { + byte str; + uint32 value = 0; + int i; + + for (i = 0; i < 4; ++i) { + str = data[0]; + ++data; + value = (value << 7) | (str & 0x7F); + if (!(str & 0x80)) + break; + } + return value; +} + +void MidiParser_SMF::onTimer() { + if (_lock++) { + --_lock; + return; + } + + if (!_play_pos || !_driver) + return; + + playToTime (_play_time + _timer_rate, true); + _lock = 0; +} + +void MidiParser_SMF::playToTime (uint32 psec, bool transmit) { + uint32 delta; + uint32 end_time; + uint32 event_time; + byte *pos; + byte *oldpos; + byte event; + uint32 length; + + end_time = psec; + pos = _play_pos; + + while (true) { + oldpos = pos; + delta = readVLQ (pos); + event_time = _last_event_time + delta * _psec_per_tick; + if (event_time > end_time) { + pos = oldpos; + break; + } + + // Process the next event. + if ((pos[0] & 0xF0) >= 0x80) + event = *pos++; + else + event = _running_status; + + if (event < 0x80) { + printf ("ERROR! Bad command or running status %02X", event); + _play_pos = 0; + return; + } + + _running_status = event; + switch (event >> 4) { + case 0xC: // Program Change + case 0xD: // Channel Aftertouch + if (transmit) + _driver->send (event | (pos[0] << 8)); + ++pos; + break; + + case 0x9: // Note On + case 0x8: // Note Off + case 0xA: // Key Aftertouch + case 0xB: // Control Change + case 0xE: // Pitch Bender Change + if (transmit) + _driver->send (event | (pos[0] << 8) | (pos[1] << 16)); + pos += 2; + break; + + case 0xF: // Meta or SysEx event + switch (event & 0x0F) { + case 0x2: // Song Position Pointer + if (transmit) + _driver->send (event | (pos[0] << 8) | (pos[1] << 16)); + pos += 2; + break; + + case 0x3: // Song Select + if (transmit) + _driver->send (event | (pos[0] << 8)); + ++pos; + break; + + case 0x6: // Tune Request + case 0x8: // MIDI Timing Clock + case 0xA: // Sequencer Start + case 0xB: // Sequencer Continue + case 0xC: // Sequencer Stop + case 0xE: // Active Sensing + if (transmit) + _driver->send (event); + break; + + case 0x0: // SysEx + length = readVLQ (pos); + if (transmit) + _driver->sysEx (pos, (uint16)(length - 1)); + pos += length; + break; + + case 0xF: // META event + event = *pos++; + length = readVLQ (pos); + + if (event == 0x2F) { + // End of Track must be processed by us, + // as well as sending it to the output device. + _play_pos = 0; + if (transmit) { + _driver->metaEvent (event, pos, (uint16) length); + } + _lock = 0; + return; + } else if (event == 0x51) { + if (length >= 3) { + delta = pos[0] << 16 | pos[1] << 8 | pos[2]; + _psec_per_tick = (delta + (_ppqn >> 2)) / _ppqn; + } + } + + if (transmit) + _driver->metaEvent (event, pos, (uint16) length); + pos += length; + break; + } + } + + _last_event_time = event_time; + } + + _play_time = end_time; + _play_pos = pos; +} + +// This code was adapted from the exult methods +// XMIDI::ExtractTracks and XMIDI::ExtractTracksFromXmi +bool MidiParser_SMF::loadMusic (byte *data, uint32 size) { + uint32 len; + bool isGMD = false; // Indicates an older GMD file without block headers + byte midi_type; + uint32 total_size; + + unloadMusic(); + byte *pos = data; + + if (!memcmp (pos, "RIFF", 4)) { + // Skip the outer RIFF header. + pos += 8; + } + + if (!memcmp (pos, "MThd", 4)) { + // SMF with MTHd information. + pos += 4; + len = read4high (pos); + if (len != 6) { + printf ("Warning: MThd length 6 expected but found %d\n", (int) len); + return false; + } + + // Verify that this MIDI either is a Type 2 + // or has only 1 track. We do not support + // multitrack Type 1 files. + _num_tracks = pos[2] << 8 | pos[3]; + midi_type = pos[1]; + if (midi_type > 2 /*|| (midi_type < 2 && _num_tracks > 1)*/) { + printf ("Warning: No support for a Type %d MIDI with %d tracks\n", (int) midi_type, (int) _num_tracks); + return false; + } + _ppqn = pos[4] << 8 | pos[5]; + pos += len; + } else if (!memcmp (pos, "GMF\x1", 4)) { + // Older GMD/MUS file with no header info. + // Assume 1 track, 192 PPQN, and no MTrk headers. + isGMD = true; + midi_type = 0; + _num_tracks = 1; + _ppqn = 192; + pos += 7; // 'GMD\x1' + 3 bytes of useless (translate: unknown) information + } else { + printf ("Expected MThd or GMD header but found '%c%c%c%c' instead.\n", pos[0], pos[1], pos[2], pos[3]); + return false; + } + + // Now we identify and store the location for each track. + if (_num_tracks > 16) { + printf ("Can only handle 16 tracks but was handed %d\n", (int) _num_tracks); + return false; + } + + total_size = 0; + int tracks_read = 0; + while (tracks_read < _num_tracks) { + if (memcmp (pos, "MTrk", 4) && !isGMD) { + printf ("Position: %p ('%c')\n", pos, *pos); + printf ("Hit invalid block '%c%c%c%c' while scanning for track locations\n", pos[0], pos[1], pos[2], pos[3]); + return false; + } + + // If needed, skip the MTrk and length bytes + _tracks[tracks_read] = pos + (isGMD ? 0 : 8); + if (!isGMD) { + pos += 4; + len = read4high (pos); + total_size += len; + pos += len; + } else { + // An SMF End of Track meta event must be placed + // at the end of the stream. + data[size] = 0xFF; + data[size+1] = 0x2F; + data[size+2] = 0x00; + data[size+3] = 0x00; + } + ++tracks_read; + } + + // If this is a Type 1 MIDI, we need to now compress + // our tracks down into a single Type 0 track. + if (_buffer) { + free (_buffer); + } + + if (midi_type == 1) { + _buffer = (byte *) calloc (size, 1); + compressToType0(); + _data = _buffer; + _num_tracks = 1; + _tracks[0] = _buffer; + } else { + _data = data; + } + + // 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.... + _active_track = 255; + _psec_per_tick = (500000 + (_ppqn >> 2)) / _ppqn; // Default to 120 BPM + setTrack (0); + return true; +} + +void MidiParser_SMF::compressToType0() { + // We assume that _buffer has been allocated + // to sufficient size for this operation. + byte command_lengths[8] = { 3, 3, 3, 3, 2, 2, 3, 0 }; + byte special_lengths[16] = { 0, 1, 1, 3, 2, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0 }; + byte *track_pos[16]; + byte running_status[16]; + uint32 track_timer[16]; + uint32 delta; + int i; + + for (i = 0; i < _num_tracks; ++i) { + running_status[i] = 0; + track_pos[i] = _tracks[i]; + track_timer[i] = readVLQ (track_pos[i]); + } + + int best_i; + uint32 length; + byte *output = _buffer; + byte *pos, *pos2; + byte event; + uint32 copy_bytes; + bool write; + byte active_tracks = (byte) _num_tracks; + + while (active_tracks) { + write = true; + best_i = 255; + for (i = 0; i < _num_tracks; ++i) { + if (track_pos[i] && (best_i == 255 || track_timer[i] < track_timer[best_i])) + best_i = i; + } + if (best_i == 255) { + printf ("Premature end of tracks!\n"); + break; + } + + // Initial VLQ delta computation + delta = 0; + length = track_timer[best_i]; + for (i = 0; length; ++i) { + delta = (delta << 8) | (length & 0x7F); + length >>= 7; + } + + // Process MIDI event. + copy_bytes = 0; + pos = track_pos[best_i]; + event = *(pos++); + if (event < 0x80) + event = running_status[best_i]; + running_status[best_i] = event; + + if (command_lengths [(event >> 4) - 8] > 0) { + copy_bytes = command_lengths [(event >> 4) - 8]; + } else if (special_lengths [(event & 0x0F)] > 0) { + copy_bytes = special_lengths [(event & 0x0F)]; + } else if (event == 0xF0) { + // SysEx + pos2 = pos; + length = readVLQ (pos); + copy_bytes = 1 + (pos - pos2) + length; + } else if (event == 0xFF) { + // META + event = *(pos++); + if (event == 0x2F && active_tracks > 1) { + track_pos[best_i] = 0; + write = false; + } else { + pos2 = pos; + length = readVLQ (pos); + copy_bytes = 2 + (pos - pos2) + length; + } + if (event == 0x2F) + --active_tracks; + } else { + printf ("Bad MIDI command %02X!\n", (int) event); + track_pos[best_i] = 0; + } + + if (track_pos[best_i]) { + // Update all tracks' deltas + for (i = 0; i < _num_tracks; ++i) { + if (track_pos[i] && i != best_i) + track_timer[i] -= track_timer[best_i]; + } + + if (write) { + // Write VLQ delta + do { + *output++ = (byte) (delta & 0xFF | (delta > 0xFF ? 0x80 : 0)); + delta >>= 8; + } while (delta); + + // Write MIDI data + memcpy (output, track_pos[best_i], copy_bytes); + output += copy_bytes; + } + + // Fetch new VLQ delta for winning track + track_pos[best_i] += copy_bytes; + track_timer[best_i] = readVLQ (track_pos[best_i]); + } + } + + *output++ = 0x00; +} + +void MidiParser_SMF::allNotesOff() { + if (!_driver) + return; + + int i; + for (i = 0; i < 15; ++i) { + _driver->send (0x007BB0 | i); + } +} + +void MidiParser_SMF::unloadMusic() { + while (_lock); + ++_lock; + + _play_pos = NULL; + _data = NULL; + _num_tracks = 0; + _active_track = 255; + _play_time = 0; + _last_event_time = 0; + _running_status = 0; + allNotesOff(); + + _lock = 0; +} + +void MidiParser_SMF::setTrack (byte track) { + if (track >= _num_tracks || track == _active_track) + return; + _active_track = track; + _play_time = 0; + _last_event_time = 0; + _play_pos = _tracks[track]; + _running_status = 0; + allNotesOff(); +} + +void MidiParser_SMF::jumpToTick (uint32 tick) { + if (_active_track >= _num_tracks) + return; + _play_pos = _tracks[_active_track]; + _play_time = 0; + _last_event_time = 0; + if (tick > 0) { + printf ("jumpToTick (%ld) not completely implemented!\n", tick); + playToTime (tick * _psec_per_tick - 1, false); + } + allNotesOff(); +} + +MidiParser *MidiParser::createParser_SMF() { return new MidiParser_SMF; } |