summaryrefslogtreecommitdiffstats
path: root/gst/real/gstrealaudiodec.c
diff options
context:
space:
mode:
Diffstat (limited to 'gst/real/gstrealaudiodec.c')
-rw-r--r--gst/real/gstrealaudiodec.c509
1 files changed, 509 insertions, 0 deletions
diff --git a/gst/real/gstrealaudiodec.c b/gst/real/gstrealaudiodec.c
new file mode 100644
index 00000000..a7f37bae
--- /dev/null
+++ b/gst/real/gstrealaudiodec.c
@@ -0,0 +1,509 @@
+/* GStreamer
+ *
+ * Copyright (C) 2006 Lutz Mueller <lutz@topfrose.de>
+ * 2006 Edward Hervey <bilboed@bilboed.com>
+ *
+ * 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 "gstrealaudiodec.h"
+
+#include <dlfcn.h>
+#include <string.h>
+
+GST_DEBUG_CATEGORY_STATIC (real_audio_dec_debug);
+#define GST_CAT_DEFAULT real_audio_dec_debug
+
+static GstElementDetails real_audio_dec_details =
+GST_ELEMENT_DETAILS ("RealAudio decoder",
+ "Codec/Decoder", "Decoder for RealAudio streams",
+ "Lutz Mueller <lutz@topfrose.de>");
+
+static GstStaticPadTemplate snk_t =
+ GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS,
+ GST_STATIC_CAPS ("audio/x-pn-realaudio; " "audio/x-sipro "));
+static GstStaticPadTemplate src_t =
+GST_STATIC_PAD_TEMPLATE ("src", GST_PAD_SRC, GST_PAD_ALWAYS,
+ GST_STATIC_CAPS ("audio/x-raw-int, "
+ "width = (int) [ 1, MAX ], "
+ "depth = (int) [ 1, MAX ], "
+ "rate = (int) [ 1, MAX ], " "channels = (int) [ 1, MAX ]"));
+
+#ifdef HAVE_CPU_I386
+#define DEFAULT_PATH "/usr/lib/win32/"
+#endif
+#ifdef HAVE_CPU_X86_64
+#define DEFAULT_PATH "/usr/lib/"
+#endif
+
+#define DEFAULT_PATH_RACOOK DEFAULT_PATH "cook.so.6.0"
+#define DEFAULT_PATH_RAATRK DEFAULT_PATH "atrk.so.6.0"
+#define DEFAULT_PATH_RA14_4 DEFAULT_PATH "14_4.so.6.0"
+#define DEFAULT_PATH_RA28_8 DEFAULT_PATH "28_8.so.6.0"
+#define DEFAULT_PATH_RASIPR DEFAULT_PATH "sipr.so.6.0"
+#define DEFAULT_PWD "Ardubancel Quazanga"
+
+enum
+{
+ PROP_0,
+ PROP_PATH_RACOOK,
+ PROP_PATH_RAATRK,
+ PROP_PATH_RA14_4,
+ PROP_PATH_RA28_8,
+ PROP_PATH_RASIPR,
+ PROP_PASSWORD
+};
+
+typedef enum
+{
+ GST_REAL_AUDIO_DEC_VERSION_COOK = 8,
+ GST_REAL_AUDIO_DEC_VERSION_ATRK = 3,
+ GST_REAL_AUDIO_DEC_VERSION_14_4 = 4,
+ GST_REAL_AUDIO_DEC_VERSION_28_8 = 5,
+ GST_REAL_AUDIO_DEC_VERSION_SIPR = 6
+} GstRealAudioDecVersion;
+
+typedef struct
+{
+ guint16 (*RADecode) (gpointer, guint8 *, guint32, guint8 *, guint32 *,
+ guint32);
+ guint16 (*RACloseCodec) (gpointer);
+ guint16 (*RAFreeDecoder) (gpointer);
+ guint16 (*RAInitDecoder) (gpointer, gpointer);
+ guint16 (*RAOpenCodec2) (gpointer, const gchar *);
+ guint16 (*RASetFlavor) (gpointer, guint16);
+ void (*SetDLLAccessPath) (gchar *);
+ void (*RASetPwd) (gpointer, gchar *);
+} RealFunctions;
+
+typedef struct
+{
+ guint32 samplerate;
+ guint16 width;
+ guint16 channels;
+ guint16 quality;
+ guint32 leaf_size;
+ guint32 packet_size;
+ guint32 datalen;
+ gpointer data;
+} RAInit;
+
+struct _GstRealAudioDec
+{
+ GstElement parent;
+
+ GstPad *src, *snk;
+
+ /* Caps */
+ guint width, height, leaf_size;
+
+ /* Hooks */
+ gpointer handle;
+ RealFunctions funcs;
+
+ /* Used by the REAL library. */
+ gpointer context;
+
+ /* Properties */
+ gchar *path_racook, *path_raatrk, *path_ra14_4, *path_ra28_8, *path_rasipr;
+ gchar *pwd;
+};
+
+struct _GstRealAudioDecClass
+{
+ GstElementClass parent_class;
+};
+
+GST_BOILERPLATE (GstRealAudioDec, gst_real_audio_dec, GstElement,
+ GST_TYPE_ELEMENT);
+
+static GstFlowReturn
+gst_real_audio_dec_chain (GstPad * pad, GstBuffer * in)
+{
+ GstRealAudioDec *dec = GST_REAL_AUDIO_DEC (GST_PAD_PARENT (pad));
+ guint len;
+ GstBuffer *out = NULL;
+ guint16 res = 0;
+ GstFlowReturn ret = GST_FLOW_OK;
+ GstClockTime timestamp = GST_BUFFER_TIMESTAMP (in);
+
+ if ((ret = gst_pad_alloc_buffer (dec->src, GST_BUFFER_OFFSET_NONE,
+ dec->width * dec->leaf_size * dec->height * 16,
+ GST_PAD_CAPS (dec->src), &out)) != GST_FLOW_OK)
+ return ret;
+ res = dec->funcs.RADecode (dec->context, GST_BUFFER_DATA (in),
+ GST_BUFFER_SIZE (in), GST_BUFFER_DATA (out), &len, -1);
+ if (res)
+ goto could_not_decode;
+ GST_BUFFER_SIZE (out) = len;
+ GST_BUFFER_TIMESTAMP (out) = timestamp;
+ return gst_pad_push (dec->src, out);
+
+ /* Errors */
+could_not_decode:
+ gst_buffer_unref (out);
+ GST_ELEMENT_ERROR (dec, STREAM, DECODE, ("Could not decode buffer (%i).",
+ res), (NULL));
+ return GST_FLOW_ERROR;
+}
+
+static gboolean
+gst_real_audio_dec_setcaps (GstPad * pad, GstCaps * caps)
+{
+ GstRealAudioDec *dec = GST_REAL_AUDIO_DEC (GST_PAD_PARENT (pad));
+ GstStructure *s = gst_caps_get_structure (caps, 0);
+ gchar *path;
+ gint version, flavor, channels, rate, leaf_size, packet_size, width, height;
+ guint16 res;
+ RAInit data;
+ gboolean bres;
+ const GValue *v;
+ GstBuffer *buf = NULL;
+ const gchar *name = gst_structure_get_name (s);
+ gpointer context = NULL, handle;
+ RealFunctions funcs;
+
+ if (!strcmp (name, "audio/x-sipro"))
+ version = GST_REAL_AUDIO_DEC_VERSION_SIPR;
+ else {
+ if (!gst_structure_get_int (s, "raversion", &version))
+ goto missing_keys;
+ }
+
+ if (!gst_structure_get_int (s, "flavor", &flavor) ||
+ !gst_structure_get_int (s, "channels", &channels) ||
+ !gst_structure_get_int (s, "width", &width) ||
+ !gst_structure_get_int (s, "rate", &rate) ||
+ !gst_structure_get_int (s, "height", &height) ||
+ !gst_structure_get_int (s, "leaf_size", &leaf_size) ||
+ !gst_structure_get_int (s, "packet_size", &packet_size))
+ goto missing_keys;
+ if ((v = gst_structure_get_value (s, "codec_data")))
+ buf = g_value_peek_pointer (v);
+
+ switch (version) {
+ case GST_REAL_AUDIO_DEC_VERSION_COOK:
+ path = dec->path_racook ? dec->path_racook : DEFAULT_PATH_RACOOK;
+ break;
+ case GST_REAL_AUDIO_DEC_VERSION_ATRK:
+ path = dec->path_raatrk ? dec->path_raatrk : DEFAULT_PATH_RAATRK;
+ break;
+ case GST_REAL_AUDIO_DEC_VERSION_14_4:
+ path = dec->path_ra14_4 ? dec->path_ra14_4 : DEFAULT_PATH_RA14_4;
+ break;
+ case GST_REAL_AUDIO_DEC_VERSION_28_8:
+ path = dec->path_ra28_8 ? dec->path_ra28_8 : DEFAULT_PATH_RA28_8;
+ break;
+ case GST_REAL_AUDIO_DEC_VERSION_SIPR:
+ path = dec->path_rasipr ? dec->path_rasipr : DEFAULT_PATH_RASIPR;
+ break;
+ default:
+ goto unknown_version;
+ }
+
+ handle = dlopen (path, RTLD_LAZY);
+ if (!handle)
+ goto could_not_open;
+ funcs.RACloseCodec = dlsym (handle, "RACloseCodec");
+ funcs.RADecode = dlsym (handle, "RADecode");
+ funcs.RAFreeDecoder = dlsym (handle, "RAFreeDecoder");
+ funcs.RAOpenCodec2 = dlsym (handle, "RAOpenCodec2");
+ funcs.RAInitDecoder = dlsym (handle, "RAInitDecoder");
+ funcs.RASetFlavor = dlsym (handle, "RASetFlavor");
+ funcs.SetDLLAccessPath = dlsym (handle, "SetDLLAccessPath");
+ funcs.RASetPwd = dlsym (handle, "RASetPwd");
+ if (!(funcs.RACloseCodec && funcs.RADecode &&
+ funcs.RAFreeDecoder && funcs.RAOpenCodec2 &&
+ funcs.RAInitDecoder && funcs.RASetFlavor))
+ goto could_not_load;
+
+ if (funcs.SetDLLAccessPath)
+ funcs.SetDLLAccessPath (DEFAULT_PATH);
+
+ if ((res = funcs.RAOpenCodec2 (&context, NULL)))
+ goto could_not_initialize;
+
+ data.samplerate = rate;
+ data.width = width;
+ data.channels = channels;
+ data.quality = 100;
+ data.leaf_size = leaf_size;
+ data.packet_size = packet_size;
+ data.datalen = buf ? GST_BUFFER_SIZE (buf) : 0;
+ data.data = buf ? GST_BUFFER_DATA (buf) : NULL;
+
+ if ((res = funcs.RAInitDecoder (context, &data)))
+ goto could_not_initialize;
+
+ if (funcs.RASetPwd)
+ funcs.RASetPwd (dec->context, dec->pwd ? dec->pwd : DEFAULT_PWD);
+
+ res = funcs.RASetFlavor (context, flavor);
+ if (res)
+ goto could_not_initialize;
+
+ caps = gst_caps_new_simple ("audio/x-raw-int",
+ "endianness", G_TYPE_INT, G_BYTE_ORDER,
+ "width", G_TYPE_INT, width,
+ "depth", G_TYPE_INT, height,
+ "rate", G_TYPE_INT, rate,
+ "channels", G_TYPE_INT, channels, "signed", G_TYPE_BOOLEAN, TRUE, NULL);
+ bres = gst_pad_set_caps (GST_PAD (dec->src), caps);
+ gst_caps_unref (caps);
+ if (!bres)
+ goto could_not_set_caps;
+
+ dec->width = width;
+ dec->height = height;
+ dec->leaf_size = leaf_size;
+ if (dec->context) {
+ dec->funcs.RACloseCodec (dec->context);
+ dec->funcs.RAFreeDecoder (dec->context);
+ }
+ dec->context = context;
+ if (dec->handle)
+ dlclose (dec->handle);
+ dec->handle = handle;
+ dec->funcs = funcs;
+
+ return TRUE;
+
+missing_keys:
+ GST_DEBUG_OBJECT (dec, "Could not find all necessary keys in structure.");
+ return FALSE;
+unknown_version:
+ GST_DEBUG_OBJECT (dec, "Cannot handle version %i.", version);
+ return FALSE;
+could_not_open:
+ GST_DEBUG_OBJECT (dec, "Could not open library '%s'.", path);
+ return FALSE;
+could_not_load:
+ dlclose (handle);
+ GST_DEBUG_OBJECT (dec, "Could not load all symbols.");
+ return FALSE;
+could_not_initialize:
+ if (context) {
+ funcs.RACloseCodec (context);
+ funcs.RAFreeDecoder (context);
+ }
+ dlclose (handle);
+ GST_DEBUG_OBJECT (dec, "Initialization of REAL driver failed (%i).", res);
+ return FALSE;
+could_not_set_caps:
+ if (context) {
+ funcs.RACloseCodec (context);
+ funcs.RAFreeDecoder (context);
+ }
+ dlclose (handle);
+ GST_DEBUG_OBJECT (dec, "Could not convince peer to accept caps.");
+ return FALSE;
+}
+
+static void
+gst_real_audio_dec_init (GstRealAudioDec * dec, GstRealAudioDecClass * klass)
+{
+ dec->snk =
+ gst_pad_new_from_template (gst_static_pad_template_get (&snk_t), "sink");
+ gst_pad_set_setcaps_function (dec->snk, gst_real_audio_dec_setcaps);
+ gst_pad_set_chain_function (dec->snk, gst_real_audio_dec_chain);
+ gst_element_add_pad (GST_ELEMENT (dec), dec->snk);
+ dec->src =
+ gst_pad_new_from_template (gst_static_pad_template_get (&src_t), "src");
+ gst_element_add_pad (GST_ELEMENT (dec), dec->src);
+}
+
+static void
+gst_real_audio_dec_base_init (gpointer g_class)
+{
+ GstElementClass *ec = GST_ELEMENT_CLASS (g_class);
+
+ gst_element_class_add_pad_template (ec, gst_static_pad_template_get (&snk_t));
+ gst_element_class_add_pad_template (ec, gst_static_pad_template_get (&src_t));
+ gst_element_class_set_details (ec, &real_audio_dec_details);
+}
+
+static GstStateChangeReturn
+gst_real_audio_dec_change_state (GstElement * element,
+ GstStateChange transition)
+{
+ GstStateChangeReturn ret;
+
+ ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
+ switch (transition) {
+ case GST_STATE_CHANGE_PAUSED_TO_READY:
+ break;
+ default:
+ break;
+ }
+ return ret;
+}
+
+static void
+gst_real_audio_dec_finalize (GObject * object)
+{
+ GstRealAudioDec *dec = GST_REAL_AUDIO_DEC (object);
+
+ if (dec->context) {
+ dec->funcs.RACloseCodec (dec->context);
+ dec->funcs.RAFreeDecoder (dec->context);
+ dec->context = NULL;
+ }
+ if (dec->handle) {
+ dlclose (dec->handle);
+ dec->handle = NULL;
+ }
+
+ if (dec->path_racook) {
+ g_free (dec->path_racook);
+ dec->path_racook = NULL;
+ }
+ if (dec->path_raatrk) {
+ g_free (dec->path_raatrk);
+ dec->path_raatrk = NULL;
+ }
+ if (dec->path_ra14_4) {
+ g_free (dec->path_ra14_4);
+ dec->path_ra14_4 = NULL;
+ }
+ if (dec->path_ra28_8) {
+ g_free (dec->path_ra28_8);
+ dec->path_ra28_8 = NULL;
+ }
+ if (dec->path_rasipr) {
+ g_free (dec->path_rasipr);
+ dec->path_rasipr = NULL;
+ }
+
+ G_OBJECT_CLASS (parent_class)->finalize (object);
+}
+
+static void
+gst_real_audio_dec_set_property (GObject * object, guint prop_id,
+ const GValue * value, GParamSpec * pspec)
+{
+ GstRealAudioDec *dec = GST_REAL_AUDIO_DEC (object);
+
+ switch (prop_id) {
+ case PROP_PATH_RACOOK:
+ if (dec->path_racook)
+ g_free (dec->path_racook);
+ dec->path_racook = g_value_dup_string (value);
+ break;
+ case PROP_PATH_RAATRK:
+ if (dec->path_raatrk)
+ g_free (dec->path_raatrk);
+ dec->path_raatrk = g_value_dup_string (value);
+ break;
+ case PROP_PATH_RA14_4:
+ if (dec->path_ra14_4)
+ g_free (dec->path_ra14_4);
+ dec->path_ra14_4 = g_value_dup_string (value);
+ break;
+ case PROP_PATH_RA28_8:
+ if (dec->path_ra28_8)
+ g_free (dec->path_ra28_8);
+ dec->path_ra28_8 = g_value_dup_string (value);
+ break;
+ case PROP_PATH_RASIPR:
+ if (dec->path_rasipr)
+ g_free (dec->path_rasipr);
+ dec->path_rasipr = g_value_dup_string (value);
+ break;
+ case PROP_PASSWORD:
+ if (dec->pwd)
+ g_free (dec->pwd);
+ dec->pwd = g_value_dup_string (value);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+gst_real_audio_dec_get_property (GObject * object, guint prop_id,
+ GValue * value, GParamSpec * pspec)
+{
+ GstRealAudioDec *dec = GST_REAL_AUDIO_DEC (object);
+
+ switch (prop_id) {
+ case PROP_PATH_RACOOK:
+ g_value_set_string (value, dec->path_racook ? dec->path_racook :
+ DEFAULT_PATH_RACOOK);
+ break;
+ case PROP_PATH_RAATRK:
+ g_value_set_string (value, dec->path_raatrk ? dec->path_raatrk :
+ DEFAULT_PATH_RAATRK);
+ break;
+ case PROP_PATH_RA14_4:
+ g_value_set_string (value, dec->path_ra14_4 ? dec->path_ra14_4 :
+ DEFAULT_PATH_RA14_4);
+ break;
+ case PROP_PATH_RA28_8:
+ g_value_set_string (value, dec->path_ra28_8 ? dec->path_ra28_8 :
+ DEFAULT_PATH_RA28_8);
+ break;
+ case PROP_PATH_RASIPR:
+ g_value_set_string (value, dec->path_rasipr ? dec->path_rasipr :
+ DEFAULT_PATH_RASIPR);
+ break;
+ case PROP_PASSWORD:
+ g_value_set_string (value, dec->pwd ? dec->pwd : DEFAULT_PWD);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+gst_real_audio_dec_class_init (GstRealAudioDecClass * klass)
+{
+ GstElementClass *element_class = GST_ELEMENT_CLASS (klass);
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ object_class->set_property = gst_real_audio_dec_set_property;
+ object_class->get_property = gst_real_audio_dec_get_property;
+ object_class->finalize = gst_real_audio_dec_finalize;
+
+ element_class->change_state = gst_real_audio_dec_change_state;
+
+ g_object_class_install_property (object_class, PROP_PATH_RACOOK,
+ g_param_spec_string ("path_racook", "Path to cook driver",
+ "Path to cook driver", DEFAULT_PATH_RACOOK, G_PARAM_READWRITE));
+ g_object_class_install_property (object_class, PROP_PATH_RAATRK,
+ g_param_spec_string ("path_raatrk", "Path to atrk driver",
+ "Path to atrk driver", DEFAULT_PATH_RAATRK, G_PARAM_READWRITE));
+ g_object_class_install_property (object_class, PROP_PATH_RA14_4,
+ g_param_spec_string ("path_ra14_4", "Path to 14_4 driver",
+ "Path to 14_4 driver", DEFAULT_PATH_RA14_4, G_PARAM_READWRITE));
+ g_object_class_install_property (object_class, PROP_PATH_RA28_8,
+ g_param_spec_string ("path_ra28_8", "Path to 28_8 driver",
+ "Path to 28_8 driver", DEFAULT_PATH_RA28_8, G_PARAM_READWRITE));
+ g_object_class_install_property (object_class, PROP_PATH_RASIPR,
+ g_param_spec_string ("path_rasipr", "Path to sipr driver",
+ "Path to sipr driver", DEFAULT_PATH_RASIPR, G_PARAM_READWRITE));
+ g_object_class_install_property (object_class, PROP_PASSWORD,
+ g_param_spec_string ("password", "Password",
+ "Password", DEFAULT_PWD, G_PARAM_READWRITE));
+
+ GST_DEBUG_CATEGORY_INIT (real_audio_dec_debug, "realaudiodec", 0,
+ "RealAudio decoder");
+}