#include #include #include #include #include #include "openal_output.hpp" #include "sound_decoder.hpp" #include "sound.hpp" #include "soundmanagerimp.hpp" #ifndef ALC_ALL_DEVICES_SPECIFIER #define ALC_ALL_DEVICES_SPECIFIER 0x1013 #endif namespace MWSound { static void fail(const std::string &msg) { throw std::runtime_error("OpenAL exception: " + msg); } static void throwALCerror(ALCdevice *device) { ALCenum err = alcGetError(device); if(err != ALC_NO_ERROR) { const ALCchar *errstring = alcGetString(device, err); fail(errstring ? errstring : ""); } } static void throwALerror() { ALenum err = alGetError(); if(err != AL_NO_ERROR) { const ALchar *errstring = alGetString(err); fail(errstring ? errstring : ""); } } static ALenum getALFormat(ChannelConfig chans, SampleType type) { 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++) { if(fmtlist[i].chans == chans && fmtlist[i].type == type) return fmtlist[i].format; } fail(std::string("Unsupported sound format (")+getChannelConfigName(chans)+", "+getSampleTypeName(type)+")"); return AL_NONE; } static ALint getBufferSampleCount(ALuint buf) { ALint size, bits, channels; alGetBufferi(buf, AL_SIZE, &size); alGetBufferi(buf, AL_BITS, &bits); alGetBufferi(buf, AL_CHANNELS, &channels); throwALerror(); return size / channels * 8 / bits; } // // A streaming OpenAL sound. // class OpenAL_SoundStream : public Sound { static const ALuint sNumBuffers = 6; static const ALfloat sBufferLength; OpenAL_Output &mOutput; ALuint mSource; ALuint mBuffers[sNumBuffers]; ALenum mFormat; ALsizei mSampleRate; ALuint mBufferSize; ALuint mSamplesTotal; ALuint mSamplesQueued; DecoderPtr mDecoder; volatile bool mIsFinished; OpenAL_SoundStream(const OpenAL_SoundStream &rhs); OpenAL_SoundStream& operator=(const OpenAL_SoundStream &rhs); public: OpenAL_SoundStream(OpenAL_Output &output, ALuint src, DecoderPtr decoder); virtual ~OpenAL_SoundStream(); virtual void stop(); virtual bool isPlaying(); virtual double getTimeOffset(); virtual void update(); void play(); bool process(); }; const ALfloat OpenAL_SoundStream::sBufferLength = 0.125f; // // A background streaming thread (keeps active streams processed) // struct OpenAL_Output::StreamThread { typedef std::vector StreamVec; StreamVec mStreams; boost::mutex mMutex; boost::thread mThread; StreamThread() : mThread(boost::ref(*this)) { } ~StreamThread() { mThread.interrupt(); } // boost::thread entry point void operator()() { while(1) { mMutex.lock(); StreamVec::iterator iter = mStreams.begin(); while(iter != mStreams.end()) { if((*iter)->process() == false) iter = mStreams.erase(iter); else iter++; } mMutex.unlock(); boost::this_thread::sleep(boost::posix_time::milliseconds(50)); } } void add(OpenAL_SoundStream *stream) { mMutex.lock(); if(std::find(mStreams.begin(), mStreams.end(), stream) == mStreams.end()) mStreams.push_back(stream); mMutex.unlock(); } void remove(OpenAL_SoundStream *stream) { mMutex.lock(); StreamVec::iterator iter = std::find(mStreams.begin(), mStreams.end(), stream); if(iter != mStreams.end()) mStreams.erase(iter); mMutex.unlock(); } void removeAll() { mMutex.lock(); mStreams.clear(); mMutex.unlock(); } private: StreamThread(const StreamThread &rhs); StreamThread& operator=(const StreamThread &rhs); }; OpenAL_SoundStream::OpenAL_SoundStream(OpenAL_Output &output, ALuint src, DecoderPtr decoder) : mOutput(output), mSource(src), mSamplesTotal(0), mSamplesQueued(0), mDecoder(decoder), mIsFinished(true) { throwALerror(); alGenBuffers(sNumBuffers, mBuffers); throwALerror(); try { int srate; ChannelConfig chans; SampleType type; mDecoder->getInfo(&srate, &chans, &type); mFormat = getALFormat(chans, type); mSampleRate = srate; mBufferSize = static_cast(sBufferLength*srate); mBufferSize = framesToBytes(mBufferSize, chans, type); } catch(std::exception &e) { alDeleteBuffers(sNumBuffers, mBuffers); alGetError(); throw; } } OpenAL_SoundStream::~OpenAL_SoundStream() { mOutput.mStreamThread->remove(this); alSourceStop(mSource); alSourcei(mSource, AL_BUFFER, 0); mOutput.mFreeSources.push_back(mSource); alDeleteBuffers(sNumBuffers, mBuffers); alGetError(); mDecoder->close(); } void OpenAL_SoundStream::play() { alSourceStop(mSource); alSourcei(mSource, AL_BUFFER, 0); throwALerror(); mSamplesQueued = 0; for(ALuint i = 0;i < sNumBuffers;i++) alBufferData(mBuffers[i], mFormat, this, 0, mSampleRate); throwALerror(); alSourceQueueBuffers(mSource, sNumBuffers, mBuffers); alSourcePlay(mSource); throwALerror(); mIsFinished = false; mOutput.mStreamThread->add(this); } void OpenAL_SoundStream::stop() { mOutput.mStreamThread->remove(this); mIsFinished = true; alSourceStop(mSource); alSourcei(mSource, AL_BUFFER, 0); throwALerror(); mSamplesQueued = 0; mDecoder->rewind(); mSamplesTotal = 0; } bool OpenAL_SoundStream::isPlaying() { ALint state; alGetSourcei(mSource, AL_SOURCE_STATE, &state); throwALerror(); if(state == AL_PLAYING || state == AL_PAUSED) return true; return !mIsFinished; } double OpenAL_SoundStream::getTimeOffset() { ALint state = AL_STOPPED; ALfloat offset = 0.0f; double t; mOutput.mStreamThread->mMutex.lock(); alGetSourcef(mSource, AL_SEC_OFFSET, &offset); alGetSourcei(mSource, AL_SOURCE_STATE, &state); if(state == AL_PLAYING || state == AL_PAUSED) t = (double)(mSamplesTotal - mSamplesQueued)/(double)mSampleRate + offset; else t = (double)mSamplesTotal / (double)mSampleRate; mOutput.mStreamThread->mMutex.unlock(); throwALerror(); return t; } void OpenAL_SoundStream::update() { ALfloat gain = mVolume*mBaseVolume; ALfloat pitch = mPitch; if(!(mFlags&MWBase::SoundManager::Play_NoEnv) && mOutput.mLastEnvironment == Env_Underwater) { gain *= 0.9f; pitch *= 0.7f; } alSourcef(mSource, AL_GAIN, gain); alSourcef(mSource, AL_PITCH, pitch); alSource3f(mSource, AL_POSITION, mPos[0], mPos[2], -mPos[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() { try { bool finished = mIsFinished; ALint samples_unqueued = 0; ALint samples_queued = 0; ALint processed, state; alGetSourcei(mSource, AL_SOURCE_STATE, &state); alGetSourcei(mSource, AL_BUFFERS_PROCESSED, &processed); throwALerror(); if(processed > 0) { std::vector data(mBufferSize); do { ALuint bufid = 0; size_t got; alSourceUnqueueBuffers(mSource, 1, &bufid); samples_unqueued += getBufferSampleCount(bufid); processed--; if(finished) continue; got = mDecoder->read(&data[0], data.size()); finished = (got < data.size()); if(got > 0) { alBufferData(bufid, mFormat, &data[0], got, mSampleRate); alSourceQueueBuffers(mSource, 1, &bufid); samples_queued += getBufferSampleCount(bufid); } } while(processed > 0); throwALerror(); } if(state != AL_PLAYING && state != AL_PAUSED) { ALint queued = 0; alGetSourcei(mSource, AL_BUFFERS_QUEUED, &queued); if(queued > 0) alSourcePlay(mSource); throwALerror(); } mSamplesQueued -= samples_unqueued; mSamplesQueued += samples_queued; mSamplesTotal += samples_queued; mIsFinished = finished; } catch(std::exception &e) { std::cout<< "Error updating stream \""<getName()<<"\"" < mMaxDistance*mMaxDistance) gain = 0.0f; else if(!(mFlags&MWBase::SoundManager::Play_NoEnv) && mOutput.mLastEnvironment == Env_Underwater) { gain *= 0.9f; pitch *= 0.7f; } alSourcef(mSource, AL_GAIN, gain); alSourcef(mSource, AL_PITCH, pitch); alSource3f(mSource, AL_POSITION, mPos[0], mPos[2], -mPos[1]); alSource3f(mSource, AL_DIRECTION, 0.0f, 0.0f, 0.0f); alSource3f(mSource, AL_VELOCITY, 0.0f, 0.0f, 0.0f); throwALerror(); } // // An OpenAL output device // std::vector OpenAL_Output::enumerate() { std::vector devlist; const ALCchar *devnames; if(alcIsExtensionPresent(NULL, "ALC_ENUMERATE_ALL_EXT")) devnames = alcGetString(NULL, ALC_ALL_DEVICES_SPECIFIER); else devnames = alcGetString(NULL, ALC_DEVICE_SPECIFIER); while(devnames && *devnames) { devlist.push_back(devnames); devnames += strlen(devnames)+1; } return devlist; } void OpenAL_Output::init(const std::string &devname) { deinit(); mDevice = alcOpenDevice(devname.c_str()); if(!mDevice) { if(devname.empty()) fail("Failed to open default device"); else fail("Failed to open \""+devname+"\""); } else { const ALCchar *name = NULL; if(alcIsExtensionPresent(mDevice, "ALC_ENUMERATE_ALL_EXT")) name = alcGetString(mDevice, ALC_ALL_DEVICES_SPECIFIER); if(alcGetError(mDevice) != AL_NO_ERROR || !name) name = alcGetString(mDevice, ALC_DEVICE_SPECIFIER); std::cout << "Opened \""<(maxmono+maxstereo, 256); if (maxtotal == 0) // workaround for broken implementations maxtotal = 256; for(size_t i = 0;i < maxtotal;i++) { ALuint src = 0; alGenSources(1, &src); throwALerror(); mSources.push_back(src); } mFreeSources.insert(mFreeSources.begin(), mSources.begin(), mSources.end()); } catch(std::exception &e) { std::cout <<"Error: "<removeAll(); mFreeSources.clear(); if(mSources.size() > 0) alDeleteSources(mSources.size(), &mSources[0]); mSources.clear(); mBufferRefs.clear(); mUnusedBuffers.clear(); while(!mBufferCache.empty()) { alDeleteBuffers(1, &mBufferCache.begin()->second); mBufferCache.erase(mBufferCache.begin()); } alcMakeContextCurrent(0); if(mContext) alcDestroyContext(mContext); mContext = 0; if(mDevice) alcCloseDevice(mDevice); mDevice = 0; mInitialized = false; } ALuint OpenAL_Output::getBuffer(const std::string &fname) { ALuint buf = 0; NameMap::iterator iditer = mBufferCache.find(fname); if(iditer != mBufferCache.end()) { buf = iditer->second; if(mBufferRefs[buf]++ == 0) { IDDq::iterator iter = std::find(mUnusedBuffers.begin(), mUnusedBuffers.end(), buf); if(iter != mUnusedBuffers.end()) mUnusedBuffers.erase(iter); } return buf; } throwALerror(); std::vector data; ChannelConfig chans; SampleType type; ALenum format; int srate; DecoderPtr decoder = mManager.getDecoder(); try { decoder->open(fname); } catch(Ogre::FileNotFoundException &e) { std::string::size_type pos = fname.rfind('.'); if(pos == std::string::npos) throw; decoder->open(fname.substr(0, pos)+".mp3"); } decoder->getInfo(&srate, &chans, &type); format = getALFormat(chans, type); decoder->readAll(data); decoder->close(); alGenBuffers(1, &buf); throwALerror(); alBufferData(buf, format, &data[0], data.size(), srate); mBufferCache[fname] = buf; mBufferRefs[buf] = 1; ALint bufsize = 0; alGetBufferi(buf, AL_SIZE, &bufsize); mBufferCacheMemSize += bufsize; // NOTE: Max buffer cache: 15MB while(mBufferCacheMemSize > 15*1024*1024) { if(mUnusedBuffers.empty()) { std::cout <<"No more unused buffers to clear!"<< std::endl; break; } ALuint oldbuf = mUnusedBuffers.front(); mUnusedBuffers.pop_front(); NameMap::iterator nameiter = mBufferCache.begin(); while(nameiter != mBufferCache.end()) { if(nameiter->second == oldbuf) mBufferCache.erase(nameiter++); else nameiter++; } bufsize = 0; alGetBufferi(oldbuf, AL_SIZE, &bufsize); alDeleteBuffers(1, &oldbuf); mBufferCacheMemSize -= bufsize; } return buf; } void OpenAL_Output::bufferFinished(ALuint buf) { if(mBufferRefs.at(buf)-- == 1) { mBufferRefs.erase(mBufferRefs.find(buf)); mUnusedBuffers.push_back(buf); } } MWBase::SoundPtr OpenAL_Output::playSound(const std::string &fname, float volume, float pitch, int flags) { boost::shared_ptr sound; ALuint src=0, buf=0; if(mFreeSources.empty()) fail("No free sources"); src = mFreeSources.front(); mFreeSources.pop_front(); try { buf = getBuffer(fname); sound.reset(new OpenAL_Sound(*this, src, buf)); } catch(std::exception &e) { mFreeSources.push_back(src); if(buf && alIsBuffer(buf)) bufferFinished(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); if(!(flags&MWBase::SoundManager::Play_NoEnv) && mLastEnvironment == Env_Underwater) { volume *= 0.9f; pitch *= 0.7f; } alSourcef(src, AL_GAIN, volume); alSourcef(src, AL_PITCH, pitch); alSourcei(src, AL_SOURCE_RELATIVE, AL_TRUE); alSourcei(src, AL_LOOPING, (flags&MWBase::SoundManager::Play_Loop) ? AL_TRUE : AL_FALSE); throwALerror(); alSourcei(src, AL_BUFFER, buf); alSourcePlay(src); throwALerror(); return sound; } MWBase::SoundPtr OpenAL_Output::playSound3D(const std::string &fname, const Ogre::Vector3 &pos, float volume, float pitch, float min, float max, int flags) { boost::shared_ptr sound; ALuint src=0, buf=0; if(mFreeSources.empty()) fail("No free sources"); src = mFreeSources.front(); mFreeSources.pop_front(); try { buf = getBuffer(fname); sound.reset(new OpenAL_Sound3D(*this, src, buf)); } catch(std::exception &e) { mFreeSources.push_back(src); if(buf && alIsBuffer(buf)) bufferFinished(buf); alGetError(); throw; } alSource3f(src, AL_POSITION, pos.x, pos.z, -pos.y); 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); if(!(flags&MWBase::SoundManager::Play_NoEnv) && mLastEnvironment == Env_Underwater) { volume *= 0.9f; pitch *= 0.7f; } alSourcef(src, AL_GAIN, (pos.squaredDistance(mPos) > max*max) ? 0.0f : volume); alSourcef(src, AL_PITCH, pitch); alSourcei(src, AL_SOURCE_RELATIVE, AL_FALSE); alSourcei(src, AL_LOOPING, (flags&MWBase::SoundManager::Play_Loop) ? AL_TRUE : AL_FALSE); throwALerror(); alSourcei(src, AL_BUFFER, buf); alSourcePlay(src); throwALerror(); return sound; } MWBase::SoundPtr OpenAL_Output::streamSound(DecoderPtr decoder, float volume, float pitch, int flags) { boost::shared_ptr sound; ALuint src; if(mFreeSources.empty()) fail("No free sources"); src = mFreeSources.front(); mFreeSources.pop_front(); if((flags&MWBase::SoundManager::Play_Loop)) std::cout <<"Warning: cannot loop stream \""<getName()<<"\""<< std::endl; try { sound.reset(new OpenAL_SoundStream(*this, src, decoder)); } catch(std::exception &e) { mFreeSources.push_back(src); 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); if(!(flags&MWBase::SoundManager::Play_NoEnv) && mLastEnvironment == Env_Underwater) { volume *= 0.9f; pitch *= 0.7f; } alSourcef(src, AL_GAIN, volume); alSourcef(src, AL_PITCH, pitch); alSourcei(src, AL_SOURCE_RELATIVE, AL_TRUE); alSourcei(src, AL_LOOPING, AL_FALSE); throwALerror(); sound->play(); return sound; } void OpenAL_Output::updateListener(const Ogre::Vector3 &pos, const Ogre::Vector3 &atdir, const Ogre::Vector3 &updir, Environment env) { mPos = pos; mLastEnvironment = env; if(mContext) { ALfloat orient[6] = { atdir.x, atdir.z, -atdir.y, updir.x, updir.z, -updir.y }; alListener3f(AL_POSITION, mPos.x, mPos.z, -mPos.y); alListenerfv(AL_ORIENTATION, orient); throwALerror(); } } void OpenAL_Output::pauseAllSounds() { IDVec sources = mSources; IDDq::const_iterator iter = mFreeSources.begin(); while(iter != mFreeSources.end()) { sources.erase(std::find(sources.begin(), sources.end(), *iter)); iter++; } if(sources.size() > 0) alSourcePausev(sources.size(), &sources[0]); } void OpenAL_Output::resumeAllSounds() { IDVec sources = mSources; IDDq::const_iterator iter = mFreeSources.begin(); while(iter != mFreeSources.end()) { sources.erase(std::find(sources.begin(), sources.end(), *iter)); iter++; } if(sources.size() > 0) alSourcePlayv(sources.size(), &sources[0]); } OpenAL_Output::OpenAL_Output(SoundManager &mgr) : Sound_Output(mgr), mDevice(0), mContext(0), mBufferCacheMemSize(0), mLastEnvironment(Env_Normal), mStreamThread(new StreamThread) { } OpenAL_Output::~OpenAL_Output() { deinit(); } }