]> Gitweb @ Texas Instruments - Open Source Git Repositories - git.TI.com/gitweb - glsdk/gst-plugins-ugly0-10.git/commitdiff
added new element to add Xing headers to MP3 files (this allows decoder to figure...
authorChristophe Fergeau <teuf@gnome.org>
Sat, 11 Mar 2006 11:12:30 +0000 (11:12 +0000)
committerChristophe Fergeau <teuf@gnome.org>
Sat, 11 Mar 2006 11:12:30 +0000 (11:12 +0000)
Original commit message from CVS:
2006-03-11  Christophe Fergeau  <teuf@gnome.org>

Reviewed by: Tim-Philipp Müller  <tim at centricular dot net>

* configure.ac:
* gst/xingheader/Makefile.am:
* gst/xingheader/gstxingmux.c:
* gst/xingheader/gstxingmux.h: added new element to add Xing headers
to MP3 files (this allows decoder to figure out the length of VBR
files)

common
gst/mpegaudioparse/gstxingmux.c [new file with mode: 0644]
gst/mpegaudioparse/gstxingmux.h [new file with mode: 0644]

diff --git a/common b/common
index c09cd18d328f740ac532377fa5605b0f712cc6fd..9200457d08a57f0d7eaeb56915804fa8faf14418 160000 (submodule)
--- a/common
+++ b/common
@@ -1 +1 @@
-Subproject commit c09cd18d328f740ac532377fa5605b0f712cc6fd
+Subproject commit 9200457d08a57f0d7eaeb56915804fa8faf14418
diff --git a/gst/mpegaudioparse/gstxingmux.c b/gst/mpegaudioparse/gstxingmux.c
new file mode 100644 (file)
index 0000000..e9e55b9
--- /dev/null
@@ -0,0 +1,380 @@
+/*
+ * (c) 2006 Christophe Fergeau  <teuf@gnome.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 <string.h>
+#include "gstxingmux.h"
+
+GST_DEBUG_CATEGORY_STATIC (xing_mux_debug);
+#define GST_CAT_DEFAULT xing_mux_debug
+
+GST_BOILERPLATE (GstXingMux, gst_xing_mux, GstElement, GST_TYPE_ELEMENT);
+
+/* Xing Header stuff */
+struct _GstXingMuxPriv
+{
+  guint64 duration;
+  guint64 byte_count;
+  GList *seek_table;
+  gboolean flush;
+};
+
+#define GST_XING_FRAME_FIELD   (1 << 0)
+#define GST_XING_BYTES_FIELD   (1 << 1)
+#define GST_XING_TOC_FIELD     (1 << 2)
+#define GST_XING_QUALITY_FIELD (1 << 3)
+
+static const int XING_FRAME_SIZE = 418;
+
+static GstStateChangeReturn
+gst_xing_mux_change_state (GstElement * element, GstStateChange transition);
+static GstFlowReturn gst_xing_mux_chain (GstPad * pad, GstBuffer * buffer);
+static gboolean gst_xing_mux_sink_event (GstPad * pad, GstEvent * event);
+
+
+static void
+gst_xing_mux_finalize (GObject * obj)
+{
+  GstXingMux *xing = GST_XING_MUX (obj);
+
+  g_free (xing->priv);
+  xing->priv = NULL;
+  G_OBJECT_CLASS (parent_class)->finalize (obj);
+}
+
+
+static GstStaticPadTemplate gst_xing_mux_sink_template =
+GST_STATIC_PAD_TEMPLATE ("sink",
+    GST_PAD_SINK,
+    GST_PAD_ALWAYS,
+    GST_STATIC_CAPS ("audio/mpeg, "
+        "mpegversion = (int) 1, " "layer = (int) 3"));
+
+
+static GstStaticPadTemplate gst_xing_mux_src_template =
+GST_STATIC_PAD_TEMPLATE ("src",
+    GST_PAD_SRC,
+    GST_PAD_ALWAYS,
+    GST_STATIC_CAPS ("audio/mpeg, "
+        "mpegversion = (int) 1, " "layer = (int) 3"));
+
+
+static void
+gst_xing_mux_base_init (gpointer g_class)
+{
+  GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
+
+  static GstElementDetails gst_xing_mux_details = {
+    "MP3 Xing Muxer",
+    "Formatter/Metadata",
+    "Adds a Xing header to the beginning of a VBR MP3 file",
+    "Christophe Fergeau <teuf@gnome.org>"
+  };
+
+
+  gst_element_class_add_pad_template (element_class,
+      gst_static_pad_template_get (&gst_xing_mux_src_template));
+  gst_element_class_add_pad_template (element_class,
+      gst_static_pad_template_get (&gst_xing_mux_sink_template));
+  gst_element_class_set_details (element_class, &gst_xing_mux_details);
+}
+
+static void
+gst_xing_mux_class_init (GstXingMuxClass * klass)
+{
+  GObjectClass *gobject_class;
+  GstElementClass *gstelement_class;
+
+  gobject_class = (GObjectClass *) klass;
+  gstelement_class = (GstElementClass *) klass;
+
+  gobject_class->finalize = GST_DEBUG_FUNCPTR (gst_xing_mux_finalize);
+  gstelement_class->change_state =
+      GST_DEBUG_FUNCPTR (gst_xing_mux_change_state);
+}
+
+static void
+xing_set_flush (GstXingMux * xing, gboolean flush)
+{
+  if (xing->priv == NULL) {
+    return;
+  }
+  xing->priv->flush = flush;
+}
+
+static void
+gst_xing_mux_init (GstXingMux * xing, GstXingMuxClass * xingmux_class)
+{
+  GstElementClass *klass = GST_ELEMENT_CLASS (xingmux_class);
+
+  /* pad through which data comes in to the element */
+  xing->sinkpad =
+      gst_pad_new_from_template (gst_element_class_get_pad_template (klass,
+          "sink"), "sink");
+  gst_pad_set_setcaps_function (xing->sinkpad,
+      GST_DEBUG_FUNCPTR (gst_pad_proxy_setcaps));
+  gst_pad_set_chain_function (xing->sinkpad,
+      GST_DEBUG_FUNCPTR (gst_xing_mux_chain));
+  gst_pad_set_event_function (xing->sinkpad,
+      GST_DEBUG_FUNCPTR (gst_xing_mux_sink_event));
+  gst_element_add_pad (GST_ELEMENT (xing), xing->sinkpad);
+
+  /* pad through which data goes out of the element */
+  xing->srcpad =
+      gst_pad_new_from_template (gst_element_class_get_pad_template (klass,
+          "src"), "src");
+  gst_element_add_pad (GST_ELEMENT (xing), xing->srcpad);
+
+  xing->priv = g_malloc0 (sizeof (GstXingMuxPriv));
+  xing_set_flush (xing, TRUE);
+  xing->priv->duration = GST_CLOCK_TIME_NONE;
+
+}
+
+G_GNUC_UNUSED static void
+xing_update_data (GstXingMux * xing, gint bytes, guint64 duration)
+{
+  if (xing->priv == NULL) {
+    return;
+  }
+  xing->priv->byte_count += bytes;
+
+  if (duration == GST_CLOCK_TIME_NONE) {
+    return;
+  }
+  if (xing->priv->duration == GST_CLOCK_TIME_NONE) {
+    xing->priv->duration = duration;
+  } else {
+    xing->priv->duration += duration;
+  }
+}
+
+static GstBuffer *
+xing_generate_header (GstXingMux * xing)
+{
+  guint32 xing_flags;
+  GstBuffer *header;
+  guint32 *data;
+
+  /* Dummy header that we will stick at the beginning of our frame
+   *
+   * 0xffe => synchronization bits
+   * 0x1b  => 11010b (11b == MPEG1 | 01b == Layer III | 0b == no CRC)
+   * 0x9   => 128kbps
+   * 0x00  => 00b == 44100 Hz | 0b == no padding | 0b == private bit
+   * 0x44  => 0010b 0010b (00b == stereo | 10b == (unused) mode extension)
+   *                      (0b == no copyright bit | 0b == original bit)
+   *                      (00b == no emphasis)
+   *
+   * Such a frame (MPEG1 Layer III) contains 1152 samples, its size is thus:
+   * (1152*(128000/8))/44100 = 417.96
+   * 
+   * There are also 32 bytes (ie 8 32 bits values) to skip after the header 
+   * for such frames
+   */
+  const guint8 mp3_header[4] = { 0xff, 0xfb, 0x90, 0x44 };
+  const int SIDE_INFO_SIZE = 32 / sizeof (guint32);
+
+  header = gst_buffer_new_and_alloc (XING_FRAME_SIZE);
+
+  data = (guint32 *) GST_BUFFER_DATA (header);
+  memset (data, 0, XING_FRAME_SIZE);
+  memcpy (data, mp3_header, 4);
+  memcpy (&data[8 + 1], "Xing", 4);
+
+  xing_flags = 0;
+  if (xing->priv->duration != GST_CLOCK_TIME_NONE) {
+    guint number_of_frames;
+
+    /* The Xing Header contains a NumberOfFrames field, which verifies to:
+     * Duration = NumberOfFrames *SamplesPerFrame/SamplingRate
+     * SamplesPerFrame and SamplingRate are values for the current frame, 
+     * ie 1152 and 44100 in our case.
+     */
+    number_of_frames = (44100 * xing->priv->duration / GST_SECOND) / 1152;
+    data[SIDE_INFO_SIZE + 3] = GUINT32_TO_BE (number_of_frames);
+
+    xing_flags |= GST_XING_FRAME_FIELD;
+  }
+
+  if (xing->priv->byte_count != 0) {
+    xing_flags |= GST_XING_BYTES_FIELD;
+    data[SIDE_INFO_SIZE + 4] = GUINT32_TO_BE (xing->priv->byte_count);
+  }
+
+  /* Un-#ifdef when it's implemented :) xing code in VbrTag.c looks like
+   * it could be stolen
+   */
+#if 0
+  if (xing->priv->seek_table != NULL) {
+    GList *it;
+
+    xing_flags |= GST_XING_TOC_FIELD;
+    for (it = xing->priv->seek_table; it != NULL; it = it->next) {
+      /* do something */
+    }
+  }
+#endif
+
+  data[SIDE_INFO_SIZE + 2] = GUINT32_TO_BE (xing_flags);
+  gst_buffer_set_caps (header, GST_PAD_CAPS (xing->srcpad));
+  //  gst_util_dump_mem ((guchar *)data, XING_FRAME_SIZE);
+  return header;
+}
+
+static gboolean
+xing_ready_to_flush (GstXingMux * xing)
+{
+  if (xing->priv == NULL) {
+    return FALSE;
+  }
+  return xing->priv->flush;
+}
+
+static void
+xing_push_header (GstXingMux * xing)
+{
+  GstBuffer *header;
+  GstEvent *event;
+
+  event = gst_event_new_new_segment (FALSE, 1.0, GST_FORMAT_BYTES,
+      0, GST_CLOCK_TIME_NONE, 0);
+
+  gst_pad_push_event (xing->srcpad, event);
+
+  header = xing_generate_header (xing);
+  xing_set_flush (xing, FALSE);
+  GST_INFO ("Writing real Xing header to beginning of stream");
+  gst_pad_push (xing->srcpad, header);
+}
+
+static GstFlowReturn
+gst_xing_mux_chain (GstPad * pad, GstBuffer * buffer)
+{
+  GstXingMux *xing = GST_XING_MUX (GST_OBJECT_PARENT (pad));
+
+  xing_update_data (xing, GST_BUFFER_SIZE (buffer),
+      GST_BUFFER_DURATION (buffer));
+
+  if (xing_ready_to_flush (xing)) {
+    GST_INFO ("Writing empty Xing header to stream");
+    gst_pad_push (xing->srcpad, xing_generate_header (xing));
+    xing_set_flush (xing, FALSE);
+  }
+
+  return gst_pad_push (xing->srcpad, buffer);
+}
+
+static gboolean
+gst_xing_mux_sink_event (GstPad * pad, GstEvent * event)
+{
+  GstXingMux *xing;
+  gboolean result;
+
+  xing = GST_XING_MUX (gst_pad_get_parent (pad));
+  result = FALSE;
+
+  switch (GST_EVENT_TYPE (event)) {
+    case GST_EVENT_NEWSEGMENT:
+    {
+      gboolean update;
+      gdouble rate;
+      GstFormat format;
+      gint64 value, end_value, base;
+
+      gst_event_parse_new_segment (event, &update, &rate, &format,
+          &value, &end_value, &base);
+      gst_event_unref (event);
+      if (format == GST_FORMAT_BYTES && gst_pad_is_linked (xing->srcpad)) {
+        GstEvent *new_event;
+
+        GST_INFO ("Adjusting NEW_SEGMENT event by %d", XING_FRAME_SIZE);
+        value += XING_FRAME_SIZE;
+        if (end_value != -1) {
+          end_value += XING_FRAME_SIZE;
+        }
+
+        new_event = gst_event_new_new_segment (update, rate, format,
+            value, end_value, base);
+        result = gst_pad_push_event (xing->srcpad, new_event);
+      } else {
+        result = FALSE;
+      }
+    }
+      break;
+
+    case GST_EVENT_EOS:
+      GST_DEBUG_OBJECT (xing, "handling EOS event");
+      xing_push_header (xing);
+      result = gst_pad_push_event (xing->srcpad, event);
+      break;
+    default:
+      result = gst_pad_event_default (pad, event);
+      break;
+  }
+  gst_object_unref (GST_OBJECT (xing));
+
+  return result;
+}
+
+
+static GstStateChangeReturn
+gst_xing_mux_change_state (GstElement * element, GstStateChange transition)
+{
+  GstXingMux *xing;
+  GstStateChangeReturn result;
+
+  xing = GST_XING_MUX (element);
+
+  result = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
+
+  switch (transition) {
+    case GST_STATE_CHANGE_PAUSED_TO_READY:
+      memset (xing->priv, 0, sizeof (GstXingMuxPriv));
+      xing_set_flush (xing, TRUE);
+      break;
+    default:
+      break;
+  }
+
+  return result;
+}
+
+
+static gboolean
+plugin_init (GstPlugin * plugin)
+{
+  if (!gst_element_register (plugin, "xingmux", GST_RANK_NONE,
+          GST_TYPE_XING_MUX))
+    return FALSE;
+
+  GST_DEBUG_CATEGORY_INIT (xing_mux_debug, "xingmux", 0, "Xing Header Muxer");
+
+  return TRUE;
+}
+
+GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,
+    GST_VERSION_MINOR,
+    "xingheader",
+    "Add a xing header to mp3 encoded data",
+    plugin_init, VERSION, "LGPL", GST_PACKAGE, GST_ORIGIN)
diff --git a/gst/mpegaudioparse/gstxingmux.h b/gst/mpegaudioparse/gstxingmux.h
new file mode 100644 (file)
index 0000000..6217aa1
--- /dev/null
@@ -0,0 +1,52 @@
+/*
+ * (c) 2006 Christophe Fergeau  <teuf@gnome.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.
+ */
+
+#include <gst/gst.h>
+
+typedef struct _GstXingMuxPriv GstXingMuxPriv;
+
+/* Definition of structure storing data for this element. */
+typedef struct _GstXingMux {
+  GstElement element;
+
+  GstPad *sinkpad, *srcpad;
+  
+  GstXingMuxPriv *priv;
+
+} GstXingMux;
+
+/* Standard definition defining a class for this element. */
+typedef struct _GstXingMuxClass {
+  GstElementClass parent_class;
+} GstXingMuxClass;
+
+/* Standard macros for defining types for this element.  */
+#define GST_TYPE_XING_MUX \
+  (gst_xing_mux_get_type())
+#define GST_XING_MUX(obj) \
+  (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_XING_MUX,GstXingMux))
+#define GST_XING_MUX_CLASS(klass) \
+  (G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_XING_MUX,GstXingMuxClass))
+#define GST_IS_XING_MUX(obj) \
+  (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_XING_MUX))
+#define GST_IS_XING_MUX_CLASS(obj) \
+  (G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_XING_MUX))
+
+/* Standard function returning type information. */
+GType gst_my_filter_get_type (void);