summaryrefslogtreecommitdiffstats
path: root/gst/cdxaparse/gstcdxaparse.c
diff options
context:
space:
mode:
Diffstat (limited to 'gst/cdxaparse/gstcdxaparse.c')
-rw-r--r--gst/cdxaparse/gstcdxaparse.c346
1 files changed, 190 insertions, 156 deletions
diff --git a/gst/cdxaparse/gstcdxaparse.c b/gst/cdxaparse/gstcdxaparse.c
index d0029dff..b7b4dc68 100644
--- a/gst/cdxaparse/gstcdxaparse.c
+++ b/gst/cdxaparse/gstcdxaparse.c
@@ -1,3 +1,4 @@
+/* -*- Mode: C; tab-width: 2; indent-tabs-mode: t; c-basic-offset: 2 -*- */
/* GStreamer
* Copyright (C) <1999> Erik Walthinsen <omega@cse.ogi.edu>
* <2002> Wim Taymans <wim.taymans@chello.be>
@@ -25,27 +26,37 @@
#include <string.h>
#include "gstcdxaparse.h"
+#include "gst/riff/riff-ids.h"
+#include "gst/riff/riff-media.h"
-#define MAKE_FOUR_CC(a,b,c,d) ( ((guint32)a) | (((guint32)b)<< 8) | \
- ((guint32)c)<<16 | (((guint32)d)<<24) )
-
+static void gst_cdxa_parse_base_init (gpointer g_class);
+static void gst_cdxa_parse_class_init (GstCDXAParseClass * klass);
+static void gst_cdxa_parse_init (GstCDXAParse * cdxa_parse);
-/* RIFF types */
-#define GST_RIFF_TAG_RIFF MAKE_FOUR_CC('R','I','F','F')
-#define GST_RIFF_RIFF_CDXA MAKE_FOUR_CC('C','D','X','A')
+static GstElementStateReturn gst_cdxa_parse_change_state (GstElement * element);
+static void gst_cdxa_parse_loop (GstElement * element);
-#define GST_RIFF_TAG_fmt MAKE_FOUR_CC('f','m','t',' ')
-#define GST_RIFF_TAG_data MAKE_FOUR_CC('d','a','t','a')
+/* elementfactory information */
+static GstElementDetails gst_cdxa_parse_details =
+GST_ELEMENT_DETAILS (".dat parser",
+ "Codec/Parser",
+ "Parse a .dat file (VCD) into raw mpeg1",
+ "Wim Taymans <wim.taymans@tvd.be>");
+static GstStaticPadTemplate sink_template_factory =
+GST_STATIC_PAD_TEMPLATE ("sink",
+ GST_PAD_SINK,
+ GST_PAD_ALWAYS,
+ GST_STATIC_CAPS ("video/x-cdxa")
+ );
-/* elementfactory information */
-static GstElementDetails gst_cdxa_parse_details = {
- ".dat parser",
- "Codec/Parser",
- "Parse a .dat file (VCD) into raw mpeg1",
- "Wim Taymans <wim.taymans@tvd.be>",
-};
+static GstStaticPadTemplate src_template_factory =
+GST_STATIC_PAD_TEMPLATE ("src",
+ GST_PAD_SRC,
+ GST_PAD_ALWAYS,
+ GST_STATIC_CAPS ("video/mpeg, " "systemstream = (boolean) TRUE")
+ );
/* CDXAParse signals and args */
enum
@@ -60,28 +71,6 @@ enum
/* FILL ME */
};
-static GstStaticPadTemplate sink_templ = GST_STATIC_PAD_TEMPLATE ("sink",
- GST_PAD_SINK,
- GST_PAD_ALWAYS,
- GST_STATIC_CAPS ("video/x-cdxa")
- );
-
-static GstStaticPadTemplate src_templ = GST_STATIC_PAD_TEMPLATE ("src",
- GST_PAD_SRC,
- GST_PAD_ALWAYS,
- GST_STATIC_CAPS ("video/mpeg, " "systemstream = (boolean) TRUE, "
- "mpegversion = (int) 1")
- );
-
-static void gst_cdxa_parse_base_init (gpointer g_class);
-static void gst_cdxa_parse_class_init (GstCDXAParseClass * klass);
-static void gst_cdxa_parse_init (GstCDXAParse * cdxa_parse);
-
-static void gst_cdxa_parse_loop (GstElement * element);
-
-static GstElementStateReturn gst_cdxa_parse_change_state (GstElement * element);
-
-
static GstElementClass *parent_class = NULL;
/*static guint gst_cdxa_parse_signals[LAST_SIGNAL] = { 0 }; */
@@ -105,34 +94,37 @@ gst_cdxa_parse_get_type (void)
};
cdxa_parse_type =
- g_type_register_static (GST_TYPE_ELEMENT, "GstCDXAParse",
+ g_type_register_static (GST_TYPE_RIFF_READ, "GstCDXAParse",
&cdxa_parse_info, 0);
}
return cdxa_parse_type;
}
+
static void
gst_cdxa_parse_base_init (gpointer g_class)
{
GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
+ gst_element_class_set_details (element_class, &gst_cdxa_parse_details);
+
+ /* register src pads */
gst_element_class_add_pad_template (element_class,
- gst_static_pad_template_get (&src_templ));
+ gst_static_pad_template_get (&sink_template_factory));
gst_element_class_add_pad_template (element_class,
- gst_static_pad_template_get (&sink_templ));
- gst_element_class_set_details (element_class, &gst_cdxa_parse_details);
+ gst_static_pad_template_get (&src_template_factory));
}
static void
gst_cdxa_parse_class_init (GstCDXAParseClass * klass)
{
- GObjectClass *gobject_class;
GstElementClass *gstelement_class;
+ GObjectClass *object_class;
- gobject_class = (GObjectClass *) klass;
gstelement_class = (GstElementClass *) klass;
+ object_class = (GObjectClass *) klass;
- parent_class = g_type_class_ref (GST_TYPE_ELEMENT);
+ parent_class = g_type_class_ref (GST_TYPE_RIFF_READ);
gstelement_class->change_state = gst_cdxa_parse_change_state;
}
@@ -140,144 +132,181 @@ gst_cdxa_parse_class_init (GstCDXAParseClass * klass)
static void
gst_cdxa_parse_init (GstCDXAParse * cdxa_parse)
{
- GST_FLAG_SET (cdxa_parse, GST_ELEMENT_EVENT_AWARE);
-
+ /* sink */
cdxa_parse->sinkpad =
- gst_pad_new_from_template (gst_static_pad_template_get (&sink_templ),
- "sink");
+ gst_pad_new_from_template (gst_static_pad_template_get
+ (&sink_template_factory), "sink");
gst_element_add_pad (GST_ELEMENT (cdxa_parse), cdxa_parse->sinkpad);
+ GST_RIFF_READ (cdxa_parse)->sinkpad = cdxa_parse->sinkpad;
+
+
+ gst_element_set_loop_function (GST_ELEMENT (cdxa_parse), gst_cdxa_parse_loop);
+
cdxa_parse->srcpad =
- gst_pad_new_from_template (gst_static_pad_template_get (&src_templ),
- "src");
+ gst_pad_new_from_template (gst_static_pad_template_get
+ (&src_template_factory), "src");
gst_element_add_pad (GST_ELEMENT (cdxa_parse), cdxa_parse->srcpad);
- gst_element_set_loop_function (GST_ELEMENT (cdxa_parse), gst_cdxa_parse_loop);
+ cdxa_parse->state = GST_CDXA_PARSE_START;
+ cdxa_parse->seek_pending = FALSE;
+ cdxa_parse->seek_offset = 0;
}
static gboolean
-gst_cdxa_parse_handle_event (GstCDXAParse * cdxa_parse)
+gst_cdxa_parse_stream_init (GstCDXAParse * cdxa_parse)
{
- guint32 remaining;
- GstEvent *event;
- GstEventType type;
+ GstRiffRead *riff = GST_RIFF_READ (cdxa_parse);
+ guint32 doctype;
- gst_bytestream_get_status (cdxa_parse->bs, &remaining, &event);
+ if (!gst_riff_read_header (riff, &doctype))
+ return FALSE;
- type = event ? GST_EVENT_TYPE (event) : GST_EVENT_UNKNOWN;
+ if (doctype != GST_RIFF_RIFF_CDXA) {
+ GST_ELEMENT_ERROR (cdxa_parse, STREAM, WRONG_TYPE, (NULL), (NULL));
+ return FALSE;
+ }
- switch (type) {
- case GST_EVENT_EOS:
- gst_pad_event_default (cdxa_parse->sinkpad, event);
- break;
- case GST_EVENT_SEEK:
- g_warning ("seek event\n");
- break;
- case GST_EVENT_FLUSH:
- g_warning ("flush event\n");
- break;
- case GST_EVENT_DISCONTINUOUS:
- g_warning ("discont event\n");
+ return TRUE;
+}
+
+/* Read 'fmt ' header */
+static gboolean
+gst_cdxa_parse_fmt (GstCDXAParse * cdxa_parse)
+{
+ GstRiffRead *riff = GST_RIFF_READ (cdxa_parse);
+ 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
+gst_cdxa_parse_other (GstCDXAParse * cdxa_parse)
+{
+ GstRiffRead *riff = GST_RIFF_READ (cdxa_parse);
+ guint32 tag, length;
+
+ /* Fixme, need to handle a seek...can you seek in cdxa? */
+
+ if (!gst_riff_peek_head (riff, &tag, &length, NULL)) {
+ return FALSE;
+ }
+
+ switch (tag) {
+ case GST_RIFF_TAG_data:
+ gst_bytestream_flush (riff->bs, 8);
+
+ cdxa_parse->state = GST_CDXA_PARSE_DATA;
+ cdxa_parse->dataleft = (guint64) length;
break;
+
default:
- g_warning ("unhandled event %d\n", type);
+ gst_riff_read_skip (riff);
break;
}
return TRUE;
}
-/*
+#define MAX_BUFFER_SIZE 4096
+
+static void
+gst_cdxa_parse_loop (GstElement * element)
+{
+ GstCDXAParse *cdxa_parse = GST_CDXA_PARSE (element);
+ GstRiffRead *riff = GST_RIFF_READ (cdxa_parse);
-CDXA starts with the following header:
-
-! RIFF:4 ! size:4 ! "CDXA" ! "fmt " ! size:4 ! (size+1)&~1 bytes of crap !
-! "data" ! data_size:4 ! (data_size/2352) sectors...
+ if (cdxa_parse->state == GST_CDXA_PARSE_DATA) {
+ if (cdxa_parse->dataleft > 0) {
+ guint32 got_bytes, desired;
+ GstBuffer *buf, *outbuf;
-*/
+ desired = GST_CDXA_SECTOR_SIZE;
-typedef struct
-{
- gchar RIFF_tag[4];
- guint32 riff_size;
- gchar CDXA_tag[4];
- gchar fmt_tag[4];
- guint32 fmt_size;
-}
-CDXAParseHeader;
+ buf = gst_riff_read_element_data (riff, desired, &got_bytes);
/*
-A sectors is 2352 bytes long and is composed of:
+
+A sector is 2352 bytes long and is composed of:
! sync ! header ! subheader ! data ... ! edc !
! 12 bytes ! 4 bytes ! 8 bytes ! 2324 bytes ! 4 bytes !
!-------------------------------------------------------!
We parse the data out of it and send it to the srcpad.
+
+sync : 00 FF FF FF FF FF FF FF FF FF FF 00
+header : hour minute second mode
+sub-header : track channel sub_mode coding repeat (4 bytes)
+edc : checksum
+
*/
-static void
-gst_cdxa_parse_loop (GstElement * element)
-{
- GstCDXAParse *cdxa_parse;
- CDXAParseHeader *header;
- guint8 *headerdata;
-
- g_return_if_fail (element != NULL);
- g_return_if_fail (GST_IS_CDXA_PARSE (element));
-
- cdxa_parse = GST_CDXA_PARSE (element);
-
- if (cdxa_parse->state == CDXA_PARSE_HEADER) {
- guint32 fmt_size;
- guint8 *buf;
- guint32 got_bytes;
-
- got_bytes = gst_bytestream_peek_bytes (cdxa_parse->bs, &headerdata, 20);
- header = (CDXAParseHeader *) headerdata;
- if (got_bytes < 20)
- return;
-
- cdxa_parse->riff_size = GUINT32_FROM_LE (header->riff_size);
- fmt_size = (GUINT32_FROM_LE (header->fmt_size) + 1) & ~1;
-
- /* flush the header + fmt_size bytes + 4 bytes "data" */
- if (!gst_bytestream_flush (cdxa_parse->bs, 20 + fmt_size + 4))
- return;
-
- /* get the data size */
- got_bytes =
- gst_bytestream_peek_bytes (cdxa_parse->bs, (guint8 **) & buf, 4);
- if (got_bytes < 4)
- return;
- cdxa_parse->data_size = GST_READ_UINT32_LE (buf);
-
- /* flush the data size */
- if (!gst_bytestream_flush (cdxa_parse->bs, 4))
- return;
-
- if (cdxa_parse->data_size % CDXA_SECTOR_SIZE)
- g_warning ("cdxa_parse: size not multiple of %d bytes", CDXA_SECTOR_SIZE);
-
- cdxa_parse->sectors = cdxa_parse->data_size / CDXA_SECTOR_SIZE;
-
- cdxa_parse->state = CDXA_PARSE_DATA;
- } else {
- GstBuffer *buf;
- GstBuffer *outbuf;
- guint32 got_bytes;
-
- got_bytes = gst_bytestream_read (cdxa_parse->bs, &buf, CDXA_SECTOR_SIZE);
- if (got_bytes < CDXA_SECTOR_SIZE) {
- gst_cdxa_parse_handle_event (cdxa_parse);
- return;
+/*
+ if (got_bytes < GST_CDXA_SECTOR_SIZE) {
+ gst_cdxa_parse_handle_event (cdxa_parse);
+ return;
+ }
+*/
+
+ /* Extract time from CDXA header */
+/* printf( "%02u:%02u:%02u\n", (unsigned char) *(GST_BUFFER_DATA(buf)+12), (unsigned char) *(GST_BUFFER_DATA(buf)+13), (unsigned char) *(GST_BUFFER_DATA(buf)+14) );*/
+
+ /* Jump CDXA headers, only keep data */
+ outbuf = gst_buffer_create_sub (buf, 24, GST_CDXA_DATA_SIZE);
+ gst_buffer_unref (buf);
+
+ gst_pad_push (cdxa_parse->srcpad, GST_DATA (outbuf));
+
+ cdxa_parse->byteoffset += got_bytes;
+ if (got_bytes < cdxa_parse->dataleft) {
+ cdxa_parse->dataleft -= got_bytes;
+ return;
+ } else {
+ cdxa_parse->dataleft = 0;
+ cdxa_parse->state = GST_CDXA_PARSE_OTHER;
+ }
+ } else {
+ cdxa_parse->state = GST_CDXA_PARSE_OTHER;
}
+ }
+
+ switch (cdxa_parse->state) {
+ case GST_CDXA_PARSE_START:
+ if (!gst_cdxa_parse_stream_init (cdxa_parse)) {
+ return;
+ }
+
+ cdxa_parse->state = GST_CDXA_PARSE_FMT;
+ /* fall-through */
+
+ case GST_CDXA_PARSE_FMT:
+ if (!gst_cdxa_parse_fmt (cdxa_parse)) {
+ return;
+ }
+
+ cdxa_parse->state = GST_CDXA_PARSE_OTHER;
+ /* fall-through */
+
+ case GST_CDXA_PARSE_OTHER:
+ if (!gst_cdxa_parse_other (cdxa_parse)) {
+ return;
+ }
+
+ break;
- outbuf = gst_buffer_create_sub (buf, 24, CDXA_DATA_SIZE);
- gst_buffer_unref (buf);
+ case GST_CDXA_PARSE_DATA:
- gst_pad_push (cdxa_parse->srcpad, GST_DATA (outbuf));
+ default:
+ g_assert_not_reached ();
}
}
@@ -289,24 +318,31 @@ gst_cdxa_parse_change_state (GstElement * element)
switch (GST_STATE_TRANSITION (element)) {
case GST_STATE_NULL_TO_READY:
break;
+
case GST_STATE_READY_TO_PAUSED:
- cdxa_parse->state = CDXA_PARSE_HEADER;
- cdxa_parse->bs = gst_bytestream_new (cdxa_parse->sinkpad);
+ cdxa_parse->state = GST_CDXA_PARSE_START;
break;
+
case GST_STATE_PAUSED_TO_PLAYING:
break;
+
case GST_STATE_PLAYING_TO_PAUSED:
break;
+
case GST_STATE_PAUSED_TO_READY:
- gst_bytestream_destroy (cdxa_parse->bs);
+ cdxa_parse->state = GST_CDXA_PARSE_START;
+
+
+ cdxa_parse->seek_pending = FALSE;
+ cdxa_parse->seek_offset = 0;
break;
+
case GST_STATE_READY_TO_NULL:
break;
- default:
- break;
}
- parent_class->change_state (element);
+ if (GST_ELEMENT_CLASS (parent_class)->change_state)
+ return GST_ELEMENT_CLASS (parent_class)->change_state (element);
return GST_STATE_SUCCESS;
}
@@ -314,14 +350,12 @@ gst_cdxa_parse_change_state (GstElement * element)
static gboolean
plugin_init (GstPlugin * plugin)
{
- if (!gst_library_load ("gstbytestream"))
- return FALSE;
-
- if (!gst_element_register (plugin, "cdxaparse", GST_RANK_SECONDARY,
- GST_TYPE_CDXA_PARSE))
+ if (!gst_library_load ("riff")) {
return FALSE;
+ }
- return TRUE;
+ return gst_element_register (plugin, "cdxaparse", GST_RANK_SECONDARY,
+ GST_TYPE_CDXA_PARSE);
}
GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,