mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-03 22:18:50 +00:00
2dded0fceb
Following the ed4d08189ea6e19a50e029e60da52d3583c39fbb commit, this one fixes rtpasfpay to use packet length as the payloaded data length, but also accepting it as the full packet size for compatibility with other implementations due to the lack of clarity of the spec in this part.
624 lines
19 KiB
C
624 lines
19 KiB
C
/* 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 = 0;
|
|
|
|
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,
|
|
asfparse->asfinfo->packet_size))
|
|
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);
|
|
}
|