diff options
Diffstat (limited to 'gst/tta/gstttaparse.c')
-rw-r--r-- | gst/tta/gstttaparse.c | 418 |
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; +} |