diff --git a/doc/user.xml b/doc/user.xml index 30fcae696..e1e62eb92 100644 --- a/doc/user.xml +++ b/doc/user.xml @@ -337,6 +337,18 @@ cd mpd-version enabled. + + + always_on + yes|no + + + If set to "yes", then MPD attempts to keep this audio + output always open. This may be useful for streaming + servers, when you don't want to disconnect all + listeners even when playback is accidently stopped. + + mixer_type diff --git a/src/output_all.c b/src/output_all.c index 4b7701144..dbd5a6ce6 100644 --- a/src/output_all.c +++ b/src/output_all.c @@ -558,6 +558,29 @@ audio_output_all_close(void) audio_output_all_elapsed_time = -1.0; } +void +audio_output_all_release(void) +{ + unsigned int i; + + for (i = 0; i < num_audio_outputs; ++i) + audio_output_release(&audio_outputs[i]); + + if (g_mp != NULL) { + assert(g_music_buffer != NULL); + + music_pipe_clear(g_mp, g_music_buffer); + music_pipe_free(g_mp); + g_mp = NULL; + } + + g_music_buffer = NULL; + + audio_format_clear(&input_audio_format); + + audio_output_all_elapsed_time = -1.0; +} + void audio_output_all_song_border(void) { diff --git a/src/output_all.h b/src/output_all.h index 8c3f1e80d..a579bf5f1 100644 --- a/src/output_all.h +++ b/src/output_all.h @@ -91,6 +91,13 @@ audio_output_all_open(const struct audio_format *audio_format, void audio_output_all_close(void); +/** + * Closes all audio outputs. Outputs with the "always_on" flag are + * put into pause mode. + */ +void +audio_output_all_release(void); + /** * Enqueue a #music_chunk object for playing, i.e. pushes it to a * #music_pipe. diff --git a/src/output_control.c b/src/output_control.c index b5d983ae1..17edc3d72 100644 --- a/src/output_control.c +++ b/src/output_control.c @@ -118,9 +118,13 @@ audio_output_open(struct audio_output *ao, if (ao->open && audio_format_equals(audio_format, &ao->in_audio_format)) { - assert(ao->pipe == mp); + assert(ao->pipe == mp || + (ao->always_on && ao->pause)); if (ao->pause) { + ao->chunk = NULL; + ao->pipe = mp; + /* unpause with the CANCEL command; this is a hack, but suits well for forcing the thread to leave the ao_pause() thread, and we need @@ -266,6 +270,15 @@ void audio_output_close(struct audio_output *ao) g_mutex_unlock(ao->mutex); } +void +audio_output_release(struct audio_output *ao) +{ + if (ao->always_on) + audio_output_pause(ao); + else + audio_output_close(ao); +} + void audio_output_finish(struct audio_output *ao) { audio_output_close(ao); diff --git a/src/output_control.h b/src/output_control.h index 9c2171a45..7f4f4a53c 100644 --- a/src/output_control.h +++ b/src/output_control.h @@ -71,7 +71,16 @@ void audio_output_drain_async(struct audio_output *ao); void audio_output_cancel(struct audio_output *ao); + void audio_output_close(struct audio_output *ao); + +/** + * Closes the audio output, but if the "always_on" flag is set, put it + * into pause mode instead. + */ +void +audio_output_release(struct audio_output *ao); + void audio_output_finish(struct audio_output *ao); #endif diff --git a/src/output_init.c b/src/output_init.c index f3d22ace1..6ee340edc 100644 --- a/src/output_init.c +++ b/src/output_init.c @@ -184,6 +184,7 @@ audio_output_init(struct audio_output *ao, const struct config_param *param, } ao->plugin = plugin; + ao->always_on = config_get_block_bool(param, "always_on", false); ao->enabled = config_get_block_bool(param, "enabled", true); ao->really_enabled = false; ao->open = false; diff --git a/src/output_internal.h b/src/output_internal.h index 2a438c62f..06df9531b 100644 --- a/src/output_internal.h +++ b/src/output_internal.h @@ -75,6 +75,12 @@ struct audio_output { */ struct mixer *mixer; + /** + * Shall this output always play something (i.e. silence), + * even when playback is stopped? + */ + bool always_on; + /** * Has the user enabled this device? */ diff --git a/src/player_thread.c b/src/player_thread.c index 3e234c4ff..2496f0cdb 100644 --- a/src/player_thread.c +++ b/src/player_thread.c @@ -978,7 +978,7 @@ static gpointer player_task(G_GNUC_UNUSED gpointer arg) case PLAYER_COMMAND_CLOSE_AUDIO: player_unlock(); - audio_output_all_close(); + audio_output_all_release(); player_lock(); player_command_finished_locked();