aboutsummaryrefslogtreecommitdiffstats
path: root/Alc/backends
diff options
context:
space:
mode:
authorChris Robinson <[email protected]>2018-02-01 18:54:13 -0800
committerChris Robinson <[email protected]>2018-02-01 18:54:13 -0800
commit975c682ec367f61e6e700b21da38e2c59cf69c3b (patch)
treeec3ca50e81986e71b2680b4f15ffb9fb3fb01f06 /Alc/backends
parente240351d81105e233f02d006f6bf42d03c0dd09b (diff)
Use semaphores to signal for more samples with JACK and OpenSL
Diffstat (limited to 'Alc/backends')
-rw-r--r--Alc/backends/jack.c34
-rw-r--r--Alc/backends/opensl.c36
2 files changed, 16 insertions, 54 deletions
diff --git a/Alc/backends/jack.c b/Alc/backends/jack.c
index f1141ce9..09f778c7 100644
--- a/Alc/backends/jack.c
+++ b/Alc/backends/jack.c
@@ -150,7 +150,7 @@ typedef struct ALCjackPlayback {
jack_port_t *Port[MAX_OUTPUT_CHANNELS];
ll_ringbuffer_t *Ring;
- alcnd_t Cond;
+ alsem_t Sem;
volatile int killNow;
althrd_t thread;
@@ -184,7 +184,7 @@ static void ALCjackPlayback_Construct(ALCjackPlayback *self, ALCdevice *device)
ALCbackend_Construct(STATIC_CAST(ALCbackend, self), device);
SET_VTABLE2(ALCjackPlayback, ALCbackend, self);
- alcnd_init(&self->Cond);
+ alsem_init(&self->Sem, 0);
self->Client = NULL;
for(i = 0;i < MAX_OUTPUT_CHANNELS;i++)
@@ -210,7 +210,7 @@ static void ALCjackPlayback_Destruct(ALCjackPlayback *self)
self->Client = NULL;
}
- alcnd_destroy(&self->Cond);
+ alsem_destroy(&self->Sem);
ALCbackend_Destruct(STATIC_CAST(ALCbackend, self));
}
@@ -287,7 +287,7 @@ static int ALCjackPlayback_process(jack_nframes_t numframes, void *arg)
}
ll_ringbuffer_read_advance(self->Ring, total);
- alcnd_signal(&self->Cond);
+ alsem_post(&self->Sem);
if(numframes > total)
{
@@ -316,23 +316,11 @@ static int ALCjackPlayback_mixerProc(void *arg)
{
ALuint todo, len1, len2;
- /* NOTE: Unfortunately, there is an unavoidable race condition here.
- * It's possible for the process() method to run, updating the read
- * pointer and signaling the condition variable, in between the mixer
- * loop checking the write size and waiting for the condition variable.
- * This will cause the mixer loop to wait until the *next* process()
- * invocation, most likely writing silence for it.
- *
- * However, this should only happen if the mixer is running behind
- * anyway (as ideally we'll be asleep in alcnd_wait by the time the
- * process() method is invoked), so this behavior is not unwarranted.
- * It's unfortunate since it'll be wasting time sleeping that could be
- * used to catch up, but there's no way around it without blocking in
- * the process() method.
- */
if(ll_ringbuffer_write_space(self->Ring) < device->UpdateSize)
{
- alcnd_wait(&self->Cond, &STATIC_CAST(ALCbackend,self)->mMutex);
+ ALCjackPlayback_unlock(self);
+ alsem_wait(&self->Sem);
+ ALCjackPlayback_lock(self);
continue;
}
@@ -509,13 +497,7 @@ static void ALCjackPlayback_stop(ALCjackPlayback *self)
return;
self->killNow = 1;
- /* Lock the backend to ensure we don't flag the mixer to die and signal the
- * mixer to wake up in between it checking the flag and going to sleep and
- * wait for a wakeup (potentially leading to it never waking back up to see
- * the flag). */
- ALCjackPlayback_lock(self);
- ALCjackPlayback_unlock(self);
- alcnd_signal(&self->Cond);
+ alsem_post(&self->Sem);
althrd_join(self->thread, &res);
jack_deactivate(self->Client);
diff --git a/Alc/backends/opensl.c b/Alc/backends/opensl.c
index 7edc9ff8..fb56b67c 100644
--- a/Alc/backends/opensl.c
+++ b/Alc/backends/opensl.c
@@ -146,7 +146,7 @@ typedef struct ALCopenslPlayback {
SLObjectItf mBufferQueueObj;
ll_ringbuffer_t *mRing;
- alcnd_t mCond;
+ alsem_t mSem;
ALsizei mFrameSize;
@@ -184,7 +184,7 @@ static void ALCopenslPlayback_Construct(ALCopenslPlayback *self, ALCdevice *devi
self->mBufferQueueObj = NULL;
self->mRing = NULL;
- alcnd_init(&self->mCond);
+ alsem_init(&self->mSem, 0);
self->mFrameSize = 0;
@@ -206,7 +206,7 @@ static void ALCopenslPlayback_Destruct(ALCopenslPlayback* self)
self->mEngineObj = NULL;
self->mEngine = NULL;
- alcnd_destroy(&self->mCond);
+ alsem_destroy(&self->mSem);
ALCbackend_Destruct(STATIC_CAST(ALCbackend, self));
}
@@ -227,7 +227,7 @@ static void ALCopenslPlayback_process(SLAndroidSimpleBufferQueueItf UNUSED(bq),
*/
ll_ringbuffer_read_advance(self->mRing, 1);
- alcnd_signal(&self->mCond);
+ alsem_post(&self->mSem);
}
@@ -287,24 +287,11 @@ static int ALCopenslPlayback_mixerProc(void *arg)
break;
}
- /* NOTE: Unfortunately, there is an unavoidable race condition
- * here. It's possible for the process() method to run, updating
- * the read pointer and signaling the condition variable, in
- * between checking the write size and waiting for the condition
- * variable here. This will cause alcnd_wait to wait until the
- * *next* process() invocation signals the condition variable
- * again.
- *
- * However, this should only happen if the mixer is running behind
- * anyway (as ideally we'll be asleep in alcnd_wait by the time the
- * process() method is invoked), so this behavior is not completely
- * unwarranted. It's unfortunate since it'll be wasting time
- * sleeping that could be used to catch up, but there's no way
- * around it without blocking in the process() method.
- */
if(ll_ringbuffer_write_space(self->mRing) <= padding)
{
- alcnd_wait(&self->mCond, &STATIC_CAST(ALCbackend,self)->mMutex);
+ ALCopenslPlayback_unlock(self);
+ alsem_wait(&self->mSem);
+ ALCopenslPlayback_lock(self);
continue;
}
}
@@ -623,14 +610,7 @@ static void ALCopenslPlayback_stop(ALCopenslPlayback *self)
if(ATOMIC_EXCHANGE_SEQ(&self->mKillNow, AL_TRUE))
return;
- /* Lock the backend to ensure we don't flag the mixer to die and signal the
- * mixer to wake up in between it checking the flag and going to sleep and
- * wait for a wakeup (potentially leading to it never waking back up to see
- * the flag).
- */
- ALCopenslPlayback_lock(self);
- ALCopenslPlayback_unlock(self);
- alcnd_signal(&self->mCond);
+ alsem_post(&self->mSem);
althrd_join(self->mThread, &res);
result = VCALL(self->mBufferQueueObj,GetInterface)(SL_IID_PLAY, &player);