2012-03-18 21:27:22 +00:00
|
|
|
#include <algorithm>
|
2012-03-17 06:40:07 +00:00
|
|
|
#include <stdexcept>
|
|
|
|
#include <iostream>
|
2012-03-17 09:55:08 +00:00
|
|
|
#include <vector>
|
2012-03-17 06:40:07 +00:00
|
|
|
|
2012-03-19 09:15:08 +00:00
|
|
|
#include <boost/thread.hpp>
|
|
|
|
|
2012-03-17 05:12:17 +00:00
|
|
|
#include "openal_output.hpp"
|
2012-03-17 09:55:08 +00:00
|
|
|
#include "sound_decoder.hpp"
|
|
|
|
#include "sound.hpp"
|
2012-03-18 06:30:43 +00:00
|
|
|
#include "soundmanager.hpp"
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-17 05:12:17 +00:00
|
|
|
|
|
|
|
namespace MWSound
|
|
|
|
{
|
|
|
|
|
|
|
|
static void fail(const std::string &msg)
|
|
|
|
{ throw std::runtime_error("OpenAL exception: " + msg); }
|
|
|
|
|
2012-03-19 14:11:01 +00:00
|
|
|
static void throwALCerror(ALCdevice *device)
|
|
|
|
{
|
|
|
|
ALCenum err = alcGetError(device);
|
|
|
|
if(err != ALC_NO_ERROR)
|
|
|
|
fail(alcGetString(device, err));
|
|
|
|
}
|
|
|
|
|
2012-03-17 09:55:08 +00:00
|
|
|
static void throwALerror()
|
|
|
|
{
|
|
|
|
ALenum err = alGetError();
|
|
|
|
if(err != AL_NO_ERROR)
|
|
|
|
fail(alGetString(err));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-18 18:56:54 +00:00
|
|
|
static ALenum getALFormat(ChannelConfig chans, SampleType type)
|
2012-03-17 09:55:08 +00:00
|
|
|
{
|
2012-03-19 07:49:52 +00:00
|
|
|
static const struct {
|
|
|
|
ALenum format;
|
|
|
|
ChannelConfig chans;
|
|
|
|
SampleType type;
|
|
|
|
} fmtlist[] = {
|
|
|
|
{ AL_FORMAT_MONO16, ChannelConfig_Mono, SampleType_Int16 },
|
|
|
|
{ AL_FORMAT_MONO8, ChannelConfig_Mono, SampleType_UInt8 },
|
|
|
|
{ AL_FORMAT_STEREO16, ChannelConfig_Stereo, SampleType_Int16 },
|
|
|
|
{ AL_FORMAT_STEREO8, ChannelConfig_Stereo, SampleType_UInt8 },
|
|
|
|
};
|
|
|
|
static const size_t fmtlistsize = sizeof(fmtlist)/sizeof(fmtlist[0]);
|
|
|
|
|
|
|
|
for(size_t i = 0;i < fmtlistsize;i++)
|
2012-03-17 09:55:08 +00:00
|
|
|
{
|
2012-03-19 07:49:52 +00:00
|
|
|
if(fmtlist[i].chans == chans && fmtlist[i].type == type)
|
|
|
|
return fmtlist[i].format;
|
2012-03-17 09:55:08 +00:00
|
|
|
}
|
2012-03-19 09:31:40 +00:00
|
|
|
fail(std::string("Unsupported sound format (")+getChannelConfigName(chans)+", "+getSampleTypeName(type)+")");
|
2012-03-17 09:55:08 +00:00
|
|
|
return AL_NONE;
|
|
|
|
}
|
|
|
|
|
2012-03-19 08:33:33 +00:00
|
|
|
//
|
|
|
|
// A streaming OpenAL sound.
|
|
|
|
//
|
2012-03-17 09:55:08 +00:00
|
|
|
class OpenAL_SoundStream : public Sound
|
|
|
|
{
|
2012-03-19 12:29:04 +00:00
|
|
|
static const ALuint sNumBuffers = 6;
|
|
|
|
static const ALfloat sBufferLength = 0.125f;
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-19 09:15:08 +00:00
|
|
|
OpenAL_Output &mOutput;
|
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
ALuint mSource;
|
|
|
|
ALuint mBuffers[sNumBuffers];
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
ALenum mFormat;
|
|
|
|
ALsizei mSampleRate;
|
2012-03-19 12:29:04 +00:00
|
|
|
ALuint mBufferSize;
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
DecoderPtr mDecoder;
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-18 21:27:22 +00:00
|
|
|
volatile bool mIsFinished;
|
|
|
|
|
2012-03-17 09:55:08 +00:00
|
|
|
public:
|
2012-03-19 14:11:01 +00:00
|
|
|
OpenAL_SoundStream(OpenAL_Output &output, ALuint src, DecoderPtr decoder);
|
2012-03-17 10:15:07 +00:00
|
|
|
virtual ~OpenAL_SoundStream();
|
|
|
|
|
2012-03-18 19:03:15 +00:00
|
|
|
virtual void stop();
|
2012-03-17 10:15:07 +00:00
|
|
|
virtual bool isPlaying();
|
2012-03-18 19:03:15 +00:00
|
|
|
virtual void update(const float *pos);
|
2012-03-18 21:27:22 +00:00
|
|
|
|
2012-03-19 14:11:01 +00:00
|
|
|
void play();
|
2012-03-18 21:27:22 +00:00
|
|
|
bool process();
|
2012-03-17 10:15:07 +00:00
|
|
|
};
|
|
|
|
|
2012-03-19 08:33:33 +00:00
|
|
|
//
|
|
|
|
// A background streaming thread (keeps active streams processed)
|
|
|
|
//
|
2012-03-19 09:15:08 +00:00
|
|
|
struct OpenAL_Output::StreamThread {
|
2012-03-18 21:27:22 +00:00
|
|
|
typedef std::vector<OpenAL_SoundStream*> StreamVec;
|
2012-03-19 09:15:08 +00:00
|
|
|
StreamVec mStreams;
|
|
|
|
boost::mutex mMutex;
|
|
|
|
boost::thread mThread;
|
|
|
|
|
|
|
|
StreamThread()
|
|
|
|
: mThread(boost::ref(*this))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
~StreamThread()
|
|
|
|
{
|
|
|
|
mThread.interrupt();
|
|
|
|
}
|
2012-03-18 21:27:22 +00:00
|
|
|
|
2012-03-19 08:33:33 +00:00
|
|
|
// boost::thread entry point
|
2012-03-18 21:27:22 +00:00
|
|
|
void operator()()
|
|
|
|
{
|
|
|
|
while(1)
|
|
|
|
{
|
2012-03-19 09:15:08 +00:00
|
|
|
mMutex.lock();
|
|
|
|
StreamVec::iterator iter = mStreams.begin();
|
|
|
|
while(iter != mStreams.end())
|
2012-03-18 21:27:22 +00:00
|
|
|
{
|
|
|
|
if((*iter)->process() == false)
|
2012-03-19 09:15:08 +00:00
|
|
|
iter = mStreams.erase(iter);
|
2012-03-18 21:27:22 +00:00
|
|
|
else
|
|
|
|
iter++;
|
|
|
|
}
|
2012-03-19 09:15:08 +00:00
|
|
|
mMutex.unlock();
|
2012-03-19 09:19:13 +00:00
|
|
|
boost::this_thread::sleep(boost::posix_time::milliseconds(50));
|
2012-03-18 21:27:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-19 09:15:08 +00:00
|
|
|
void add(OpenAL_SoundStream *stream)
|
2012-03-18 21:27:22 +00:00
|
|
|
{
|
2012-03-19 09:15:08 +00:00
|
|
|
mMutex.lock();
|
|
|
|
if(std::find(mStreams.begin(), mStreams.end(), stream) == mStreams.end())
|
|
|
|
mStreams.push_back(stream);
|
|
|
|
mMutex.unlock();
|
2012-03-18 21:27:22 +00:00
|
|
|
}
|
|
|
|
|
2012-03-19 09:15:08 +00:00
|
|
|
void remove(OpenAL_SoundStream *stream)
|
2012-03-18 21:27:22 +00:00
|
|
|
{
|
2012-03-19 09:15:08 +00:00
|
|
|
mMutex.lock();
|
|
|
|
StreamVec::iterator iter = std::find(mStreams.begin(), mStreams.end(), stream);
|
|
|
|
if(iter != mStreams.end())
|
|
|
|
mStreams.erase(iter);
|
|
|
|
mMutex.unlock();
|
2012-03-18 21:27:22 +00:00
|
|
|
}
|
2012-03-19 07:38:56 +00:00
|
|
|
|
2012-03-19 09:15:08 +00:00
|
|
|
void removeAll()
|
2012-03-19 07:38:56 +00:00
|
|
|
{
|
2012-03-19 09:15:08 +00:00
|
|
|
mMutex.lock();
|
|
|
|
mStreams.clear();
|
|
|
|
mMutex.unlock();
|
2012-03-19 07:38:56 +00:00
|
|
|
}
|
2012-03-18 21:27:22 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2012-03-19 14:11:01 +00:00
|
|
|
OpenAL_SoundStream::OpenAL_SoundStream(OpenAL_Output &output, ALuint src, DecoderPtr decoder)
|
|
|
|
: mOutput(output), mSource(src), mDecoder(decoder), mIsFinished(true)
|
2012-03-17 10:15:07 +00:00
|
|
|
{
|
|
|
|
throwALerror();
|
|
|
|
|
2012-03-19 14:11:01 +00:00
|
|
|
alGenBuffers(sNumBuffers, mBuffers);
|
2012-03-17 10:15:07 +00:00
|
|
|
throwALerror();
|
|
|
|
try
|
|
|
|
{
|
|
|
|
int srate;
|
2012-03-18 18:56:54 +00:00
|
|
|
ChannelConfig chans;
|
|
|
|
SampleType type;
|
2012-03-17 10:15:07 +00:00
|
|
|
|
2012-03-18 18:47:15 +00:00
|
|
|
mDecoder->getInfo(&srate, &chans, &type);
|
2012-03-18 18:30:53 +00:00
|
|
|
mFormat = getALFormat(chans, type);
|
|
|
|
mSampleRate = srate;
|
2012-03-19 12:29:04 +00:00
|
|
|
|
|
|
|
mBufferSize = static_cast<ALuint>(sBufferLength*srate);
|
|
|
|
mBufferSize = framesToBytes(mBufferSize, chans, type);
|
2012-03-17 09:55:08 +00:00
|
|
|
}
|
2012-03-17 10:15:07 +00:00
|
|
|
catch(std::exception &e)
|
2012-03-17 09:55:08 +00:00
|
|
|
{
|
2012-03-19 14:11:01 +00:00
|
|
|
mOutput.mFreeSources.push_back(mSource);
|
2012-03-18 18:30:53 +00:00
|
|
|
alDeleteBuffers(sNumBuffers, mBuffers);
|
2012-03-17 09:55:08 +00:00
|
|
|
alGetError();
|
2012-03-17 10:15:07 +00:00
|
|
|
throw;
|
2012-03-17 09:55:08 +00:00
|
|
|
}
|
2012-03-17 10:15:07 +00:00
|
|
|
}
|
|
|
|
OpenAL_SoundStream::~OpenAL_SoundStream()
|
|
|
|
{
|
2012-03-19 09:15:08 +00:00
|
|
|
mOutput.mStreamThread->remove(this);
|
2012-03-18 21:27:22 +00:00
|
|
|
|
2012-03-19 14:11:01 +00:00
|
|
|
alSourceStop(mSource);
|
|
|
|
alSourcei(mSource, AL_BUFFER, 0);
|
|
|
|
|
|
|
|
mOutput.mFreeSources.push_back(mSource);
|
2012-03-18 18:30:53 +00:00
|
|
|
alDeleteBuffers(sNumBuffers, mBuffers);
|
2012-03-17 10:15:07 +00:00
|
|
|
alGetError();
|
2012-03-18 21:27:22 +00:00
|
|
|
|
2012-03-18 18:47:15 +00:00
|
|
|
mDecoder->close();
|
2012-03-17 10:15:07 +00:00
|
|
|
}
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-19 14:11:01 +00:00
|
|
|
void OpenAL_SoundStream::play()
|
2012-03-17 10:15:07 +00:00
|
|
|
{
|
2012-03-19 12:29:04 +00:00
|
|
|
std::vector<char> data(mBufferSize);
|
2012-03-17 10:15:07 +00:00
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
alSourceStop(mSource);
|
|
|
|
alSourcei(mSource, AL_BUFFER, 0);
|
2012-03-17 10:15:07 +00:00
|
|
|
throwALerror();
|
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
for(ALuint i = 0;i < sNumBuffers;i++)
|
2012-03-17 09:55:08 +00:00
|
|
|
{
|
2012-03-17 10:15:07 +00:00
|
|
|
size_t got;
|
2012-03-18 18:47:15 +00:00
|
|
|
got = mDecoder->read(&data[0], data.size());
|
2012-03-18 18:30:53 +00:00
|
|
|
alBufferData(mBuffers[i], mFormat, &data[0], got, mSampleRate);
|
2012-03-17 10:15:07 +00:00
|
|
|
}
|
|
|
|
throwALerror();
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
alSourceQueueBuffers(mSource, sNumBuffers, mBuffers);
|
|
|
|
alSourcePlay(mSource);
|
2012-03-17 10:15:07 +00:00
|
|
|
throwALerror();
|
2012-03-18 21:27:22 +00:00
|
|
|
|
2012-03-19 09:15:08 +00:00
|
|
|
mOutput.mStreamThread->add(this);
|
2012-03-18 21:27:22 +00:00
|
|
|
mIsFinished = false;
|
2012-03-17 10:15:07 +00:00
|
|
|
}
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-18 19:03:15 +00:00
|
|
|
void OpenAL_SoundStream::stop()
|
2012-03-17 10:15:07 +00:00
|
|
|
{
|
2012-03-19 09:15:08 +00:00
|
|
|
mOutput.mStreamThread->remove(this);
|
2012-03-19 08:33:33 +00:00
|
|
|
mIsFinished = true;
|
2012-03-18 21:27:22 +00:00
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
alSourceStop(mSource);
|
|
|
|
alSourcei(mSource, AL_BUFFER, 0);
|
2012-03-17 10:15:07 +00:00
|
|
|
throwALerror();
|
|
|
|
// FIXME: Rewind decoder
|
|
|
|
}
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-17 10:15:07 +00:00
|
|
|
bool OpenAL_SoundStream::isPlaying()
|
2012-03-18 21:27:22 +00:00
|
|
|
{
|
|
|
|
return !mIsFinished;
|
|
|
|
}
|
|
|
|
|
|
|
|
void OpenAL_SoundStream::update(const float *pos)
|
|
|
|
{
|
|
|
|
alSource3f(mSource, AL_POSITION, pos[0], pos[2], -pos[1]);
|
|
|
|
alSource3f(mSource, AL_DIRECTION, 0.0f, 0.0f, 0.0f);
|
|
|
|
alSource3f(mSource, AL_VELOCITY, 0.0f, 0.0f, 0.0f);
|
|
|
|
throwALerror();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool OpenAL_SoundStream::process()
|
2012-03-17 10:15:07 +00:00
|
|
|
{
|
|
|
|
ALint processed, state;
|
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
alGetSourcei(mSource, AL_SOURCE_STATE, &state);
|
|
|
|
alGetSourcei(mSource, AL_BUFFERS_PROCESSED, &processed);
|
2012-03-17 10:15:07 +00:00
|
|
|
throwALerror();
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-17 10:15:07 +00:00
|
|
|
if(processed > 0)
|
2012-03-17 09:55:08 +00:00
|
|
|
{
|
2012-03-19 12:29:04 +00:00
|
|
|
std::vector<char> data(mBufferSize);
|
2012-03-17 10:15:07 +00:00
|
|
|
do {
|
|
|
|
ALuint bufid;
|
|
|
|
size_t got;
|
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
alSourceUnqueueBuffers(mSource, 1, &bufid);
|
2012-03-17 10:15:07 +00:00
|
|
|
processed--;
|
|
|
|
|
2012-03-18 18:47:15 +00:00
|
|
|
got = mDecoder->read(&data[0], data.size());
|
2012-03-17 10:15:07 +00:00
|
|
|
if(got > 0)
|
|
|
|
{
|
2012-03-18 18:30:53 +00:00
|
|
|
alBufferData(bufid, mFormat, &data[0], got, mSampleRate);
|
|
|
|
alSourceQueueBuffers(mSource, 1, &bufid);
|
2012-03-17 10:15:07 +00:00
|
|
|
}
|
|
|
|
} while(processed > 0);
|
2012-03-17 09:55:08 +00:00
|
|
|
throwALerror();
|
|
|
|
}
|
|
|
|
|
2012-03-17 10:15:07 +00:00
|
|
|
if(state != AL_PLAYING && state != AL_PAUSED)
|
2012-03-17 09:55:08 +00:00
|
|
|
{
|
2012-03-17 10:15:07 +00:00
|
|
|
ALint queued;
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
alGetSourcei(mSource, AL_BUFFERS_QUEUED, &queued);
|
2012-03-17 09:55:08 +00:00
|
|
|
throwALerror();
|
2012-03-17 10:15:07 +00:00
|
|
|
if(queued == 0)
|
2012-03-18 21:27:22 +00:00
|
|
|
{
|
|
|
|
mIsFinished = true;
|
2012-03-17 10:15:07 +00:00
|
|
|
return false;
|
2012-03-18 21:27:22 +00:00
|
|
|
}
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
alSourcePlay(mSource);
|
2012-03-17 10:15:07 +00:00
|
|
|
throwALerror();
|
2012-03-17 09:55:08 +00:00
|
|
|
}
|
2012-03-17 10:15:07 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-19 08:33:33 +00:00
|
|
|
//
|
|
|
|
// A regular OpenAL sound
|
|
|
|
//
|
|
|
|
class OpenAL_Sound : public Sound
|
|
|
|
{
|
2012-03-19 14:11:01 +00:00
|
|
|
OpenAL_Output &mOutput;
|
|
|
|
|
2012-03-19 08:33:33 +00:00
|
|
|
ALuint mSource;
|
|
|
|
ALuint mBuffer;
|
|
|
|
public:
|
2012-03-19 14:11:01 +00:00
|
|
|
OpenAL_Sound(OpenAL_Output &output, ALuint src, ALuint buf);
|
2012-03-19 08:33:33 +00:00
|
|
|
virtual ~OpenAL_Sound();
|
|
|
|
|
|
|
|
virtual void stop();
|
|
|
|
virtual bool isPlaying();
|
|
|
|
virtual void update(const float *pos);
|
|
|
|
|
|
|
|
static ALuint LoadBuffer(DecoderPtr decoder);
|
|
|
|
};
|
|
|
|
|
|
|
|
ALuint OpenAL_Sound::LoadBuffer(DecoderPtr decoder)
|
|
|
|
{
|
|
|
|
int srate;
|
|
|
|
ChannelConfig chans;
|
|
|
|
SampleType type;
|
|
|
|
ALenum format;
|
|
|
|
|
|
|
|
decoder->getInfo(&srate, &chans, &type);
|
|
|
|
format = getALFormat(chans, type);
|
|
|
|
|
|
|
|
std::vector<char> data(32768);
|
|
|
|
size_t got, total = 0;
|
|
|
|
while((got=decoder->read(&data[total], data.size()-total)) > 0)
|
|
|
|
{
|
|
|
|
total += got;
|
|
|
|
data.resize(total*2);
|
|
|
|
}
|
|
|
|
data.resize(total);
|
|
|
|
|
2012-03-19 14:11:01 +00:00
|
|
|
ALuint buf=0;
|
2012-03-19 08:33:33 +00:00
|
|
|
alGenBuffers(1, &buf);
|
|
|
|
alBufferData(buf, format, &data[0], total, srate);
|
|
|
|
return buf;
|
|
|
|
}
|
2012-03-17 05:12:17 +00:00
|
|
|
|
2012-03-19 14:11:01 +00:00
|
|
|
OpenAL_Sound::OpenAL_Sound(OpenAL_Output &output, ALuint src, ALuint buf)
|
|
|
|
: mOutput(output), mSource(src), mBuffer(buf)
|
2012-03-17 16:15:47 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
OpenAL_Sound::~OpenAL_Sound()
|
|
|
|
{
|
2012-03-19 14:11:01 +00:00
|
|
|
alSourceStop(mSource);
|
|
|
|
alSourcei(mSource, AL_BUFFER, 0);
|
|
|
|
|
|
|
|
mOutput.mFreeSources.push_back(mSource);
|
2012-03-18 18:30:53 +00:00
|
|
|
alDeleteBuffers(1, &mBuffer);
|
2012-03-17 16:15:47 +00:00
|
|
|
alGetError();
|
|
|
|
}
|
|
|
|
|
2012-03-18 19:03:15 +00:00
|
|
|
void OpenAL_Sound::stop()
|
2012-03-17 16:15:47 +00:00
|
|
|
{
|
2012-03-18 18:30:53 +00:00
|
|
|
alSourceStop(mSource);
|
2012-03-17 16:15:47 +00:00
|
|
|
throwALerror();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool OpenAL_Sound::isPlaying()
|
|
|
|
{
|
|
|
|
ALint state;
|
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
alGetSourcei(mSource, AL_SOURCE_STATE, &state);
|
2012-03-17 16:15:47 +00:00
|
|
|
throwALerror();
|
|
|
|
|
|
|
|
return state==AL_PLAYING;
|
|
|
|
}
|
|
|
|
|
2012-03-18 19:03:15 +00:00
|
|
|
void OpenAL_Sound::update(const float *pos)
|
2012-03-17 16:51:03 +00:00
|
|
|
{
|
2012-03-18 18:30:53 +00:00
|
|
|
alSource3f(mSource, AL_POSITION, pos[0], pos[2], -pos[1]);
|
|
|
|
alSource3f(mSource, AL_DIRECTION, 0.0f, 0.0f, 0.0f);
|
|
|
|
alSource3f(mSource, AL_VELOCITY, 0.0f, 0.0f, 0.0f);
|
2012-03-17 16:51:03 +00:00
|
|
|
throwALerror();
|
|
|
|
}
|
|
|
|
|
2012-03-17 16:15:47 +00:00
|
|
|
|
2012-03-19 08:33:33 +00:00
|
|
|
//
|
|
|
|
// An OpenAL output device
|
|
|
|
//
|
2012-03-18 19:19:54 +00:00
|
|
|
void OpenAL_Output::init(const std::string &devname)
|
2012-03-17 05:12:17 +00:00
|
|
|
{
|
2012-03-18 19:19:54 +00:00
|
|
|
if(mDevice || mContext)
|
|
|
|
fail("Device already open");
|
2012-03-17 05:12:17 +00:00
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
mDevice = alcOpenDevice(devname.c_str());
|
|
|
|
if(!mDevice)
|
2012-03-18 19:19:54 +00:00
|
|
|
fail("Failed to open \""+devname+"\"");
|
2012-03-18 18:30:53 +00:00
|
|
|
std::cout << "Opened \""<<alcGetString(mDevice, ALC_DEVICE_SPECIFIER)<<"\"" << std::endl;
|
2012-03-17 05:12:17 +00:00
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
mContext = alcCreateContext(mDevice, NULL);
|
|
|
|
if(!mContext || alcMakeContextCurrent(mContext) == ALC_FALSE)
|
2012-03-18 19:19:54 +00:00
|
|
|
fail(std::string("Failed to setup context: ")+alcGetString(mDevice, alcGetError(mDevice)));
|
|
|
|
|
2012-03-18 16:05:38 +00:00
|
|
|
alDistanceModel(AL_LINEAR_DISTANCE_CLAMPED);
|
|
|
|
throwALerror();
|
2012-03-19 14:11:01 +00:00
|
|
|
|
|
|
|
ALCint maxmono, maxstereo;
|
|
|
|
alcGetIntegerv(mDevice, ALC_MONO_SOURCES, 1, &maxmono);
|
|
|
|
alcGetIntegerv(mDevice, ALC_STEREO_SOURCES, 1, &maxstereo);
|
|
|
|
throwALCerror(mDevice);
|
|
|
|
|
|
|
|
mFreeSources.resize(std::min(maxmono+maxstereo, 256));
|
|
|
|
for(size_t i = 0;i < mFreeSources.size();i++)
|
|
|
|
{
|
|
|
|
ALuint src;
|
|
|
|
alGenSources(1, &src);
|
|
|
|
if(alGetError() != AL_NO_ERROR)
|
|
|
|
{
|
|
|
|
mFreeSources.resize(i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
mFreeSources[i] = src;
|
|
|
|
}
|
|
|
|
if(mFreeSources.size() == 0)
|
|
|
|
fail("Could not allocate any sources");
|
2012-03-17 05:12:17 +00:00
|
|
|
}
|
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
void OpenAL_Output::deinit()
|
2012-03-17 05:12:17 +00:00
|
|
|
{
|
2012-03-19 14:11:01 +00:00
|
|
|
mStreamThread->removeAll();
|
2012-03-19 07:38:56 +00:00
|
|
|
|
2012-03-19 14:11:01 +00:00
|
|
|
if(!mFreeSources.empty())
|
|
|
|
{
|
|
|
|
alDeleteSources(mFreeSources.size(), mFreeSources.data());
|
|
|
|
mFreeSources.clear();
|
|
|
|
}
|
2012-03-17 05:12:17 +00:00
|
|
|
alcMakeContextCurrent(0);
|
2012-03-18 18:30:53 +00:00
|
|
|
if(mContext)
|
|
|
|
alcDestroyContext(mContext);
|
|
|
|
mContext = 0;
|
|
|
|
if(mDevice)
|
|
|
|
alcCloseDevice(mDevice);
|
|
|
|
mDevice = 0;
|
2012-03-17 05:12:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
Sound* OpenAL_Output::playSound(const std::string &fname, float volume, float pitch, bool loop)
|
2012-03-17 13:51:44 +00:00
|
|
|
{
|
2012-03-17 16:15:47 +00:00
|
|
|
throwALerror();
|
|
|
|
|
2012-03-19 14:11:01 +00:00
|
|
|
std::auto_ptr<OpenAL_Sound> sound;
|
2012-03-17 16:15:47 +00:00
|
|
|
ALuint src=0, buf=0;
|
2012-03-19 14:11:01 +00:00
|
|
|
|
|
|
|
if(mFreeSources.empty())
|
|
|
|
fail("No free sources");
|
|
|
|
src = mFreeSources.back();
|
|
|
|
mFreeSources.pop_back();
|
|
|
|
|
2012-03-17 16:15:47 +00:00
|
|
|
try
|
|
|
|
{
|
2012-03-19 14:11:01 +00:00
|
|
|
DecoderPtr decoder = mManager.getDecoder();
|
|
|
|
decoder->open(fname);
|
2012-03-19 08:33:33 +00:00
|
|
|
buf = OpenAL_Sound::LoadBuffer(decoder);
|
2012-03-17 16:15:47 +00:00
|
|
|
throwALerror();
|
2012-03-19 14:11:01 +00:00
|
|
|
decoder->close();
|
|
|
|
|
|
|
|
sound.reset(new OpenAL_Sound(*this, src, buf));
|
2012-03-17 16:15:47 +00:00
|
|
|
}
|
|
|
|
catch(std::exception &e)
|
|
|
|
{
|
2012-03-19 14:11:01 +00:00
|
|
|
mFreeSources.push_back(src);
|
2012-03-17 16:15:47 +00:00
|
|
|
if(alIsBuffer(buf))
|
|
|
|
alDeleteBuffers(1, &buf);
|
|
|
|
alGetError();
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
|
|
|
|
alSource3f(src, AL_POSITION, 0.0f, 0.0f, 0.0f);
|
|
|
|
alSource3f(src, AL_DIRECTION, 0.0f, 0.0f, 0.0f);
|
|
|
|
alSource3f(src, AL_VELOCITY, 0.0f, 0.0f, 0.0f);
|
|
|
|
|
|
|
|
alSourcef(src, AL_REFERENCE_DISTANCE, 1.0f);
|
|
|
|
alSourcef(src, AL_MAX_DISTANCE, 1000.0f);
|
|
|
|
alSourcef(src, AL_ROLLOFF_FACTOR, 0.0f);
|
|
|
|
|
|
|
|
alSourcef(src, AL_GAIN, volume);
|
|
|
|
alSourcef(src, AL_PITCH, pitch);
|
|
|
|
|
|
|
|
alSourcei(src, AL_SOURCE_RELATIVE, AL_TRUE);
|
|
|
|
alSourcei(src, AL_LOOPING, (loop?AL_TRUE:AL_FALSE));
|
|
|
|
throwALerror();
|
|
|
|
|
|
|
|
alSourcei(src, AL_BUFFER, buf);
|
|
|
|
alSourcePlay(src);
|
|
|
|
throwALerror();
|
|
|
|
|
|
|
|
return sound.release();
|
2012-03-17 13:51:44 +00:00
|
|
|
}
|
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
Sound* OpenAL_Output::playSound3D(const std::string &fname, const float *pos, float volume, float pitch,
|
2012-03-17 16:15:47 +00:00
|
|
|
float min, float max, bool loop)
|
2012-03-17 13:51:44 +00:00
|
|
|
{
|
2012-03-17 16:15:47 +00:00
|
|
|
throwALerror();
|
|
|
|
|
2012-03-19 14:11:01 +00:00
|
|
|
std::auto_ptr<OpenAL_Sound> sound;
|
2012-03-17 16:15:47 +00:00
|
|
|
ALuint src=0, buf=0;
|
2012-03-19 14:11:01 +00:00
|
|
|
|
|
|
|
if(mFreeSources.empty())
|
|
|
|
fail("No free sources");
|
|
|
|
src = mFreeSources.back();
|
|
|
|
mFreeSources.pop_back();
|
|
|
|
|
2012-03-17 16:15:47 +00:00
|
|
|
try
|
|
|
|
{
|
2012-03-19 14:11:01 +00:00
|
|
|
DecoderPtr decoder = mManager.getDecoder();
|
|
|
|
decoder->open(fname);
|
2012-03-19 08:33:33 +00:00
|
|
|
buf = OpenAL_Sound::LoadBuffer(decoder);
|
2012-03-17 16:15:47 +00:00
|
|
|
throwALerror();
|
2012-03-19 14:11:01 +00:00
|
|
|
decoder->close();
|
|
|
|
|
|
|
|
sound.reset(new OpenAL_Sound(*this, src, buf));
|
2012-03-17 16:15:47 +00:00
|
|
|
}
|
|
|
|
catch(std::exception &e)
|
|
|
|
{
|
2012-03-19 14:11:01 +00:00
|
|
|
mFreeSources.push_back(src);
|
2012-03-17 16:15:47 +00:00
|
|
|
if(alIsBuffer(buf))
|
|
|
|
alDeleteBuffers(1, &buf);
|
|
|
|
alGetError();
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
|
|
|
|
alSource3f(src, AL_POSITION, pos[0], pos[2], -pos[1]);
|
|
|
|
alSource3f(src, AL_DIRECTION, 0.0f, 0.0f, 0.0f);
|
|
|
|
alSource3f(src, AL_VELOCITY, 0.0f, 0.0f, 0.0f);
|
|
|
|
|
|
|
|
alSourcef(src, AL_REFERENCE_DISTANCE, min);
|
|
|
|
alSourcef(src, AL_MAX_DISTANCE, max);
|
|
|
|
alSourcef(src, AL_ROLLOFF_FACTOR, 1.0f);
|
|
|
|
|
|
|
|
alSourcef(src, AL_GAIN, volume);
|
|
|
|
alSourcef(src, AL_PITCH, pitch);
|
|
|
|
|
|
|
|
alSourcei(src, AL_SOURCE_RELATIVE, AL_FALSE);
|
|
|
|
alSourcei(src, AL_LOOPING, (loop?AL_TRUE:AL_FALSE));
|
|
|
|
throwALerror();
|
|
|
|
|
|
|
|
alSourcei(src, AL_BUFFER, buf);
|
|
|
|
alSourcePlay(src);
|
|
|
|
throwALerror();
|
|
|
|
|
|
|
|
return sound.release();
|
2012-03-17 13:51:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
Sound* OpenAL_Output::streamSound(const std::string &fname, float volume, float pitch)
|
2012-03-17 09:55:08 +00:00
|
|
|
{
|
2012-03-19 14:11:01 +00:00
|
|
|
throwALerror();
|
|
|
|
|
2012-03-17 09:55:08 +00:00
|
|
|
std::auto_ptr<OpenAL_SoundStream> sound;
|
2012-03-19 14:11:01 +00:00
|
|
|
ALuint src;
|
|
|
|
|
|
|
|
if(mFreeSources.empty())
|
|
|
|
fail("No free sources");
|
|
|
|
src = mFreeSources.back();
|
|
|
|
mFreeSources.pop_back();
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
DecoderPtr decoder = mManager.getDecoder();
|
|
|
|
decoder->open(fname);
|
|
|
|
sound.reset(new OpenAL_SoundStream(*this, src, decoder));
|
|
|
|
}
|
|
|
|
catch(std::exception &e)
|
|
|
|
{
|
|
|
|
mFreeSources.push_back(src);
|
|
|
|
throw;
|
|
|
|
}
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-19 14:11:01 +00:00
|
|
|
alSource3f(src, AL_POSITION, 0.0f, 0.0f, 0.0f);
|
|
|
|
alSource3f(src, AL_DIRECTION, 0.0f, 0.0f, 0.0f);
|
|
|
|
alSource3f(src, AL_VELOCITY, 0.0f, 0.0f, 0.0f);
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-19 14:11:01 +00:00
|
|
|
alSourcef(src, AL_REFERENCE_DISTANCE, 1.0f);
|
|
|
|
alSourcef(src, AL_MAX_DISTANCE, 1000.0f);
|
|
|
|
alSourcef(src, AL_ROLLOFF_FACTOR, 0.0f);
|
|
|
|
|
|
|
|
alSourcef(src, AL_GAIN, volume);
|
|
|
|
alSourcef(src, AL_PITCH, pitch);
|
|
|
|
|
|
|
|
alSourcei(src, AL_SOURCE_RELATIVE, AL_TRUE);
|
|
|
|
alSourcei(src, AL_LOOPING, AL_FALSE);
|
|
|
|
throwALerror();
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-19 14:11:01 +00:00
|
|
|
sound->play();
|
2012-03-17 09:55:08 +00:00
|
|
|
return sound.release();
|
|
|
|
}
|
|
|
|
|
2012-03-19 14:28:03 +00:00
|
|
|
Sound* OpenAL_Output::streamSound3D(const std::string &fname, const float *pos, float volume, float pitch,
|
|
|
|
float min, float max)
|
|
|
|
{
|
|
|
|
throwALerror();
|
|
|
|
|
|
|
|
std::auto_ptr<OpenAL_SoundStream> sound;
|
|
|
|
ALuint src;
|
|
|
|
|
|
|
|
if(mFreeSources.empty())
|
|
|
|
fail("No free sources");
|
|
|
|
src = mFreeSources.back();
|
|
|
|
mFreeSources.pop_back();
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
DecoderPtr decoder = mManager.getDecoder();
|
|
|
|
decoder->open(fname);
|
|
|
|
sound.reset(new OpenAL_SoundStream(*this, src, decoder));
|
|
|
|
}
|
|
|
|
catch(std::exception &e)
|
|
|
|
{
|
|
|
|
mFreeSources.push_back(src);
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
|
|
|
|
alSource3f(src, AL_POSITION, pos[0], pos[2], -pos[1]);
|
|
|
|
alSource3f(src, AL_DIRECTION, 0.0f, 0.0f, 0.0f);
|
|
|
|
alSource3f(src, AL_VELOCITY, 0.0f, 0.0f, 0.0f);
|
|
|
|
|
|
|
|
alSourcef(src, AL_REFERENCE_DISTANCE, min);
|
|
|
|
alSourcef(src, AL_MAX_DISTANCE, max);
|
|
|
|
alSourcef(src, AL_ROLLOFF_FACTOR, 1.0f);
|
|
|
|
|
|
|
|
alSourcef(src, AL_GAIN, volume);
|
|
|
|
alSourcef(src, AL_PITCH, pitch);
|
|
|
|
|
|
|
|
alSourcei(src, AL_SOURCE_RELATIVE, AL_FALSE);
|
|
|
|
alSourcei(src, AL_LOOPING, AL_FALSE);
|
|
|
|
throwALerror();
|
|
|
|
|
|
|
|
sound->play();
|
|
|
|
return sound.release();
|
|
|
|
}
|
|
|
|
|
2012-03-17 09:55:08 +00:00
|
|
|
|
2012-03-18 18:30:53 +00:00
|
|
|
void OpenAL_Output::updateListener(const float *pos, const float *atdir, const float *updir)
|
2012-03-17 11:22:54 +00:00
|
|
|
{
|
2012-03-17 17:36:34 +00:00
|
|
|
float orient[6] = {
|
|
|
|
atdir[0], atdir[2], -atdir[1],
|
|
|
|
updir[0], updir[2], -updir[1]
|
|
|
|
};
|
|
|
|
|
|
|
|
alListener3f(AL_POSITION, pos[0], pos[2], -pos[1]);
|
2012-03-17 11:22:54 +00:00
|
|
|
alListenerfv(AL_ORIENTATION, orient);
|
|
|
|
throwALerror();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-17 05:12:17 +00:00
|
|
|
OpenAL_Output::OpenAL_Output(SoundManager &mgr)
|
2012-03-19 09:15:08 +00:00
|
|
|
: Sound_Output(mgr), mDevice(0), mContext(0), mStreamThread(new StreamThread)
|
2012-03-17 05:12:17 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
OpenAL_Output::~OpenAL_Output()
|
|
|
|
{
|
2012-03-18 18:30:53 +00:00
|
|
|
deinit();
|
2012-03-17 05:12:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|