blob: 06e09862227f288e2d809891f389736701ebfd74 [file] [log] [blame]
/*-*- Mode: C; c-basic-offset: 2 -*-*/
/*
* GStreamer pulseaudio plugin
*
* Copyright (c) 2004-2008 Lennart Poettering
*
* gst-pulse is free software; you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of the
* License, or (at your option) any later version.
*
* gst-pulse is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with gst-pulse; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
* USA.
*/
/* FIXME 0.11: suppress warnings for deprecated API such as GValueArray
* with newer GLib versions (>= 2.31.0) */
#define GLIB_DISABLE_DEPRECATION_WARNINGS
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include "pulseprobe.h"
#include "pulseutil.h"
GST_DEBUG_CATEGORY_EXTERN (pulse_debug);
#define GST_CAT_DEFAULT pulse_debug
static void
gst_pulseprobe_context_state_cb (pa_context * context, void *userdata)
{
GstPulseProbe *c = (GstPulseProbe *) userdata;
/* Called from the background thread! */
switch (pa_context_get_state (context)) {
case PA_CONTEXT_READY:
case PA_CONTEXT_TERMINATED:
case PA_CONTEXT_FAILED:
pa_threaded_mainloop_signal (c->mainloop, 0);
break;
case PA_CONTEXT_UNCONNECTED:
case PA_CONTEXT_CONNECTING:
case PA_CONTEXT_AUTHORIZING:
case PA_CONTEXT_SETTING_NAME:
break;
}
}
static void
gst_pulseprobe_sink_info_cb (pa_context * context, const pa_sink_info * i,
int eol, void *userdata)
{
GstPulseProbe *c = (GstPulseProbe *) userdata;
/* Called from the background thread! */
if (eol || !i) {
c->operation_success = eol > 0;
pa_threaded_mainloop_signal (c->mainloop, 0);
}
if (i)
c->devices = g_list_append (c->devices, g_strdup (i->name));
}
static void
gst_pulseprobe_source_info_cb (pa_context * context, const pa_source_info * i,
int eol, void *userdata)
{
GstPulseProbe *c = (GstPulseProbe *) userdata;
/* Called from the background thread! */
if (eol || !i) {
c->operation_success = eol > 0;
pa_threaded_mainloop_signal (c->mainloop, 0);
}
if (i)
c->devices = g_list_append (c->devices, g_strdup (i->name));
}
static void
gst_pulseprobe_invalidate (GstPulseProbe * c)
{
g_list_foreach (c->devices, (GFunc) g_free, NULL);
g_list_free (c->devices);
c->devices = NULL;
c->devices_valid = FALSE;
}
static gboolean
gst_pulseprobe_open (GstPulseProbe * c)
{
int e;
gchar *name;
g_assert (c);
GST_DEBUG_OBJECT (c->object, "probe open");
c->mainloop = pa_threaded_mainloop_new ();
if (!c->mainloop)
return FALSE;
e = pa_threaded_mainloop_start (c->mainloop);
if (e < 0)
return FALSE;
name = gst_pulse_client_name ();
pa_threaded_mainloop_lock (c->mainloop);
if (!(c->context =
pa_context_new (pa_threaded_mainloop_get_api (c->mainloop), name))) {
GST_WARNING_OBJECT (c->object, "Failed to create context");
goto unlock_and_fail;
}
pa_context_set_state_callback (c->context, gst_pulseprobe_context_state_cb,
c);
if (pa_context_connect (c->context, c->server, 0, NULL) < 0) {
GST_WARNING_OBJECT (c->object, "Failed to connect context: %s",
pa_strerror (pa_context_errno (c->context)));
goto unlock_and_fail;
}
for (;;) {
pa_context_state_t state;
state = pa_context_get_state (c->context);
if (!PA_CONTEXT_IS_GOOD (state)) {
GST_WARNING_OBJECT (c->object, "Failed to connect context: %s",
pa_strerror (pa_context_errno (c->context)));
goto unlock_and_fail;
}
if (state == PA_CONTEXT_READY)
break;
/* Wait until the context is ready */
pa_threaded_mainloop_wait (c->mainloop);
}
pa_threaded_mainloop_unlock (c->mainloop);
g_free (name);
gst_pulseprobe_invalidate (c);
return TRUE;
unlock_and_fail:
if (c->mainloop)
pa_threaded_mainloop_unlock (c->mainloop);
g_free (name);
return FALSE;
}
static gboolean
gst_pulseprobe_is_dead (GstPulseProbe * pulseprobe)
{
if (!pulseprobe->context ||
!PA_CONTEXT_IS_GOOD (pa_context_get_state (pulseprobe->context))) {
const gchar *err_str =
pulseprobe->context ?
pa_strerror (pa_context_errno (pulseprobe->context)) : NULL;
GST_ELEMENT_ERROR ((pulseprobe), RESOURCE, FAILED,
("Disconnected: %s", err_str), (NULL));
return TRUE;
}
return FALSE;
}
static gboolean
gst_pulseprobe_enumerate (GstPulseProbe * c)
{
pa_operation *o = NULL;
GST_DEBUG_OBJECT (c->object, "probe enumerate");
pa_threaded_mainloop_lock (c->mainloop);
if (c->enumerate_sinks) {
/* Get sink info */
if (!(o =
pa_context_get_sink_info_list (c->context,
gst_pulseprobe_sink_info_cb, c))) {
GST_WARNING_OBJECT (c->object, "Failed to get sink info: %s",
pa_strerror (pa_context_errno (c->context)));
goto unlock_and_fail;
}
c->operation_success = FALSE;
while (pa_operation_get_state (o) == PA_OPERATION_RUNNING) {
if (gst_pulseprobe_is_dead (c))
goto unlock_and_fail;
pa_threaded_mainloop_wait (c->mainloop);
}
if (!c->operation_success) {
GST_WARNING_OBJECT (c->object, "Failed to get sink info: %s",
pa_strerror (pa_context_errno (c->context)));
goto unlock_and_fail;
}
pa_operation_unref (o);
o = NULL;
}
if (c->enumerate_sources) {
/* Get source info */
if (!(o =
pa_context_get_source_info_list (c->context,
gst_pulseprobe_source_info_cb, c))) {
GST_WARNING_OBJECT (c->object, "Failed to get source info: %s",
pa_strerror (pa_context_errno (c->context)));
goto unlock_and_fail;
}
c->operation_success = FALSE;
while (pa_operation_get_state (o) == PA_OPERATION_RUNNING) {
if (gst_pulseprobe_is_dead (c))
goto unlock_and_fail;
pa_threaded_mainloop_wait (c->mainloop);
}
if (!c->operation_success) {
GST_WARNING_OBJECT (c->object, "Failed to get sink info: %s",
pa_strerror (pa_context_errno (c->context)));
goto unlock_and_fail;
}
pa_operation_unref (o);
o = NULL;
}
c->devices_valid = TRUE;
pa_threaded_mainloop_unlock (c->mainloop);
return TRUE;
unlock_and_fail:
if (o)
pa_operation_unref (o);
pa_threaded_mainloop_unlock (c->mainloop);
return FALSE;
}
static void
gst_pulseprobe_close (GstPulseProbe * c)
{
g_assert (c);
GST_DEBUG_OBJECT (c->object, "probe close");
if (c->mainloop)
pa_threaded_mainloop_stop (c->mainloop);
if (c->context) {
pa_context_disconnect (c->context);
pa_context_set_state_callback (c->context, NULL, NULL);
pa_context_unref (c->context);
c->context = NULL;
}
if (c->mainloop) {
pa_threaded_mainloop_free (c->mainloop);
c->mainloop = NULL;
}
}
GstPulseProbe *
gst_pulseprobe_new (GObject * object, GObjectClass * klass,
guint prop_id, const gchar * server, gboolean sinks, gboolean sources)
{
GstPulseProbe *c = NULL;
c = g_new (GstPulseProbe, 1);
c->object = object; /* We don't inc the ref counter here to avoid a ref loop */
c->server = g_strdup (server);
c->enumerate_sinks = sinks;
c->enumerate_sources = sources;
c->mainloop = NULL;
c->context = NULL;
c->prop_id = prop_id;
c->properties =
g_list_append (NULL, g_object_class_find_property (klass, "device"));
c->devices = NULL;
c->devices_valid = FALSE;
c->operation_success = FALSE;
return c;
}
void
gst_pulseprobe_free (GstPulseProbe * c)
{
g_assert (c);
gst_pulseprobe_close (c);
g_list_free (c->properties);
g_free (c->server);
g_list_foreach (c->devices, (GFunc) g_free, NULL);
g_list_free (c->devices);
g_free (c);
}
const GList *
gst_pulseprobe_get_properties (GstPulseProbe * c)
{
return c->properties;
}
gboolean
gst_pulseprobe_needs_probe (GstPulseProbe * c, guint prop_id,
const GParamSpec * pspec)
{
if (prop_id == c->prop_id)
return !c->devices_valid;
G_OBJECT_WARN_INVALID_PROPERTY_ID (c->object, prop_id, pspec);
return FALSE;
}
void
gst_pulseprobe_probe_property (GstPulseProbe * c, guint prop_id,
const GParamSpec * pspec)
{
if (prop_id != c->prop_id) {
G_OBJECT_WARN_INVALID_PROPERTY_ID (c->object, prop_id, pspec);
return;
}
if (gst_pulseprobe_open (c)) {
gst_pulseprobe_enumerate (c);
gst_pulseprobe_close (c);
}
}
#if 0
GValueArray *
gst_pulseprobe_get_values (GstPulseProbe * c, guint prop_id,
const GParamSpec * pspec)
{
GValueArray *array;
GValue value = {
0
};
GList *item;
if (prop_id != c->prop_id) {
G_OBJECT_WARN_INVALID_PROPERTY_ID (c->object, prop_id, pspec);
return NULL;
}
if (!c->devices_valid)
return NULL;
array = g_value_array_new (g_list_length (c->devices));
g_value_init (&value, G_TYPE_STRING);
for (item = c->devices; item != NULL; item = item->next) {
GST_WARNING_OBJECT (c->object, "device found: %s",
(const gchar *) item->data);
g_value_set_string (&value, (const gchar *) item->data);
g_value_array_append (array, &value);
}
g_value_unset (&value);
return array;
}
#endif
void
gst_pulseprobe_set_server (GstPulseProbe * c, const gchar * server)
{
g_assert (c);
gst_pulseprobe_invalidate (c);
g_free (c->server);
c->server = g_strdup (server);
}