1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-09 03:39:14 +00:00
OpenMW/sound/music.d
2009-05-06 06:39:41 +00:00

395 lines
8.7 KiB
D

/*
OpenMW - The completely unofficial reimplementation of Morrowind
Copyright (C) 2008 Nicolay Korslund
Email: < korslund@gmail.com >
WWW: http://openmw.snaptoad.com/
This file (music.d) is part of the OpenMW package.
OpenMW is distributed as free software: you can redistribute it
and/or modify it under the terms of the GNU General Public License
version 3, as published by the Free Software Foundation.
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
version 3 along with this program. If not, see
http://www.gnu.org/licenses/ .
*/
module sound.music;
import sound.avcodec;
import sound.audio;
import sound.al;
import monster.monster;
import std.stdio;
import std.string;
import core.config;
import core.resource;
class Idle_waitUntilFinished : IdleFunction
{
override:
IS initiate(Thread*) { return IS.Poll; }
bool hasFinished(Thread* trd)
{
Jukebox mgr = cast(Jukebox)trd.extraData.obj;
assert(mgr !is null);
// Return when the music is no longer playing
return !mgr.isPlaying();
}
}
struct Music
{
private:
static:
// The Monster classes and objects
MonsterClass jukeC, controlC;
public:
MonsterObject *controlM;
void init()
{
assert(jukeC is null);
assert(controlC is null);
assert(controlM is null);
jukeC = vm.load("sound.Jukebox");
jukeC.bind("waitUntilFinished",
new Idle_waitUntilFinished);
jukeC.bind("setSound", { Jukebox.get().setSound(); });
jukeC.bind("setVolume", { Jukebox.get().setVolume(); });
jukeC.bind("playSound", { Jukebox.get().playSound(); });
jukeC.bind("stopSound", { Jukebox.get().stopSound(); });
jukeC.bindConst({new Jukebox(params.obj()); });
controlC = vm.load("sound.Music");
controlM = controlC.getSing();
controlM.call("setup");
}
private void pushSArray(char[][] strs)
{
AIndex arr[];
arr.length = strs.length;
// Create the array indices for each element string
foreach(i, ref elm; arr)
elm = arrays.create(strs[i]).getIndex();
// Push the final array
stack.pushArray(arr);
}
void setPlaylists(char[][] normal, char[][] battle)
{
if(controlM is null) return;
pushSArray(normal);
pushSArray(battle);
controlM.call("setPlaylists");
}
void play()
{
if(controlM !is null)
controlM.call("play");
}
void toggle()
{
if(controlM !is null)
controlM.call("toggle");
}
void toggleMute()
{
if(controlM !is null)
controlM.call("toggleMute");
}
void updateBuffers()
{
if(controlM is null) return;
foreach(ref MonsterObject b; jukeC)
Jukebox.get(b).updateBuffers();
}
void shutdown()
{
if(controlM is null) return;
foreach(ref MonsterObject b; jukeC)
Jukebox.get(b).shutdown();
}
}
// Simple music player, has a playlist and can pause/resume
// music. Most of the high-level code is in mscripts/jukebox.mn.
class Jukebox
{
private:
// Maximum buffer length, divided up among OpenAL buffers
const uint bufLength = 128*1024;
const uint numBufs = 4;
char[] getName()
{
if(mo is null) return "(no name)";
else return mo.getString8("name");
}
void fail(char[] msg)
{
throw new SoundException(getName() ~ " Jukebox", msg);
}
ALuint sID; // Sound id
ALuint bIDs[numBufs]; // Buffers
ALenum bufFormat;
ALint bufRate;
AVFile fileHandle;
AVAudio audioHandle;
static ubyte[] outData;
static this()
{
outData.length = bufLength / numBufs;
}
// The jukebox Monster object
MonsterObject *mo;
public:
this(MonsterObject *m)
{
mo = m;
m.getExtra(Music.jukeC).obj = this;
sID = 0;
bIDs[] = 0;
}
static Jukebox get(MonsterObject *m)
{
auto j = cast(Jukebox)m.getExtra(Music.jukeC).obj;
assert(j !is null);
assert(j.mo == m);
return j;
}
static Jukebox get(ref MonsterObject m)
{ return get(&m); }
static Jukebox get()
{ return get(params.obj()); }
private:
// Disable music
void shutdown()
{
mo.call("stop");
if(fileHandle) avc_closeAVFile(fileHandle);
fileHandle = null;
audioHandle = null;
if(sID)
{
alSourceStop(sID);
alDeleteSources(1, &sID);
checkALError("disabling music");
sID = 0;
alDeleteBuffers(bIDs.length, bIDs.ptr);
checkALError("deleting buffers");
bIDs[] = 0;
}
}
void setSound()
{
char[] fname = stack.popString8();
// Generate a source to play back with if needed
if(sID == 0)
{
alGenSources(1, &sID);
checkALError("generating buffers");
// Set listner relative coordinates (sound follows the player)
alSourcei(sID, AL_SOURCE_RELATIVE, AL_TRUE);
alGenBuffers(bIDs.length, bIDs.ptr);
}
else
{
// Kill current track, but keep the sID source.
alSourceStop(sID);
checkALError("stopping current track");
alSourcei(sID, AL_BUFFER, 0);
//alDeleteBuffers(bIDs.length, bIDs.ptr);
//bIDs[] = 0;
checkALError("resetting buffer");
}
if(fileHandle) avc_closeAVFile(fileHandle);
fileHandle = null;
audioHandle = null;
//alGenBuffers(bIDs.length, bIDs.ptr);
// If something fails, clean everything up.
scope(failure) shutdown();
fileHandle = avc_openAVFile(toStringz(fname));
if(!fileHandle)
fail("Unable to open " ~ fname);
audioHandle = avc_getAVAudioStream(fileHandle, 0);
if(!audioHandle)
fail("Unable to load music track " ~ fname);
int rate, ch, bits;
if(avc_getAVAudioInfo(audioHandle, &rate, &ch, &bits) != 0)
fail("Unable to get info for music track " ~ fname);
// Translate format from avformat to OpenAL
bufRate = rate;
bufFormat = 0;
// TODO: These don't really fail gracefully for 4 and 6 channels
// if these aren't supported.
if(bits == 8)
{
if(ch == 1) bufFormat = AL_FORMAT_MONO8;
if(ch == 2) bufFormat = AL_FORMAT_STEREO8;
if(alIsExtensionPresent("AL_EXT_MCFORMATS"))
{
if(ch == 4) bufFormat = alGetEnumValue("AL_FORMAT_QUAD8");
if(ch == 6) bufFormat = alGetEnumValue("AL_FORMAT_51CHN8");
}
}
if(bits == 16)
{
if(ch == 1) bufFormat = AL_FORMAT_MONO16;
if(ch == 2) bufFormat = AL_FORMAT_STEREO16;
if(alIsExtensionPresent("AL_EXT_MCFORMATS"))
{
if(ch == 4) bufFormat = alGetEnumValue("AL_FORMAT_QUAD16");
if(ch == 6) bufFormat = alGetEnumValue("AL_FORMAT_51CHN16");
}
}
if(bufFormat == 0)
fail(format("Unhandled format (%d channels, %d bits) for music track %s",
ch, bits, fname));
// Fill the buffers
foreach(int i, ref b; bIDs)
{
int length = avc_getAVAudioData(audioHandle, outData.ptr, outData.length);
if(length) alBufferData(b, bufFormat, outData.ptr, length, bufRate);
if(length == 0 || !noALError())
{
if(i == 0)
fail("No audio data in music track " ~ fname);
alDeleteBuffers(bIDs.length-i, bIDs.ptr+i);
checkALError("running alDeleteBuffers");
bIDs[i..$] = 0;
break;
}
}
// Associate the buffers with the sound id
alSourceQueueBuffers(sID, bIDs.length, bIDs.ptr);
}
void setVolume()
{
float volume = stack.popFloat();
volume = saneVol(volume);
// Set the new volume
if(sID) alSourcef(sID, AL_GAIN, volume);
}
void playSound()
{
if(!sID) return;
alSourcePlay(sID);
checkALError("starting music");
}
void stopSound()
{
// How to stop / pause music
if(sID) alSourcePause(sID);
}
bool isPlaying()
{
ALint state;
alGetSourcei(sID, AL_SOURCE_STATE, &state);
checkALError("getting status");
return state == AL_PLAYING;
}
void updateBuffers()
{
if(!sID || !isPlaying)
return;
// Get the number of processed buffers
ALint count;
alGetSourcei(sID, AL_BUFFERS_PROCESSED, &count);
checkALError("getting number of unprocessed buffers");
for(int i = 0;i < count;i++)
{
int length = avc_getAVAudioData(audioHandle, outData.ptr, outData.length);
if(length <= 0)
break;
ALuint bid;
alSourceUnqueueBuffers(sID, 1, &bid);
if(noALError())
{
alBufferData(bid, bufFormat, outData.ptr, length, bufRate);
alSourceQueueBuffers(sID, 1, &bid);
checkALError("queueing buffers");
}
}
}
}