diff options
Diffstat (limited to 'gst/mpeg4videoparse/mpeg4videoparse.c')
-rw-r--r-- | gst/mpeg4videoparse/mpeg4videoparse.c | 281 |
1 files changed, 281 insertions, 0 deletions
diff --git a/gst/mpeg4videoparse/mpeg4videoparse.c b/gst/mpeg4videoparse/mpeg4videoparse.c new file mode 100644 index 00000000..21be173c --- /dev/null +++ b/gst/mpeg4videoparse/mpeg4videoparse.c @@ -0,0 +1,281 @@ +/* GStreamer + * Copyright (C) <2007> Julien Moutte <julien@fluendo.com> + * + * 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 "mpeg4videoparse.h" + +GST_DEBUG_CATEGORY_STATIC (mpeg4v_parse_debug); +#define GST_CAT_DEFAULT mpeg4v_parse_debug + +/* elementfactory information */ +static GstElementDetails mpeg4vparse_details = +GST_ELEMENT_DETAILS ("MPEG 4 video elementary stream parser", + "Codec/Parser/Video", + "Parses MPEG-4 Part 2 elementary video streams", + "Julien Moutte <julien@fluendo.com>"); + +static GstStaticPadTemplate src_template = +GST_STATIC_PAD_TEMPLATE ("src", GST_PAD_SRC, + GST_PAD_ALWAYS, + GST_STATIC_CAPS ("video/mpeg, " + "mpegversion = (int) 4, " + "parsed = (boolean) true, " "systemstream = (boolean) false") + ); + +static GstStaticPadTemplate sink_template = +GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, + GST_PAD_ALWAYS, + GST_STATIC_CAPS ("video/mpeg, " + "mpegversion = (int) 4, " + "parsed = (boolean) false, " "systemstream = (boolean) false") + ); + +GST_BOILERPLATE (GstMpeg4VParse, gst_mpeg4vparse, GstElement, GST_TYPE_ELEMENT); + +static GstFlowReturn +gst_mpeg4vparse_drain (GstMpeg4VParse * parse) +{ + GstFlowReturn ret = GST_FLOW_OK; + const guint8 *data = NULL; + guint i = 0, available = 0; + + while (gst_adapter_available (parse->adapter) >= 4) { + /* If we have enough data, ensure we're aligned to a start code */ + data = gst_adapter_peek (parse->adapter, 4); + if (data[0] == 0 && data[1] == 0 && data[2] == 1) { + GST_LOG_OBJECT (parse, "found start code with type %02X", data[3]); + parse->found_start = TRUE; + break; + } else { + GST_LOG_OBJECT (parse, "flushing 1 byte"); + gst_adapter_flush (parse->adapter, 1); + } + } + + if (G_UNLIKELY (!parse->found_start)) { + GST_DEBUG_OBJECT (parse, "start code not found, need more data"); + goto beach; + } + + if (G_UNLIKELY (gst_adapter_available (parse->adapter) < 8)) { + GST_DEBUG_OBJECT (parse, "start code found, need more data to find next"); + goto beach; + } + + /* Found a start code, search for the next one */ + available = gst_adapter_available (parse->adapter); + data = gst_adapter_peek (parse->adapter, available); + for (i = 4; i < available - 4; i++) { + /* We generate packets based on VOP start code */ + if (data[i] == 0 && data[i + 1] == 0 && data[i + 2] == 1 && + data[i + 3] == 0xB6) { + GstBuffer *out_buf = gst_adapter_take_buffer (parse->adapter, i); + + GST_LOG_OBJECT (parse, "found next start code at %u", i); + if (out_buf) { + gst_buffer_set_caps (out_buf, GST_PAD_CAPS (parse->srcpad)); + gst_pad_push (parse->srcpad, out_buf); + } + parse->found_start = FALSE; + } + } + +beach: + return ret; +} + +static GstFlowReturn +gst_mpeg4vparse_chain (GstPad * pad, GstBuffer * buffer) +{ + GstMpeg4VParse *parse = GST_MPEG4VIDEOPARSE (gst_pad_get_parent (pad)); + GstFlowReturn ret = GST_FLOW_OK; + + GST_DEBUG_OBJECT (parse, "received buffer of %u bytes with ts %" + GST_TIME_FORMAT " and offset %" G_GINT64_FORMAT, GST_BUFFER_SIZE (buffer), + GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buffer)), + GST_BUFFER_OFFSET (buffer)); + + gst_adapter_push (parse->adapter, buffer); + + ret = gst_mpeg4vparse_drain (parse); + + gst_object_unref (parse); + + return ret; +} + +static gboolean +gst_mpeg4vparse_sink_setcaps (GstPad * pad, GstCaps * caps) +{ + gboolean res = TRUE; + GstCaps *out_caps = NULL; + GstMpeg4VParse *parse = GST_MPEG4VIDEOPARSE (gst_pad_get_parent (pad)); + + GST_DEBUG_OBJECT (parse, "setcaps called with %" GST_PTR_FORMAT, caps); + + out_caps = gst_caps_new_simple ("video/mpeg", + "mpegversion", G_TYPE_INT, 4, + "systemstream", G_TYPE_BOOLEAN, FALSE, + "parsed", G_TYPE_BOOLEAN, TRUE, NULL); + + if (out_caps) { + GST_DEBUG_OBJECT (parse, "setting downstream caps to %" GST_PTR_FORMAT, + caps); + res = gst_pad_set_caps (parse->srcpad, out_caps); + gst_caps_unref (out_caps); + } + + gst_object_unref (parse); + + return res; +} + +static gboolean +gst_mpeg4vparse_sink_event (GstPad * pad, GstEvent * event) +{ + gboolean res = TRUE; + GstMpeg4VParse *parse = GST_MPEG4VIDEOPARSE (gst_pad_get_parent (pad)); + + GST_DEBUG_OBJECT (parse, "handling event type %s", + GST_EVENT_TYPE_NAME (event)); + + switch (GST_EVENT_TYPE (event)) { + default: + res = gst_pad_event_default (pad, event); + break; + } + + gst_object_unref (parse); + + return res; +} + +static void +gst_mpeg4vparse_cleanup (GstMpeg4VParse * parse) +{ + if (parse->adapter) { + gst_adapter_clear (parse->adapter); + } + + parse->found_start = FALSE; +} + +static GstStateChangeReturn +gst_mpeg4vparse_change_state (GstElement * element, GstStateChange transition) +{ + GstMpeg4VParse *parse = GST_MPEG4VIDEOPARSE (element); + GstStateChangeReturn ret; + + ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition); + + switch (transition) { + case GST_STATE_CHANGE_PAUSED_TO_READY: + gst_mpeg4vparse_cleanup (parse); + break; + default: + break; + } + + return ret; +} + +static void +gst_mpeg4vparse_dispose (GObject * object) +{ + GstMpeg4VParse *parse = GST_MPEG4VIDEOPARSE (object); + + if (parse->adapter) { + g_object_unref (parse->adapter); + parse->adapter = NULL; + } + + GST_CALL_PARENT (G_OBJECT_CLASS, dispose, (object)); +} + +static void +gst_mpeg4vparse_base_init (gpointer klass) +{ + GstElementClass *element_class = GST_ELEMENT_CLASS (klass); + + gst_element_class_add_pad_template (element_class, + gst_static_pad_template_get (&src_template)); + gst_element_class_add_pad_template (element_class, + gst_static_pad_template_get (&sink_template)); + + gst_element_class_set_details (element_class, &mpeg4vparse_details); +} + +static void +gst_mpeg4vparse_class_init (GstMpeg4VParseClass * klass) +{ + GObjectClass *gobject_class; + GstElementClass *gstelement_class; + + gstelement_class = (GstElementClass *) klass; + gobject_class = G_OBJECT_CLASS (klass); + + parent_class = g_type_class_peek_parent (klass); + + gobject_class->dispose = GST_DEBUG_FUNCPTR (gst_mpeg4vparse_dispose); + gstelement_class->change_state = + GST_DEBUG_FUNCPTR (gst_mpeg4vparse_change_state); +} + +static void +gst_mpeg4vparse_init (GstMpeg4VParse * parse, GstMpeg4VParseClass * g_class) +{ + parse->sinkpad = gst_pad_new_from_static_template (&sink_template, "sink"); + gst_pad_set_chain_function (parse->sinkpad, + GST_DEBUG_FUNCPTR (gst_mpeg4vparse_chain)); + gst_pad_set_event_function (parse->sinkpad, + GST_DEBUG_FUNCPTR (gst_mpeg4vparse_sink_event)); + gst_pad_set_setcaps_function (parse->sinkpad, + GST_DEBUG_FUNCPTR (gst_mpeg4vparse_sink_setcaps)); + gst_element_add_pad (GST_ELEMENT (parse), parse->sinkpad); + + parse->srcpad = gst_pad_new_from_static_template (&src_template, "src"); + gst_pad_use_fixed_caps (parse->srcpad); + gst_element_add_pad (GST_ELEMENT (parse), parse->srcpad); + + parse->adapter = gst_adapter_new (); + + gst_mpeg4vparse_cleanup (parse); +} + +static gboolean +plugin_init (GstPlugin * plugin) +{ + GST_DEBUG_CATEGORY_INIT (mpeg4v_parse_debug, "mpeg4videoparse", 0, + "MPEG-4 video parser"); + + if (!gst_element_register (plugin, "mpeg4videoparse", GST_RANK_SECONDARY, + gst_mpeg4vparse_get_type ())) + return FALSE; + + return TRUE; +} + +GST_PLUGIN_DEFINE (GST_VERSION_MAJOR, + GST_VERSION_MINOR, + "mpeg4videoparse", + "MPEG-4 video parser", + plugin_init, VERSION, GST_LICENSE, GST_PACKAGE_NAME, GST_PACKAGE_ORIGIN) |