summaryrefslogtreecommitdiffstats
path: root/gst/tta/gstttaparse.c
diff options
context:
space:
mode:
Diffstat (limited to 'gst/tta/gstttaparse.c')
-rw-r--r--gst/tta/gstttaparse.c418
1 files changed, 418 insertions, 0 deletions
diff --git a/gst/tta/gstttaparse.c b/gst/tta/gstttaparse.c
new file mode 100644
index 00000000..3c480d9c
--- /dev/null
+++ b/gst/tta/gstttaparse.c
@@ -0,0 +1,418 @@
+/* GStreamer TTA plugin
+ * (c) 2004 Arwed v. Merkatz <v.merkatz@gmx.net>
+ *
+ * gstttaparse.c: TTA file parser
+ *
+ * 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>
+
+#include <math.h>
+
+#include "gstttaparse.h"
+#include "ttadec.h"
+#include "crc32.h"
+
+GST_DEBUG_CATEGORY_STATIC (gst_tta_parse_debug);
+#define GST_CAT_DEFAULT gst_tta_parse_debug
+
+/* Filter signals and args */
+enum
+{
+ LAST_SIGNAL
+};
+
+enum
+{
+ ARG_0
+};
+
+static GstStaticPadTemplate sink_factory = GST_STATIC_PAD_TEMPLATE ("sink",
+ GST_PAD_SINK,
+ GST_PAD_ALWAYS,
+ GST_STATIC_CAPS ("audio/x-ttafile")
+ );
+
+static GstStaticPadTemplate src_factory = GST_STATIC_PAD_TEMPLATE ("src",
+ GST_PAD_SRC,
+ GST_PAD_ALWAYS,
+ GST_STATIC_CAPS ("audio/x-tta, "
+ "width = (int) { 8, 16, 24 }, "
+ "channels = (int) { 1, 2 }, " "rate = (int) [ 8000, 96000 ]")
+ );
+
+static void gst_tta_parse_class_init (GstTtaParseClass * klass);
+static void gst_tta_parse_base_init (GstTtaParseClass * klass);
+static void gst_tta_parse_init (GstTtaParse * ttaparse);
+
+static void gst_tta_parse_chain (GstPad * pad, GstData * in);
+
+static GstElementClass *parent = NULL;
+
+GType
+gst_tta_parse_get_type (void)
+{
+ static GType plugin_type = 0;
+
+ if (!plugin_type) {
+ static const GTypeInfo plugin_info = {
+ sizeof (GstTtaParseClass),
+ (GBaseInitFunc) gst_tta_parse_base_init,
+ NULL,
+ (GClassInitFunc) gst_tta_parse_class_init,
+ NULL,
+ NULL,
+ sizeof (GstTtaParse),
+ 0,
+ (GInstanceInitFunc) gst_tta_parse_init,
+ };
+ plugin_type = g_type_register_static (GST_TYPE_ELEMENT,
+ "GstTtaParse", &plugin_info, 0);
+ }
+ return plugin_type;
+}
+
+static void
+gst_tta_parse_base_init (GstTtaParseClass * klass)
+{
+ static GstElementDetails plugin_details = {
+ "TTA file parser",
+ "Codec/Demuxer/Audio",
+ "Parses TTA files",
+ "Arwed v. Merkatz <v.merkatz@gmx.net>"
+ };
+ GstElementClass *element_class = GST_ELEMENT_CLASS (klass);
+
+ gst_element_class_add_pad_template (element_class,
+ gst_static_pad_template_get (&src_factory));
+ gst_element_class_add_pad_template (element_class,
+ gst_static_pad_template_get (&sink_factory));
+ gst_element_class_set_details (element_class, &plugin_details);
+}
+
+static void
+gst_tta_parse_dispose (GObject * object)
+{
+ GstTtaParse *ttaparse = GST_TTA_PARSE (object);
+
+ g_free (ttaparse->index);
+
+ G_OBJECT_CLASS (parent)->dispose (object);
+}
+
+static void
+gst_tta_parse_class_init (GstTtaParseClass * klass)
+{
+ GObjectClass *gobject_class;
+ GstElementClass *gstelement_class;
+
+ gobject_class = (GObjectClass *) klass;
+ gstelement_class = (GstElementClass *) klass;
+
+ parent = g_type_class_ref (GST_TYPE_ELEMENT);
+
+ gobject_class->dispose = gst_tta_parse_dispose;
+}
+
+static gboolean
+gst_tta_src_query (GstPad * pad, GstQueryType type,
+ GstFormat * format, gint64 * value)
+{
+ GstTtaParse *ttaparse = GST_TTA_PARSE (gst_pad_get_parent (pad));
+
+ if (type == GST_QUERY_TOTAL) {
+ if (*format == GST_FORMAT_TIME) {
+ if ((ttaparse->data_length == 0) || (ttaparse->samplerate == 0)) {
+ *value = 0;
+ return FALSE;
+ }
+ *value =
+ ((gdouble) ttaparse->data_length / (gdouble) ttaparse->samplerate) *
+ GST_SECOND;
+ GST_DEBUG_OBJECT (ttaparse, "got queried for time, returned %lli",
+ *value);
+ return TRUE;
+ }
+ } else {
+ return gst_pad_query_default (pad, type, format, value);
+ }
+ return FALSE;
+}
+
+static gboolean
+gst_tta_src_event (GstPad * pad, GstEvent * event)
+{
+ GstTtaParse *ttaparse = GST_TTA_PARSE (gst_pad_get_parent (pad));
+
+ switch (GST_EVENT_TYPE (event)) {
+ case GST_EVENT_SEEK:
+ {
+ if (GST_EVENT_SEEK_FORMAT (event) == GST_FORMAT_TIME) {
+ GST_DEBUG_OBJECT (ttaparse, "got seek event");
+ GstEvent *seek_event;
+ guint64 time = GST_EVENT_SEEK_OFFSET (event);
+ guint64 seek_frame = time / (FRAME_TIME * 1000000000);
+ guint64 seekpos = ttaparse->index[seek_frame].pos;
+
+ GST_DEBUG_OBJECT (ttaparse, "seeking to %u", (guint) seekpos);
+ seek_event =
+ gst_event_new_seek (GST_FORMAT_BYTES | GST_SEEK_METHOD_SET |
+ GST_SEEK_FLAG_ACCURATE, seekpos);
+ gst_event_unref (event);
+ if (gst_pad_send_event (GST_PAD_PEER (ttaparse->sinkpad), seek_event)) {
+ gst_pad_event_default (ttaparse->srcpad,
+ gst_event_new (GST_EVENT_FLUSH));
+ return TRUE;
+ } else {
+ GST_LOG_OBJECT (ttaparse, "seek failed");
+ return FALSE;
+ }
+ } else {
+ return gst_pad_send_event (pad, event);
+ }
+ break;
+ }
+ default:
+ return gst_pad_send_event (pad, event);
+ break;
+ }
+}
+
+static void
+gst_tta_parse_init (GstTtaParse * ttaparse)
+{
+ GstElementClass *klass = GST_ELEMENT_GET_CLASS (ttaparse);
+
+ ttaparse->sinkpad =
+ gst_pad_new_from_template (gst_element_class_get_pad_template (klass,
+ "sink"), "sink");
+
+ ttaparse->srcpad =
+ gst_pad_new_from_template (gst_element_class_get_pad_template (klass,
+ "src"), "src");
+ gst_pad_use_explicit_caps (ttaparse->srcpad);
+ gst_pad_set_query_function (ttaparse->srcpad, gst_tta_src_query);
+ gst_pad_set_event_function (ttaparse->srcpad, gst_tta_src_event);
+
+ gst_element_add_pad (GST_ELEMENT (ttaparse), ttaparse->sinkpad);
+ gst_element_add_pad (GST_ELEMENT (ttaparse), ttaparse->srcpad);
+ gst_pad_set_chain_function (ttaparse->sinkpad, gst_tta_parse_chain);
+
+ ttaparse->silent = FALSE;
+ ttaparse->header_parsed = FALSE;
+ ttaparse->partialbuf = NULL;
+ ttaparse->seek_ok = FALSE;
+ ttaparse->current_frame = 0;
+ ttaparse->data_length = 0;
+ ttaparse->samplerate = 0;
+
+ GST_FLAG_SET (ttaparse, GST_ELEMENT_EVENT_AWARE);
+}
+
+static void
+gst_tta_handle_event (GstPad * pad, GstBuffer * buffer)
+{
+ GstEvent *event = GST_EVENT (buffer);
+ GstTtaParse *ttaparse = GST_TTA_PARSE (gst_pad_get_parent (pad));
+
+ GST_DEBUG_OBJECT (ttaparse, "got some event");
+ switch (GST_EVENT_TYPE (event)) {
+ case GST_EVENT_DISCONTINUOUS:
+ {
+ GstEvent *discont;
+ guint64 offset = GST_EVENT_DISCONT_OFFSET (event, 0).value;
+ int i;
+
+ GST_DEBUG_OBJECT (ttaparse, "discont with offset: %u", offset);
+ for (i = 0; i < ttaparse->num_frames; i++) {
+ if (offset == ttaparse->index[i].pos) {
+ GST_DEBUG_OBJECT (ttaparse, "setting current frame to %i", i);
+ discont = gst_event_new_discontinuous (FALSE,
+ GST_FORMAT_TIME, ttaparse->index[i].time, NULL);
+ gst_event_unref (event);
+ gst_buffer_unref (ttaparse->partialbuf);
+ ttaparse->partialbuf = NULL;
+ ttaparse->current_frame = i;
+ gst_pad_event_default (pad, gst_event_new (GST_EVENT_FLUSH));
+ gst_pad_event_default (pad, discont);
+ GST_DEBUG_OBJECT (ttaparse, "sent discont event");
+ return;
+ }
+ }
+ break;
+ }
+ default:
+ gst_pad_event_default (pad, event);
+ break;
+ }
+}
+
+static void
+gst_tta_parse_chain (GstPad * pad, GstData * in)
+{
+ GstTtaParse *ttaparse;
+ GstBuffer *outbuf, *buf = GST_BUFFER (in);
+ guchar *data;
+ gint i;
+ guint64 size, offset = 0;
+ GstCaps *caps;
+
+ g_return_if_fail (GST_IS_PAD (pad));
+ g_return_if_fail (buf != NULL);
+
+ ttaparse = GST_TTA_PARSE (GST_OBJECT_PARENT (pad));
+ g_return_if_fail (GST_IS_TTA_PARSE (ttaparse));
+
+ if (GST_IS_EVENT (buf)) {
+ gst_tta_handle_event (pad, buf);
+ return;
+ }
+
+ if (ttaparse->partialbuf) {
+ GstBuffer *newbuf;
+
+ newbuf = gst_buffer_merge (ttaparse->partialbuf, buf);
+ gst_buffer_unref (buf);
+ gst_buffer_unref (ttaparse->partialbuf);
+ ttaparse->partialbuf = newbuf;
+ } else {
+ ttaparse->partialbuf = buf;
+ }
+
+ size = GST_BUFFER_SIZE (ttaparse->partialbuf);
+ data = GST_BUFFER_DATA (ttaparse->partialbuf);
+ if (!ttaparse->header_parsed) {
+ if ((*data == 'T') && (*(data + 1) == 'T') && (*(data + 2) == 'A')) {
+ double frame_length;
+ int num_frames;
+ guint32 datasize = 0;
+ guint32 crc;
+
+ offset = offset + 4;
+ offset = offset + 2;
+ ttaparse->channels = GST_READ_UINT16_LE (data + offset);
+ offset = offset + 2;
+ ttaparse->bits = GST_READ_UINT16_LE (data + offset);
+ offset += 2;
+ ttaparse->samplerate = GST_READ_UINT32_LE (data + offset);
+ frame_length = FRAME_TIME * ttaparse->samplerate;
+ offset += 4;
+ ttaparse->data_length = GST_READ_UINT32_LE (data + offset);
+ offset += 4;
+ num_frames = (ttaparse->data_length / frame_length) + 1;
+ crc = crc32 (data, 18);
+ if (crc != GST_READ_UINT32_LE (data + offset)) {
+ GST_WARNING_OBJECT (ttaparse, "Header CRC wrong!");
+ }
+ offset += 4;
+ GST_INFO_OBJECT (ttaparse,
+ "channels: %u, bits: %u, samplerate: %u, data_length: %u, num_frames: %u",
+ ttaparse->channels, ttaparse->bits, ttaparse->samplerate,
+ ttaparse->data_length, num_frames);
+ ttaparse->index =
+ (GstTtaIndex *) g_malloc (num_frames * sizeof (GstTtaIndex));
+ ttaparse->num_frames = num_frames;
+ for (i = 0; i < num_frames; i++) {
+ ttaparse->index[i].size = GST_READ_UINT32_LE (data + offset);
+ ttaparse->index[i].pos = GST_BUFFER_OFFSET (ttaparse->partialbuf) + (num_frames) * 4 + 4 + datasize + 22; // 22 == header size, +4 for the TTA1
+ ttaparse->index[i].time = i * FRAME_TIME * 1000000000;
+ offset += 4;
+ datasize += ttaparse->index[i].size;
+ }
+ GST_DEBUG_OBJECT (ttaparse, "Datasize: %u", datasize);
+ crc = crc32 (data + 22, num_frames * 4);
+ if (crc != GST_READ_UINT32_LE (data + offset)) {
+ GST_WARNING_OBJECT (ttaparse, "Seek table CRC wrong!");
+ } else {
+ ttaparse->seek_ok = TRUE;
+ /*
+ g_print("allowing seeking!\n");
+ g_print("dumping index:\n");
+ for (i = 0; i < ttaparse->num_frames; i++) {
+ g_print("frame %u: offset = %llu, time=%llu, size=%u\n",
+ i,
+ ttaparse->index[i].pos,
+ ttaparse->index[i].time,
+ ttaparse->index[i].size);
+ }
+ */
+ }
+ offset += 4;
+ ttaparse->header_parsed = TRUE;
+ caps = gst_caps_new_simple ("audio/x-tta",
+ "width", G_TYPE_INT, ttaparse->bits,
+ "channels", G_TYPE_INT, ttaparse->channels,
+ "rate", G_TYPE_INT, ttaparse->samplerate, NULL);
+ gst_pad_set_explicit_caps (ttaparse->srcpad, caps);
+ }
+ }
+
+ i = ttaparse->current_frame;
+ while (size - offset >= ttaparse->index[i].size) {
+ guint32 crc;
+
+ crc = crc32 (data + offset, ttaparse->index[i].size - 4);
+ if (crc != GST_READ_UINT32_LE (data + offset + ttaparse->index[i].size - 4)) {
+ GST_WARNING_OBJECT (ttaparse, "Frame %u corrupted :(", i);
+ GST_WARNING_OBJECT (ttaparse, "calculated crc: %u, got crc: %u", crc,
+ GST_READ_UINT32_LE (data + offset + ttaparse->index[i].size - 4));
+ }
+ outbuf =
+ gst_buffer_create_sub (ttaparse->partialbuf, offset,
+ ttaparse->index[i].size - 4);
+ GST_BUFFER_TIMESTAMP (outbuf) = ttaparse->index[i].time;
+ if (ttaparse->current_frame + 1 == ttaparse->num_frames) {
+ guint32 samples =
+ ttaparse->data_length % (gint64) (ttaparse->samplerate * FRAME_TIME);
+ gdouble frametime = (gdouble) samples / (gdouble) ttaparse->samplerate;
+
+ GST_BUFFER_DURATION (outbuf) = (guint64) (frametime * GST_SECOND);
+ } else {
+ GST_BUFFER_DURATION (outbuf) = FRAME_TIME * 1000000000;
+ }
+ gst_pad_push (ttaparse->srcpad, GST_DATA (outbuf));
+ offset += ttaparse->index[i].size;
+ ttaparse->current_frame++;
+ i = ttaparse->current_frame;
+ }
+
+ if (size - offset > 0) {
+ glong remainder = size - offset;
+
+ outbuf = gst_buffer_create_sub (ttaparse->partialbuf, offset, remainder);
+ gst_buffer_unref (ttaparse->partialbuf);
+ ttaparse->partialbuf = outbuf;
+ } else {
+ gst_buffer_unref (ttaparse->partialbuf);
+ ttaparse->partialbuf = NULL;
+ }
+
+}
+
+gboolean
+gst_tta_parse_plugin_init (GstPlugin * plugin)
+{
+ if (!gst_element_register (plugin, "ttaparse",
+ GST_RANK_PRIMARY, GST_TYPE_TTA_PARSE)) {
+ return FALSE;
+ }
+
+ GST_DEBUG_CATEGORY_INIT (gst_tta_parse_debug, "ttaparse", 0,
+ "tta file parser");
+
+ return TRUE;
+}