/* GStreamer * Copyright (C) <1999> Erik Walthinsen <omega@cse.ogi.edu> * <2002> Wim Taymans <wim.taymans@chello.be> * * 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 "gstcdxaparse.h" #include "gstcdxastrip.h" #include "gst/riff/riff-ids.h" #include "gst/riff/riff-media.h" static void gst_cdxaparse_base_init (gpointer g_class); static void gst_cdxaparse_class_init (GstCDXAParseClass * klass); static void gst_cdxaparse_init (GstCDXAParse * cdxaparse); static void gst_cdxaparse_loop (GstElement * element); static GstElementStateReturn gst_cdxaparse_change_state (GstElement * element); static GstStaticPadTemplate sink_template_factory = GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS, GST_STATIC_CAPS ("video/x-cdxa") ); static GstStaticPadTemplate src_template_factory = GST_STATIC_PAD_TEMPLATE ("src", GST_PAD_SRC, GST_PAD_ALWAYS, GST_STATIC_CAPS ("video/mpeg, " "systemstream = (boolean) TRUE") ); static GstRiffReadClass *parent_class = NULL; GType gst_cdxaparse_get_type (void) { static GType cdxaparse_type = 0; if (!cdxaparse_type) { static const GTypeInfo cdxaparse_info = { sizeof (GstCDXAParseClass), gst_cdxaparse_base_init, NULL, (GClassInitFunc) gst_cdxaparse_class_init, NULL, NULL, sizeof (GstCDXAParse), 0, (GInstanceInitFunc) gst_cdxaparse_init, }; cdxaparse_type = g_type_register_static (GST_TYPE_RIFF_READ, "GstCDXAParse", &cdxaparse_info, 0); } return cdxaparse_type; } static void gst_cdxaparse_base_init (gpointer g_class) { GstElementClass *element_class = GST_ELEMENT_CLASS (g_class); static GstElementDetails gst_cdxaparse_details = GST_ELEMENT_DETAILS (".dat parser", "Codec/Parser", "Parse a .dat file (VCD) into raw mpeg1", "Wim Taymans <wim.taymans@tvd.be>"); gst_element_class_set_details (element_class, &gst_cdxaparse_details); /* register src pads */ gst_element_class_add_pad_template (element_class, gst_static_pad_template_get (&sink_template_factory)); gst_element_class_add_pad_template (element_class, gst_static_pad_template_get (&src_template_factory)); } static void gst_cdxaparse_class_init (GstCDXAParseClass * klass) { GstElementClass *gstelement_class = GST_ELEMENT_CLASS (klass); parent_class = g_type_class_ref (GST_TYPE_RIFF_READ); gstelement_class->change_state = gst_cdxaparse_change_state; } static void gst_cdxaparse_init (GstCDXAParse * cdxaparse) { /* FIXME: this element needs to be event aware */ cdxaparse->sinkpad = gst_pad_new_from_template (gst_static_pad_template_get (&sink_template_factory), "sink"); GST_RIFF_READ (cdxaparse)->sinkpad = cdxaparse->sinkpad; gst_element_add_pad (GST_ELEMENT (cdxaparse), cdxaparse->sinkpad); cdxaparse->srcpad = gst_pad_new_from_template (gst_static_pad_template_get (&src_template_factory), "src"); /* FIXME: event, query */ gst_element_add_pad (GST_ELEMENT (cdxaparse), cdxaparse->srcpad); gst_element_set_loop_function (GST_ELEMENT (cdxaparse), gst_cdxaparse_loop); cdxaparse->state = GST_CDXAPARSE_START; cdxaparse->seek_pending = FALSE; cdxaparse->seek_offset = 0; GST_FLAG_SET (cdxaparse, GST_ELEMENT_EVENT_AWARE); } static gboolean gst_cdxaparse_stream_init (GstCDXAParse * cdxa) { GstRiffRead *riff = GST_RIFF_READ (cdxa); guint32 doctype; if (!gst_riff_read_header (riff, &doctype)) return FALSE; if (doctype != GST_RIFF_RIFF_CDXA) { GST_ELEMENT_ERROR (cdxa, STREAM, WRONG_TYPE, (NULL), (NULL)); return FALSE; } return TRUE; } /* Read 'fmt ' header */ static gboolean G_GNUC_UNUSED gst_cdxaparse_fmt (GstCDXAParse * cdxa) { GstRiffRead *riff = GST_RIFF_READ (cdxa); gst_riff_strf_auds *header; if (!gst_riff_read_strf_auds (riff, &header)) { g_warning ("Not fmt"); return FALSE; } /* As we don't know what is in this fmt field, we do nothing */ return TRUE; } static gboolean G_GNUC_UNUSED gst_cdxaparse_other (GstCDXAParse * cdxa) { GstRiffRead *riff = GST_RIFF_READ (cdxa); guint32 tag, length; if (!gst_riff_peek_head (riff, &tag, &length, NULL)) { return FALSE; } switch (tag) { case GST_RIFF_TAG_data: if (!gst_bytestream_flush (riff->bs, 8)) return FALSE; cdxa->state = GST_CDXAPARSE_DATA; cdxa->dataleft = cdxa->datasize = (guint64) length; cdxa->datastart = gst_bytestream_tell (riff->bs); break; default: if (!gst_riff_read_skip (riff)) return FALSE; break; } return TRUE; } static void gst_cdxaparse_loop (GstElement * element) { GstCDXAParse *cdxa = GST_CDXAPARSE (element); GstRiffRead *riff = GST_RIFF_READ (cdxa); if (cdxa->state == GST_CDXAPARSE_DATA) { if (cdxa->dataleft > 0) { gint sync; guint got_bytes, desired; GstBuffer *buf = NULL; GstBuffer *outbuf = NULL; /* resync */ desired = cdxa->dataleft; if (desired > 1024) desired = 1024; if (!(buf = gst_riff_peek_element_data (riff, desired, &got_bytes))) return; sync = gst_cdxastrip_sync (buf); gst_buffer_unref (buf); if (sync == -1) { gst_bytestream_flush_fast (riff->bs, desired); cdxa->dataleft -= desired; return; } if (sync > 0) { if (cdxa->dataleft < sync) sync = cdxa->dataleft; gst_bytestream_flush_fast (riff->bs, sync); cdxa->dataleft -= sync; if (cdxa->dataleft == 0) return; } /* get data */ desired = GST_CDXA_SECTOR_SIZE; if (!(buf = gst_riff_read_element_data (riff, desired, &got_bytes))) return; /* Skip CDXA headers, only keep data */ outbuf = gst_cdxastrip_strip (buf); GST_DEBUG ("Pushing one buffer"); gst_pad_push (cdxa->srcpad, GST_DATA (outbuf)); if (got_bytes < cdxa->dataleft) cdxa->dataleft -= got_bytes; else cdxa->dataleft = 0; return; } else { cdxa->state = GST_CDXAPARSE_OTHER; } } switch (cdxa->state) { case GST_CDXAPARSE_START: if (!gst_cdxaparse_stream_init (cdxa)) return; cdxa->state = GST_CDXAPARSE_DATA; cdxa->dataleft = cdxa->datasize = (guint64) gst_bytestream_length (riff->bs); cdxa->datastart = gst_bytestream_tell (riff->bs); break; #if 0 cdxa->state = GST_CDXAPARSE_FMT; /* fall-through */ case GST_CDXAPARSE_FMT: if (0 && !gst_cdxaparse_fmt (cdxa)) return; cdxa->state = GST_CDXAPARSE_OTHER; /* fall-through */ #endif case GST_CDXAPARSE_OTHER: if (!gst_cdxaparse_other (cdxa)) return; break; case GST_CDXAPARSE_DATA: default: g_assert_not_reached (); } } static GstElementStateReturn gst_cdxaparse_change_state (GstElement * element) { GstCDXAParse *cdxa = GST_CDXAPARSE (element); switch (GST_STATE_TRANSITION (element)) { case GST_STATE_READY_TO_PAUSED: cdxa->state = GST_CDXAPARSE_START; break; case GST_STATE_PAUSED_TO_READY: cdxa->state = GST_CDXAPARSE_START; cdxa->seek_pending = FALSE; cdxa->seek_offset = 0; break; default: break; } if (GST_ELEMENT_CLASS (parent_class)->change_state) return GST_ELEMENT_CLASS (parent_class)->change_state (element); return GST_STATE_SUCCESS; } static gboolean plugin_init (GstPlugin * plugin) { if (!gst_library_load ("riff") || !gst_library_load ("gstbytestream")) { return FALSE; } return gst_element_register (plugin, "cdxaparse", GST_RANK_PRIMARY, GST_TYPE_CDXAPARSE) && gst_element_register (plugin, "cdxastrip", GST_RANK_PRIMARY, GST_TYPE_CDXASTRIP); } GST_PLUGIN_DEFINE (GST_VERSION_MAJOR, GST_VERSION_MINOR, "cdxaparse", "Parse a .dat file (VCD) into raw mpeg1", plugin_init, VERSION, "LGPL", GST_PACKAGE, GST_ORIGIN)