2017-05-26 03:24:38 +00:00
|
|
|
/* 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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2017-06-07 07:20:42 +00:00
|
|
|
#include "common/debug.h"
|
2017-05-27 18:16:54 +00:00
|
|
|
#include "common/file.h"
|
2017-06-09 21:58:54 +00:00
|
|
|
#include "common/list.h"
|
2017-06-09 05:55:03 +00:00
|
|
|
#include "common/memstream.h"
|
2017-05-27 18:16:54 +00:00
|
|
|
|
2017-06-07 07:20:42 +00:00
|
|
|
#include "audio/audiostream.h"
|
|
|
|
#include "audio/mixer.h"
|
|
|
|
#include "audio/decoders/wave.h"
|
|
|
|
#include "audio/decoders/vorbis.h"
|
2017-06-09 05:32:00 +00:00
|
|
|
#include "audio/mods/protracker.h"
|
2017-06-07 07:20:42 +00:00
|
|
|
|
2017-06-05 17:10:47 +00:00
|
|
|
#include "sludge/allfiles.h"
|
|
|
|
#include "sludge/debug.h"
|
|
|
|
#include "sludge/newfatal.h"
|
|
|
|
#include "sludge/sound.h"
|
|
|
|
#include "sludge/moreio.h"
|
|
|
|
#include "sludge/fileset.h"
|
2017-06-07 07:20:42 +00:00
|
|
|
#include "sludge/sludge.h"
|
2017-06-05 17:10:47 +00:00
|
|
|
|
2017-05-26 03:24:38 +00:00
|
|
|
#define MAX_SAMPLES 8
|
|
|
|
#define MAX_MODS 3
|
|
|
|
#define NUM_BUFS 3
|
|
|
|
|
2017-05-26 19:25:11 +00:00
|
|
|
namespace Sludge {
|
|
|
|
|
2017-05-26 03:24:38 +00:00
|
|
|
bool soundOK = false;
|
|
|
|
bool SilenceIKillYou = false;
|
2017-06-10 17:27:13 +00:00
|
|
|
bool isHandlingSoundList = false;
|
2017-06-09 21:58:54 +00:00
|
|
|
// there's possibility that several sound list played at the same time
|
|
|
|
typedef Common::List<soundList *> SoundListHandles;
|
|
|
|
SoundListHandles soundListHandles;
|
|
|
|
|
2017-05-26 03:24:38 +00:00
|
|
|
struct soundThing {
|
2017-06-09 05:55:03 +00:00
|
|
|
Audio::SoundHandle handle;
|
|
|
|
int fileLoaded, vol; //Used for sounds only. (sound saving/loading)
|
|
|
|
bool looping; //Used for sounds only. (sound saving/loading)
|
2017-06-10 17:27:13 +00:00
|
|
|
bool inSoundList;
|
2017-05-26 03:24:38 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
soundThing soundCache[MAX_SAMPLES];
|
2017-06-09 05:55:03 +00:00
|
|
|
#if 0
|
2017-05-26 03:24:38 +00:00
|
|
|
soundThing modCache[MAX_MODS];
|
2017-06-09 05:55:03 +00:00
|
|
|
#endif
|
2017-05-26 03:24:38 +00:00
|
|
|
|
|
|
|
int defVol = 128;
|
|
|
|
int defSoundVol = 255;
|
|
|
|
const float modLoudness = 0.95f;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set up, tear down:
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool initSoundStuff(HWND hwnd) {
|
2017-06-09 05:55:03 +00:00
|
|
|
for (int a = 0; a < MAX_SAMPLES; a ++) {
|
2017-05-26 03:24:38 +00:00
|
|
|
soundCache[a].fileLoaded = -1;
|
|
|
|
soundCache[a].looping = false;
|
2017-06-10 17:27:13 +00:00
|
|
|
soundCache[a].inSoundList = false;
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
2017-06-09 05:55:03 +00:00
|
|
|
#if 0
|
|
|
|
for (int a = 0; a < MAX_MODS; a ++) {
|
2017-05-26 03:24:38 +00:00
|
|
|
modCache[a].stream = NULL;
|
|
|
|
modCache[a].playing = false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return soundOK = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void killSoundStuff() {
|
2017-05-29 06:02:59 +00:00
|
|
|
if (!soundOK)
|
|
|
|
return;
|
2017-06-09 05:55:03 +00:00
|
|
|
|
2017-05-26 03:24:38 +00:00
|
|
|
SilenceIKillYou = true;
|
|
|
|
for (int i = 0; i < MAX_SAMPLES; i ++) {
|
2017-06-09 05:55:03 +00:00
|
|
|
if (g_sludge->_mixer->isSoundHandleActive(soundCache[i].handle)) {
|
|
|
|
g_sludge->_mixer->stopHandle(soundCache[i].handle);
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
}
|
2017-06-09 05:55:03 +00:00
|
|
|
#if 0
|
2017-05-26 03:24:38 +00:00
|
|
|
for (int i = 0; i < MAX_MODS; i ++) {
|
|
|
|
if (modCache[i].playing) {
|
|
|
|
|
|
|
|
if (! alureStopSource(modCache[i].playingOnSource, AL_TRUE)) {
|
|
|
|
debugOut("Failed to stop source: %s\n",
|
2017-05-29 06:02:59 +00:00
|
|
|
alureGetErrorString());
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (modCache[i].stream != NULL) {
|
|
|
|
|
|
|
|
if (! alureDestroyStream(modCache[i].stream, 0, NULL)) {
|
|
|
|
debugOut("Failed to destroy stream: %s\n",
|
2017-05-29 06:02:59 +00:00
|
|
|
alureGetErrorString());
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2017-06-09 05:55:03 +00:00
|
|
|
SilenceIKillYou = false;
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Some setters:
|
|
|
|
*/
|
|
|
|
|
|
|
|
void setMusicVolume(int a, int v) {
|
2017-05-29 06:02:59 +00:00
|
|
|
if (!soundOK)
|
|
|
|
return;
|
2017-05-26 03:24:38 +00:00
|
|
|
#if 0
|
2017-06-09 05:55:03 +00:00
|
|
|
if (modCache[a].playing) {
|
2017-05-26 03:24:38 +00:00
|
|
|
alSourcef(modCache[a].playingOnSource, AL_GAIN, (float) modLoudness * v / 256);
|
|
|
|
}
|
2017-06-09 05:55:03 +00:00
|
|
|
#endif
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void setDefaultMusicVolume(int v) {
|
|
|
|
defVol = v;
|
|
|
|
}
|
|
|
|
|
|
|
|
void setSoundVolume(int a, int v) {
|
2017-05-29 06:02:59 +00:00
|
|
|
if (!soundOK)
|
|
|
|
return;
|
2017-05-26 03:24:38 +00:00
|
|
|
int ch = findInSoundCache(a);
|
|
|
|
if (ch != -1) {
|
2017-06-09 05:55:03 +00:00
|
|
|
if (g_sludge->_mixer->isSoundHandleActive(soundCache[ch].handle)) {
|
2017-05-26 03:24:38 +00:00
|
|
|
soundCache[ch].vol = v;
|
2017-06-09 05:55:03 +00:00
|
|
|
g_sludge->_mixer->setChannelVolume(soundCache[ch].handle, v);
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void setDefaultSoundVolume(int v) {
|
|
|
|
defSoundVol = v;
|
|
|
|
}
|
|
|
|
|
|
|
|
void setSoundLoop(int a, int s, int e) {
|
|
|
|
//#pragma unused (a,s,e)
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Stopping things:
|
|
|
|
*/
|
|
|
|
|
|
|
|
int findInSoundCache(int a) {
|
|
|
|
int i;
|
2017-05-29 06:02:59 +00:00
|
|
|
for (i = 0; i < MAX_SAMPLES; i++) {
|
2017-05-26 03:24:38 +00:00
|
|
|
if (soundCache[i].fileLoaded == a) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void stopMOD(int i) {
|
2017-05-29 06:02:59 +00:00
|
|
|
if (!soundOK)
|
|
|
|
return;
|
2017-05-26 03:24:38 +00:00
|
|
|
#if 0
|
|
|
|
alGetError();
|
|
|
|
if (modCache[i].playing) {
|
|
|
|
if (! alureStopSource(modCache[i].playingOnSource, AL_TRUE)) {
|
|
|
|
debugOut("Failed to stop source: %s\n", alureGetErrorString());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void huntKillSound(int filenum) {
|
2017-05-29 06:02:59 +00:00
|
|
|
if (!soundOK)
|
|
|
|
return;
|
2017-05-26 03:24:38 +00:00
|
|
|
|
|
|
|
int gotSlot = findInSoundCache(filenum);
|
|
|
|
if (gotSlot == -1) return;
|
|
|
|
|
|
|
|
SilenceIKillYou = true;
|
|
|
|
|
2017-07-05 07:35:50 +00:00
|
|
|
if (g_sludge->_mixer->isSoundHandleActive(soundCache[gotSlot].handle)) {
|
|
|
|
g_sludge->_mixer->stopHandle(soundCache[gotSlot].handle);
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
2017-07-05 07:35:50 +00:00
|
|
|
|
2017-05-26 03:24:38 +00:00
|
|
|
SilenceIKillYou = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void freeSound(int a) {
|
2017-05-29 06:02:59 +00:00
|
|
|
if (!soundOK)
|
|
|
|
return;
|
2017-05-26 03:24:38 +00:00
|
|
|
|
|
|
|
SilenceIKillYou = true;
|
|
|
|
|
2017-06-09 05:55:03 +00:00
|
|
|
if (g_sludge->_mixer->isSoundHandleActive(soundCache[a].handle)) {
|
|
|
|
g_sludge->_mixer->stopHandle(soundCache[a].handle);
|
2017-06-10 17:27:13 +00:00
|
|
|
if (soundCache[a].inSoundList)
|
|
|
|
handleSoundLists();
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
soundCache[a].fileLoaded = -1;
|
2017-06-09 05:55:03 +00:00
|
|
|
|
2017-05-26 03:24:38 +00:00
|
|
|
SilenceIKillYou = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void huntKillFreeSound(int filenum) {
|
2017-05-29 06:02:59 +00:00
|
|
|
if (!soundOK)
|
|
|
|
return;
|
2017-05-26 03:24:38 +00:00
|
|
|
int gotSlot = findInSoundCache(filenum);
|
2017-05-29 06:02:59 +00:00
|
|
|
if (gotSlot == -1)
|
|
|
|
return;
|
2017-05-26 03:24:38 +00:00
|
|
|
freeSound(gotSlot);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Loading and playing:
|
|
|
|
*/
|
|
|
|
bool playMOD(int f, int a, int fromTrack) {
|
2017-06-09 05:32:00 +00:00
|
|
|
#if 0
|
|
|
|
// load sound
|
|
|
|
setResourceForFatal(f);
|
|
|
|
uint32 length = openFileFromNum(f);
|
|
|
|
if (length == 0) {
|
|
|
|
finishAccess();
|
|
|
|
setResourceForFatal(-1);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::SeekableReadStream *memImage = bigDataFile->readStream(length);
|
|
|
|
if (memImage->size() != length || bigDataFile->err())
|
|
|
|
debug("Sound reading failed");
|
|
|
|
Audio::AudioStream *stream = Audio::makeProtrackerStream(memImage);
|
|
|
|
//TODO: replace by xm file decoders
|
|
|
|
if (!stream)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// play sound
|
|
|
|
Audio::SoundHandle soundHandle;
|
|
|
|
g_sludge->_mixer->playStream(Audio::Mixer::kSFXSoundType, &soundHandle,
|
|
|
|
stream, -1, Audio::Mixer::kMaxChannelVolume);
|
|
|
|
#endif
|
|
|
|
#if 0
|
2017-05-29 06:02:59 +00:00
|
|
|
if (!soundOK)
|
|
|
|
return true;
|
2017-05-26 03:24:38 +00:00
|
|
|
stopMOD(a);
|
|
|
|
|
|
|
|
setResourceForFatal(f);
|
2017-06-05 17:35:03 +00:00
|
|
|
uint32 length = openFileFromNum(f);
|
2017-05-26 03:24:38 +00:00
|
|
|
if (length == 0) {
|
|
|
|
finishAccess();
|
|
|
|
setResourceForFatal(-1);
|
|
|
|
return false;
|
|
|
|
}
|
2017-06-09 05:32:00 +00:00
|
|
|
|
2017-05-26 03:24:38 +00:00
|
|
|
unsigned char *memImage;
|
|
|
|
memImage = (unsigned char *) loadEntireFileToMemory(bigDataFile, length);
|
|
|
|
if (! memImage) return fatal(ERROR_MUSIC_MEMORY_LOW);
|
|
|
|
|
|
|
|
modCache[a].stream = alureCreateStreamFromMemory(memImage, length, 19200, 0, NULL);
|
|
|
|
|
|
|
|
delete memImage;
|
|
|
|
|
|
|
|
if (modCache[a].stream != NULL) {
|
|
|
|
setMusicVolume(a, defVol);
|
|
|
|
|
|
|
|
if (! alureSetStreamOrder(modCache[a].stream, fromTrack)) {
|
|
|
|
debugOut("Failed to set stream order: %s\n",
|
2017-05-29 06:02:59 +00:00
|
|
|
alureGetErrorString());
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
playStream(a, true, true);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
debugOut("Failed to create stream from MOD: %s\n",
|
2017-05-29 06:02:59 +00:00
|
|
|
alureGetErrorString());
|
2017-05-26 03:24:38 +00:00
|
|
|
|
|
|
|
warning(ERROR_MUSIC_ODDNESS);
|
|
|
|
soundCache[a].stream = NULL;
|
|
|
|
soundCache[a].playing = false;
|
|
|
|
soundCache[a].playingOnSource = 0;
|
|
|
|
}
|
|
|
|
setResourceForFatal(-1);
|
|
|
|
#endif
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool stillPlayingSound(int ch) {
|
|
|
|
if (soundOK)
|
|
|
|
if (ch != -1)
|
|
|
|
if (soundCache[ch].fileLoaded != -1)
|
2017-06-09 05:55:03 +00:00
|
|
|
if (g_sludge->_mixer->isSoundHandleActive(soundCache[ch].handle))
|
2017-05-26 03:24:38 +00:00
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool forceRemoveSound() {
|
2017-05-29 06:02:59 +00:00
|
|
|
for (int a = 0; a < MAX_SAMPLES; a++) {
|
|
|
|
if (soundCache[a].fileLoaded != -1 && !stillPlayingSound(a)) {
|
2017-05-26 03:24:38 +00:00
|
|
|
// soundWarning ("Deleting silent sound", a);
|
|
|
|
freeSound(a);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-29 06:02:59 +00:00
|
|
|
for (int a = 0; a < MAX_SAMPLES; a++) {
|
2017-05-26 03:24:38 +00:00
|
|
|
if (soundCache[a].fileLoaded != -1) {
|
|
|
|
// soundWarning ("Deleting playing sound", a);
|
|
|
|
freeSound(a);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// soundWarning ("Cache is empty!", 0);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int emptySoundSlot = 0;
|
|
|
|
|
|
|
|
int findEmptySoundSlot() {
|
2017-06-09 05:55:03 +00:00
|
|
|
for (int t = 0; t < MAX_SAMPLES; t++) {
|
2017-05-29 06:02:59 +00:00
|
|
|
emptySoundSlot++;
|
2017-05-26 03:24:38 +00:00
|
|
|
emptySoundSlot %= MAX_SAMPLES;
|
2017-06-10 17:27:13 +00:00
|
|
|
if (!g_sludge->_mixer->isSoundHandleActive(soundCache[emptySoundSlot].handle) && !soundCache[emptySoundSlot].inSoundList)
|
2017-05-26 03:24:38 +00:00
|
|
|
return emptySoundSlot;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Argh! They're all playing! Let's trash the oldest that's not looping...
|
|
|
|
|
2017-06-09 05:55:03 +00:00
|
|
|
for (int t = 0; t < MAX_SAMPLES; t++) {
|
2017-05-29 06:02:59 +00:00
|
|
|
emptySoundSlot++;
|
2017-05-26 03:24:38 +00:00
|
|
|
emptySoundSlot %= MAX_SAMPLES;
|
2017-06-10 17:27:13 +00:00
|
|
|
if (!soundCache[emptySoundSlot].looping && !soundCache[emptySoundSlot].inSoundList)
|
2017-05-29 06:02:59 +00:00
|
|
|
return emptySoundSlot;
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Holy crap, they're all looping! What's this twat playing at?
|
|
|
|
|
2017-05-29 06:02:59 +00:00
|
|
|
emptySoundSlot++;
|
2017-05-26 03:24:38 +00:00
|
|
|
emptySoundSlot %= MAX_SAMPLES;
|
|
|
|
return emptySoundSlot;
|
|
|
|
}
|
|
|
|
|
|
|
|
int cacheSound(int f) {
|
2017-06-09 05:55:03 +00:00
|
|
|
return 0; // don't load source in advance
|
|
|
|
}
|
2017-05-26 03:24:38 +00:00
|
|
|
|
2017-06-09 05:55:03 +00:00
|
|
|
int makeSoundAudioStream(int f, Audio::AudioStream *&audiostream, bool loopy) {
|
2017-05-29 06:02:59 +00:00
|
|
|
if (!soundOK)
|
|
|
|
return -1;
|
2017-05-26 03:24:38 +00:00
|
|
|
|
|
|
|
int a = findInSoundCache(f);
|
2017-06-09 05:55:03 +00:00
|
|
|
if (a != -1) { // if this sound has been loaded before
|
|
|
|
// still playing
|
|
|
|
if (g_sludge->_mixer->isSoundHandleActive(soundCache[a].handle)) {
|
|
|
|
g_sludge->_mixer->stopHandle(soundCache[a].handle); // stop it
|
2017-06-10 17:27:13 +00:00
|
|
|
if (soundCache[a].inSoundList) {
|
|
|
|
handleSoundLists();
|
|
|
|
}
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
2017-06-09 05:55:03 +00:00
|
|
|
} else {
|
|
|
|
if (f == -2)
|
|
|
|
return -1;
|
|
|
|
a = findEmptySoundSlot();
|
|
|
|
freeSound(a);
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
|
2017-06-09 05:55:03 +00:00
|
|
|
setResourceForFatal(f);
|
2017-06-05 17:35:03 +00:00
|
|
|
uint32 length = openFileFromNum(f);
|
2017-06-09 05:55:03 +00:00
|
|
|
if (!length)
|
|
|
|
return -1;
|
2017-05-26 03:24:38 +00:00
|
|
|
|
2017-06-09 05:55:03 +00:00
|
|
|
uint curr_ptr = bigDataFile->pos();
|
|
|
|
Audio::RewindableAudioStream *stream = Audio::makeWAVStream(bigDataFile->readStream(length), DisposeAfterUse::NO);
|
2017-05-26 03:24:38 +00:00
|
|
|
|
2017-06-09 05:55:03 +00:00
|
|
|
#ifdef USE_VORBIS
|
|
|
|
if (!stream) {
|
|
|
|
bigDataFile->seek(curr_ptr);
|
|
|
|
stream = Audio::makeVorbisStream(bigDataFile->readStream(length), DisposeAfterUse::NO);
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
2017-06-09 05:55:03 +00:00
|
|
|
#endif
|
|
|
|
finishAccess();
|
2017-05-26 03:24:38 +00:00
|
|
|
|
2017-06-09 05:55:03 +00:00
|
|
|
if (stream) {
|
|
|
|
audiostream = Audio::makeLoopingAudioStream(stream, loopy ? 0 : 1);
|
2017-05-26 03:24:38 +00:00
|
|
|
soundCache[a].fileLoaded = f;
|
|
|
|
setResourceForFatal(-1);
|
|
|
|
} else {
|
2017-06-09 05:55:03 +00:00
|
|
|
audiostream = nullptr;
|
2017-05-26 03:24:38 +00:00
|
|
|
warning(ERROR_SOUND_ODDNESS);
|
|
|
|
soundCache[a].fileLoaded = -1;
|
|
|
|
soundCache[a].looping = false;
|
2017-06-09 05:55:03 +00:00
|
|
|
return -1;
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
2017-06-09 05:55:03 +00:00
|
|
|
|
|
|
|
return a;
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool startSound(int f, bool loopy) {
|
2017-06-07 07:20:42 +00:00
|
|
|
if (soundOK) {
|
2017-06-09 05:55:03 +00:00
|
|
|
// Load sound
|
|
|
|
Audio::AudioStream *stream = nullptr;
|
|
|
|
int a = makeSoundAudioStream(f, stream, loopy);
|
2017-05-26 03:24:38 +00:00
|
|
|
if (a == -1) {
|
|
|
|
debugOut("Failed to cache sound!\n");
|
|
|
|
return false;
|
|
|
|
}
|
2017-06-09 05:55:03 +00:00
|
|
|
|
|
|
|
// play sound
|
2017-05-26 03:24:38 +00:00
|
|
|
soundCache[a].looping = loopy;
|
|
|
|
soundCache[a].vol = defSoundVol;
|
2017-06-09 05:55:03 +00:00
|
|
|
g_sludge->_mixer->playStream(Audio::Mixer::kSFXSoundType, &soundCache[a].handle, stream, -1, soundCache[a].vol);
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-05-27 18:16:54 +00:00
|
|
|
void saveSounds(Common::WriteStream *stream) {
|
2017-05-26 03:24:38 +00:00
|
|
|
if (soundOK) {
|
2017-05-29 06:02:59 +00:00
|
|
|
for (int i = 0; i < MAX_SAMPLES; i++) {
|
2017-05-26 03:24:38 +00:00
|
|
|
if (soundCache[i].looping) {
|
2017-05-30 07:59:56 +00:00
|
|
|
stream->writeByte(1);
|
|
|
|
stream->writeUint16BE(soundCache[i].fileLoaded);
|
|
|
|
stream->writeUint16BE(soundCache[i].vol);
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-05-30 07:59:56 +00:00
|
|
|
stream->writeByte(0);
|
|
|
|
stream->writeUint16BE(defSoundVol);
|
|
|
|
stream->writeUint16BE(defVol);
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
|
2017-05-27 18:16:54 +00:00
|
|
|
void loadSounds(Common::SeekableReadStream *stream) {
|
2017-05-29 06:02:59 +00:00
|
|
|
for (int i = 0; i < MAX_SAMPLES; i++)
|
|
|
|
freeSound(i);
|
2017-05-26 03:24:38 +00:00
|
|
|
|
2017-05-30 07:59:56 +00:00
|
|
|
while (stream->readByte()) {
|
|
|
|
int fileLoaded = stream->readUint16BE();
|
|
|
|
defSoundVol = stream->readUint16BE();
|
2017-05-26 03:24:38 +00:00
|
|
|
startSound(fileLoaded, 1);
|
|
|
|
}
|
|
|
|
|
2017-05-30 07:59:56 +00:00
|
|
|
defSoundVol = stream->readUint16BE();
|
|
|
|
defVol = stream->readUint16BE();
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool getSoundCacheStack(stackHandler *sH) {
|
|
|
|
variable newFileHandle;
|
|
|
|
newFileHandle.varType = SVT_NULL;
|
|
|
|
|
2017-05-29 06:02:59 +00:00
|
|
|
for (int a = 0; a < MAX_SAMPLES; a++) {
|
2017-05-26 03:24:38 +00:00
|
|
|
if (soundCache[a].fileLoaded != -1) {
|
|
|
|
setVariable(newFileHandle, SVT_FILE, soundCache[a].fileLoaded);
|
2017-05-29 06:02:59 +00:00
|
|
|
if (!addVarToStackQuick(newFileHandle, sH->first))
|
|
|
|
return false;
|
|
|
|
if (sH->last == NULL)
|
|
|
|
sH->last = sH->first;
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-06-09 21:58:54 +00:00
|
|
|
bool deleteSoundFromList(soundList *&s) {
|
2017-05-26 03:24:38 +00:00
|
|
|
// Don't delete a playing sound.
|
2017-05-29 06:02:59 +00:00
|
|
|
if (s->cacheIndex)
|
2017-06-09 21:58:54 +00:00
|
|
|
return false;
|
2017-05-26 03:24:38 +00:00
|
|
|
|
|
|
|
soundList *o = NULL;
|
2017-05-29 06:02:59 +00:00
|
|
|
if (!s->next) {
|
2017-05-26 03:24:38 +00:00
|
|
|
o = s->prev;
|
2017-05-29 06:02:59 +00:00
|
|
|
if (o)
|
|
|
|
o->next = NULL;
|
2017-05-26 03:24:38 +00:00
|
|
|
delete s;
|
2017-06-09 21:58:54 +00:00
|
|
|
s = o;
|
|
|
|
return (s != NULL);
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
if (s != s->next) {
|
|
|
|
o = s->next;
|
|
|
|
o->prev = s->prev;
|
2017-05-29 06:02:59 +00:00
|
|
|
if (o->prev)
|
|
|
|
o->prev->next = o;
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
delete s;
|
2017-06-09 21:58:54 +00:00
|
|
|
s = o;
|
|
|
|
return (s != NULL);
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
|
2017-06-09 21:58:54 +00:00
|
|
|
void handleSoundLists() {
|
2017-06-10 17:27:13 +00:00
|
|
|
if (isHandlingSoundList)
|
|
|
|
return;
|
|
|
|
isHandlingSoundList = true;
|
2017-06-09 21:58:54 +00:00
|
|
|
for (SoundListHandles::iterator it = soundListHandles.begin(); it != soundListHandles.end(); ++it) {
|
|
|
|
soundList *s = (*it);
|
|
|
|
int a = s->cacheIndex;
|
|
|
|
bool remove = false;
|
|
|
|
if (!g_sludge->_mixer->isSoundHandleActive(soundCache[a].handle)) { // reach the end of stream
|
|
|
|
s->cacheIndex = false;
|
2017-06-10 17:27:13 +00:00
|
|
|
soundCache[a].inSoundList = false;
|
2017-06-09 21:58:54 +00:00
|
|
|
if (SilenceIKillYou) {
|
|
|
|
while (deleteSoundFromList(s))
|
|
|
|
;
|
|
|
|
remove = (s == NULL); // s not null if still playing
|
|
|
|
} else {
|
|
|
|
if (s->next) {
|
|
|
|
if (s->next == s) { // loop the same sound
|
|
|
|
int v = defSoundVol;
|
|
|
|
defSoundVol = soundCache[a].vol;
|
|
|
|
startSound(s->sound, true);
|
|
|
|
defSoundVol = v;
|
|
|
|
while (deleteSoundFromList(s))
|
|
|
|
;
|
|
|
|
remove = (s == NULL); // s not null if still playing
|
|
|
|
} else { // repush the next sound list
|
|
|
|
s->next->vol = soundCache[a].vol;
|
|
|
|
playSoundList(s->next);
|
|
|
|
remove = true; // remove this one
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
while (deleteSoundFromList(s))
|
|
|
|
;
|
|
|
|
remove = (s == NULL); // s not null if still playing
|
|
|
|
}
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
2017-06-09 21:58:54 +00:00
|
|
|
}
|
|
|
|
if (remove) {
|
|
|
|
it = soundListHandles.reverse_erase(it);
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
}
|
2017-06-10 17:27:13 +00:00
|
|
|
isHandlingSoundList = false;
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
|
2017-06-09 05:55:03 +00:00
|
|
|
// loop a list of sound
|
2017-05-26 03:24:38 +00:00
|
|
|
void playSoundList(soundList *s) {
|
|
|
|
if (soundOK) {
|
2017-06-09 05:55:03 +00:00
|
|
|
// Load sound
|
|
|
|
Audio::AudioStream *stream;
|
2017-06-09 21:58:54 +00:00
|
|
|
int a = makeSoundAudioStream(s->sound, stream, false);
|
2017-05-26 03:24:38 +00:00
|
|
|
if (a == -1) {
|
|
|
|
debugOut("Failed to cache sound!\n");
|
|
|
|
return;
|
|
|
|
}
|
2017-06-09 05:55:03 +00:00
|
|
|
|
|
|
|
// Play sound
|
2017-05-26 03:24:38 +00:00
|
|
|
soundCache[a].looping = false;
|
|
|
|
if (s->vol < 0)
|
2017-06-09 05:55:03 +00:00
|
|
|
soundCache[a].vol = defSoundVol;
|
2017-05-26 03:24:38 +00:00
|
|
|
else
|
2017-06-09 05:55:03 +00:00
|
|
|
soundCache[a].vol = s->vol;
|
2017-05-26 03:24:38 +00:00
|
|
|
s-> cacheIndex = a;
|
2017-06-09 05:55:03 +00:00
|
|
|
g_sludge->_mixer->playStream(Audio::Mixer::kSFXSoundType, &soundCache[a].handle, stream, -1, soundCache[a].vol);
|
2017-06-10 17:27:13 +00:00
|
|
|
soundCache[a].inSoundList = true;
|
2017-06-09 21:58:54 +00:00
|
|
|
|
|
|
|
// push sound list
|
|
|
|
soundListHandles.push_back(s);
|
|
|
|
|
2017-05-26 03:24:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void playMovieStream(int a) {
|
2017-05-27 18:16:54 +00:00
|
|
|
#if 0
|
2017-05-26 03:24:38 +00:00
|
|
|
if (! soundOK) return;
|
|
|
|
ALboolean ok;
|
|
|
|
ALuint src;
|
2017-05-27 18:16:54 +00:00
|
|
|
|
2017-05-26 03:24:38 +00:00
|
|
|
alGenSources(1, &src);
|
|
|
|
if (alGetError() != AL_NO_ERROR) {
|
|
|
|
debugOut("Failed to create OpenAL source!\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
alSourcef(src, AL_GAIN, (float) soundCache[a].vol / 256);
|
|
|
|
|
|
|
|
ok = alurePlaySourceStream(src, soundCache[a].stream,
|
2017-05-29 06:02:59 +00:00
|
|
|
10, 0, sound_eos_callback, &intpointers[a]);
|
2017-05-26 03:24:38 +00:00
|
|
|
if (!ok) {
|
|
|
|
debugOut("Failed to play stream: %s\n", alureGetErrorString());
|
|
|
|
alDeleteSources(1, &src);
|
|
|
|
if (alGetError() != AL_NO_ERROR) {
|
|
|
|
debugOut("Failed to delete OpenAL source!\n");
|
|
|
|
}
|
2017-05-27 18:16:54 +00:00
|
|
|
|
2017-05-26 03:24:38 +00:00
|
|
|
soundCache[a].playingOnSource = 0;
|
|
|
|
} else {
|
|
|
|
soundCache[a].playingOnSource = src;
|
|
|
|
soundCache[a].playing = true;
|
|
|
|
}
|
|
|
|
#endif
|
2017-05-27 18:16:54 +00:00
|
|
|
}
|
2017-05-26 03:24:38 +00:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
int initMovieSound(int f, ALenum format, int audioChannels, ALuint samplerate,
|
2017-05-29 06:02:59 +00:00
|
|
|
ALuint(*callback)(void *userdata, ALubyte *data, ALuint bytes)) {
|
2017-05-26 03:24:38 +00:00
|
|
|
if (! soundOK) return 0;
|
|
|
|
|
|
|
|
int retval;
|
|
|
|
int a = findEmptySoundSlot();
|
|
|
|
freeSound(a);
|
|
|
|
|
|
|
|
soundCache[a].looping = false;
|
|
|
|
#if 0
|
|
|
|
// audioChannel * sampleRate gives us a buffer of half a second. Not much, but it should be enough.
|
|
|
|
soundCache[a].stream = alureCreateStreamFromCallback(
|
2017-05-29 06:02:59 +00:00
|
|
|
callback,
|
|
|
|
&intpointers[a], format, samplerate,
|
|
|
|
audioChannels * samplerate, 0, NULL);
|
2017-05-26 03:24:38 +00:00
|
|
|
#endif
|
|
|
|
if (soundCache[a].stream != NULL) {
|
|
|
|
soundCache[a].fileLoaded = f;
|
|
|
|
soundCache[a].vol = defSoundVol;
|
|
|
|
retval = a;
|
|
|
|
} else {
|
|
|
|
#if 0
|
|
|
|
debugOut("Failed to create stream from sound: %s\n",
|
2017-05-29 06:02:59 +00:00
|
|
|
alureGetErrorString());
|
2017-05-26 03:24:38 +00:00
|
|
|
#endif
|
|
|
|
warning(ERROR_SOUND_ODDNESS);
|
|
|
|
soundCache[a].stream = NULL;
|
|
|
|
soundCache[a].playing = false;
|
|
|
|
soundCache[a].playingOnSource = 0;
|
|
|
|
soundCache[a].fileLoaded = -1;
|
|
|
|
retval = -1;
|
|
|
|
}
|
|
|
|
//fprintf (stderr, "Stream %d created. Sample rate: %d Channels: %d\n", retval, samplerate, audioChannels);
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
unsigned int getSoundSource(int index) {
|
|
|
|
return 0; /*soundCache[index].playingOnSource;*/ //TODO:false value
|
|
|
|
}
|
2017-05-26 19:25:11 +00:00
|
|
|
|
|
|
|
} // End of namespace Sludge
|