summaryrefslogtreecommitdiffstats
path: root/ext/nas/nassink.c
diff options
context:
space:
mode:
Diffstat (limited to 'ext/nas/nassink.c')
-rw-r--r--ext/nas/nassink.c643
1 files changed, 643 insertions, 0 deletions
diff --git a/ext/nas/nassink.c b/ext/nas/nassink.c
new file mode 100644
index 00000000..45862750
--- /dev/null
+++ b/ext/nas/nassink.c
@@ -0,0 +1,643 @@
+/* GStreamer
+ * Copyright (C) <2003> Laurent Vivier <Laurent.Vivier@bull.net>
+ * Copyright (C) <2004> Arwed v. Merkatz <v.merkatz@gmx.net>
+ *
+ * Based on esdsink.c:
+ * Copyright (C) <2001> Richard Boulton <richard-gst@tartarus.org>
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This library 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; 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 <gst/gst.h>
+#include <string.h>
+#include <audio/audiolib.h>
+#include <audio/soundlib.h>
+#include "nassink.h"
+
+#define NAS_SOUND_PORT_DURATION (2)
+
+/* Signals and args */
+enum {
+ /* FILL ME */
+ LAST_SIGNAL
+};
+
+enum {
+ ARG_0,
+ ARG_MUTE,
+ ARG_HOST
+};
+
+static GstStaticPadTemplate sink_factory =
+GST_STATIC_PAD_TEMPLATE (
+ "sink",
+ GST_PAD_SINK,
+ GST_PAD_ALWAYS,
+ GST_STATIC_CAPS(
+ "audio/x-raw-int, "
+ "endianess = (int) BYTE_ORDER, "
+ "signed = (boolean) TRUE, "
+ "width = (int) 16, "
+ "depth = (int) 16, "
+ "rate = (int) [ 8000, 96000 ], "
+ "channels = (int) [ 1, 2 ]; "
+ "audio/x-raw-int, "
+ "signed = (boolean) FALSE, "
+ "width = (int) 8, "
+ "depth = (int) 8, "
+ "rate = (int) [ 8000, 96000 ], "
+ "channels = (int) [ 1, 2 ]"
+ )
+);
+
+static void gst_nassink_base_init (gpointer g_class);
+static void gst_nassink_class_init (GstNassinkClass *klass);
+static void gst_nassink_init (GstNassink *nassink);
+
+static gboolean gst_nassink_open_audio (GstNassink *sink);
+static void gst_nassink_close_audio (GstNassink *sink);
+static GstElementStateReturn gst_nassink_change_state (GstElement *element);
+static gboolean gst_nassink_sync_parms (GstNassink *nassink);
+static GstPadLinkReturn gst_nassink_sinkconnect (GstPad *pad, const GstCaps *caps);
+
+static void gst_nassink_chain (GstPad *pad, GstData *_data);
+
+static void gst_nassink_set_property (GObject *object, guint prop_id,
+ const GValue *value, GParamSpec *pspec);
+static void gst_nassink_get_property (GObject *object, guint prop_id,
+ GValue *value, GParamSpec *pspec);
+
+static void NAS_flush (GstNassink *sink);
+static void NAS_sendData (GstNassink *sink, AuUint32 numBytes);
+static AuBool NAS_EventHandler (AuServer *aud, AuEvent *ev, AuEventHandlerRec *handler);
+static AuDeviceID NAS_getDevice (AuServer* aud, int numTracks);
+static int NAS_allocBuffer (GstNassink *sink);
+static int NAS_createFlow (GstNassink *sink, unsigned char format, unsigned short rate, int numTracks);
+
+static GstElementClass *parent_class = NULL;
+
+GType
+gst_nassink_get_type (void)
+{
+ static GType nassink_type = 0;
+
+ if (!nassink_type) {
+ static const GTypeInfo nassink_info = {
+ sizeof(GstNassinkClass),
+ gst_nassink_base_init,
+ NULL,
+ (GClassInitFunc)gst_nassink_class_init,
+ NULL,
+ NULL,
+ sizeof(GstNassink),
+ 0,
+ (GInstanceInitFunc)gst_nassink_init,
+ };
+ nassink_type = g_type_register_static(GST_TYPE_ELEMENT, "GstNassink", &nassink_info, 0);
+ }
+
+ return nassink_type;
+}
+
+static void
+gst_nassink_base_init (gpointer g_class)
+{
+ static GstElementDetails nassink_details = {
+ "NAS sink",
+ "Sink/Audio",
+ "Plays audio to a Network Audio Server",
+ "Laurent Vivier <Laurent.Vivier@bull.net>, "
+ "Arwed v. Merkatz <v.merkatz@gmx.net>"
+ };
+ GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
+
+ gst_element_class_add_pad_template (element_class,
+ gst_static_pad_template_get (&sink_factory));
+ gst_element_class_set_details (element_class, &nassink_details);
+}
+
+static void
+gst_nassink_class_init (GstNassinkClass *klass)
+{
+ GObjectClass *gobject_class;
+ GstElementClass *gstelement_class;
+
+ gobject_class = (GObjectClass*)klass;
+ gstelement_class = (GstElementClass*)klass;
+
+ if (parent_class == NULL)
+ parent_class = g_type_class_ref(GST_TYPE_ELEMENT);
+
+ gobject_class->set_property = gst_nassink_set_property;
+ gobject_class->get_property = gst_nassink_get_property;
+
+ g_object_class_install_property(G_OBJECT_CLASS(klass), ARG_MUTE,
+ g_param_spec_boolean("mute","mute","mute",
+ TRUE,G_PARAM_READWRITE)); /* CHECKME */
+ g_object_class_install_property(G_OBJECT_CLASS(klass), ARG_HOST,
+ g_param_spec_string("host","host","host",
+ NULL, G_PARAM_READWRITE)); /* CHECKME */
+
+ gstelement_class->change_state = gst_nassink_change_state;
+}
+
+static void
+gst_nassink_init(GstNassink *nassink)
+{
+ nassink->sinkpad = gst_pad_new_from_template (
+ gst_static_pad_template_get (&sink_factory), "sink");
+ gst_element_add_pad(GST_ELEMENT(nassink), nassink->sinkpad);
+ gst_pad_set_chain_function(nassink->sinkpad, GST_DEBUG_FUNCPTR(gst_nassink_chain));
+ gst_pad_set_link_function(nassink->sinkpad, gst_nassink_sinkconnect);
+
+ nassink->mute = FALSE;
+ nassink->depth = 16;
+ nassink->tracks = 2;
+ nassink->rate = 44100;
+ nassink->host = g_strdup (getenv("AUDIOSERVER"));
+ if (nassink->host == NULL)
+ nassink->host = g_strdup (getenv("DISPLAY"));
+
+ nassink->audio = NULL;
+ nassink->flow = AuNone;
+ nassink->size = 0;
+ nassink->pos = 0;
+ nassink->buf = NULL;
+}
+
+static gboolean
+gst_nassink_sync_parms (GstNassink *nassink)
+{
+ gint ret;
+ unsigned char format;
+ g_return_val_if_fail (nassink != NULL, FALSE);
+ g_return_val_if_fail (GST_IS_NASSINK (nassink), FALSE);
+
+ if (nassink->audio == NULL) return TRUE;
+
+ if (nassink->flow != AuNone)
+ {
+ while (nassink->pos && nassink->buf)
+ NAS_flush(nassink);
+ AuStopFlow( nassink->audio, nassink->flow, NULL);
+ AuReleaseScratchFlow(nassink->audio, nassink->flow, NULL);
+ nassink->flow = AuNone;
+ }
+
+ if (nassink->depth == 16)
+#if G_BYTE_ORDER == G_BIG_ENDIAN
+ format = AuFormatLinearSigned16MSB;
+#else
+ format = AuFormatLinearSigned16LSB;
+#endif
+ else
+ format = AuFormatLinearUnsigned8;
+
+ ret = NAS_createFlow(nassink, format, nassink->rate, nassink->tracks);
+
+ return ret >= 0;
+}
+
+static GstPadLinkReturn
+gst_nassink_sinkconnect (GstPad *pad, const GstCaps *caps)
+{
+ GstNassink *nassink;
+ GstStructure *structure;
+
+ nassink = GST_NASSINK (gst_pad_get_parent (pad));
+
+ structure = gst_caps_get_structure (caps, 0);
+
+ gst_structure_get_int (structure, "depth", &nassink->depth);
+ gst_structure_get_int (structure, "channels", &nassink->tracks);
+ gst_structure_get_int (structure, "rate", &nassink->rate);
+
+ if (!gst_nassink_sync_parms(nassink))
+ return GST_PAD_LINK_REFUSED;
+
+ return GST_PAD_LINK_OK;
+}
+
+static void
+gst_nassink_chain (GstPad *pad, GstData *_data)
+{
+ GstBuffer *buf = GST_BUFFER (_data);
+ int pos = 0;
+ int remaining;
+ int available;
+ GstNassink *nassink;
+
+ g_return_if_fail(pad != NULL);
+ g_return_if_fail(GST_IS_PAD(pad));
+ g_return_if_fail(buf != NULL);
+
+ nassink = GST_NASSINK (gst_pad_get_parent (pad));
+
+ g_return_if_fail(nassink->buf != NULL);
+
+ if (GST_BUFFER_DATA (buf) != NULL) {
+ if (!nassink->mute && nassink->audio != NULL) {
+ GST_DEBUG ("nassink: data=%p size=%d", GST_BUFFER_DATA (buf), GST_BUFFER_SIZE (buf));
+
+ remaining = GST_BUFFER_SIZE (buf);
+ while ((nassink->flow != AuNone) && ( remaining > 0)) {
+
+ /* number of bytes we can copy to buffer */
+
+ available = remaining > nassink->size - nassink->pos ?
+ nassink->size - nassink->pos : remaining;
+
+ /* fill the buffer */
+
+ memcpy (nassink->buf + nassink->pos, GST_BUFFER_DATA (buf) + pos, available);
+
+ nassink->pos += available;
+ pos += available;
+
+ remaining -= available;
+
+ /* if we have more bytes, need to flush the buffer */
+
+ if (remaining > 0) {
+ while ((nassink->flow != AuNone) && (nassink->pos == nassink->size)) {
+ NAS_flush(nassink);
+ }
+ }
+ }
+
+ /* give some time to event handler */
+
+ AuSync(nassink->audio, AuFalse);
+
+ }
+ }
+ gst_buffer_unref (buf);
+}
+
+static void
+gst_nassink_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec)
+{
+ GstNassink *nassink;
+
+ /* it's not null if we got it, but it might not be ours */
+ g_return_if_fail(GST_IS_NASSINK(object));
+ nassink = GST_NASSINK(object);
+
+ switch (prop_id) {
+ case ARG_MUTE:
+ nassink->mute = g_value_get_boolean (value);
+ break;
+ case ARG_HOST:
+ if (nassink->host != NULL) g_free(nassink->host);
+ if (g_value_get_string (value) == NULL)
+ nassink->host = NULL;
+ else
+ nassink->host = g_strdup (g_value_get_string (value));
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+gst_nassink_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec)
+{
+ GstNassink *nassink;
+
+ g_return_if_fail(GST_IS_NASSINK(object));
+
+ nassink = GST_NASSINK(object);
+
+ switch (prop_id) {
+ case ARG_MUTE:
+ g_value_set_boolean (value, nassink->mute);
+ break;
+ case ARG_HOST:
+ g_value_set_string (value, nassink->host);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static gboolean
+plugin_init (GstPlugin *plugin)
+{
+ if (!gst_element_register (plugin, "nassink", GST_RANK_NONE,
+ GST_TYPE_NASSINK)){
+ return FALSE;
+ }
+
+ return TRUE;
+}
+
+GST_PLUGIN_DEFINE (
+ GST_VERSION_MAJOR,
+ GST_VERSION_MINOR,
+ "nassink",
+ "uses NAS for audio output",
+ plugin_init,
+ VERSION,
+ "LGPL",
+ GST_PACKAGE,
+ GST_ORIGIN
+);
+
+static gboolean
+gst_nassink_open_audio (GstNassink *sink)
+{
+ /* Open Server */
+
+ sink->audio = AuOpenServer(sink->host, 0, NULL, 0, NULL, NULL);
+ if (sink->audio == NULL)
+ return FALSE;
+
+ sink->flow = AuNone;
+ sink->size = 0;
+ sink->pos = 0;
+ sink->buf = NULL;
+
+ /* Start a flow */
+
+ GST_FLAG_SET (sink, GST_NASSINK_OPEN);
+
+ return TRUE;
+}
+
+static void
+gst_nassink_close_audio (GstNassink *sink)
+{
+ if (sink->audio == NULL) return;
+
+ if (sink->flow != AuNone) {
+ while (sink->pos && sink->buf) {
+ NAS_flush(sink);
+ }
+
+ AuStopFlow( sink->audio, sink->flow, NULL);
+ AuReleaseScratchFlow(sink->audio, sink->flow, NULL);
+ sink->flow = AuNone;
+ }
+
+ if (sink->buf != NULL)
+ {
+ free(sink->buf);
+ sink->buf = NULL;
+ }
+
+ AuCloseServer(sink->audio);
+ sink->audio = NULL;
+
+ GST_FLAG_UNSET (sink, GST_NASSINK_OPEN);
+
+ GST_DEBUG ("nassink: closed sound device");
+}
+
+static GstElementStateReturn
+gst_nassink_change_state (GstElement *element)
+{
+ GstNassink *nassink;
+ g_return_val_if_fail (GST_IS_NASSINK (element), FALSE);
+
+ nassink = GST_NASSINK (element);
+
+ switch (GST_STATE_PENDING (element)) {
+ case GST_STATE_NULL:
+ if (GST_FLAG_IS_SET (element, GST_NASSINK_OPEN))
+ gst_nassink_close_audio (nassink);
+ break;
+
+ case GST_STATE_READY:
+ if (!GST_FLAG_IS_SET (element, GST_NASSINK_OPEN))
+ gst_nassink_open_audio (nassink);
+ break;
+
+ case GST_STATE_PAUSED:
+ while (nassink->pos && nassink->buf)
+ NAS_flush(nassink);
+ break;
+
+ case GST_STATE_PLAYING:
+ break;
+ }
+
+ if (GST_ELEMENT_CLASS (parent_class)->change_state)
+ return GST_ELEMENT_CLASS (parent_class)->change_state (element);
+
+ return GST_STATE_SUCCESS;
+}
+
+static void
+NAS_flush(GstNassink *sink)
+{
+ AuEvent ev;
+
+ AuNextEvent(sink->audio, AuTrue, &ev);
+ AuDispatchEvent(sink->audio, &ev);
+}
+
+static void
+NAS_sendData(GstNassink *sink, AuUint32 numBytes)
+{
+ if (numBytes < (sink->pos)) {
+
+ AuWriteElement(sink->audio, sink->flow, 0,
+ numBytes, sink->buf, AuFalse, NULL);
+
+ memmove(sink->buf, sink->buf + numBytes,
+ sink->pos - numBytes);
+
+ sink->pos = sink->pos - numBytes;
+
+ } else
+ {
+ AuWriteElement(sink->audio, sink->flow, 0,
+ sink->pos, sink->buf,
+ (numBytes > sink->pos), NULL);
+ sink->pos = 0;
+ }
+}
+
+static AuBool
+NAS_EventHandler(AuServer *aud, AuEvent *ev, AuEventHandlerRec *handler)
+{
+ GstNassink *sink = (GstNassink *)handler->data;
+ AuElementNotifyEvent *notify;
+
+ switch (ev->type) {
+
+ case AuEventTypeElementNotify:
+
+ notify = (AuElementNotifyEvent *) ev;
+
+ switch(notify->kind) {
+
+ case AuElementNotifyKindLowWater:
+ NAS_sendData(sink, notify->num_bytes);
+ break;
+
+ case AuElementNotifyKindState:
+
+ switch(notify->cur_state) {
+
+ case AuStateStop:
+
+ if (sink->flow != AuNone) {
+ if (notify->reason == AuReasonEOF)
+ AuStopFlow(handler->aud, sink->flow, NULL);
+ AuReleaseScratchFlow(handler->aud, sink->flow, NULL);
+ sink->flow = AuNone;
+ }
+ AuUnregisterEventHandler(handler->aud, handler);
+ break;
+
+ case AuStatePause:
+
+ switch(notify->reason) {
+ case AuReasonUnderrun:
+ case AuReasonOverrun:
+ case AuReasonEOF:
+ case AuReasonWatermark:
+
+ NAS_sendData(sink, notify->num_bytes);
+
+ break;
+
+ case AuReasonHardware:
+
+ if (AuSoundRestartHardwarePauses)
+ AuStartFlow(handler->aud, sink->flow, NULL);
+ else
+ AuStopFlow(handler->aud, sink->flow, NULL);
+
+ break;
+ }
+ break;
+ }
+ break;
+ }
+ break;
+ }
+
+ return AuTrue;
+}
+
+static AuDeviceID
+NAS_getDevice(AuServer* aud, int numTracks)
+{
+ int i;
+
+ for (i = 0; i < AuServerNumDevices(aud); i++) {
+ if ( (AuDeviceKind(AuServerDevice(aud, i))
+ == AuComponentKindPhysicalOutput) &&
+ (AuDeviceNumTracks(AuServerDevice(aud, i)) == numTracks )) {
+
+ return AuDeviceIdentifier(AuServerDevice(aud, i));
+
+ }
+ }
+
+ return AuNone;
+}
+
+static int
+NAS_allocBuffer(GstNassink *sink)
+{
+ if (sink->buf != NULL) {
+ free(sink->buf);
+ }
+
+ sink->buf = (char *) malloc(sink->size);
+ if (sink->buf == NULL) {
+ return -1;
+ }
+
+ sink->pos = 0;
+
+ return 0;
+}
+
+static int
+NAS_createFlow(GstNassink *sink, unsigned char format, unsigned short rate, int numTracks)
+{
+ AuDeviceID device;
+ AuElement elements[2];
+ AuUint32 buf_samples;
+
+ device = NAS_getDevice(sink->audio, numTracks);
+ if (device == AuNone) {
+ return -1;
+ }
+
+ sink->flow = AuGetScratchFlow(sink->audio, NULL);
+ if (sink->flow == 0) {
+ return -1;
+ }
+
+ buf_samples = rate * NAS_SOUND_PORT_DURATION;
+
+ AuMakeElementImportClient( &elements[0], /* element */
+ rate, /* rate */
+ format, /* format */
+ numTracks, /* number of tracks */
+ AuTrue, /* discart */
+ buf_samples, /* max samples */
+ (AuUint32) (buf_samples / 100
+ * AuSoundPortLowWaterMark),
+ /* low water mark */
+ 0, /* num actions */
+ NULL);
+
+ AuMakeElementExportDevice( &elements[1], /* element */
+ 0, /* input */
+ device, /* device */
+ rate, /* rate */
+ AuUnlimitedSamples, /* num samples */
+ 0, /* num actions */
+ NULL); /* actions */
+
+ AuSetElements( sink->audio, /* server */
+ sink->flow, /* flow ID */
+ AuTrue, /* clocked */
+ 2, /* num elements */
+ elements, /* elements */
+ NULL);
+
+ AuRegisterEventHandler( sink->audio, /* server */
+ AuEventHandlerIDMask, /* value mask */
+ 0, /* type */
+ sink->flow, /* flow ID */
+ NAS_EventHandler, /* callback */
+ (AuPointer)sink); /* data */
+
+ sink->size = buf_samples * numTracks * AuSizeofFormat(format);
+
+ if (NAS_allocBuffer(sink) < 0) {
+
+ AuReleaseScratchFlow(sink->audio, sink->flow, NULL);
+
+ return -1;
+ }
+
+ AuStartFlow(sink->audio, sink->flow, NULL);
+
+ return 0;
+}