pulsesink: share the PA context between all clients with the same name

Avoid to create a new PA context for each new client by using a hash
table containing the list of ring-buffers and the shared PA context
for each client. Doing this will improve application memory usage in
the cases where multiple pipelines involving multiple pulsesink
elements are used.

Fixes bug #624338.
This commit is contained in:
Philippe Normand 2010-08-16 09:12:04 +02:00 committed by Sebastian Dröge
parent 2c5976d355
commit 69a397c32f

View file

@ -100,6 +100,19 @@ enum
typedef struct _GstPulseRingBuffer GstPulseRingBuffer; typedef struct _GstPulseRingBuffer GstPulseRingBuffer;
typedef struct _GstPulseRingBufferClass GstPulseRingBufferClass; typedef struct _GstPulseRingBufferClass GstPulseRingBufferClass;
typedef struct _GstPulseContext GstPulseContext;
struct _GstPulseContext
{
pa_context *context;
GSList *ring_buffers;
};
/* Store the PA contexts in a hash table to allow easy sharing among
* multiple instances of the sink. Keys are $context_name@$server_name
* (strings) and values should be GstPulseContext pointers. */
static GHashTable *gst_pulse_shared_contexts;
/* We keep a custom ringbuffer that is backed up by data allocated by /* We keep a custom ringbuffer that is backed up by data allocated by
* pulseaudio. We must also overide the commit function to write into * pulseaudio. We must also overide the commit function to write into
* pulseaudio memory instead. */ * pulseaudio memory instead. */
@ -107,9 +120,9 @@ struct _GstPulseRingBuffer
{ {
GstRingBuffer object; GstRingBuffer object;
gchar *context_name;
gchar *stream_name; gchar *stream_name;
pa_context *context;
pa_stream *stream; pa_stream *stream;
pa_sample_spec sample_spec; pa_sample_spec sample_spec;
@ -151,6 +164,15 @@ static guint gst_pulseringbuffer_commit (GstRingBuffer * buf,
G_DEFINE_TYPE (GstPulseRingBuffer, gst_pulseringbuffer, GST_TYPE_RING_BUFFER); G_DEFINE_TYPE (GstPulseRingBuffer, gst_pulseringbuffer, GST_TYPE_RING_BUFFER);
static GMutex *pa_ring_buffer_mutex = NULL;
static void
gst_pulseringbuffer_init_contexts (void)
{
g_assert (pa_ring_buffer_mutex == NULL);
pa_ring_buffer_mutex = g_mutex_new ();
gst_pulse_shared_contexts = g_hash_table_new (g_str_hash, g_str_equal);
}
static void static void
gst_pulseringbuffer_class_init (GstPulseRingBufferClass * klass) gst_pulseringbuffer_class_init (GstPulseRingBufferClass * klass)
{ {
@ -186,7 +208,6 @@ static void
gst_pulseringbuffer_init (GstPulseRingBuffer * pbuf) gst_pulseringbuffer_init (GstPulseRingBuffer * pbuf)
{ {
pbuf->stream_name = NULL; pbuf->stream_name = NULL;
pbuf->context = NULL;
pbuf->stream = NULL; pbuf->stream = NULL;
#ifdef HAVE_PULSE_0_9_13 #ifdef HAVE_PULSE_0_9_13
@ -245,23 +266,49 @@ gst_pulsering_destroy_stream (GstPulseRingBuffer * pbuf)
pbuf->stream_name = NULL; pbuf->stream_name = NULL;
} }
static GstPulseContext *
gst_pulsering_get_context (GstPulseRingBuffer * pbuf)
{
GstPulseContext *pctx;
GstPulseSink *psink;
g_mutex_lock (pa_ring_buffer_mutex);
psink = GST_PULSESINK_CAST (GST_OBJECT_PARENT (pbuf));
pctx = g_hash_table_lookup (gst_pulse_shared_contexts, pbuf->context_name);
g_mutex_unlock (pa_ring_buffer_mutex);
return pctx;
}
static void static void
gst_pulsering_destroy_context (GstPulseRingBuffer * pbuf) gst_pulsering_destroy_context (GstPulseRingBuffer * pbuf)
{ {
GstPulseContext *pctx;
GstPulseSink *psink;
g_mutex_lock (pa_ring_buffer_mutex);
psink = GST_PULSESINK_CAST (GST_OBJECT_PARENT (pbuf));
pctx = g_hash_table_lookup (gst_pulse_shared_contexts, pbuf->context_name);
gst_pulsering_destroy_stream (pbuf); gst_pulsering_destroy_stream (pbuf);
if (pbuf->context) { if (pctx) {
pa_context_disconnect (pbuf->context); pctx->ring_buffers = g_slist_remove (pctx->ring_buffers, pbuf);
if (!g_slist_length (pctx->ring_buffers)) {
pa_context_disconnect (pctx->context);
/* Make sure we don't get any further callbacks */ /* Make sure we don't get any further callbacks */
pa_context_set_state_callback (pbuf->context, NULL, NULL); pa_context_set_state_callback (pctx->context, NULL, NULL);
#ifdef HAVE_PULSE_0_9_12 #ifdef HAVE_PULSE_0_9_12
pa_context_set_subscribe_callback (pbuf->context, NULL, NULL); pa_context_set_subscribe_callback (pctx->context, NULL, NULL);
#endif #endif
pa_context_unref (pbuf->context); pa_context_unref (pctx->context);
pbuf->context = NULL; g_hash_table_remove (gst_pulse_shared_contexts, pbuf->context_name);
g_free (pbuf->context_name);
} }
}
g_mutex_unlock (pa_ring_buffer_mutex);
} }
static void static void
@ -279,13 +326,19 @@ gst_pulseringbuffer_finalize (GObject * object)
static gboolean static gboolean
gst_pulsering_is_dead (GstPulseSink * psink, GstPulseRingBuffer * pbuf) gst_pulsering_is_dead (GstPulseSink * psink, GstPulseRingBuffer * pbuf)
{ {
if (!pbuf->context GstPulseContext *pctx = gst_pulsering_get_context (pbuf);
|| !PA_CONTEXT_IS_GOOD (pa_context_get_state (pbuf->context))
if (!pctx) {
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, ("Disconnected"), (NULL));
return TRUE;
}
if (!pctx->context
|| !PA_CONTEXT_IS_GOOD (pa_context_get_state (pctx->context))
|| !pbuf->stream || !pbuf->stream
|| !PA_STREAM_IS_GOOD (pa_stream_get_state (pbuf->stream))) { || !PA_STREAM_IS_GOOD (pa_stream_get_state (pbuf->stream))) {
const gchar *err_str = pbuf->context ? const gchar *err_str =
pa_strerror (pa_context_errno (pbuf->context)) : NULL; pctx->context ? pa_strerror (pa_context_errno (pctx->context)) : NULL;
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, ("Disconnected: %s", GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, ("Disconnected: %s",
err_str), (NULL)); err_str), (NULL));
return TRUE; return TRUE;
@ -297,19 +350,22 @@ static void
gst_pulsering_context_state_cb (pa_context * c, void *userdata) gst_pulsering_context_state_cb (pa_context * c, void *userdata)
{ {
GstPulseSink *psink; GstPulseSink *psink;
GstPulseRingBuffer *pbuf;
pa_context_state_t state; pa_context_state_t state;
pbuf = GST_PULSERING_BUFFER_CAST (userdata); GstPulseContext *pctx = (GstPulseContext *) userdata;
psink = GST_PULSESINK_CAST (GST_OBJECT_PARENT (pbuf)); GSList *walk;
state = pa_context_get_state (c); state = pa_context_get_state (c);
for (walk = pctx->ring_buffers; walk; walk = g_slist_next (walk)) {
GstPulseRingBuffer *pbuf = (GstPulseRingBuffer *) walk->data;
psink = GST_PULSESINK_CAST (GST_OBJECT_PARENT (pbuf));
GST_LOG_OBJECT (psink, "got new context state %d", state); GST_LOG_OBJECT (psink, "got new context state %d", state);
/* psink can be null when we are shutting down and the ringbuffer is already /* psink can be null when we are shutting down and the ringbuffer is already
* unparented */ * unparented */
if (psink == NULL) if (psink == NULL)
return; continue;
switch (state) { switch (state) {
case PA_CONTEXT_READY: case PA_CONTEXT_READY:
@ -325,6 +381,7 @@ gst_pulsering_context_state_cb (pa_context * c, void *userdata)
case PA_CONTEXT_SETTING_NAME: case PA_CONTEXT_SETTING_NAME:
break; break;
} }
}
} }
#ifdef HAVE_PULSE_0_9_12 #ifdef HAVE_PULSE_0_9_12
@ -333,22 +390,24 @@ gst_pulsering_context_subscribe_cb (pa_context * c,
pa_subscription_event_type_t t, uint32_t idx, void *userdata) pa_subscription_event_type_t t, uint32_t idx, void *userdata)
{ {
GstPulseSink *psink; GstPulseSink *psink;
GstPulseRingBuffer *pbuf; GstPulseContext *pctx = (GstPulseContext *) userdata;
GSList *walk;
pbuf = GST_PULSERING_BUFFER_CAST (userdata);
psink = GST_PULSESINK_CAST (GST_OBJECT_PARENT (pbuf));
GST_LOG_OBJECT (psink, "type %d, idx %u", t, idx);
if (t != (PA_SUBSCRIPTION_EVENT_SINK_INPUT | PA_SUBSCRIPTION_EVENT_CHANGE) && if (t != (PA_SUBSCRIPTION_EVENT_SINK_INPUT | PA_SUBSCRIPTION_EVENT_CHANGE) &&
t != (PA_SUBSCRIPTION_EVENT_SINK_INPUT | PA_SUBSCRIPTION_EVENT_NEW)) t != (PA_SUBSCRIPTION_EVENT_SINK_INPUT | PA_SUBSCRIPTION_EVENT_NEW))
return; return;
for (walk = pctx->ring_buffers; walk; walk = g_slist_next (walk)) {
GstPulseRingBuffer *pbuf = (GstPulseRingBuffer *) walk->data;
psink = GST_PULSESINK_CAST (GST_OBJECT_PARENT (pbuf));
GST_LOG_OBJECT (psink, "type %d, idx %u", t, idx);
if (!pbuf->stream) if (!pbuf->stream)
return; continue;
if (idx != pa_stream_get_index (pbuf->stream)) if (idx != pa_stream_get_index (pbuf->stream))
return; continue;
/* Actually this event is also triggered when other properties of /* Actually this event is also triggered when other properties of
* the stream change that are unrelated to the volume. However it is * the stream change that are unrelated to the volume. However it is
@ -357,6 +416,7 @@ gst_pulsering_context_subscribe_cb (pa_context * c,
/* inform streaming thread to notify */ /* inform streaming thread to notify */
g_atomic_int_compare_and_exchange (&psink->notify, 0, 1); g_atomic_int_compare_and_exchange (&psink->notify, 0, 1);
}
} }
#endif #endif
@ -367,44 +427,60 @@ gst_pulseringbuffer_open_device (GstRingBuffer * buf)
{ {
GstPulseSink *psink; GstPulseSink *psink;
GstPulseRingBuffer *pbuf; GstPulseRingBuffer *pbuf;
gchar *name; GstPulseContext *pctx;
pa_mainloop_api *api; pa_mainloop_api *api;
psink = GST_PULSESINK_CAST (GST_OBJECT_PARENT (buf)); psink = GST_PULSESINK_CAST (GST_OBJECT_PARENT (buf));
pbuf = GST_PULSERING_BUFFER_CAST (buf); pbuf = GST_PULSERING_BUFFER_CAST (buf);
g_assert (!pbuf->context);
g_assert (!pbuf->stream); g_assert (!pbuf->stream);
name = gst_pulse_client_name (); pbuf->context_name = g_strdup_printf ("%s@%s", gst_pulse_client_name (),
GST_STR_NULL (psink->server));
pa_threaded_mainloop_lock (psink->mainloop); pa_threaded_mainloop_lock (psink->mainloop);
g_mutex_lock (pa_ring_buffer_mutex);
pctx = g_hash_table_lookup (gst_pulse_shared_contexts, pbuf->context_name);
if (pctx == NULL) {
pctx = g_slice_new0 (GstPulseContext);
/* get the mainloop api and create a context */ /* get the mainloop api and create a context */
GST_LOG_OBJECT (psink, "new context with name %s", GST_STR_NULL (name)); GST_LOG_OBJECT (psink, "new context with name %s",
GST_STR_NULL (pbuf->context_name));
api = pa_threaded_mainloop_get_api (psink->mainloop); api = pa_threaded_mainloop_get_api (psink->mainloop);
if (!(pbuf->context = pa_context_new (api, name))) if (!(pctx->context = pa_context_new (api, pbuf->context_name)))
goto create_failed; goto create_failed;
pctx->ring_buffers = g_slist_append (pctx->ring_buffers, pbuf);
g_hash_table_insert (gst_pulse_shared_contexts, pbuf->context_name,
(gpointer) pctx);
/* register some essential callbacks */ /* register some essential callbacks */
pa_context_set_state_callback (pbuf->context, pa_context_set_state_callback (pctx->context,
gst_pulsering_context_state_cb, pbuf); gst_pulsering_context_state_cb, pctx);
#ifdef HAVE_PULSE_0_9_12 #ifdef HAVE_PULSE_0_9_12
pa_context_set_subscribe_callback (pbuf->context, pa_context_set_subscribe_callback (pctx->context,
gst_pulsering_context_subscribe_cb, pbuf); gst_pulsering_context_subscribe_cb, pctx);
#endif #endif
/* try to connect to the server and wait for completioni, we don't want to /* try to connect to the server and wait for completioni, we don't want to
* autospawn a deamon */ * autospawn a deamon */
GST_LOG_OBJECT (psink, "connect to server %s", GST_STR_NULL (psink->server)); GST_LOG_OBJECT (psink, "connect to server %s",
if (pa_context_connect (pbuf->context, psink->server, PA_CONTEXT_NOAUTOSPAWN, GST_STR_NULL (psink->server));
NULL) < 0) if (pa_context_connect (pctx->context, psink->server,
PA_CONTEXT_NOAUTOSPAWN, NULL) < 0)
goto connect_failed; goto connect_failed;
} else {
GST_LOG_OBJECT (psink, "reusing shared pulseaudio context with name %s",
GST_STR_NULL (pbuf->context_name));
pctx->ring_buffers = g_slist_append (pctx->ring_buffers, pbuf);
}
for (;;) { for (;;) {
pa_context_state_t state; pa_context_state_t state;
state = pa_context_get_state (pbuf->context); state = pa_context_get_state (pctx->context);
GST_LOG_OBJECT (psink, "context state is now %d", state); GST_LOG_OBJECT (psink, "context state is now %d", state);
@ -421,18 +497,18 @@ gst_pulseringbuffer_open_device (GstRingBuffer * buf)
GST_LOG_OBJECT (psink, "opened the device"); GST_LOG_OBJECT (psink, "opened the device");
g_mutex_unlock (pa_ring_buffer_mutex);
pa_threaded_mainloop_unlock (psink->mainloop); pa_threaded_mainloop_unlock (psink->mainloop);
g_free (name);
return TRUE; return TRUE;
/* ERRORS */ /* ERRORS */
unlock_and_fail: unlock_and_fail:
{ {
g_mutex_unlock (pa_ring_buffer_mutex);
gst_pulsering_destroy_context (pbuf); gst_pulsering_destroy_context (pbuf);
pa_threaded_mainloop_unlock (psink->mainloop); pa_threaded_mainloop_unlock (psink->mainloop);
g_free (name);
return FALSE; return FALSE;
} }
create_failed: create_failed:
@ -444,7 +520,7 @@ create_failed:
connect_failed: connect_failed:
{ {
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, ("Failed to connect: %s", GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, ("Failed to connect: %s",
pa_strerror (pa_context_errno (pbuf->context))), (NULL)); pa_strerror (pa_context_errno (pctx->context))), (NULL));
goto unlock_and_fail; goto unlock_and_fail;
} }
} }
@ -636,6 +712,7 @@ gst_pulseringbuffer_acquire (GstRingBuffer * buf, GstRingBufferSpec * spec)
{ {
GstPulseSink *psink; GstPulseSink *psink;
GstPulseRingBuffer *pbuf; GstPulseRingBuffer *pbuf;
GstPulseContext *pctx;
pa_buffer_attr wanted; pa_buffer_attr wanted;
const pa_buffer_attr *actual; const pa_buffer_attr *actual;
pa_channel_map channel_map; pa_channel_map channel_map;
@ -659,12 +736,12 @@ gst_pulseringbuffer_acquire (GstRingBuffer * buf, GstRingBufferSpec * spec)
pa_threaded_mainloop_lock (psink->mainloop); pa_threaded_mainloop_lock (psink->mainloop);
/* we need a context and a no stream */ /* we need a context and a no stream */
g_assert (pbuf->context); pctx = gst_pulsering_get_context (pbuf);
g_assert (!pbuf->stream); g_assert (!pbuf->stream);
/* enable event notifications */ /* enable event notifications */
GST_LOG_OBJECT (psink, "subscribing to context events"); GST_LOG_OBJECT (psink, "subscribing to context events");
if (!(o = pa_context_subscribe (pbuf->context, if (!(o = pa_context_subscribe (pctx->context,
PA_SUBSCRIPTION_MASK_SINK_INPUT, NULL, NULL))) PA_SUBSCRIPTION_MASK_SINK_INPUT, NULL, NULL)))
goto subscribe_failed; goto subscribe_failed;
@ -681,7 +758,7 @@ gst_pulseringbuffer_acquire (GstRingBuffer * buf, GstRingBufferSpec * spec)
/* create a stream */ /* create a stream */
GST_LOG_OBJECT (psink, "creating stream with name %s", name); GST_LOG_OBJECT (psink, "creating stream with name %s", name);
if (!(pbuf->stream = pa_stream_new (pbuf->context, if (!(pbuf->stream = pa_stream_new (pctx->context,
name, &pbuf->sample_spec, &channel_map))) name, &pbuf->sample_spec, &channel_map)))
goto stream_failed; goto stream_failed;
@ -816,21 +893,21 @@ subscribe_failed:
{ {
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, GST_ELEMENT_ERROR (psink, RESOURCE, FAILED,
("pa_context_subscribe() failed: %s", ("pa_context_subscribe() failed: %s",
pa_strerror (pa_context_errno (pbuf->context))), (NULL)); pa_strerror (pa_context_errno (pctx->context))), (NULL));
goto unlock_and_fail; goto unlock_and_fail;
} }
stream_failed: stream_failed:
{ {
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, GST_ELEMENT_ERROR (psink, RESOURCE, FAILED,
("Failed to create stream: %s", ("Failed to create stream: %s",
pa_strerror (pa_context_errno (pbuf->context))), (NULL)); pa_strerror (pa_context_errno (pctx->context))), (NULL));
goto unlock_and_fail; goto unlock_and_fail;
} }
connect_failed: connect_failed:
{ {
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, GST_ELEMENT_ERROR (psink, RESOURCE, FAILED,
("Failed to connect stream: %s", ("Failed to connect stream: %s",
pa_strerror (pa_context_errno (pbuf->context))), (NULL)); pa_strerror (pa_context_errno (pctx->context))), (NULL));
goto unlock_and_fail; goto unlock_and_fail;
} }
} }
@ -872,6 +949,7 @@ gst_pulsering_set_corked (GstPulseRingBuffer * pbuf, gboolean corked,
{ {
pa_operation *o = NULL; pa_operation *o = NULL;
GstPulseSink *psink; GstPulseSink *psink;
GstPulseContext *pctx = NULL;
gboolean res = FALSE; gboolean res = FALSE;
psink = GST_PULSESINK_CAST (GST_OBJECT_PARENT (pbuf)); psink = GST_PULSESINK_CAST (GST_OBJECT_PARENT (pbuf));
@ -907,9 +985,10 @@ server_dead:
} }
cork_failed: cork_failed:
{ {
pctx = gst_pulsering_get_context (pbuf);
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, GST_ELEMENT_ERROR (psink, RESOURCE, FAILED,
("pa_stream_cork() failed: %s", ("pa_stream_cork() failed: %s",
pa_strerror (pa_context_errno (pbuf->context))), (NULL)); pa_strerror (pa_context_errno (pctx->context))), (NULL));
goto cleanup; goto cleanup;
} }
} }
@ -1168,6 +1247,7 @@ gst_pulseringbuffer_commit (GstRingBuffer * buf, guint64 * sample,
{ {
GstPulseSink *psink; GstPulseSink *psink;
GstPulseRingBuffer *pbuf; GstPulseRingBuffer *pbuf;
GstPulseContext *pctx;
guint result; guint result;
guint8 *data_end; guint8 *data_end;
gboolean reverse; gboolean reverse;
@ -1539,16 +1619,20 @@ was_paused:
} }
writable_size_failed: writable_size_failed:
{ {
pctx = gst_pulsering_get_context (pbuf);
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, GST_ELEMENT_ERROR (psink, RESOURCE, FAILED,
("pa_stream_writable_size() failed: %s", ("pa_stream_writable_size() failed: %s",
pa_strerror (pa_context_errno (pbuf->context))), (NULL)); pa_strerror (pa_context_errno (pctx->context))), (NULL));
goto unlock_and_fail; goto unlock_and_fail;
} }
write_failed: write_failed:
{ {
pctx = gst_pulsering_get_context (pbuf);
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, GST_ELEMENT_ERROR (psink, RESOURCE, FAILED,
("pa_stream_write() failed: %s", ("pa_stream_write() failed: %s",
pa_strerror (pa_context_errno (pbuf->context))), (NULL)); pa_strerror (pa_context_errno (pctx->context))), (NULL));
goto unlock_and_fail; goto unlock_and_fail;
} }
} }
@ -1718,6 +1802,8 @@ gst_pulsesink_class_init (GstPulseSinkClass * klass)
gstelement_class->change_state = gstelement_class->change_state =
GST_DEBUG_FUNCPTR (gst_pulsesink_change_state); GST_DEBUG_FUNCPTR (gst_pulsesink_change_state);
gst_pulseringbuffer_init_contexts ();
gstaudiosink_class->create_ringbuffer = gstaudiosink_class->create_ringbuffer =
GST_DEBUG_FUNCPTR (gst_pulsesink_create_ringbuffer); GST_DEBUG_FUNCPTR (gst_pulsesink_create_ringbuffer);
@ -1846,6 +1932,7 @@ gst_pulsesink_set_volume (GstPulseSink * psink, gdouble volume)
pa_cvolume v; pa_cvolume v;
pa_operation *o = NULL; pa_operation *o = NULL;
GstPulseRingBuffer *pbuf; GstPulseRingBuffer *pbuf;
GstPulseContext *pctx;
uint32_t idx; uint32_t idx;
if (!psink->mainloop) if (!psink->mainloop)
@ -1864,7 +1951,9 @@ gst_pulsesink_set_volume (GstPulseSink * psink, gdouble volume)
gst_pulse_cvolume_from_linear (&v, pbuf->sample_spec.channels, volume); gst_pulse_cvolume_from_linear (&v, pbuf->sample_spec.channels, volume);
if (!(o = pa_context_set_sink_input_volume (pbuf->context, idx, pctx = gst_pulsering_get_context (pbuf);
if (!(o = pa_context_set_sink_input_volume (pctx->context, idx,
&v, NULL, NULL))) &v, NULL, NULL)))
goto volume_failed; goto volume_failed;
@ -1904,7 +1993,7 @@ volume_failed:
{ {
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, GST_ELEMENT_ERROR (psink, RESOURCE, FAILED,
("pa_stream_set_sink_input_volume() failed: %s", ("pa_stream_set_sink_input_volume() failed: %s",
pa_strerror (pa_context_errno (pbuf->context))), (NULL)); pa_strerror (pa_context_errno (pctx->context))), (NULL));
goto unlock; goto unlock;
} }
} }
@ -1914,6 +2003,7 @@ gst_pulsesink_set_mute (GstPulseSink * psink, gboolean mute)
{ {
pa_operation *o = NULL; pa_operation *o = NULL;
GstPulseRingBuffer *pbuf; GstPulseRingBuffer *pbuf;
GstPulseContext *pctx;
uint32_t idx; uint32_t idx;
if (!psink->mainloop) if (!psink->mainloop)
@ -1930,7 +2020,9 @@ gst_pulsesink_set_mute (GstPulseSink * psink, gboolean mute)
if ((idx = pa_stream_get_index (pbuf->stream)) == PA_INVALID_INDEX) if ((idx = pa_stream_get_index (pbuf->stream)) == PA_INVALID_INDEX)
goto no_index; goto no_index;
if (!(o = pa_context_set_sink_input_mute (pbuf->context, idx, pctx = gst_pulsering_get_context (pbuf);
if (!(o = pa_context_set_sink_input_mute (pctx->context, idx,
mute, NULL, NULL))) mute, NULL, NULL)))
goto mute_failed; goto mute_failed;
@ -1970,7 +2062,7 @@ mute_failed:
{ {
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, GST_ELEMENT_ERROR (psink, RESOURCE, FAILED,
("pa_stream_set_sink_input_mute() failed: %s", ("pa_stream_set_sink_input_mute() failed: %s",
pa_strerror (pa_context_errno (pbuf->context))), (NULL)); pa_strerror (pa_context_errno (pctx->context))), (NULL));
goto unlock; goto unlock;
} }
} }
@ -2007,6 +2099,7 @@ static gdouble
gst_pulsesink_get_volume (GstPulseSink * psink) gst_pulsesink_get_volume (GstPulseSink * psink)
{ {
GstPulseRingBuffer *pbuf; GstPulseRingBuffer *pbuf;
GstPulseContext *pctx;
pa_operation *o = NULL; pa_operation *o = NULL;
gdouble v = DEFAULT_VOLUME; gdouble v = DEFAULT_VOLUME;
uint32_t idx; uint32_t idx;
@ -2023,7 +2116,9 @@ gst_pulsesink_get_volume (GstPulseSink * psink)
if ((idx = pa_stream_get_index (pbuf->stream)) == PA_INVALID_INDEX) if ((idx = pa_stream_get_index (pbuf->stream)) == PA_INVALID_INDEX)
goto no_index; goto no_index;
if (!(o = pa_context_get_sink_input_info (pbuf->context, idx, pctx = gst_pulsering_get_context (pbuf);
if (!(o = pa_context_get_sink_input_info (pctx->context, idx,
gst_pulsesink_sink_input_info_cb, pbuf))) gst_pulsesink_sink_input_info_cb, pbuf)))
goto info_failed; goto info_failed;
@ -2069,7 +2164,7 @@ info_failed:
{ {
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, GST_ELEMENT_ERROR (psink, RESOURCE, FAILED,
("pa_context_get_sink_input_info() failed: %s", ("pa_context_get_sink_input_info() failed: %s",
pa_strerror (pa_context_errno (pbuf->context))), (NULL)); pa_strerror (pa_context_errno (pctx->context))), (NULL));
goto unlock; goto unlock;
} }
} }
@ -2078,6 +2173,7 @@ static gboolean
gst_pulsesink_get_mute (GstPulseSink * psink) gst_pulsesink_get_mute (GstPulseSink * psink)
{ {
GstPulseRingBuffer *pbuf; GstPulseRingBuffer *pbuf;
GstPulseContext *pctx;
pa_operation *o = NULL; pa_operation *o = NULL;
uint32_t idx; uint32_t idx;
gboolean mute = FALSE; gboolean mute = FALSE;
@ -2095,7 +2191,9 @@ gst_pulsesink_get_mute (GstPulseSink * psink)
if ((idx = pa_stream_get_index (pbuf->stream)) == PA_INVALID_INDEX) if ((idx = pa_stream_get_index (pbuf->stream)) == PA_INVALID_INDEX)
goto no_index; goto no_index;
if (!(o = pa_context_get_sink_input_info (pbuf->context, idx, pctx = gst_pulsering_get_context (pbuf);
if (!(o = pa_context_get_sink_input_info (pctx->context, idx,
gst_pulsesink_sink_input_info_cb, pbuf))) gst_pulsesink_sink_input_info_cb, pbuf)))
goto info_failed; goto info_failed;
@ -2106,6 +2204,7 @@ gst_pulsesink_get_mute (GstPulseSink * psink)
} }
unlock: unlock:
if (o) if (o)
pa_operation_unref (o); pa_operation_unref (o);
@ -2134,7 +2233,7 @@ info_failed:
{ {
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, GST_ELEMENT_ERROR (psink, RESOURCE, FAILED,
("pa_context_get_sink_input_info() failed: %s", ("pa_context_get_sink_input_info() failed: %s",
pa_strerror (pa_context_errno (pbuf->context))), (NULL)); pa_strerror (pa_context_errno (pctx->context))), (NULL));
goto unlock; goto unlock;
} }
} }
@ -2169,6 +2268,7 @@ static gchar *
gst_pulsesink_device_description (GstPulseSink * psink) gst_pulsesink_device_description (GstPulseSink * psink)
{ {
GstPulseRingBuffer *pbuf; GstPulseRingBuffer *pbuf;
GstPulseContext *pctx;
pa_operation *o = NULL; pa_operation *o = NULL;
gchar *t; gchar *t;
@ -2180,7 +2280,9 @@ gst_pulsesink_device_description (GstPulseSink * psink)
if (pbuf == NULL || pbuf->stream == NULL) if (pbuf == NULL || pbuf->stream == NULL)
goto no_buffer; goto no_buffer;
if (!(o = pa_context_get_sink_info_by_index (pbuf->context, pctx = gst_pulsering_get_context (pbuf);
if (!(o = pa_context_get_sink_info_by_index (pctx->context,
pa_stream_get_device_index (pbuf->stream), pa_stream_get_device_index (pbuf->stream),
gst_pulsesink_sink_info_cb, pbuf))) gst_pulsesink_sink_info_cb, pbuf)))
goto info_failed; goto info_failed;
@ -2192,6 +2294,7 @@ gst_pulsesink_device_description (GstPulseSink * psink)
} }
unlock: unlock:
if (o) if (o)
pa_operation_unref (o); pa_operation_unref (o);
@ -2215,7 +2318,7 @@ info_failed:
{ {
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, GST_ELEMENT_ERROR (psink, RESOURCE, FAILED,
("pa_context_get_sink_info_by_index() failed: %s", ("pa_context_get_sink_info_by_index() failed: %s",
pa_strerror (pa_context_errno (pbuf->context))), (NULL)); pa_strerror (pa_context_errno (pctx->context))), (NULL));
goto unlock; goto unlock;
} }
} }
@ -2287,6 +2390,7 @@ gst_pulsesink_change_title (GstPulseSink * psink, const gchar * t)
{ {
pa_operation *o = NULL; pa_operation *o = NULL;
GstPulseRingBuffer *pbuf; GstPulseRingBuffer *pbuf;
GstPulseContext *pctx;
pa_threaded_mainloop_lock (psink->mainloop); pa_threaded_mainloop_lock (psink->mainloop);
@ -2298,11 +2402,14 @@ gst_pulsesink_change_title (GstPulseSink * psink, const gchar * t)
g_free (pbuf->stream_name); g_free (pbuf->stream_name);
pbuf->stream_name = g_strdup (t); pbuf->stream_name = g_strdup (t);
pctx = gst_pulsering_get_context (pbuf);
if (!(o = pa_stream_set_name (pbuf->stream, pbuf->stream_name, NULL, NULL))) if (!(o = pa_stream_set_name (pbuf->stream, pbuf->stream_name, NULL, NULL)))
goto name_failed; goto name_failed;
/* We're not interested if this operation failed or not */ /* We're not interested if this operation failed or not */
unlock: unlock:
if (o) if (o)
pa_operation_unref (o); pa_operation_unref (o);
pa_threaded_mainloop_unlock (psink->mainloop); pa_threaded_mainloop_unlock (psink->mainloop);
@ -2319,7 +2426,7 @@ name_failed:
{ {
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, GST_ELEMENT_ERROR (psink, RESOURCE, FAILED,
("pa_stream_set_name() failed: %s", ("pa_stream_set_name() failed: %s",
pa_strerror (pa_context_errno (pbuf->context))), (NULL)); pa_strerror (pa_context_errno (pctx->context))), (NULL));
goto unlock; goto unlock;
} }
} }
@ -2345,6 +2452,7 @@ gst_pulsesink_change_props (GstPulseSink * psink, GstTagList * l)
gboolean empty = TRUE; gboolean empty = TRUE;
pa_operation *o = NULL; pa_operation *o = NULL;
GstPulseRingBuffer *pbuf; GstPulseRingBuffer *pbuf;
GstPulseContext *pctx;
pl = pa_proplist_new (); pl = pa_proplist_new ();
@ -2365,10 +2473,13 @@ gst_pulsesink_change_props (GstPulseSink * psink, GstTagList * l)
goto finish; goto finish;
pa_threaded_mainloop_lock (psink->mainloop); pa_threaded_mainloop_lock (psink->mainloop);
pbuf = GST_PULSERING_BUFFER_CAST (GST_BASE_AUDIO_SINK (psink)->ringbuffer); pbuf = GST_PULSERING_BUFFER_CAST (GST_BASE_AUDIO_SINK (psink)->ringbuffer);
if (pbuf == NULL || pbuf->stream == NULL) if (pbuf == NULL || pbuf->stream == NULL)
goto no_buffer; goto no_buffer;
pctx = gst_pulsering_get_context (pbuf);
if (!(o = pa_stream_proplist_update (pbuf->stream, PA_UPDATE_REPLACE, if (!(o = pa_stream_proplist_update (pbuf->stream, PA_UPDATE_REPLACE,
pl, NULL, NULL))) pl, NULL, NULL)))
goto update_failed; goto update_failed;
@ -2398,7 +2509,7 @@ update_failed:
{ {
GST_ELEMENT_ERROR (psink, RESOURCE, FAILED, GST_ELEMENT_ERROR (psink, RESOURCE, FAILED,
("pa_stream_proplist_update() failed: %s", ("pa_stream_proplist_update() failed: %s",
pa_strerror (pa_context_errno (pbuf->context))), (NULL)); pa_strerror (pa_context_errno (pctx->context))), (NULL));
goto unlock; goto unlock;
} }
} }