bluez: Add possible capabilities and connect to audio server

This commit is contained in:
Marcel Holtmann 2007-08-26 14:14:34 +00:00 committed by Tim-Philipp Müller
parent 8fdac7fec0
commit cb7d3b6d4e
2 changed files with 121 additions and 61 deletions

View File

@ -25,14 +25,25 @@
#include <config.h> #include <config.h>
#endif #endif
#include <unistd.h>
#include <sys/un.h>
#include <sys/socket.h>
#include "ipc.h" #include "ipc.h"
#include "sbc.h"
#include "gsta2dpsink.h" #include "gsta2dpsink.h"
GST_DEBUG_CATEGORY_STATIC (a2dp_sink_debug); GST_DEBUG_CATEGORY_STATIC (a2dp_sink_debug);
#define GST_CAT_DEFAULT a2dp_sink_debug #define GST_CAT_DEFAULT a2dp_sink_debug
#define DEFAULT_DEVICE "default"
enum
{
PROP_0,
PROP_DEVICE,
};
GST_BOILERPLATE (GstA2dpSink, gst_a2dp_sink, GstAudioSink, GST_TYPE_AUDIO_SINK); GST_BOILERPLATE (GstA2dpSink, gst_a2dp_sink, GstAudioSink, GST_TYPE_AUDIO_SINK);
static const GstElementDetails a2dp_sink_details = static const GstElementDetails a2dp_sink_details =
@ -41,109 +52,129 @@ GST_ELEMENT_DETAILS ("Bluetooth A2DP sink",
"Plays audio to an A2DP device", "Plays audio to an A2DP device",
"Marcel Holtmann <marcel@holtmann.org>"); "Marcel Holtmann <marcel@holtmann.org>");
static GstStaticPadTemplate sink_factory = static GstStaticPadTemplate a2dp_sink_factory =
GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS, GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS,
GST_STATIC_CAPS ("ANY")); GST_STATIC_CAPS ("audio/x-sbc, "
"rate = (int) { 16000, 32000, 44100, 48000 }, "
"channels = (int) [ 1, 2 ], "
"mode = (string) { mono, dual, stereo, joint }, "
"blocks = (int) { 4, 8, 12, 16 }, "
"subbands = (int) { 4, 8 }, "
"allocation = (string) { snr, loudness }; "
"audio/mpeg, "
"mpegversion = (int) 1, "
"layer = (int) [ 1, 3 ], "
"rate = (int) { 16000, 22050, 24000, 32000, 44100, 48000 }, "
"channels = (int) [ 1, 2 ]"));
static void static void
gst_a2dp_sink_base_init (gpointer g_class) gst_a2dp_sink_base_init (gpointer g_class)
{ {
GstElementClass *element_class = GST_ELEMENT_CLASS (g_class); GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
GST_DEBUG ("");
gst_element_class_add_pad_template (element_class, gst_element_class_add_pad_template (element_class,
gst_static_pad_template_get (&sink_factory)); gst_static_pad_template_get (&a2dp_sink_factory));
gst_element_class_set_details (element_class, &a2dp_sink_details); gst_element_class_set_details (element_class, &a2dp_sink_details);
} }
static void
gst_a2dp_sink_dispose (GObject * object)
{
G_OBJECT_CLASS (parent_class)->dispose (object);
}
static void static void
gst_a2dp_sink_finalize (GObject * object) gst_a2dp_sink_finalize (GObject * object)
{ {
G_OBJECT_CLASS (parent_class)->finalize (object); GstA2dpSink *sink = GST_A2DP_SINK (object);
}
static void g_io_channel_close (sink->server);
gst_a2dp_sink_get_property (GObject * object, guint prop_id,
GValue * value, GParamSpec * pspec) g_free (sink->device);
{
switch (prop_id) { G_OBJECT_CLASS (parent_class)->finalize (object);
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
} }
static void static void
gst_a2dp_sink_set_property (GObject * object, guint prop_id, gst_a2dp_sink_set_property (GObject * object, guint prop_id,
const GValue * value, GParamSpec * pspec) const GValue * value, GParamSpec * pspec)
{ {
GstA2dpSink *sink = GST_A2DP_SINK (object);
switch (prop_id) { switch (prop_id) {
case PROP_DEVICE:
g_free (sink->device);
sink->device = g_value_dup_string (value);
if (sink->device == NULL)
sink->device = g_strdup (DEFAULT_DEVICE);
break;
default: default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break; break;
} }
} }
static GstCaps * static void
gst_a2dp_sink_getcaps (GstBaseSink * basesink) gst_a2dp_sink_get_property (GObject * object, guint prop_id,
GValue * value, GParamSpec * pspec)
{ {
GST_DEBUG_OBJECT (basesink, ""); GstA2dpSink *sink = GST_A2DP_SINK (object);
return NULL; switch (prop_id) {
case PROP_DEVICE:
g_value_set_string (value, sink->device);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
} }
static gboolean static gboolean
gst_a2dp_sink_open (GstAudioSink * audiosink) gst_a2dp_sink_open (GstAudioSink * self)
{ {
GST_DEBUG_OBJECT (audiosink, ""); GstA2dpSink *sink = GST_A2DP_SINK (self);
printf ("device %s\n", sink->device);
printf ("open\n");
return TRUE; return TRUE;
} }
static gboolean static gboolean
gst_a2dp_sink_prepare (GstAudioSink * audiosink, GstRingBufferSpec * spec) gst_a2dp_sink_prepare (GstAudioSink * self, GstRingBufferSpec * spec)
{ {
GST_DEBUG_OBJECT (audiosink, "spec %p", spec); printf ("perpare\n");
printf ("rate %d\n", spec->rate);
printf ("channels %d\n", spec->channels);
return TRUE; return TRUE;
} }
static gboolean static gboolean
gst_a2dp_sink_unprepare (GstAudioSink * audiosink) gst_a2dp_sink_unprepare (GstAudioSink * self)
{ {
GST_DEBUG_OBJECT (audiosink, ""); printf ("unprepare\n");
return TRUE; return TRUE;
} }
static gboolean static gboolean
gst_a2dp_sink_close (GstAudioSink * audiosink) gst_a2dp_sink_close (GstAudioSink * self)
{ {
GST_DEBUG_OBJECT (audiosink, ""); printf ("close\n");
return TRUE; return TRUE;
} }
static guint static guint
gst_a2dp_sink_write (GstAudioSink * audiosink, gpointer data, guint length) gst_a2dp_sink_write (GstAudioSink * self, gpointer data, guint length)
{ {
GST_DEBUG_OBJECT (audiosink, "data %p length %d", data, length); return 0;
return length;
} }
static guint static guint
gst_a2dp_sink_delay (GstAudioSink * audiosink) gst_a2dp_sink_delay (GstAudioSink * audiosink)
{ {
GST_DEBUG_OBJECT (audiosink, ""); printf ("delay\n");
return 0; return 0;
} }
@ -151,40 +182,65 @@ gst_a2dp_sink_delay (GstAudioSink * audiosink)
static void static void
gst_a2dp_sink_reset (GstAudioSink * audiosink) gst_a2dp_sink_reset (GstAudioSink * audiosink)
{ {
GST_DEBUG_OBJECT (audiosink, ""); printf ("reset\n");
}
static gboolean
server_callback (GIOChannel * chan, GIOCondition cond, gpointer data)
{
printf ("callback\n");
return TRUE;
} }
static void static void
gst_a2dp_sink_class_init (GstA2dpSinkClass * klass) gst_a2dp_sink_class_init (GstA2dpSinkClass * klass)
{ {
GObjectClass *gobject_class = G_OBJECT_CLASS (klass); GObjectClass *object_class = G_OBJECT_CLASS (klass);
GstBaseSinkClass *gstbasesink_class = GST_BASE_SINK_CLASS (klass); GstAudioSinkClass *audiosink_class = GST_AUDIO_SINK_CLASS (klass);
GstAudioSinkClass *gstaudiosink_class = GST_AUDIO_SINK_CLASS (klass);
GST_DEBUG ("");
parent_class = g_type_class_peek_parent (klass); parent_class = g_type_class_peek_parent (klass);
gobject_class->dispose = GST_DEBUG_FUNCPTR (gst_a2dp_sink_dispose); object_class->finalize = GST_DEBUG_FUNCPTR (gst_a2dp_sink_finalize);
gobject_class->finalize = GST_DEBUG_FUNCPTR (gst_a2dp_sink_finalize); object_class->set_property = GST_DEBUG_FUNCPTR (gst_a2dp_sink_set_property);
gobject_class->get_property = GST_DEBUG_FUNCPTR (gst_a2dp_sink_get_property); object_class->get_property = GST_DEBUG_FUNCPTR (gst_a2dp_sink_get_property);
gobject_class->set_property = GST_DEBUG_FUNCPTR (gst_a2dp_sink_set_property);
gstbasesink_class->get_caps = GST_DEBUG_FUNCPTR (gst_a2dp_sink_getcaps); audiosink_class->open = GST_DEBUG_FUNCPTR (gst_a2dp_sink_open);
audiosink_class->prepare = GST_DEBUG_FUNCPTR (gst_a2dp_sink_prepare);
audiosink_class->unprepare = GST_DEBUG_FUNCPTR (gst_a2dp_sink_unprepare);
audiosink_class->close = GST_DEBUG_FUNCPTR (gst_a2dp_sink_close);
audiosink_class->write = GST_DEBUG_FUNCPTR (gst_a2dp_sink_write);
audiosink_class->delay = GST_DEBUG_FUNCPTR (gst_a2dp_sink_delay);
audiosink_class->reset = GST_DEBUG_FUNCPTR (gst_a2dp_sink_reset);
gstaudiosink_class->open = GST_DEBUG_FUNCPTR (gst_a2dp_sink_open); g_object_class_install_property (object_class, PROP_DEVICE,
gstaudiosink_class->prepare = GST_DEBUG_FUNCPTR (gst_a2dp_sink_prepare); g_param_spec_string ("device", "Device",
gstaudiosink_class->unprepare = GST_DEBUG_FUNCPTR (gst_a2dp_sink_unprepare); "Bluetooth remote device", DEFAULT_DEVICE, G_PARAM_READWRITE));
gstaudiosink_class->close = GST_DEBUG_FUNCPTR (gst_a2dp_sink_close);
gstaudiosink_class->write = GST_DEBUG_FUNCPTR (gst_a2dp_sink_write);
gstaudiosink_class->delay = GST_DEBUG_FUNCPTR (gst_a2dp_sink_delay);
gstaudiosink_class->reset = GST_DEBUG_FUNCPTR (gst_a2dp_sink_reset);
GST_DEBUG_CATEGORY_INIT (a2dp_sink_debug, "a2dpsink", 0, "A2DP sink element"); GST_DEBUG_CATEGORY_INIT (a2dp_sink_debug, "a2dpsink", 0, "A2DP sink element");
} }
static void static void
gst_a2dp_sink_init (GstA2dpSink * a2dpsink, GstA2dpSinkClass * klass) gst_a2dp_sink_init (GstA2dpSink * self, GstA2dpSinkClass * klass)
{ {
GST_DEBUG_OBJECT (a2dpsink, ""); struct sockaddr_un addr = { AF_UNIX, IPC_SOCKET_NAME };
int sk;
self->device = g_strdup (DEFAULT_DEVICE);
sk = socket (PF_LOCAL, SOCK_STREAM, 0);
if (sk < 0)
return;
if (connect (sk, (struct sockaddr *) &addr, sizeof (addr)) < 0) {
close (sk);
return;
}
self->server = g_io_channel_unix_new (sk);
g_io_add_watch (self->server, G_IO_IN | G_IO_HUP | G_IO_ERR | G_IO_NVAL,
server_callback, self);
g_io_channel_unref (self->server);
} }

View File

@ -42,6 +42,10 @@ typedef struct _GstA2dpSinkClass GstA2dpSinkClass;
struct _GstA2dpSink { struct _GstA2dpSink {
GstAudioSink sink; GstAudioSink sink;
gchar *device;
GIOChannel *server;
}; };
struct _GstA2dpSinkClass { struct _GstA2dpSinkClass {