From 176046d2b59c2042cd35a58848d4964563287f63 Mon Sep 17 00:00:00 2001 From: Lukasz Marek Date: Wed, 16 Apr 2014 15:23:46 +0200 Subject: [PATCH] lavd/pulse_audio_common: add context helper functions Functions allow to connect and disconnect from server. Helpfull to implement utility functions with nested loops. Signed-off-by: Lukasz Marek --- libavdevice/pulse_audio_common.c | 89 ++++++++++++++++++++++---------- libavdevice/pulse_audio_common.h | 5 ++ 2 files changed, 66 insertions(+), 28 deletions(-) diff --git a/libavdevice/pulse_audio_common.c b/libavdevice/pulse_audio_common.c index bffe9e3417..5a2568b82b 100644 --- a/libavdevice/pulse_audio_common.c +++ b/libavdevice/pulse_audio_common.c @@ -24,6 +24,7 @@ #include "libavutil/attributes.h" #include "libavutil/avstring.h" #include "libavutil/mem.h" +#include "libavutil/avassert.h" pa_sample_format_t av_cold ff_codec_id_to_pulse_format(enum AVCodecID codec_id) { @@ -73,6 +74,64 @@ static void pa_state_cb(pa_context *c, void *userdata) } } +void ff_pulse_audio_disconnect_context(pa_mainloop **pa_ml, pa_context **pa_ctx) +{ + av_assert0(pa_ml); + av_assert0(pa_ctx); + + if (*pa_ctx) { + pa_context_set_state_callback(*pa_ctx, NULL, NULL); + pa_context_disconnect(*pa_ctx); + pa_context_unref(*pa_ctx); + } + if (*pa_ml) + pa_mainloop_free(*pa_ml); + *pa_ml = NULL; + *pa_ctx = NULL; +} + +int ff_pulse_audio_connect_context(pa_mainloop **pa_ml, pa_context **pa_ctx, + const char *server, const char *description) +{ + int ret; + pa_mainloop_api *pa_mlapi = NULL; + enum PulseAudioContextState context_state = PULSE_CONTEXT_INITIALIZING; + + av_assert0(pa_ml); + av_assert0(pa_ctx); + + *pa_ml = NULL; + *pa_ctx = NULL; + + if (!(*pa_ml = pa_mainloop_new())) + return AVERROR(ENOMEM); + if (!(pa_mlapi = pa_mainloop_get_api(*pa_ml))) { + ret = AVERROR_EXTERNAL; + goto fail; + } + if (!(*pa_ctx = pa_context_new(pa_mlapi, description))) { + ret = AVERROR(ENOMEM); + goto fail; + } + pa_context_set_state_callback(*pa_ctx, pa_state_cb, &context_state); + if (pa_context_connect(*pa_ctx, server, 0, NULL) < 0) { + ret = AVERROR_EXTERNAL; + goto fail; + } + + while (context_state == PULSE_CONTEXT_INITIALIZING) + pa_mainloop_iterate(*pa_ml, 1, NULL); + if (context_state == PULSE_CONTEXT_FINISHED) { + ret = AVERROR_EXTERNAL; + goto fail; + } + return 0; + + fail: + ff_pulse_audio_disconnect_context(pa_ml, pa_ctx); + return ret; +} + static void pulse_add_detected_device(PulseAudioDeviceList *info, const char *name, const char *description) { @@ -138,11 +197,9 @@ static void pulse_server_info_cb(pa_context *c, const pa_server_info *i, void *u int ff_pulse_audio_get_devices(AVDeviceInfoList *devices, const char *server, int output) { pa_mainloop *pa_ml = NULL; - pa_mainloop_api *pa_mlapi = NULL; pa_operation *pa_op = NULL; pa_context *pa_ctx = NULL; enum pa_operation_state op_state; - enum PulseAudioContextState loop_state = PULSE_CONTEXT_INITIALIZING; PulseAudioDeviceList dev_list = { 0 }; int i; @@ -152,28 +209,9 @@ int ff_pulse_audio_get_devices(AVDeviceInfoList *devices, const char *server, in return AVERROR(EINVAL); devices->nb_devices = 0; devices->devices = NULL; - if (!(pa_ml = pa_mainloop_new())) - return AVERROR(ENOMEM); - if (!(pa_mlapi = pa_mainloop_get_api(pa_ml))) { - dev_list.error_code = AVERROR_EXTERNAL; - goto fail; - } - if (!(pa_ctx = pa_context_new(pa_mlapi, "Query devices"))) { - dev_list.error_code = AVERROR(ENOMEM); - goto fail; - } - pa_context_set_state_callback(pa_ctx, pa_state_cb, &loop_state); - if (pa_context_connect(pa_ctx, server, 0, NULL) < 0) { - dev_list.error_code = AVERROR_EXTERNAL; - goto fail; - } - while (loop_state == PULSE_CONTEXT_INITIALIZING) - pa_mainloop_iterate(pa_ml, 1, NULL); - if (loop_state == PULSE_CONTEXT_FINISHED) { - dev_list.error_code = AVERROR_EXTERNAL; + if ((dev_list.error_code = ff_pulse_audio_connect_context(&pa_ml, &pa_ctx, server, "Query devices")) < 0) goto fail; - } if (output) pa_op = pa_context_get_sink_info_list(pa_ctx, pulse_audio_sink_device_cb, &dev_list); @@ -206,11 +244,6 @@ int ff_pulse_audio_get_devices(AVDeviceInfoList *devices, const char *server, in fail: av_free(dev_list.default_device); - if(pa_ctx) - pa_context_disconnect(pa_ctx); - if (pa_ctx) - pa_context_unref(pa_ctx); - if (pa_ml) - pa_mainloop_free(pa_ml); + ff_pulse_audio_disconnect_context(&pa_ml, &pa_ctx); return dev_list.error_code; } diff --git a/libavdevice/pulse_audio_common.h b/libavdevice/pulse_audio_common.h index b049cdea1a..02534f79a1 100644 --- a/libavdevice/pulse_audio_common.h +++ b/libavdevice/pulse_audio_common.h @@ -30,4 +30,9 @@ pa_sample_format_t ff_codec_id_to_pulse_format(enum AVCodecID codec_id); int ff_pulse_audio_get_devices(AVDeviceInfoList *devices, const char *server, int output); +int ff_pulse_audio_connect_context(pa_mainloop **pa_ml, pa_context **pa_ctx, + const char *server, const char *description); + +void ff_pulse_audio_disconnect_context(pa_mainloop **pa_ml, pa_context **pa_ctx); + #endif /* AVDEVICE_PULSE_AUDIO_COMMON_H */