mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-04 14:38:48 +00:00
0037635bf2
newer pulseaudio. Fixes: #567794 * Hook pulsesink's volume property up with the stream volume -- not the sink volume in PA. * Read the device description directly from the sink instead of going via the mixer. * Properly implement _reset() methods for both sink and source to avoid deadlocks when shutting down a pipeline. * Replace all simple pa_threaded_mainloop_wait() by proper loops to guarantee that we wait for the right event in case multiple events are fired. While this is not strictly necessary in many cases it certainly is more correct and makes me sleep better at night. * Replace CHECK_DEAD_GOTO macros with proper functions * Extend the number of supported channels to 32 since that is the actual limit in PA. * Get rid of _dispose() methods since we don't need them. * Increase the volume property upper limit of the sink to 1000. * Reset function pointers after we disconnect a stream/context. Better fix for bug 556986. * Reset the state of the element properly if open/prepare fails * Cork the PA stream when the pipeline is paused. This allows the PA * daemon to close audio device on pause and thus save a bit of power. * Set PA stream properties based on GST tags such as GST_TAG_TITLE, GST_TAG_ARTIST, and so on. Signed-off-by: Lennart Poettering <lennart@poettering.net>
401 lines
9.1 KiB
C
401 lines
9.1 KiB
C
/*
|
|
* 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.
|
|
*/
|
|
|
|
#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 = gst_pulse_client_name ();
|
|
|
|
g_assert (c);
|
|
|
|
c->mainloop = pa_threaded_mainloop_new ();
|
|
g_assert (c->mainloop);
|
|
|
|
e = pa_threaded_mainloop_start (c->mainloop);
|
|
g_assert (e == 0);
|
|
|
|
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;
|
|
|
|
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 = 0;
|
|
|
|
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 = 0;
|
|
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);
|
|
|
|
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 = 0;
|
|
|
|
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);
|
|
}
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
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);
|
|
}
|