/* ASF parser plugin for GStreamer * Copyright (C) 2009 Thiago Santos * * 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 #include "gstasfparse.h" /* FIXME add this include * #include */ 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 "); 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); }