summaryrefslogtreecommitdiffstats
path: root/gst/asfmux/gstasfparse.c
diff options
context:
space:
mode:
Diffstat (limited to 'gst/asfmux/gstasfparse.c')
-rw-r--r--gst/asfmux/gstasfparse.c623
1 files changed, 623 insertions, 0 deletions
diff --git a/gst/asfmux/gstasfparse.c b/gst/asfmux/gstasfparse.c
new file mode 100644
index 00000000..17db7293
--- /dev/null
+++ b/gst/asfmux/gstasfparse.c
@@ -0,0 +1,623 @@
+/* ASF parser plugin for GStreamer
+ * Copyright (C) 2009 Thiago Santos <thiagoss@embedded.ufcg.edu.br>
+ *
+ * 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 "gstasfparse.h"
+
+/* FIXME add this include
+ * #include <gst/gst-i18n-plugin.h> */
+
+GST_DEBUG_CATEGORY_STATIC (asfparse_debug);
+#define GST_CAT_DEFAULT asfparse_debug
+
+enum
+{
+ PROP_0,
+};
+
+static const GstElementDetails gst_asf_parse_details =
+GST_ELEMENT_DETAILS ("ASF parser",
+ "Parser",
+ "Parses ASF",
+ "Thiago Santos <thiagoss@embedded.ufcg.edu.br>");
+
+static GstStaticPadTemplate src_factory = GST_STATIC_PAD_TEMPLATE ("src",
+ GST_PAD_SRC,
+ GST_PAD_ALWAYS,
+ GST_STATIC_CAPS ("video/x-ms-asf, parsed = (boolean) true")
+ );
+
+static GstStaticPadTemplate sink_factory = GST_STATIC_PAD_TEMPLATE ("sink",
+ GST_PAD_SINK,
+ GST_PAD_ALWAYS,
+ GST_STATIC_CAPS ("video/x-ms-asf, parsed = (boolean) false")
+ );
+
+static void gst_asf_parse_set_property (GObject * object,
+ guint prop_id, const GValue * value, GParamSpec * pspec);
+static void gst_asf_parse_get_property (GObject * object,
+ guint prop_id, GValue * value, GParamSpec * pspec);
+static GstStateChangeReturn gst_asf_parse_change_state (GstElement * element,
+ GstStateChange transition);
+static void gst_asf_parse_loop (GstPad * pad);
+
+GST_BOILERPLATE (GstAsfParse, gst_asf_parse, GstElement, GST_TYPE_ELEMENT);
+
+static void
+gst_asf_parse_reset (GstAsfParse * asfparse)
+{
+ gst_adapter_clear (asfparse->adapter);
+ gst_asf_file_info_reset (asfparse->asfinfo);
+ asfparse->parse_state = ASF_PARSING_HEADERS;
+ asfparse->headers_size = 0;
+ asfparse->data_size = 0;
+ asfparse->parsed_packets = 0;
+ asfparse->offset = 0;
+}
+
+static gboolean
+gst_asf_parse_sink_activate (GstPad * pad)
+{
+ if (gst_pad_check_pull_range (pad)) {
+ return gst_pad_activate_pull (pad, TRUE);
+ } else {
+ return gst_pad_activate_push (pad, TRUE);
+ }
+}
+
+static gboolean
+gst_asf_parse_sink_activate_pull (GstPad * pad, gboolean active)
+{
+ if (active) {
+ return gst_pad_start_task (pad, (GstTaskFunction) gst_asf_parse_loop, pad);
+ } else {
+ return gst_pad_stop_task (pad);
+ }
+}
+
+static GstFlowReturn
+gst_asf_parse_push (GstAsfParse * asfparse, GstBuffer * buf)
+{
+ gst_buffer_set_caps (buf, asfparse->outcaps);
+ return gst_pad_push (asfparse->srcpad, buf);
+}
+
+static GstFlowReturn
+gst_asf_parse_parse_data_object (GstAsfParse * asfparse, GstBuffer * buffer)
+{
+ GstByteReader *reader;
+ GstFlowReturn ret = GST_FLOW_OK;
+ guint64 packet_count;
+
+ GST_DEBUG_OBJECT (asfparse, "Parsing data object");
+
+ reader = gst_byte_reader_new_from_buffer (buffer);
+ /* skip to packet count */
+ if (!gst_byte_reader_skip (reader, 40))
+ goto error;
+ if (!gst_byte_reader_get_uint64_le (reader, &packet_count))
+ goto error;
+
+ if (asfparse->asfinfo->packets_count != packet_count) {
+ GST_WARNING_OBJECT (asfparse, "File properties object and data object have "
+ "different packets count, %" G_GUINT64_FORMAT " %" G_GUINT64_FORMAT,
+ asfparse->asfinfo->packets_count, packet_count);
+ } else {
+ GST_DEBUG_OBJECT (asfparse, "Total packets: %" G_GUINT64_FORMAT,
+ packet_count);
+ }
+
+ gst_byte_reader_free (reader);
+ return gst_asf_parse_push (asfparse, buffer);
+
+error:
+ ret = GST_FLOW_ERROR;
+ GST_ERROR_OBJECT (asfparse, "Error while parsing data object headers");
+ gst_byte_reader_free (reader);
+ return ret;
+}
+
+static GstFlowReturn
+gst_asf_parse_parse_packet (GstAsfParse * asfparse, GstBuffer * buffer)
+{
+ GstAsfPacketInfo *packetinfo = asfparse->packetinfo;
+
+ if (!gst_asf_parse_packet (buffer, packetinfo, FALSE))
+ goto error;
+
+ GST_DEBUG_OBJECT (asfparse, "Received packet of length %" G_GUINT32_FORMAT
+ ", padding %" G_GUINT32_FORMAT ", send time %" G_GUINT32_FORMAT
+ ", duration %" G_GUINT16_FORMAT " and %s keyframe(s)",
+ packetinfo->packet_size, packetinfo->padding,
+ packetinfo->send_time, packetinfo->duration,
+ (packetinfo->has_keyframe) ? "with" : "without");
+
+ /* set gstbuffer fields */
+ if (!packetinfo->has_keyframe) {
+ GST_BUFFER_FLAG_SET (buffer, GST_BUFFER_FLAG_DELTA_UNIT);
+ }
+ GST_BUFFER_TIMESTAMP (buffer) = ((GstClockTime) packetinfo->send_time)
+ * GST_MSECOND;
+ GST_BUFFER_DURATION (buffer) = ((GstClockTime) packetinfo->duration)
+ * GST_MSECOND;
+
+ return gst_asf_parse_push (asfparse, buffer);
+
+error:
+ GST_ERROR_OBJECT (asfparse, "Error while parsing data packet");
+ return GST_FLOW_ERROR;
+}
+
+static GstFlowReturn
+gst_asf_parse_pull_headers (GstAsfParse * asfparse)
+{
+ GstBuffer *guid_and_size = NULL;
+ GstBuffer *headers = NULL;
+ guint64 size;
+ GstFlowReturn ret;
+
+ if ((ret = gst_pad_pull_range (asfparse->sinkpad, asfparse->offset,
+ ASF_GUID_OBJSIZE_SIZE, &guid_and_size)) != GST_FLOW_OK) {
+ GST_ERROR_OBJECT (asfparse, "Failed to pull data from headers");
+ goto leave;
+ }
+ asfparse->offset += ASF_GUID_OBJSIZE_SIZE;
+ size = gst_asf_match_and_peek_obj_size (GST_BUFFER_DATA (guid_and_size),
+ &(guids[ASF_HEADER_OBJECT_INDEX]));
+
+ if (size == 0) {
+ GST_ERROR_OBJECT (asfparse, "ASF starting identifier missing");
+ goto leave;
+ }
+
+ if ((ret = gst_pad_pull_range (asfparse->sinkpad, asfparse->offset,
+ size - ASF_GUID_OBJSIZE_SIZE, &headers)) != GST_FLOW_OK) {
+ GST_ERROR_OBJECT (asfparse, "Failed to pull data from headers");
+ goto leave;
+ }
+ headers = gst_buffer_join (guid_and_size, headers);
+ guid_and_size = NULL;
+ asfparse->offset += size - ASF_GUID_OBJSIZE_SIZE;
+ if (!gst_asf_parse_headers (headers, asfparse->asfinfo)) {
+ goto leave;
+ }
+ return gst_asf_parse_push (asfparse, headers);
+
+leave:
+ if (headers)
+ gst_buffer_unref (headers);
+ if (guid_and_size)
+ gst_buffer_unref (guid_and_size);
+ return ret;
+}
+
+static GstFlowReturn
+gst_asf_parse_pull_data_header (GstAsfParse * asfparse)
+{
+ GstBuffer *buf = NULL;
+ GstFlowReturn ret;
+
+ if ((ret = gst_pad_pull_range (asfparse->sinkpad, asfparse->offset,
+ ASF_DATA_OBJECT_SIZE, &buf)) != GST_FLOW_OK) {
+ GST_ERROR_OBJECT (asfparse, "Failed to pull data header");
+ return ret;
+ }
+ asfparse->offset += ASF_DATA_OBJECT_SIZE;
+ asfparse->data_size = gst_asf_match_and_peek_obj_size (GST_BUFFER_DATA (buf),
+ &(guids[ASF_DATA_OBJECT_INDEX]));
+ if (asfparse->data_size == 0) {
+ GST_ERROR_OBJECT (asfparse, "Unexpected object, was expecting data object");
+ gst_buffer_unref (buf);
+ return GST_FLOW_ERROR;
+ }
+
+ return gst_asf_parse_parse_data_object (asfparse, buf);
+}
+
+static GstFlowReturn
+gst_asf_parse_pull_packets (GstAsfParse * asfparse)
+{
+ GstFlowReturn ret;
+ while (asfparse->asfinfo->broadcast ||
+ asfparse->parsed_packets < asfparse->asfinfo->packets_count) {
+ GstBuffer *packet = NULL;
+
+ GST_DEBUG_OBJECT (asfparse, "Parsing packet %" G_GUINT64_FORMAT,
+ asfparse->parsed_packets);
+
+ /* get the packet */
+ ret = gst_pad_pull_range (asfparse->sinkpad, asfparse->offset,
+ asfparse->asfinfo->packet_size, &packet);
+ if (ret != GST_FLOW_OK)
+ return ret;
+ asfparse->parsed_packets++;
+ asfparse->offset += asfparse->asfinfo->packet_size;
+
+ /* parse the packet */
+ ret = gst_asf_parse_parse_packet (asfparse, packet);
+ if (ret != GST_FLOW_OK)
+ return ret;
+ }
+ return GST_FLOW_OK;
+}
+
+static GstFlowReturn
+gst_asf_parse_pull_indexes (GstAsfParse * asfparse)
+{
+ GstBuffer *guid_and_size = NULL;
+ GstBuffer *buf = NULL;
+ guint64 obj_size;
+ GstFlowReturn ret = GST_FLOW_OK;
+ while (1) {
+ ret = gst_pad_pull_range (asfparse->sinkpad, asfparse->offset,
+ ASF_GUID_OBJSIZE_SIZE, &guid_and_size);
+ if (ret != GST_FLOW_OK)
+ break;
+ /* we can peek at the object size */
+ obj_size =
+ gst_asf_match_and_peek_obj_size (GST_BUFFER_DATA (guid_and_size), NULL);
+ if (obj_size == 0) {
+ GST_ERROR_OBJECT (asfparse, "Incomplete object found");
+ gst_buffer_unref (guid_and_size);
+ ret = GST_FLOW_ERROR;
+ break;
+ }
+ asfparse->offset += ASF_GUID_OBJSIZE_SIZE;
+
+ /* pull the rest of the object */
+ ret = gst_pad_pull_range (asfparse->sinkpad, asfparse->offset, obj_size,
+ &buf);
+ if (ret != GST_FLOW_OK) {
+ gst_buffer_unref (guid_and_size);
+ break;
+ }
+ asfparse->offset += obj_size - ASF_GUID_OBJSIZE_SIZE;
+
+ buf = gst_buffer_join (guid_and_size, buf);
+ ret = gst_asf_parse_push (asfparse, buf);
+ if (ret != GST_FLOW_OK)
+ break;
+ }
+ return ret;
+}
+
+static void
+gst_asf_parse_loop (GstPad * pad)
+{
+ GstFlowReturn ret = GST_FLOW_OK;
+ GstAsfParse *asfparse = GST_ASF_PARSE_CAST (GST_OBJECT_PARENT (pad));
+
+ GST_LOG_OBJECT (asfparse, "Processing data in loop function");
+ switch (asfparse->parse_state) {
+ case ASF_PARSING_HEADERS:
+ GST_INFO_OBJECT (asfparse, "Starting to parse headers");
+ ret = gst_asf_parse_pull_headers (asfparse);
+ if (ret != GST_FLOW_OK)
+ goto pause;
+ asfparse->parse_state = ASF_PARSING_DATA;
+
+ case ASF_PARSING_DATA:
+ GST_INFO_OBJECT (asfparse, "Parsing data object headers");
+ ret = gst_asf_parse_pull_data_header (asfparse);
+ if (ret != GST_FLOW_OK)
+ goto pause;
+ asfparse->parse_state = ASF_PARSING_PACKETS;
+
+ case ASF_PARSING_PACKETS:
+ GST_INFO_OBJECT (asfparse, "Starting packet parsing");
+ GST_INFO_OBJECT (asfparse, "Broadcast mode %s",
+ asfparse->asfinfo->broadcast ? "on" : "off");
+ ret = gst_asf_parse_pull_packets (asfparse);
+ if (ret != GST_FLOW_OK)
+ goto pause;
+
+ /* test if all packets have been processed */
+ if (!asfparse->asfinfo->broadcast &&
+ asfparse->parsed_packets == asfparse->asfinfo->packets_count) {
+ GST_INFO_OBJECT (asfparse,
+ "All %" G_GUINT64_FORMAT " packets processed",
+ asfparse->parsed_packets);
+ asfparse->parse_state = ASF_PARSING_INDEXES;
+ }
+
+ case ASF_PARSING_INDEXES:
+ /* we currently don't care about indexes, so just push them forward */
+ GST_INFO_OBJECT (asfparse, "Starting indexes parsing");
+ ret = gst_asf_parse_pull_indexes (asfparse);
+ if (ret != GST_FLOW_OK)
+ goto pause;
+ default:
+ break;
+ }
+
+pause:
+ {
+ const gchar *reason = gst_flow_get_name (ret);
+
+ GST_INFO_OBJECT (asfparse, "Pausing sinkpad task");
+ gst_pad_pause_task (pad);
+
+ if (GST_FLOW_IS_FATAL (ret) || ret == GST_FLOW_NOT_LINKED) {
+ if (ret == GST_FLOW_UNEXPECTED) {
+ } else {
+ GST_ELEMENT_ERROR (asfparse, STREAM, FAILED,
+ (NULL), ("streaming task paused, reason %s (%d)", reason, ret));
+ }
+ gst_pad_push_event (asfparse->srcpad, gst_event_new_eos ());
+ }
+ }
+}
+
+static GstFlowReturn
+gst_asf_parse_chain (GstPad * pad, GstBuffer * buffer)
+{
+ GstAsfParse *asfparse;
+ GstFlowReturn ret = GST_FLOW_OK;
+
+ asfparse = GST_ASF_PARSE (GST_PAD_PARENT (pad));
+ gst_adapter_push (asfparse->adapter, buffer);
+
+ switch (asfparse->parse_state) {
+ case ASF_PARSING_HEADERS:
+ if (asfparse->headers_size == 0 &&
+ gst_adapter_available (asfparse->adapter) >= ASF_GUID_OBJSIZE_SIZE) {
+
+ /* we can peek at the object size */
+ asfparse->headers_size =
+ gst_asf_match_and_peek_obj_size (gst_adapter_peek
+ (asfparse->adapter, ASF_GUID_OBJSIZE_SIZE),
+ &(guids[ASF_HEADER_OBJECT_INDEX]));
+
+ if (asfparse->headers_size == 0) {
+ /* something is wrong, this probably ain't an ASF stream */
+ GST_ERROR_OBJECT (asfparse, "ASF starting identifier missing");
+ ret = GST_FLOW_ERROR;
+ goto end;
+ }
+ }
+ if (gst_adapter_available (asfparse->adapter) >= asfparse->headers_size) {
+ GstBuffer *headers = gst_adapter_take_buffer (asfparse->adapter,
+ asfparse->headers_size);
+ if (gst_asf_parse_headers (headers, asfparse->asfinfo)) {
+ ret = gst_asf_parse_push (asfparse, headers);
+ asfparse->parse_state = ASF_PARSING_DATA;
+ } else {
+ ret = GST_FLOW_ERROR;
+ GST_ERROR_OBJECT (asfparse, "Failed to parse headers");
+ }
+ }
+ break;
+ case ASF_PARSING_DATA:
+ if (asfparse->data_size == 0 &&
+ gst_adapter_available (asfparse->adapter) >= ASF_GUID_OBJSIZE_SIZE) {
+
+ /* we can peek at the object size */
+ asfparse->data_size =
+ gst_asf_match_and_peek_obj_size (gst_adapter_peek
+ (asfparse->adapter, ASF_GUID_OBJSIZE_SIZE),
+ &(guids[ASF_DATA_OBJECT_INDEX]));
+
+ if (asfparse->data_size == 0) {
+ /* something is wrong */
+ GST_ERROR_OBJECT (asfparse, "Unexpected object after headers, was "
+ "expecting a data object");
+ ret = GST_FLOW_ERROR;
+ goto end;
+ }
+ }
+ /* if we have received the full data object headers */
+ if (gst_adapter_available (asfparse->adapter) >= ASF_DATA_OBJECT_SIZE) {
+ ret = gst_asf_parse_parse_data_object (asfparse,
+ gst_adapter_take_buffer (asfparse->adapter, ASF_DATA_OBJECT_SIZE));
+ if (ret != GST_FLOW_OK) {
+ goto end;
+ }
+ asfparse->parse_state = ASF_PARSING_PACKETS;
+ }
+ break;
+ case ASF_PARSING_PACKETS:
+ g_assert (asfparse->asfinfo->packet_size);
+ while ((asfparse->asfinfo->broadcast ||
+ asfparse->parsed_packets < asfparse->asfinfo->packets_count) &&
+ gst_adapter_available (asfparse->adapter) >=
+ asfparse->asfinfo->packet_size) {
+ GstBuffer *packet = gst_adapter_take_buffer (asfparse->adapter,
+ asfparse->asfinfo->packet_size);
+ asfparse->parsed_packets++;
+ ret = gst_asf_parse_parse_packet (asfparse, packet);
+ if (ret != GST_FLOW_OK)
+ goto end;
+ }
+ if (!asfparse->asfinfo->broadcast &&
+ asfparse->parsed_packets >= asfparse->asfinfo->packets_count) {
+ GST_INFO_OBJECT (asfparse, "Finished parsing packets");
+ asfparse->parse_state = ASF_PARSING_INDEXES;
+ }
+ break;
+ case ASF_PARSING_INDEXES:
+ /* we currently don't care about any of those objects */
+ if (gst_adapter_available (asfparse->adapter) >= ASF_GUID_OBJSIZE_SIZE) {
+ guint64 obj_size;
+ /* we can peek at the object size */
+ obj_size = gst_asf_match_and_peek_obj_size (gst_adapter_peek
+ (asfparse->adapter, ASF_GUID_OBJSIZE_SIZE), NULL);
+ if (gst_adapter_available (asfparse->adapter) >= obj_size) {
+ GST_DEBUG_OBJECT (asfparse, "Skiping object");
+ ret = gst_asf_parse_push (asfparse,
+ gst_adapter_take_buffer (asfparse->adapter, obj_size));
+ if (ret != GST_FLOW_OK) {
+ goto end;
+ }
+ }
+ }
+ break;
+ default:
+ break;
+ }
+
+end:
+ return ret;
+}
+
+static void
+gst_asf_parse_base_init (gpointer g_class)
+{
+ GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
+
+ 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, &gst_asf_parse_details);
+
+ GST_DEBUG_CATEGORY_INIT (asfparse_debug, "asfparse", 0,
+ "Parser for ASF streams");
+}
+
+static void
+gst_asf_parse_finalize (GObject * object)
+{
+ GstAsfParse *asfparse = GST_ASF_PARSE (object);
+ gst_adapter_clear (asfparse->adapter);
+ g_object_unref (G_OBJECT (asfparse->adapter));
+ gst_caps_unref (asfparse->outcaps);
+ gst_asf_file_info_free (asfparse->asfinfo);
+ g_free (asfparse->packetinfo);
+ G_OBJECT_CLASS (parent_class)->finalize (object);
+}
+
+static void
+gst_asf_parse_class_init (GstAsfParseClass * klass)
+{
+ GObjectClass *gobject_class;
+ GstElementClass *gstelement_class;
+
+ gobject_class = (GObjectClass *) klass;
+ gstelement_class = (GstElementClass *) klass;
+
+ parent_class = g_type_class_peek_parent (klass);
+
+ gobject_class->get_property = gst_asf_parse_get_property;
+ gobject_class->set_property = gst_asf_parse_set_property;
+ gobject_class->finalize = gst_asf_parse_finalize;
+
+ gstelement_class->change_state =
+ GST_DEBUG_FUNCPTR (gst_asf_parse_change_state);
+}
+
+static void
+gst_asf_parse_init (GstAsfParse * asfparse, GstAsfParseClass * klass)
+{
+ asfparse->sinkpad = gst_pad_new_from_static_template (&sink_factory, "sink");
+ gst_pad_set_chain_function (asfparse->sinkpad, gst_asf_parse_chain);
+ gst_pad_set_activate_function (asfparse->sinkpad,
+ gst_asf_parse_sink_activate);
+ gst_pad_set_activatepull_function (asfparse->sinkpad,
+ gst_asf_parse_sink_activate_pull);
+ gst_element_add_pad (GST_ELEMENT (asfparse), asfparse->sinkpad);
+
+ asfparse->srcpad = gst_pad_new_from_static_template (&src_factory, "src");
+ gst_pad_use_fixed_caps (asfparse->srcpad);
+ gst_element_add_pad (GST_ELEMENT (asfparse), asfparse->srcpad);
+
+ asfparse->adapter = gst_adapter_new ();
+ asfparse->outcaps = gst_caps_new_simple ("video/x-ms-asf", NULL);
+ asfparse->asfinfo = gst_asf_file_info_new ();
+ asfparse->packetinfo = g_new0 (GstAsfPacketInfo, 1);
+ gst_asf_parse_reset (asfparse);
+}
+
+static void
+gst_asf_parse_get_property (GObject * object,
+ guint prop_id, GValue * value, GParamSpec * pspec)
+{
+ GstAsfParse *asfparse;
+
+ asfparse = GST_ASF_PARSE (object);
+ switch (prop_id) {
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+gst_asf_parse_set_property (GObject * object,
+ guint prop_id, const GValue * value, GParamSpec * pspec)
+{
+ GstAsfParse *asfparse;
+
+ asfparse = GST_ASF_PARSE (object);
+ switch (prop_id) {
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static GstStateChangeReturn
+gst_asf_parse_change_state (GstElement * element, GstStateChange transition)
+{
+ GstAsfParse *asfparse;
+ GstStateChangeReturn ret;
+
+ asfparse = GST_ASF_PARSE (element);
+
+ switch (transition) {
+ case GST_STATE_CHANGE_READY_TO_PAUSED:
+ gst_asf_parse_reset (asfparse);
+ break;
+ case GST_STATE_CHANGE_PAUSED_TO_PLAYING:
+ break;
+ case GST_STATE_CHANGE_PAUSED_TO_READY:
+ break;
+ default:
+ break;
+ }
+
+ ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
+ if (ret == GST_STATE_CHANGE_FAILURE)
+ goto done;
+
+ switch (transition) {
+ case GST_STATE_CHANGE_PLAYING_TO_PAUSED:
+ break;
+ case GST_STATE_CHANGE_PAUSED_TO_READY:
+ break;
+ case GST_STATE_CHANGE_READY_TO_NULL:
+ break;
+ default:
+ break;
+ }
+
+done:
+ return ret;
+}
+
+gboolean
+gst_asf_parse_plugin_init (GstPlugin * plugin)
+{
+ return gst_element_register (plugin, "asfparse",
+ GST_RANK_NONE, GST_TYPE_ASF_PARSE);
+}