Refactor the audio streaming code to be a bit saner

openmw-38
Chris Robinson 9 years ago
parent 16f72886e9
commit 83721092f2

@ -150,18 +150,6 @@ static ALenum getALFormat(ChannelConfig chans, SampleType type)
return AL_NONE; 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. // A streaming OpenAL sound.
// //
@ -174,17 +162,17 @@ class OpenAL_SoundStream : public Sound
ALuint mSource; ALuint mSource;
ALuint mBuffers[sNumBuffers]; ALuint mBuffers[sNumBuffers];
ALint mCurrentBufIdx;
ALenum mFormat; ALenum mFormat;
ALsizei mSampleRate; ALsizei mSampleRate;
ALuint mBufferSize; ALuint mBufferSize;
ALuint mFrameSize;
ALuint mSamplesQueued; ALint mSilence;
DecoderPtr mDecoder; DecoderPtr mDecoder;
volatile bool mIsFinished; volatile bool mIsFinished;
volatile bool mIsInitialBatchEnqueued;
void updateAll(bool local); void updateAll(bool local);
@ -204,6 +192,7 @@ public:
void play(); void play();
bool process(); bool process();
ALint refillQueue();
}; };
const ALfloat OpenAL_SoundStream::sBufferLength = 0.125f; const ALfloat OpenAL_SoundStream::sBufferLength = 0.125f;
@ -277,7 +266,8 @@ private:
OpenAL_SoundStream::OpenAL_SoundStream(OpenAL_Output &output, ALuint src, DecoderPtr decoder, float basevol, float pitch, int flags) OpenAL_SoundStream::OpenAL_SoundStream(OpenAL_Output &output, ALuint src, DecoderPtr decoder, float basevol, float pitch, int flags)
: Sound(osg::Vec3f(0.f, 0.f, 0.f), 1.0f, basevol, pitch, 1.0f, 1000.0f, flags) : Sound(osg::Vec3f(0.f, 0.f, 0.f), 1.0f, basevol, pitch, 1.0f, 1000.0f, flags)
, mOutput(output), mSource(src), mSamplesQueued(0), mDecoder(decoder), mIsFinished(true), mIsInitialBatchEnqueued(false) , mOutput(output), mSource(src), mCurrentBufIdx(0), mFrameSize(0), mSilence(0)
, mDecoder(decoder), mIsFinished(true)
{ {
throwALerror(); throwALerror();
@ -293,8 +283,16 @@ OpenAL_SoundStream::OpenAL_SoundStream(OpenAL_Output &output, ALuint src, Decode
mFormat = getALFormat(chans, type); mFormat = getALFormat(chans, type);
mSampleRate = srate; mSampleRate = srate;
switch(type)
{
case SampleType_UInt8: mSilence = 0x80;
case SampleType_Int16: mSilence = 0x00;
case SampleType_Float32: mSilence = 0x00;
}
mFrameSize = framesToBytes(1, chans, type);
mBufferSize = static_cast<ALuint>(sBufferLength*srate); mBufferSize = static_cast<ALuint>(sBufferLength*srate);
mBufferSize = framesToBytes(mBufferSize, chans, type); mBufferSize *= mFrameSize;
mOutput.mActiveStreams.push_back(this); mOutput.mActiveStreams.push_back(this);
} }
@ -327,9 +325,8 @@ void OpenAL_SoundStream::play()
alSourceStop(mSource); alSourceStop(mSource);
alSourcei(mSource, AL_BUFFER, 0); alSourcei(mSource, AL_BUFFER, 0);
throwALerror(); throwALerror();
mSamplesQueued = 0;
mIsFinished = false; mIsFinished = false;
mIsInitialBatchEnqueued = false;
mOutput.mStreamThread->add(this); mOutput.mStreamThread->add(this);
} }
@ -337,12 +334,10 @@ void OpenAL_SoundStream::stop()
{ {
mOutput.mStreamThread->remove(this); mOutput.mStreamThread->remove(this);
mIsFinished = true; mIsFinished = true;
mIsInitialBatchEnqueued = false;
alSourceStop(mSource); alSourceStop(mSource);
alSourcei(mSource, AL_BUFFER, 0); alSourcei(mSource, AL_BUFFER, 0);
throwALerror(); throwALerror();
mSamplesQueued = 0;
mDecoder->rewind(); mDecoder->rewind();
} }
@ -351,6 +346,7 @@ bool OpenAL_SoundStream::isPlaying()
{ {
ALint state; ALint state;
boost::lock_guard<boost::recursive_mutex> lock(mOutput.mStreamThread->mMutex);
alGetSourcei(mSource, AL_SOURCE_STATE, &state); alGetSourcei(mSource, AL_SOURCE_STATE, &state);
throwALerror(); throwALerror();
@ -362,17 +358,26 @@ bool OpenAL_SoundStream::isPlaying()
double OpenAL_SoundStream::getTimeOffset() double OpenAL_SoundStream::getTimeOffset()
{ {
ALint state = AL_STOPPED; ALint state = AL_STOPPED;
ALfloat offset = 0.0f; ALint offset;
double t; double t;
mOutput.mStreamThread->mMutex.lock(); boost::unique_lock<boost::recursive_mutex> lock(mOutput.mStreamThread->mMutex);
alGetSourcef(mSource, AL_SEC_OFFSET, &offset); alGetSourcei(mSource, AL_SAMPLE_OFFSET, &offset);
alGetSourcei(mSource, AL_SOURCE_STATE, &state); alGetSourcei(mSource, AL_SOURCE_STATE, &state);
if(state == AL_PLAYING || state == AL_PAUSED) if(state == AL_PLAYING || state == AL_PAUSED)
t = (double)(mDecoder->getSampleOffset() - mSamplesQueued)/(double)mSampleRate + offset; {
ALint queued;
alGetSourcei(mSource, AL_BUFFERS_QUEUED, &queued);
ALint inqueue = mBufferSize/mFrameSize*queued + offset;
t = (double)(mDecoder->getSampleOffset() - inqueue) / (double)mSampleRate;
}
else else
{
/* Underrun, or not started yet. The decoder offset is where we'll play
* next. */
t = (double)mDecoder->getSampleOffset() / (double)mSampleRate; t = (double)mDecoder->getSampleOffset() / (double)mSampleRate;
mOutput.mStreamThread->mMutex.unlock(); }
lock.unlock();
throwALerror(); throwALerror();
return t; return t;
@ -418,77 +423,64 @@ void OpenAL_SoundStream::update()
bool OpenAL_SoundStream::process() bool OpenAL_SoundStream::process()
{ {
try { try {
bool finished = mIsFinished; if(refillQueue() > 0)
ALint processed, state; {
ALint state;
alGetSourcei(mSource, AL_SOURCE_STATE, &state); alGetSourcei(mSource, AL_SOURCE_STATE, &state);
alGetSourcei(mSource, AL_BUFFERS_PROCESSED, &processed); if(state != AL_PLAYING && state != AL_PAUSED)
{
if(refillQueue() > 0)
alSourcePlay(mSource);
throwALerror(); throwALerror();
}
}
}
catch(std::exception&) {
std::cout<< "Error updating stream \""<<mDecoder->getName()<<"\"" <<std::endl;
mIsFinished = true;
}
return !mIsFinished;
}
if(processed > 0) ALint OpenAL_SoundStream::refillQueue()
{ {
std::vector<char> data(mBufferSize); ALint processed;
do { alGetSourcei(mSource, AL_BUFFERS_PROCESSED, &processed);
ALuint bufid = 0; while(processed > 0)
size_t got;
alSourceUnqueueBuffers(mSource, 1, &bufid);
mSamplesQueued -= 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); ALuint buf;
alSourceQueueBuffers(mSource, 1, &bufid); alSourceUnqueueBuffers(mSource, 1, &buf);
mSamplesQueued += getBufferSampleCount(bufid); --processed;
} }
} while(processed > 0);
throwALerror(); throwALerror();
}
else if (!mIsInitialBatchEnqueued) { // nothing enqueued yet
std::vector<char> data(mBufferSize);
for(ALuint i = 0;i < sNumBuffers && !finished;i++) ALint queued;
alGetSourcei(mSource, AL_BUFFERS_QUEUED, &queued);
if(!mIsFinished && (ALuint)queued < sNumBuffers)
{
std::vector<char> data(mBufferSize);
for(;!mIsFinished && (ALuint)queued < sNumBuffers;++queued)
{ {
size_t got = mDecoder->read(&data[0], data.size()); size_t got = mDecoder->read(&data[0], data.size());
finished = (got < data.size()); if(got < data.size())
{
mIsFinished = true;
memset(&data[got], mSilence, data.size()-got);
}
if(got > 0) if(got > 0)
{ {
ALuint bufid = mBuffers[i]; ALuint bufid = mBuffers[mCurrentBufIdx];
alBufferData(bufid, mFormat, &data[0], got, mSampleRate); alBufferData(bufid, mFormat, &data[0], data.size(), mSampleRate);
alSourceQueueBuffers(mSource, 1, &bufid); alSourceQueueBuffers(mSource, 1, &bufid);
throwALerror(); throwALerror();
mSamplesQueued += getBufferSampleCount(bufid); mCurrentBufIdx = (mCurrentBufIdx+1) % sNumBuffers;
} }
} }
mIsInitialBatchEnqueued = true;
} }
if(state != AL_PLAYING && state != AL_PAUSED) return queued;
{
ALint queued = 0;
alGetSourcei(mSource, AL_BUFFERS_QUEUED, &queued);
if(queued > 0)
alSourcePlay(mSource);
throwALerror();
} }
mIsFinished = finished;
}
catch(std::exception&) {
std::cout<< "Error updating stream \""<<mDecoder->getName()<<"\"" <<std::endl;
mSamplesQueued = 0;
mIsFinished = true;
mIsInitialBatchEnqueued = false;
}
return !mIsFinished;
}
// //
// A regular 2D OpenAL sound // A regular 2D OpenAL sound

Loading…
Cancel
Save