/* ScummVM - Scumm Interpreter * Copyright (C) 2001 Ludvig Strigeus * 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 "stdafx.h" #include "actor.h" #include "bundle.h" #include "imuse.h" #include "imuse_digi.h" #include "player_v2.h" #include "player_v3a.h" #include "scumm.h" #include "sound.h" #include "common/config-file.h" #include "common/timer.h" #include "common/util.h" #include "sound/mididrv.h" #include "sound/midiparser.h" #include "sound/mixer.h" #include "sound/voc.h" enum { SOUND_HEADER_SIZE = 26, SOUND_HEADER_BIG_SIZE = 26 + 8 }; struct MP3OffsetTable { /* Compressed Sound (.SO3) */ int org_offset; int new_offset; int num_tags; int compressed_size; }; class DigitalTrackInfo { public: virtual bool error() = 0; virtual int play(SoundMixer *mixer, PlayingSoundHandle *handle, int startFrame, int duration) = 0; virtual ~DigitalTrackInfo() { } }; #ifdef USE_MAD class MP3TrackInfo : public DigitalTrackInfo { private: struct mad_header _mad_header; long _size; File *_file; bool _error_flag; public: MP3TrackInfo(File *file); ~MP3TrackInfo(); bool error() { return _error_flag; } int play(SoundMixer *mixer, PlayingSoundHandle *handle, int startFrame, int duration); }; #endif #ifdef USE_VORBIS class VorbisTrackInfo : public DigitalTrackInfo { private: File *_file; OggVorbis_File _ov_file; bool _error_flag; public: VorbisTrackInfo(File *file); ~VorbisTrackInfo(); bool error() { return _error_flag; } int play(SoundMixer *mixer, PlayingSoundHandle *handle, int startFrame, int duration); }; #endif Sound::Sound(Scumm *parent) { memset(this,0,sizeof(Sound)); // palmos _scumm = parent; _nameBundleMusic = ""; _musicBundleBufFinal = NULL; _musicBundleBufOutput = NULL; _musicDisk = 0; _talkChannelHandle = 0; _current_cache = 0; _currentCDSound = 0; _sfxFile = 0; _bundle = new Bundle(); } Sound::~Sound() { delete _sfxFile; delete _bundle; } void Sound::addSoundToQueue(int sound) { if (!(_scumm->_features & GF_DIGI_IMUSE)) { _scumm->VAR(_scumm->VAR_LAST_SOUND) = sound; _scumm->ensureResourceLoaded(rtSound, sound); addSoundToQueue2(sound); } else { // WARNING ! This may break something, maybe this sould be put inside if (_gameID == GID_FT) ? // But why addSoundToQueue should not queue sound ? _scumm->ensureResourceLoaded(rtSound, sound); addSoundToQueue2(sound); } } void Sound::addSoundToQueue2(int sound) { if (_soundQue2Pos < 10) { _soundQue2[_soundQue2Pos++] = sound; } } void Sound::processSoundQues() { int i = 0, d, num; int data[16]; processSfxQueues(); while (_soundQue2Pos) { d = _soundQue2[--_soundQue2Pos]; if (d) playSound(d); } while (i < _soundQuePos) { num = _soundQue[i++]; if (i + num > _soundQuePos) { warning("processSoundQues: invalid num value"); break; } memset(data, 0, sizeof(data)); if (num > 0) { for (int j = 0; j < num; j++) data[j] = _soundQue[i + j]; i += num; #if 0 debug(1, "processSoundQues(%d,%d,%d,%d,%d,%d,%d,%d,%d)", data[0] >> 8, data[0] & 0xFF, data[1], data[2], data[3], data[4], data[5], data[6], data[7] ); #endif if (_scumm->_features & GF_DIGI_IMUSE) { if (_scumm->_imuseDigital) _scumm->_imuseDigital->doCommand(data[0], data[1], data[2], data[3], data[4], data[5], data[6], data[7]); } else if (_scumm->_imuse) { _scumm->VAR(_scumm->VAR_SOUNDRESULT) = (short)_scumm->_imuse->doCommand (num, data); } } } _soundQuePos = 0; } void Sound::playSound(int soundID) { byte *ptr; char *sound; int size; int rate; byte flags = SoundMixer::FLAG_UNSIGNED | SoundMixer::FLAG_AUTOFREE; debug(3, "playSound #%d (room %d)", soundID, _scumm->getResourceRoomNr(rtSound, soundID)); ptr = _scumm->getResourceAddress(rtSound, soundID); if (!ptr) { return; } if (READ_UINT32(ptr) == MKID('iMUS')){ assert(_scumm->_musicEngine); _scumm->_musicEngine->startSound(soundID); } else if (READ_UINT32(ptr) == MKID('Crea')) { assert(_scumm->_musicEngine); _scumm->_musicEngine->startSound(soundID); } /* // XMIDI else if ((READ_UINT32(ptr) == MKID('MIDI')) && (_scumm->_features & GF_HUMONGOUS)) { // Pass XMIDI on to IMuse unprocessed. // IMuse can handle XMIDI resources now. } else if (READ_UINT32(ptr) == MKID('ADL ')) { // played as MIDI, just to make perhaps the later use // of WA possible (see "else if" with GF_OLD256 below) } */ else if (READ_UINT32(ptr) == MKID('SOUN')) { ptr += 24; int track = ptr[0]; int loops = ptr[1]; int start = (ptr[2] * 60 + ptr[3]) * 75 + ptr[4]; playCDTrack(track, loops == 0xff ? -1 : loops, start, 0); _currentCDSound = soundID; } // Support for SFX in Monkey Island 1, Mac version // This is rather hackish right now, but works OK. SFX are not sounding // 100% correct, though, not sure right now what is causing this. else if (READ_UINT32(ptr) == MKID('Mac1')) { // Read info from the header size = READ_BE_UINT32(ptr+0x60); rate = READ_BE_UINT16(ptr+0x64); // Skip over the header (fixed size) ptr += 0x72; // Allocate a sound buffer, copy the data into it, and play sound = (char *)malloc(size); memcpy(sound, ptr, size); _scumm->_mixer->playRaw(NULL, sound, size, rate, flags, soundID); } // Support for Putt-Putt sounds - very hackish, too 8-) else if (READ_UINT32(ptr) == MKID('DIGI')) { // TODO - discover what data the first chunk, HSHD, contains // it might be useful here. ptr += 8 + READ_BE_UINT32(ptr+12); if (READ_UINT32(ptr) != MKID('SDAT')) return; // abort size = READ_BE_UINT32(ptr+4) - 8; // FIXME - what value here ?!? 11025 is just a guess based on strings in w32 bin, prev guess 8000 rate = 11025; // Allocate a sound buffer, copy the data into it, and play sound = (char *)malloc(size); memcpy(sound, ptr + 8, size); _scumm->_mixer->playRaw(NULL, sound, size, rate, flags, soundID); } else if (READ_UINT32(ptr) == MKID('MRAW')) { // pcm music in 3DO humongous games // TODO play via imuse so isSoundRunning can properly report music value? ptr += 8 + READ_BE_UINT32(ptr+12); if (READ_UINT32(ptr) != MKID('SDAT')) return; size = READ_BE_UINT32(ptr+4) - 8; rate = 22050; flags = SoundMixer::FLAG_AUTOFREE; // Allocate a sound buffer, copy the data into it, and play sound = (char *)malloc(size); memcpy(sound, ptr + 8, size); _scumm->_mixer->playRaw(NULL, sound, size, rate, flags, soundID); } // Support for sampled sound effects in Monkey Island 1 and 2 else if (READ_UINT32(ptr) == MKID('SBL ')) { debug(2, "Using SBL sound effect"); // SBL resources essentially contain VOC sound data. // There are at least two main variants: in one, // there are two subchunks AUhd and AUdt, in the other // the chunks are called WVhd and WVdt. Besides that, // the two variants seem pretty similiar. // The first subchunk (AUhd resp. WVhd) seems to always // contain three bytes (00 00 80) of unknown meaning. // After that, a second subchunk contains VOC data. // Two real examples: // // 53 42 4c 20 00 00 11 ae |SBL ....| // 41 55 68 64 00 00 00 03 |AUhd....| // 00 00 80 41 55 64 74 00 |...AUdt.| // 00 11 9b 01 96 11 00 a6 |........| // 00 7f 7f 7e 7e 7e 7e 7e |...~~~~~| // 7e 7f 7f 80 80 7f 7f 7f |~.......| // 7f 80 80 7f 7e 7d 7d 7e |....~}}~| // 7e 7e 7e 7e 7e 7e 7e 7f |~~~~~~~.| // // And from the non-interactive Sam & Max demo: // // 53 42 4c 20 00 01 15 6e |SBL ...n| // 57 56 68 64 00 00 00 03 |WVhd....| // 00 00 80 57 56 64 74 00 |...WVdt.| // 01 15 5b 01 56 15 01 a6 |..[.V...| // 00 80 80 80 80 80 80 80 |........| // 80 80 80 80 80 80 80 80 |........| // 80 80 80 80 80 80 80 80 |........| // 80 80 80 80 80 80 80 80 |........| #if 1 // Fingolfin says: after eyeballing a single SEGA // SBL resource, it would seem as if the content of the // data subchunk (AUdt) is XORed with 0x16. At least // then a semi-sane VOC header is revealed, with // a sampling rate of ~25000 Hz (does that make sense?). // I'll add some code to test that theory for now. if (_scumm->_gameId == GID_MONKEY_SEGA) { size = READ_BE_UINT32(ptr + 4) - 27; for (int i = 0; i < size; i++) ptr[27 + i] ^= 0x16; } VocBlockHeader &voc_block_hdr = *(VocBlockHeader *)(ptr + 27); assert(voc_block_hdr.blocktype == 1); size = voc_block_hdr.size[0] + (voc_block_hdr.size[1] << 8) + (voc_block_hdr.size[2] << 16) - 2; rate = getSampleRateFromVOCRate(voc_block_hdr.sr); assert(voc_block_hdr.pack == 0); #else // FIXME: SBL resources are apparently horribly // distorted on segacd even though it shares the same // header etc. So don't try to play them for now. if (_scumm->_gameId == GID_MONKEY_SEGA) { return; } if (READ_UINT32(ptr + 8) == MKID('WVhd')) rate = 11025; else rate = 8000; size = READ_BE_UINT32(ptr + 4) - 27; #endif // Allocate a sound buffer, copy the data into it, and play sound = (char *)malloc(size); memcpy(sound, ptr + 33, size); _scumm->_mixer->playRaw(NULL, sound, size, rate, flags, soundID); } else if (_scumm->_features & GF_FMTOWNS) { size = READ_LE_UINT32(ptr); rate = 11025; int type = *(ptr + 0x0D); switch(type) { case 0: { // Sound effect int numInstruments = *(ptr + 0x14); ptr += 0x16; size -= 0x16; while (numInstruments--) { int waveSize = READ_LE_UINT32(ptr + 0x0C); int loopStart = READ_LE_UINT32(ptr + 0x10); int loopEnd = READ_LE_UINT32(ptr + 0x14); // it's not exactly * 10, maybe it's not even linear, but * 10 sounds ok. rate = READ_LE_UINT32(ptr + 0x18) * 10; ptr += 0x20; size -= 0x20; if (size < waveSize) { warning("Wrong wave size in sound #%i: %i", soundID, waveSize); waveSize = size; } sound = (char *)malloc(waveSize); for (int x = 0; x < waveSize; x++) { int bit = *ptr++; if (bit < 0x80) sound[x] = 0x7F - bit; else sound[x] = bit; } size -= waveSize; if (loopEnd > 0) flags |= SoundMixer::FLAG_LOOP; _scumm->_mixer->playRaw(NULL, sound, waveSize, rate, flags, soundID, 255, 0, loopStart, loopEnd); } break; } case 1: { // Music (Euphony format) if (_scumm->_musicEngine) _scumm->_musicEngine->startSound (soundID); break; } case 2: { // CD track resource ptr += 0x16; int track = ptr[0]; int loops = ptr[1]; int start = (ptr[2] * 60 + ptr[3]) * 75 + ptr[4]; int end = (ptr[5] * 60 + ptr[6]) * 75 + ptr[7]; if (soundID == _currentCDSound) if (pollCD() == 1) return; playCDTrack(track, loops == 0xff ? -1 : loops, start, end - start); _currentCDSound = soundID; break; } } } else if ((_scumm->_gameId == GID_LOOM) && (_scumm->_features & GF_MACINTOSH)) { // Mac version of Loom uses yet another sound format /* playSound #9 (room 70) 000000: 55 00 00 45 73 6f 00 64 01 00 00 00 00 00 00 00 |U..Eso.d........| 000010: 00 05 00 8e 2a 8f 2d 1c 2a 8f 2a 8f 2d 1c 00 28 |....*.-.*.*.-..(| 000020: 00 31 00 3a 00 43 00 4c 00 01 00 00 00 01 00 64 |.1.:.C.L.......d| 000030: 5a 00 01 00 00 00 01 00 64 00 00 01 00 00 00 01 |Z.......d.......| 000040: 00 64 5a 00 01 00 00 00 01 00 64 5a 00 01 00 00 |.dZ.......dZ....| 000050: 00 01 00 64 00 00 00 00 00 00 00 07 00 00 00 64 |...d...........d| 000060: 64 00 00 4e 73 6f 00 64 01 00 00 00 00 00 00 00 |d..Nso.d........| 000070: 00 05 00 89 3d 57 2d 1c 3d 57 3d 57 2d 1c 00 28 |....=W-.=W=W-..(| playSound #16 (room 69) 000000: dc 00 00 a5 73 6f 00 64 01 00 00 00 00 00 00 00 |....so.d........| 000010: 00 05 00 00 2a 8f 03 e8 03 e8 03 e8 03 e8 00 28 |....*..........(| 000020: 00 79 00 7f 00 85 00 d6 00 01 00 00 00 19 01 18 |.y..............| 000030: 2f 00 18 00 01 18 32 00 18 00 01 18 36 00 18 00 |/.....2.....6...| 000040: 01 18 3b 00 18 00 01 18 3e 00 18 00 01 18 42 00 |..;.....>.....B.| 000050: 18 00 01 18 47 00 18 00 01 18 4a 00 18 00 01 18 |....G.....J.....| 000060: 4e 00 10 00 01 18 53 00 10 00 01 18 56 00 10 00 |N.....S.....V...| 000070: 01 18 5a 00 10 00 02 28 5f 00 01 00 00 00 00 00 |..Z....(_.......| */ } else if ((_scumm->_features & GF_MACINTOSH) && (_scumm->_gameId == GID_INDY3) && (ptr[26] == 0)) { size = READ_BE_UINT16(ptr + 12); rate = 3579545 / READ_BE_UINT16(ptr + 20); sound = (char *)malloc(size); int vol = ptr[24] * 4; memcpy(sound,ptr + READ_BE_UINT16(ptr + 8), size); _scumm->_mixer->playRaw(NULL, sound, size, rate, SoundMixer::FLAG_AUTOFREE, soundID, vol, 0); } else if ((_scumm->_features & GF_AMIGA) && (_scumm->_version <= 2) && READ_BE_UINT16(ptr + 14) == 0x0880) { size = READ_BE_UINT16(ptr + 6); int start = READ_BE_UINT16(ptr + 8); start += 10; rate = 11000; int vol = 255; int i = 0; while (i < start) { if ((READ_BE_UINT16(ptr) == 0x357c) && (READ_BE_UINT16(ptr + 4) == 6)) rate = 3579545 / READ_BE_UINT16(ptr + 2); if ((READ_BE_UINT16(ptr) == 0x357c) && (READ_BE_UINT16(ptr + 4) == 8)) vol = READ_BE_UINT16(ptr + 2) * 4; ptr += 2; i += 2; } sound = (char *)malloc(size); memcpy(sound, ptr, size); _scumm->_mixer->playRaw(NULL, sound, size, rate, SoundMixer::FLAG_AUTOFREE, soundID, vol, 0); } else { if (_scumm->_gameId == GID_MONKEY_VGA || _scumm->_gameId == GID_MONKEY_EGA) { // Sound is currently not supported at all in the amiga versions of these games if (_scumm->_features & GF_AMIGA) return; // Works around the fact that in some places in MonkeyEGA/VGA, // the music is never explicitly stopped. // Rather it seems that starting a new music is supposed to // automatically stop the old song. if (_scumm->_imuse) { if (READ_UINT32(ptr) != MKID('ASFX')) _scumm->_imuse->stopAllSounds(); } } if (_scumm->_musicEngine) { _scumm->_musicEngine->startSound(soundID); } } } void Sound::processSfxQueues() { Actor *a; int act; bool b, finished; if (_talk_sound_mode != 0) { if (_talk_sound_mode & 1) startTalkSound(_talk_sound_a1, _talk_sound_b1, 1); if (_talk_sound_mode & 2) { startTalkSound(_talk_sound_a2, _talk_sound_b2, 2, &_talkChannelHandle); } _talk_sound_mode = 0; } if ((_sfxMode & 2) && _scumm->VAR(_scumm->VAR_TALK_ACTOR)) { act = _scumm->VAR(_scumm->VAR_TALK_ACTOR); finished = !_talkChannelHandle; if (act != 0 && (uint) act < 0x80 && !_scumm->_string[0].no_talk_anim) { a = _scumm->derefActor(act, "processSfxQueues"); if (a->isInCurrentRoom() && (finished || !_endOfMouthSync)) { b = true; if (!finished) b = isMouthSyncOff(_curSoundPos); if (_mouthSyncMode != b) { _mouthSyncMode = b; if (_talk_sound_frame != -1) { a->startAnimActor(_talk_sound_frame); _talk_sound_frame = -1; } else a->startAnimActor(b ? a->talkStopFrame : a->talkStartFrame); } } } if (finished && _scumm->_talkDelay == 0) { _scumm->stopTalk(); } } if (_sfxMode & 1) { if (isSfxFinished()) { _sfxMode &= ~1; } } } static int compareMP3OffsetTable(const void *a, const void *b) { return ((const MP3OffsetTable *)a)->org_offset - ((const MP3OffsetTable *)b)->org_offset; } void Sound::startTalkSound(uint32 offset, uint32 b, int mode, PlayingSoundHandle *handle) { int num = 0, i; int size; byte *sound; if (_sfxFile->isOpen() == false) { warning("startTalkSound: SFX file is not open"); return; } // FIXME hack until more is known // the size of the data after the sample isn't known // 64 is just a guess if (_scumm->_features & GF_HUMONGOUS) { // SKIP TLKB (8) TALK (8) HSHD (24) and SDAT (8) _sfxFile->seek(offset + 48, SEEK_SET); sound = (byte *)malloc(b - 64); _sfxFile->read(sound, b - 64); _scumm->_mixer->playRaw(handle, sound, b - 64, 11025, SoundMixer::FLAG_UNSIGNED | SoundMixer::FLAG_AUTOFREE); return; } // Some games frequently assume that starting one sound effect will // automatically stop any other that may be playing at that time. So // that is what we do here, but we make an exception for speech. // // Do any other games than these need this hack? // // HACK: Checking for script 99 in Sam & Max is to keep Conroy's song // from being interrupted. int talkChannel = (_talkChannelHandle - 1); if (mode == 1 && (_scumm->_gameId == GID_TENTACLE || (_scumm->_gameId == GID_SAMNMAX && !_scumm->isScriptRunning(99)))) { for (i = 0; i < _scumm->_mixer->NUM_CHANNELS; i++) { if (i != talkChannel) { _scumm->_mixer->stopChannel(i); } } } if (b > 8) { num = (b - 8) >> 1; } if (offset_table != NULL) { MP3OffsetTable *result = NULL, key; key.org_offset = offset; result = (MP3OffsetTable *)bsearch(&key, offset_table, num_sound_effects, sizeof(MP3OffsetTable), compareMP3OffsetTable); if (result == NULL) { warning("startTalkSound: did not find sound at offset %d !", offset); return; } if (2 * num != result->num_tags) { warning("startTalkSound: number of tags do not match (%d - %d) !", b, result->num_tags); num = result->num_tags; } offset = result->new_offset; size = result->compressed_size; } else { offset += 8; size = -1; } _sfxFile->seek(offset, SEEK_SET); assert(num+1 < (int)ARRAYSIZE(_mouthSyncTimes)); for (i = 0; i < num; i++) _mouthSyncTimes[i] = _sfxFile->readUint16BE(); _mouthSyncTimes[i] = 0xFFFF; _sfxMode |= mode; _curSoundPos = 0; _mouthSyncMode = true; startSfxSound(_sfxFile, size, handle); } void Sound::stopTalkSound() { if (_sfxMode & 2) { _scumm->_mixer->stopHandle(_talkChannelHandle); _sfxMode &= ~2; } } bool Sound::isMouthSyncOff(uint pos) { uint j; bool val = true; uint16 *ms = _mouthSyncTimes; _endOfMouthSync = false; do { val = !val; j = *ms++; if (j == 0xFFFF) { _endOfMouthSync = true; break; } } while (pos > j); return val; } int Sound::isSoundRunning(int sound) const { if (sound == _currentCDSound) return pollCD(); if (_scumm->_features & GF_HUMONGOUS) { if (sound == -2) { return isSfxFinished(); } else if (sound == -1) { // getSoundStatus(), with a -1, will return the // ID number of the first active music it finds. // TODO handle MRAW (pcm music) in humongous games return _scumm->_imuse->getSoundStatus(sound); } } if (isSoundInQueue(sound)) return 1; if (!_scumm->isResourceLoaded(rtSound, sound)) return 0; if (_scumm->_musicEngine) return _scumm->_musicEngine->getSoundStatus(sound); return 0; } /** * Check whether the sound resource with the specified ID is still * used. This is invoked by Scumm::isResourceInUse, to determine * which resources can be expired from memory. * Technically, this works very similar to isSoundRunning, however it * calls IMuse::get_sound_active() instead of IMuse::getSoundStatus(). * The difference between those two is in how they treat sounds which * are being faded out: get_sound_active() returns true even when the * sound is being faded out, while getSoundStatus() returns false in * that case. */ bool Sound::isSoundInUse(int sound) const { if (sound == _currentCDSound) return pollCD() != 0; if (isSoundInQueue(sound)) return true; if (!_scumm->isResourceLoaded(rtSound, sound)) return false; if (_scumm->_imuseDigital) return (_scumm->_imuseDigital->getSoundStatus(sound) != 0); if (_scumm->_imuse) return _scumm->_imuse->get_sound_active(sound); return false; } bool Sound::isSoundInQueue(int sound) const { int i, j, num; int16 table[16]; i = _soundQue2Pos; while (i--) { if (_soundQue2[i] == sound) return 1; } i = 0; while (i < _soundQuePos) { num = _soundQue[i++]; memset(table, 0, sizeof(table)); if (num > 0) { for (j = 0; j < num; j++) table[j] = _soundQue[i + j]; i += num; if (table[0] == 0x10F && table[1] == 8 && table[2] == sound) return 1; } } return 0; } void Sound::stopSound(int a) { int i; if (a != 0 && a == _currentCDSound) { _currentCDSound = 0; stopCD(); stopCDTimer(); } if (_scumm->_features & GF_FMTOWNS) { _scumm->_mixer->stopID(a); } else if (_scumm->_musicEngine) { _scumm->_musicEngine->stopSound(a); } for (i = 0; i < 10; i++) if (_soundQue2[i] == a) _soundQue2[i] = 0; } void Sound::stopAllSounds() { if (_currentCDSound != 0) { _currentCDSound = 0; stopCD(); stopCDTimer(); } // Clear the (secondary) sound queue _soundQue2Pos = 0; memset(_soundQue2, 0, sizeof(_soundQue2)); if (_scumm->_musicEngine) { _scumm->_musicEngine->stopAllSounds(); } if (_scumm->_imuse) { // FIXME: Maybe we could merge this call to clear_queue() // into IMuse::stopAllSounds() ? _scumm->_imuse->clear_queue(); } // Stop all SFX if (!_scumm->_imuseDigital) { _scumm->_mixer->stopAll(); } } void Sound::soundKludge(int *list, int num) { int i; if (list[0] == -1) { processSoundQues(); return; } if ((_soundQuePos + num) > 0x100) { // FIXME: temporarily changed this to an error to help track down what // is causing the sound queue overflows(in particular, to figure out // the room/script/offset where the bug occurs). Please report your // findings to Fingolfin. // Reverting to warning for now room 11, script 2016 offset 0x7Af9 was // what it error'd on here warning("Sound queue buffer overflow (%d + %d = %d)", _soundQuePos, num, _soundQuePos+num); return; } _soundQue[_soundQuePos++] = num; for (i = 0; i < num; i++) { _soundQue[_soundQuePos++] = list[i]; } } void Sound::talkSound(uint32 a, uint32 b, int mode, int frame) { if (mode == 1) { _talk_sound_a1 = a; _talk_sound_b1 = b; } else { _talk_sound_a2 = a; _talk_sound_b2 = b; } _talk_sound_frame = frame; _talk_sound_mode |= mode; } /* The sound code currently only supports General Midi. * General Midi is used in Day Of The Tentacle. * Roland music is also playable, but doesn't sound well. * A mapping between roland instruments and GM instruments * is needed. */ void Sound::setupSound() { if (_scumm->_imuse) { _scumm->_imuse->setBase(_scumm->res.address[rtSound]); _scumm->_imuse->setMasterVolume(_sound_volume_master); _scumm->_imuse->set_music_volume(_sound_volume_music); } _scumm->_mixer->setVolume(_sound_volume_sfx * _sound_volume_master / 255); _scumm->_mixer->setMusicVolume(_sound_volume_music); delete _sfxFile; _sfxFile = openSfxFile(); } void Sound::pauseSounds(bool pause) { if (_scumm->_imuse) _scumm->_imuse->pause(pause); // Don't pause sounds if the game isn't active // FIXME - this is quite a nasty hack, replace with something cleaner, and w/o // having to access member vars directly! if (!_scumm->_roomResource) return; _soundsPaused = pause; _scumm->_mixer->pauseAll(pause); _scumm->_sound->pauseBundleMusic(pause); if (_scumm->_imuseDigital) { _scumm->_imuseDigital->pause(pause); } if ((_scumm->_features & GF_AUDIOTRACKS) && _scumm->VAR(_scumm->VAR_MUSIC_TIMER) > 0) { if (pause) stopCDTimer(); else startCDTimer(); } } void Sound::startSfxSound(File *file, int file_size, PlayingSoundHandle *handle) { char ident[8]; uint size = 0; int rate, comp; byte *data; if (_soundsPaused || _scumm->_noDigitalSamples) return; if (file_size > 0) { if (_vorbis_mode) { playSfxSound_Vorbis(file, file_size, handle); } else { #ifdef USE_MAD _scumm->_mixer->playMP3(handle, file, file_size); #endif } return; } if (file->read(ident, 8) != 8) goto invalid; if (!memcmp(ident, "VTLK", 4)) { file->seek(SOUND_HEADER_BIG_SIZE - 8, SEEK_CUR); } else if (!memcmp(ident, "Creative", 8)) { file->seek(SOUND_HEADER_SIZE - 8, SEEK_CUR); } else { invalid:; warning("startSfxSound: invalid header"); return; } VocBlockHeader voc_block_hdr; file->read(&voc_block_hdr, sizeof(voc_block_hdr)); if (voc_block_hdr.blocktype != 1) { warning("startSfxSound: Expecting block_type == 1, got %d", voc_block_hdr.blocktype); return; } size = voc_block_hdr.size[0] + (voc_block_hdr.size[1] << 8) + (voc_block_hdr.size[2] << 16) - 2; rate = getSampleRateFromVOCRate(voc_block_hdr.sr); comp = voc_block_hdr.pack; if (comp != 0) { warning("startSfxSound: Unsupported compression type %d", comp); return; } data = (byte *)malloc(size); if (data == NULL) { error("startSfxSound: out of memory"); } if (file->read(data, size) != size) { /* no need to free the memory since error will shut down */ error("startSfxSound: cannot read %d bytes", size); } playSfxSound(data, size, rate, true, handle); } File *Sound::openSfxFile() { char buf[256]; File *file = new File(); /* Try opening the file <_exe_name>.sou first, eg tentacle.sou. * That way, you can keep .sou files for multiple games in the * same directory */ offset_table = NULL; #ifdef USE_MAD sprintf(buf, "%s.so3", _scumm->getExeName()); if (!file->open(buf, _scumm->getGameDataPath())) { file->open("monster.so3", _scumm->getGameDataPath()); } if (file->isOpen()) _vorbis_mode = false; #endif #ifdef USE_VORBIS if (!file->isOpen()) { sprintf(buf, "%s.sog", _scumm->getExeName()); if (!file->open(buf, _scumm->getGameDataPath())) file->open("monster.sog", _scumm->getGameDataPath()); if (file->isOpen()) _vorbis_mode = true; } #endif if (file->isOpen()) { /* Now load the 'offset' index in memory to be able to find the MP3 data The format of the .SO3 file is easy : - number of bytes of the 'index' part - N times the following fields (4 bytes each) : + offset in the original sound file + offset of the MP3 data in the .SO3 file WITHOUT taking into account the index field and the 'size' field + the number of 'tags' + the size of the MP3 data - and then N times : + the tags + the MP3 data */ int size, compressed_offset; MP3OffsetTable *cur; compressed_offset = file->readUint32BE(); offset_table = (MP3OffsetTable *) malloc(compressed_offset); num_sound_effects = compressed_offset / 16; size = compressed_offset; cur = offset_table; while (size > 0) { cur[0].org_offset = file->readUint32BE(); cur[0].new_offset = file->readUint32BE() + compressed_offset + 4; /* The + 4 is to take into accound the 'size' field */ cur[0].num_tags = file->readUint32BE(); cur[0].compressed_size = file->readUint32BE(); size -= 4 * 4; cur++; } return file; } sprintf(buf, "%s.sou", _scumm->getExeName()); if (!file->open(buf, _scumm->getGameDataPath())) { file->open("monster.sou", _scumm->getGameDataPath()); } if (!file->isOpen()) { sprintf(buf, "%s.tlk", _scumm->getExeName()); file->open(buf, _scumm->getGameDataPath(), 1, 0x69); } return file; } bool Sound::isSfxFinished() const { return !_scumm->_mixer->hasActiveSFXChannel(); } uint32 Sound::decode12BitsSample(byte *src, byte **dst, uint32 size, bool stereo) { uint32 s_size = (size / 3) * 4; uint32 loop_size = s_size / 4; if (stereo) { s_size *= 2; } byte *ptr = *dst = (byte *)malloc(s_size); uint32 tmp; while (loop_size--) { byte v1 = *src++; byte v2 = *src++; byte v3 = *src++; tmp = ((((v2 & 0x0f) << 8) | v1) << 4) - 0x8000; *ptr++ = (byte)((tmp >> 8) & 0xff); *ptr++ = (byte)(tmp & 0xff); if (stereo) { *ptr++ = (byte)((tmp >> 8) & 0xff); *ptr++ = (byte)(tmp & 0xff); } tmp = ((((v2 & 0xf0) << 4) | v3) << 4) - 0x8000; *ptr++ = (byte)((tmp >> 8) & 0xff); *ptr++ = (byte)(tmp & 0xff); if (stereo) { *ptr++ = (byte)((tmp >> 8) & 0xff); *ptr++ = (byte)(tmp & 0xff); } } return s_size; } static void music_handler(void *refCon) { Sound *sound = (Sound *)refCon; sound->bundleMusicHandler(g_scumm); } void Sound::playBundleMusic(const char *song) { if (_scumm->_silentDigitalImuse) { return; } if (_nameBundleMusic[0] == 0) { _outputMixerSize = 66150; // ((22050 * 2 * 2) / 4) * 3 if (_scumm->_gameId == GID_CMI) { char bunfile[20]; sprintf(bunfile, "musdisk%d.bun", _scumm->VAR(_scumm->VAR_CURRENTDISK)); if (_musicDisk != _scumm->VAR(_scumm->VAR_CURRENTDISK)) _bundle->closeMusicFile(); if (_bundle->openMusicFile(bunfile, _scumm->getGameDataPath()) == false) { if (_bundle->openMusicFile("music.bun", _scumm->getGameDataPath()) == false) { _outputMixerSize = 0; return; } } _musicDisk = (byte)_scumm->VAR(_scumm->VAR_CURRENTDISK); _outputMixerSize = 88140; // ((22050 * 2 * 2) } else { if (_bundle->openMusicFile("digmusic.bun", _scumm->getGameDataPath()) == false) return; } _musicBundleBufFinal = (byte *)malloc(_outputMixerSize); _musicBundleBufOutput = (byte *)malloc(((_outputMixerSize / 0x2000) + 1) * _outputMixerSize); _currentSampleBundleMusic = 0; _offsetSampleBundleMusic = 0; _offsetBufBundleMusic = 0; _bundleMusicPosition = 0; _pauseBundleMusic = false; _musicBundleToBeRemoved = false; _musicBundleToBeChanged = false; _bundleMusicTrack = 0; _numberSamplesBundleMusic = _bundle->getNumberOfMusicSamplesByName(song); _nameBundleMusic = song; _scumm->_timer->installProcedure(&music_handler, 1000000, this); return; } if (strcmp(_nameBundleMusic, song) != 0) { _newNameBundleMusic = song; _musicBundleToBeRemoved = false; _musicBundleToBeChanged = true; } } void Sound::pauseBundleMusic(bool state) { _pauseBundleMusic = state; } void Sound::stopBundleMusic() { _musicBundleToBeRemoved = true; } void Sound::bundleMusicHandler(Scumm *scumm) { byte *ptr; int32 l, num = _numberSamplesBundleMusic, length, k; int32 rate = 22050; int32 tag, size = -1, header_size = 0; if (_pauseBundleMusic) return; if (_musicBundleToBeRemoved) { _scumm->_timer->releaseProcedure(&music_handler); _nameBundleMusic = ""; _scumm->_mixer->stopChannel(_bundleMusicTrack); if (_musicBundleBufFinal) { free(_musicBundleBufFinal); _musicBundleBufFinal = NULL; } if (_musicBundleBufOutput) { free(_musicBundleBufOutput); _musicBundleBufOutput = NULL; } return; } if (_musicBundleToBeChanged) { _nameBundleMusic = _newNameBundleMusic; _numberSamplesBundleMusic = _bundle->getNumberOfMusicSamplesByName(_nameBundleMusic); _currentSampleBundleMusic = 0; _offsetSampleBundleMusic = 0; _offsetBufBundleMusic = 0; _musicBundleToBeChanged = false; _bundleMusicPosition = 0; } ptr = _musicBundleBufOutput; for (k = 0, l = _currentSampleBundleMusic; l < num; k++) { length = _bundle->decompressMusicSampleByName(_nameBundleMusic, l, (_musicBundleBufOutput + ((k * 0x2000) + _offsetBufBundleMusic))); _offsetSampleBundleMusic += length; if (l == 0) { tag = READ_BE_UINT32(ptr); ptr += 4; if (tag != MKID_BE('iMUS')) { warning("Decompression of bundle song failed"); _musicBundleToBeRemoved = true; return; } ptr += 12; while (tag != MKID_BE('DATA')) { tag = READ_BE_UINT32(ptr); ptr += 4; switch(tag) { case MKID_BE('FRMT'): ptr += 12; _bundleMusicSampleBits = READ_BE_UINT32(ptr); ptr += 4; rate = READ_BE_UINT32(ptr); ptr += 4; _bundleSampleChannels = READ_BE_UINT32(ptr); ptr += 4; break; case MKID_BE('TEXT'): case MKID_BE('REGN'): case MKID_BE('STOP'): case MKID_BE('JUMP'): case MKID_BE('SYNC'): size = READ_BE_UINT32(ptr); ptr += size + 4; break; case MKID_BE('DATA'): size = READ_BE_UINT32(ptr); ptr += 4; break; default: error("Unknown sound header %c%c%c%c", (byte)(tag >> 24), (byte)(tag >> 16), (byte)(tag >> 8), (byte)tag); } } if (size < 0) { warning("Decompression sound failed (no size field)"); _musicBundleToBeRemoved = true; return; } header_size = (ptr - _musicBundleBufOutput); } l++; _currentSampleBundleMusic = l; if (_offsetSampleBundleMusic >= _outputMixerSize + header_size) { memcpy(_musicBundleBufFinal, (_musicBundleBufOutput + header_size), _outputMixerSize); _offsetBufBundleMusic = _offsetSampleBundleMusic - _outputMixerSize - header_size; memcpy(_musicBundleBufOutput, (_musicBundleBufOutput + (_outputMixerSize + header_size)), _offsetBufBundleMusic); _offsetSampleBundleMusic = _offsetBufBundleMusic; break; } } if (_currentSampleBundleMusic == num) { _currentSampleBundleMusic = 0; _offsetSampleBundleMusic = 0; _offsetBufBundleMusic = 0; _bundleMusicPosition = 0; } ptr = _musicBundleBufFinal; byte *buffer = NULL; uint32 final_size; if (_bundleMusicSampleBits == 12) { final_size = decode12BitsSample(ptr, &buffer, _outputMixerSize, false); } else if (_bundleMusicSampleBits == 16) { buffer = (byte *)malloc(_outputMixerSize); final_size = _outputMixerSize; memcpy(buffer, ptr, _outputMixerSize); } else { warning("Sound::bundleMusicHandler TODO: more newStream options..."); return; } _bundleMusicPosition += final_size; if (_bundleMusicTrack == 0) { _scumm->_mixer->newStream(&_bundleMusicTrack, buffer, final_size, rate, SoundMixer::FLAG_16BITS | SoundMixer::FLAG_STEREO, 300000); } else { _scumm->_mixer->appendStream(_bundleMusicTrack, buffer, final_size); } free(buffer); } void Sound::playBundleSound(char *sound, PlayingSoundHandle *handle) { byte *ptr = 0, *orig_ptr = 0; byte *final; bool result; if (_scumm->_noDigitalSamples) return; if (_scumm->_gameId == GID_CMI) { char voxfile[20]; sprintf(voxfile, "voxdisk%d.bun", _scumm->VAR(_scumm->VAR_CURRENTDISK)); if (_voiceDisk != _scumm->VAR(_scumm->VAR_CURRENTDISK)) _bundle->closeVoiceFile(); result = _bundle->openVoiceFile(voxfile, _scumm->getGameDataPath()); if (result == false) result = _bundle->openVoiceFile("voice.bun", _scumm->getGameDataPath()); _voiceDisk = (byte)_scumm->VAR(_scumm->VAR_CURRENTDISK); } else if (_scumm->_gameId == GID_DIG) result = _bundle->openVoiceFile("digvoice.bun", _scumm->getGameDataPath()); else error("Don't know which bundle file to load"); if (!result) return; int32 rate = 22050, pan = 0, channels, output_size = 0; int32 tag, size = -1, bits = 0; if (_scumm->_gameId == GID_CMI) { char name[20]; strcpy(name, sound); if (_scumm->_maxRooms != 6) // CMI demo does not have .IMX for voice but does for music... strcat(name, ".IMX"); output_size = _bundle->decompressVoiceSampleByName(name, &ptr); } else { output_size = _bundle->decompressVoiceSampleByName(sound, &ptr); } orig_ptr = ptr; if (output_size == 0 || orig_ptr == 0) { goto bail; } tag = READ_BE_UINT32(ptr); ptr += 4; if (tag != MKID_BE('iMUS')) { warning("Decompression of bundle sound failed"); goto bail; } ptr += 12; while (tag != MKID_BE('DATA')) { tag = READ_BE_UINT32(ptr); ptr += 4; switch(tag) { case MKID_BE('FRMT'): ptr += 12; bits = READ_BE_UINT32(ptr); ptr += 4; rate = READ_BE_UINT32(ptr); ptr += 4; channels = READ_BE_UINT32(ptr); ptr += 4; break; case MKID_BE('TEXT'): case MKID_BE('REGN'): case MKID_BE('STOP'): case MKID_BE('JUMP'): case MKID_BE('SYNC'): size = READ_BE_UINT32(ptr); ptr += size + 4; break; case MKID_BE('DATA'): size = READ_BE_UINT32(ptr); ptr += 4; break; default: error("Unknown sound header %c%c%c%c", (byte)(tag >> 24), (byte)(tag >> 16), (byte)(tag >> 8), (byte)tag); } } if (size < 0) { warning("Decompression sound failed (no size field)"); goto bail; } final = (byte *)malloc(size); memcpy(final, ptr, size); if (_scumm->_actorToPrintStrFor != 0xFF && _scumm->_actorToPrintStrFor != 0) { Actor *a = _scumm->derefActor(_scumm->_actorToPrintStrFor, "playBundleSound"); rate = (rate * a->talkFrequency) / 256; // Adjust to fit the mixer's notion of panning. if (pan != 64) pan = 2 * a->talkPan - 127; } // Stop any sound currently playing on the given handle if (handle) _scumm->_mixer->stopHandle(*handle); if (bits == 8) { _scumm->_mixer->playRaw(handle, final, size, rate, SoundMixer::FLAG_UNSIGNED | SoundMixer::FLAG_AUTOFREE, -1, 255, pan); } else if (bits == 16) { // FIXME: For some weird reasons, sometimes we get an odd size, even though // the data is supposed to be in 16 bit format... that makes no sense... size &= ~1; _scumm->_mixer->playRaw(handle, final, size, rate, SoundMixer::FLAG_16BITS | SoundMixer::FLAG_AUTOFREE, -1, 255, pan); } else { warning("Sound::playBundleSound() to do more options to playRaw..."); } bail: free(orig_ptr); } void Sound::playSfxSound(void *sound, uint32 size, uint rate, bool isUnsigned, PlayingSoundHandle *handle) { byte flags = SoundMixer::FLAG_AUTOFREE; if (isUnsigned) flags |= SoundMixer::FLAG_UNSIGNED; _scumm->_mixer->playRaw(handle, sound, size, rate, flags); } #ifdef USE_VORBIS // These are wrapper functions to allow using a File object to // provide data to the OggVorbis_File object. struct file_info { File *file; int start, curr_pos; size_t len; }; static size_t read_wrap(void *ptr, size_t size, size_t nmemb, void *datasource) { file_info *f = (file_info *) datasource; int result; nmemb *= size; if (f->curr_pos > (int) f->len) nmemb = 0; else if (nmemb > f->len - f->curr_pos) nmemb = f->len - f->curr_pos; result = f->file->read(ptr, nmemb); if (result == -1) { f->curr_pos = f->file->pos() - f->start; return (size_t) -1; } else { f->curr_pos += result; return result / size; } } static int seek_wrap(void *datasource, ogg_int64_t offset, int whence) { file_info *f = (file_info *) datasource; if (whence == SEEK_SET) offset += f->start; else if (whence == SEEK_END) { offset += f->start + f->len; whence = SEEK_SET; } f->file->seek(offset, whence); f->curr_pos = f->file->pos() - f->start; return f->curr_pos; } static int close_wrap(void *datasource) { file_info *f = (file_info *) datasource; f->file->close(); delete f; return 0; } static long tell_wrap(void *datasource) { file_info *f = (file_info *) datasource; return f->curr_pos; } static ov_callbacks g_File_wrap = { read_wrap, seek_wrap, close_wrap, tell_wrap }; #endif void Sound::playSfxSound_Vorbis(File *file, uint32 size, PlayingSoundHandle *handle) { #ifdef USE_VORBIS OggVorbis_File *ov_file = new OggVorbis_File; file_info *f = new file_info; f->file = file; f->start = file->pos(); f->len = size; f->curr_pos = 0; if (ov_open_callbacks((void *) f, ov_file, NULL, 0, g_File_wrap) < 0) { warning("Invalid file format"); delete ov_file; delete f; } else _scumm->_mixer->playVorbis(handle, ov_file, 0, false); #endif } // We use a real timer in an attempt to get better sync with CD tracks. This is // necessary for games like Loom CD. static void cd_timer_handler(void *refCon) { Scumm *scumm = (Scumm *)refCon; // FIXME: Turn off the timer when it's no longer needed. In theory, it // should be possible to check with pollCD(), but since CD sound isn't // properly restarted when reloading a saved game, I don't dare to. scumm->VAR(scumm->VAR_MUSIC_TIMER) += 6; } void Sound::startCDTimer() { int timer_interval; // The timer interval has been tuned for Loom CD and the Monkey 1 // intro. I have to use 100 for Loom, or there will be a nasty stutter // when Chaos first appears, and I have to use 101 for Monkey 1 or the // intro music will be cut short. if (_scumm->_gameId == GID_LOOM256) timer_interval = 100; else timer_interval = 101; _scumm->_timer->releaseProcedure(&cd_timer_handler); _scumm->_timer->installProcedure(&cd_timer_handler, 1000 * timer_interval, _scumm); } void Sound::stopCDTimer() { _scumm->_timer->releaseProcedure(&cd_timer_handler); } void Sound::playCDTrack(int track, int numLoops, int startFrame, int duration) { // Reset the music timer variable at the start of a new track _scumm->VAR(_scumm->VAR_MUSIC_TIMER) = 0; if (!_soundsPaused && (numLoops != 0 || startFrame != 0)) { // Try to load the track from a .mp3/.ogg file, and if found, use // that. If not found, attempt to do regular Audio CD playback of // the requested track. int index = getCachedTrack(track); if (index >= 0) { _scumm->_mixer->stopHandle(_dig_cd.handle); _dig_cd.playing = true; _dig_cd.track = track; _dig_cd.numLoops = numLoops; _dig_cd.start = startFrame; _dig_cd.duration = duration; _track_info[index]->play(_scumm->_mixer, &_dig_cd.handle, _dig_cd.start, _dig_cd.duration); } else { _scumm->_system->play_cdrom(track, numLoops, startFrame, duration); } } // Start the timer after starting the track. Starting an MP3 track is // almost instantaneous, but a CD player may take some time. Hopefully // play_cdrom() will block during that delay. startCDTimer(); } void Sound::stopCD() { if (_dig_cd.playing) { _scumm->_mixer->stopHandle(_dig_cd.handle); _dig_cd.playing = false; } else { _scumm->_system->stop_cdrom(); } } int Sound::pollCD() const { return _dig_cd.playing || _scumm->_system->poll_cdrom(); } void Sound::updateCD() { if (_dig_cd.playing) { // If the sound handle is 0, then playback stopped. if (!_dig_cd.handle) { // If playback just stopped, check if the current track is supposed // to be repeated, and if that's the case, play it again. Else, stop // the CD explicitly. if (_dig_cd.numLoops == -1 || --_dig_cd.numLoops > 0) { _scumm->VAR(_scumm->VAR_MUSIC_TIMER) = 0; if (!_soundsPaused) { int index = getCachedTrack(_dig_cd.track); assert(index >= 0); _track_info[index]->play(_scumm->_mixer, &_dig_cd.handle, _dig_cd.start, _dig_cd.duration); } } else { _scumm->_mixer->stopHandle(_dig_cd.handle); _dig_cd.playing = false; } } } else { _scumm->_system->update_cdrom(); } } int Sound::getCachedTrack(int track) { int i; #if defined(USE_MAD) || defined(USE_VORBIS) char track_name[1024]; File *file = new File(); #endif int current_index; // See if we find the track in the cache for (i = 0; i < CACHE_TRACKS; i++) if (_cached_tracks[i] == track) { if (_track_info[i]) return i; else return -1; } current_index = _current_cache++; _current_cache %= CACHE_TRACKS; // Not found, see if it exists // First, delete the previous track info object delete _track_info[current_index]; _track_info[current_index] = NULL; _cached_tracks[current_index] = track; #ifdef USE_MAD sprintf(track_name, "track%d.mp3", track); file->open(track_name, _scumm->getGameDataPath()); if (file->isOpen()) { _track_info[current_index] = new MP3TrackInfo(file); if (_track_info[current_index]->error()) { delete _track_info[current_index]; _track_info[current_index] = NULL; return -1; } return current_index; } #endif #ifdef USE_VORBIS sprintf(track_name, "track%d.ogg", track); file->open(track_name, _scumm->getGameDataPath()); if (file->isOpen()) { _track_info[current_index] = new VorbisTrackInfo(file); if (_track_info[current_index]->error()) { delete _track_info[current_index]; _track_info[current_index] = NULL; return -1; } return current_index; } #endif debug(2, "Track %d not available in compressed format", track); return -1; } #ifdef USE_MAD MP3TrackInfo::MP3TrackInfo(File *file) { struct mad_stream stream; struct mad_frame frame; unsigned char buffer[8192]; unsigned int buflen = 0; int count = 0; // Check the format and bitrate mad_stream_init(&stream); mad_frame_init(&frame); while (1) { if (buflen < sizeof(buffer)) { int bytes; bytes = file->read(buffer + buflen, sizeof(buffer) - buflen); if (bytes <= 0) { if (bytes == -1) { warning("Invalid file format"); goto error; } break; } buflen += bytes; } mad_stream_buffer(&stream, buffer, buflen); while (1) { if (mad_frame_decode(&frame, &stream) == -1) { if (!MAD_RECOVERABLE(stream.error)) break; if (stream.error != MAD_ERROR_BADCRC) continue; } if (count++) break; } if (count || stream.error != MAD_ERROR_BUFLEN) break; memmove(buffer, stream.next_frame, buflen = &buffer[buflen] - stream.next_frame); } if (count) memcpy(&_mad_header, &frame.header, sizeof(mad_header)); else { warning("Invalid file format"); goto error; } mad_frame_finish(&frame); mad_stream_finish(&stream); // Get file size _size = file->size(); _file = file; _error_flag = false; return; error: mad_frame_finish(&frame); mad_stream_finish(&stream); _error_flag = true; delete file; } int MP3TrackInfo::play(SoundMixer *mixer, PlayingSoundHandle *handle, int startFrame, int duration) { unsigned int offset; mad_timer_t durationTime; // Calc offset. As all bitrates are in kilobit per seconds, the division by 200 is always exact offset = (startFrame * (_mad_header.bitrate / (8 * 25))) / 3; _file->seek(offset, SEEK_SET); // Calc delay if (!duration) { // FIXME: Using _size here is a problem if offset (or equivalently // startFrame) is non-zero. mad_timer_set(&durationTime, (_size * 8) / _mad_header.bitrate, (_size * 8) % _mad_header.bitrate, _mad_header.bitrate); } else { mad_timer_set(&durationTime, duration / 75, duration % 75, 75); } // Play it return mixer->playMP3CDTrack(handle, _file, durationTime); } MP3TrackInfo::~MP3TrackInfo() { if (! _error_flag) _file->close(); } #endif #ifdef USE_VORBIS VorbisTrackInfo::VorbisTrackInfo(File *file) { file_info *f = new file_info; f->file = file; f->start = 0; f->len = file->size(); f->curr_pos = file->pos(); if (ov_open_callbacks((void *) f, &_ov_file, NULL, 0, g_File_wrap) < 0) { warning("Invalid file format"); _error_flag = true; delete f; delete file; } else { _error_flag = false; _file = file; } } #ifdef CHUNKSIZE #define VORBIS_TREMOR #endif int VorbisTrackInfo::play(SoundMixer *mixer, PlayingSoundHandle *handle, int startFrame, int duration) { #ifdef VORBIS_TREMOR ov_time_seek(&_ov_file, (ogg_int64_t)(startFrame / 75.0 * 1000)); #else ov_time_seek(&_ov_file, startFrame / 75.0); #endif return mixer->playVorbis(handle, &_ov_file, duration * ov_info(&_ov_file, -1)->rate / 75, true); } VorbisTrackInfo::~VorbisTrackInfo() { if (! _error_flag) { ov_clear(&_ov_file); delete _file; } } #endif