From 7648fdf6824f67aa72c3c82f3accde94434ecede Mon Sep 17 00:00:00 2001 From: Pavel Sountsov Date: Sun, 23 Jun 2024 19:25:38 -0700 Subject: [PATCH] Fix a data race between the feeder thread and attaching of the stream to the mixer. This is done by by ensuring the stream is pre-filled before attaching is possible. Possibly related to #1561 --- addons/acodec/helper.c | 16 ++++++++++------ addons/audio/kcm_stream.c | 13 +++++++------ 2 files changed, 17 insertions(+), 12 deletions(-) diff --git a/addons/acodec/helper.c b/addons/acodec/helper.c index 26194ca7d0..bcf16c05cb 100644 --- a/addons/acodec/helper.c +++ b/addons/acodec/helper.c @@ -11,19 +11,23 @@ void _al_acodec_start_feed_thread(ALLEGRO_AUDIO_STREAM *stream) stream->feed_thread_started_cond = al_create_cond(); stream->feed_thread_started_mutex = al_create_mutex(); al_start_thread(stream->feed_thread); -} - -void _al_acodec_stop_feed_thread(ALLEGRO_AUDIO_STREAM *stream) -{ - ALLEGRO_EVENT quit_event; /* Need to wait for the thread to start, otherwise the quit event may be - * sent before the event source is registered with the queue. */ + * sent before the event source is registered with the queue. + * + * This also makes the pre-fill system thread safe, as it needs to operate + * before the mutexes are set up. + */ al_lock_mutex(stream->feed_thread_started_mutex); while (!stream->feed_thread_started) { al_wait_cond(stream->feed_thread_started_cond, stream->feed_thread_started_mutex); } al_unlock_mutex(stream->feed_thread_started_mutex); +} + +void _al_acodec_stop_feed_thread(ALLEGRO_AUDIO_STREAM *stream) +{ + ALLEGRO_EVENT quit_event; quit_event.type = _KCM_STREAM_FEEDER_QUIT_EVENT_TYPE; al_emit_user_event(al_get_audio_stream_event_source(stream), &quit_event, NULL); diff --git a/addons/audio/kcm_stream.c b/addons/audio/kcm_stream.c index 8d784fa9ad..2b8487c27d 100644 --- a/addons/audio/kcm_stream.c +++ b/addons/audio/kcm_stream.c @@ -692,11 +692,6 @@ void *_al_kcm_feed_stream(ALLEGRO_THREAD *self, void *vstream) queue = al_create_event_queue(); al_register_event_source(queue, &stream->spl.es); - al_lock_mutex(stream->feed_thread_started_mutex); - stream->feed_thread_started = true; - al_broadcast_cond(stream->feed_thread_started_cond); - al_unlock_mutex(stream->feed_thread_started_mutex); - stream->quit_feed_thread = false; while (!stream->quit_feed_thread) { @@ -708,7 +703,6 @@ void *_al_kcm_feed_stream(ALLEGRO_THREAD *self, void *vstream) if ((prefill || event.type == ALLEGRO_EVENT_AUDIO_STREAM_FRAGMENT) && !stream->is_draining) { - prefill = false; unsigned long bytes; unsigned long bytes_written; ALLEGRO_MUTEX *stream_mutex; @@ -784,6 +778,13 @@ void *_al_kcm_feed_stream(ALLEGRO_THREAD *self, void *vstream) fin_event.user.timestamp = al_get_time(); al_emit_user_event(&stream->spl.es, &fin_event, NULL); } + if (prefill) { + al_lock_mutex(stream->feed_thread_started_mutex); + stream->feed_thread_started = true; + al_broadcast_cond(stream->feed_thread_started_cond); + al_unlock_mutex(stream->feed_thread_started_mutex); + } + prefill = false; } al_destroy_event_queue(queue);