/* GStreamer * Copyright (C) 1999,2000 Erik Walthinsen * 2000 Wim Taymans * 2005 Wim Taymans * * gstidentity.c: * * 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. */ #include #ifdef HAVE_CONFIG_H # include "config.h" #endif #include "../gst-i18n-lib.h" #include "gstidentity.h" #include static GstStaticPadTemplate sinktemplate = GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS, GST_STATIC_CAPS_ANY); static GstStaticPadTemplate srctemplate = GST_STATIC_PAD_TEMPLATE ("src", GST_PAD_SRC, GST_PAD_ALWAYS, GST_STATIC_CAPS_ANY); GST_DEBUG_CATEGORY_STATIC (gst_identity_debug); #define GST_CAT_DEFAULT gst_identity_debug GstElementDetails gst_identity_details = GST_ELEMENT_DETAILS ("Identity", "Generic", "Pass data without modification", "Erik Walthinsen "); /* Identity signals and args */ enum { SIGNAL_HANDOFF, /* FILL ME */ LAST_SIGNAL }; #define DEFAULT_SLEEP_TIME 0 #define DEFAULT_DUPLICATE 1 #define DEFAULT_ERROR_AFTER -1 #define DEFAULT_DROP_PROBABILITY 0.0 #define DEFAULT_DATARATE 0 #define DEFAULT_SILENT FALSE #define DEFAULT_DUMP FALSE #define DEFAULT_SYNC FALSE #define DEFAULT_CHECK_PERFECT FALSE enum { PROP_0, PROP_SLEEP_TIME, PROP_DUPLICATE, PROP_ERROR_AFTER, PROP_DROP_PROBABILITY, PROP_DATARATE, PROP_SILENT, PROP_LAST_MESSAGE, PROP_DUMP, PROP_SYNC, PROP_CHECK_PERFECT }; #define _do_init(bla) \ GST_DEBUG_CATEGORY_INIT (gst_identity_debug, "identity", 0, "identity element"); GST_BOILERPLATE_FULL (GstIdentity, gst_identity, GstBaseTransform, GST_TYPE_BASE_TRANSFORM, _do_init); static void gst_identity_finalize (GObject * object); static void gst_identity_set_property (GObject * object, guint prop_id, const GValue * value, GParamSpec * pspec); static void gst_identity_get_property (GObject * object, guint prop_id, GValue * value, GParamSpec * pspec); static gboolean gst_identity_event (GstBaseTransform * trans, GstEvent * event); static GstFlowReturn gst_identity_transform (GstBaseTransform * trans, GstBuffer * inbuf, GstBuffer ** outbuf); static gboolean gst_identity_start (GstBaseTransform * trans); static gboolean gst_identity_stop (GstBaseTransform * trans); static guint gst_identity_signals[LAST_SIGNAL] = { 0 }; static void gst_identity_base_init (gpointer g_class) { GstElementClass *gstelement_class = GST_ELEMENT_CLASS (g_class); gst_element_class_add_pad_template (gstelement_class, gst_static_pad_template_get (&srctemplate)); gst_element_class_add_pad_template (gstelement_class, gst_static_pad_template_get (&sinktemplate)); gst_element_class_set_details (gstelement_class, &gst_identity_details); } static void gst_identity_finalize (GObject * object) { GstIdentity *identity; identity = GST_IDENTITY (object); g_free (identity->last_message); G_OBJECT_CLASS (parent_class)->finalize (object); } static void gst_identity_class_init (GstIdentityClass * klass) { GObjectClass *gobject_class; GstElementClass *gstelement_class; GstBaseTransformClass *gstbasetrans_class; gobject_class = G_OBJECT_CLASS (klass); gstelement_class = GST_ELEMENT_CLASS (klass); gstbasetrans_class = GST_BASE_TRANSFORM_CLASS (klass); gobject_class->set_property = GST_DEBUG_FUNCPTR (gst_identity_set_property); gobject_class->get_property = GST_DEBUG_FUNCPTR (gst_identity_get_property); g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_SLEEP_TIME, g_param_spec_uint ("sleep-time", "Sleep time", "Microseconds to sleep between processing", 0, G_MAXUINT, DEFAULT_SLEEP_TIME, G_PARAM_READWRITE)); g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_DUPLICATE, g_param_spec_uint ("duplicate", "Duplicate Buffers", "Push the buffers N times", 0, G_MAXUINT, DEFAULT_DUPLICATE, G_PARAM_READWRITE)); g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_ERROR_AFTER, g_param_spec_int ("error_after", "Error After", "Error after N buffers", G_MININT, G_MAXINT, DEFAULT_ERROR_AFTER, G_PARAM_READWRITE)); g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_DROP_PROBABILITY, g_param_spec_float ("drop_probability", "Drop Probability", "The Probability a buffer is dropped", 0.0, 1.0, DEFAULT_DROP_PROBABILITY, G_PARAM_READWRITE)); g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_DATARATE, g_param_spec_int ("datarate", "Datarate", "(Re)timestamps buffers with number of bytes per second (0 = inactive)", 0, G_MAXINT, DEFAULT_DATARATE, G_PARAM_READWRITE)); g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_SILENT, g_param_spec_boolean ("silent", "silent", "silent", DEFAULT_SILENT, G_PARAM_READWRITE)); g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_LAST_MESSAGE, g_param_spec_string ("last-message", "last-message", "last-message", NULL, G_PARAM_READABLE)); g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_DUMP, g_param_spec_boolean ("dump", "Dump", "Dump buffer contents", DEFAULT_DUMP, G_PARAM_READWRITE)); g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_SYNC, g_param_spec_boolean ("sync", "Synchronize", "Synchronize to pipeline clock", DEFAULT_SYNC, G_PARAM_READWRITE)); g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_CHECK_PERFECT, g_param_spec_boolean ("check-perfect", "Check For Perfect Stream", "Verify that the stream is time- and data-contiguous", DEFAULT_CHECK_PERFECT, G_PARAM_READWRITE)); gst_identity_signals[SIGNAL_HANDOFF] = g_signal_new ("handoff", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GstIdentityClass, handoff), NULL, NULL, gst_marshal_VOID__OBJECT, G_TYPE_NONE, 1, G_TYPE_POINTER); gobject_class->finalize = GST_DEBUG_FUNCPTR (gst_identity_finalize); gstbasetrans_class->event = GST_DEBUG_FUNCPTR (gst_identity_event); gstbasetrans_class->transform = GST_DEBUG_FUNCPTR (gst_identity_transform); gstbasetrans_class->start = GST_DEBUG_FUNCPTR (gst_identity_start); gstbasetrans_class->stop = GST_DEBUG_FUNCPTR (gst_identity_stop); } static void gst_identity_init (GstIdentity * identity) { identity->sleep_time = DEFAULT_SLEEP_TIME; identity->duplicate = DEFAULT_DUPLICATE; identity->error_after = DEFAULT_ERROR_AFTER; identity->drop_probability = DEFAULT_DROP_PROBABILITY; identity->datarate = DEFAULT_DATARATE; identity->silent = DEFAULT_SILENT; identity->sync = DEFAULT_SYNC; identity->check_perfect = DEFAULT_CHECK_PERFECT; identity->dump = DEFAULT_DUMP; identity->last_message = NULL; } static gboolean gst_identity_event (GstBaseTransform * trans, GstEvent * event) { GstIdentity *identity; identity = GST_IDENTITY (trans); if (!identity->silent) { g_free (identity->last_message); identity->last_message = g_strdup_printf ("chain ******* (%s:%s)E (type: %d) %p", GST_DEBUG_PAD_NAME (trans->sinkpad), GST_EVENT_TYPE (event), event); g_object_notify (G_OBJECT (identity), "last_message"); } return TRUE; } static void gst_identity_check_perfect (GstIdentity * identity, GstBuffer * buf) { GstClockTime timestamp; timestamp = GST_BUFFER_TIMESTAMP (buf); /* see if we need to do perfect stream checking */ /* invalid timestamp drops us out of check. FIXME: maybe warn ? */ if (timestamp != GST_CLOCK_TIME_NONE) { /* check if we had a previous buffer to compare to */ if (identity->prev_timestamp != GST_CLOCK_TIME_NONE) { guint64 offset; if (identity->prev_timestamp + identity->prev_duration != timestamp) { GST_WARNING_OBJECT (identity, "Buffer not time-contiguous with previous one: " "prev ts %" GST_TIME_FORMAT ", prev dur %" GST_TIME_FORMAT ", new ts %" GST_TIME_FORMAT, GST_TIME_ARGS (identity->prev_timestamp), GST_TIME_ARGS (identity->prev_duration), GST_TIME_ARGS (timestamp)); } offset = GST_BUFFER_OFFSET (buf); if (identity->prev_offset_end != offset) { GST_WARNING_OBJECT (identity, "Buffer not data-contiguous with previous one: " "prev offset_end %" G_GINT64_FORMAT ", new offset %" G_GINT64_FORMAT, identity->prev_offset_end, offset); } } /* update prev values */ identity->prev_timestamp = timestamp; identity->prev_duration = GST_BUFFER_DURATION (buf); identity->prev_offset_end = GST_BUFFER_OFFSET_END (buf); } } static GstFlowReturn gst_identity_transform (GstBaseTransform * trans, GstBuffer * inbuf, GstBuffer ** outbuf) { GstFlowReturn ret = GST_FLOW_OK; GstIdentity *identity = GST_IDENTITY (trans); guint i; if (identity->check_perfect) gst_identity_check_perfect (identity, inbuf); if (identity->error_after >= 0) { identity->error_after--; if (identity->error_after == 0) { GST_ELEMENT_ERROR (identity, CORE, FAILED, (_("Failed after iterations as requested.")), (NULL)); return GST_FLOW_ERROR; } } if (identity->drop_probability > 0.0) { if ((gfloat) (1.0 * rand () / (RAND_MAX)) < identity->drop_probability) { g_free (identity->last_message); identity->last_message = g_strdup_printf ("dropping ******* (%s:%s)i (%d bytes, timestamp: %" GST_TIME_FORMAT ", duration: %" GST_TIME_FORMAT ", offset: %" G_GINT64_FORMAT ", offset_end: % " G_GINT64_FORMAT ", flags: %d) %p", GST_DEBUG_PAD_NAME (trans->sinkpad), GST_BUFFER_SIZE (inbuf), GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (inbuf)), GST_TIME_ARGS (GST_BUFFER_DURATION (inbuf)), GST_BUFFER_OFFSET (inbuf), GST_BUFFER_OFFSET_END (inbuf), GST_BUFFER_FLAGS (inbuf), inbuf); g_object_notify (G_OBJECT (identity), "last-message"); return GST_FLOW_OK; } } if (identity->dump) { gst_util_dump_mem (GST_BUFFER_DATA (inbuf), GST_BUFFER_SIZE (inbuf)); } for (i = identity->duplicate; i; i--) { GstClockTime time; if (!identity->silent) { g_free (identity->last_message); identity->last_message = g_strdup_printf ("chain ******* (%s:%s)i (%d bytes, timestamp: %" GST_TIME_FORMAT ", duration: %" GST_TIME_FORMAT ", offset: %" G_GINT64_FORMAT ", offset_end: % " G_GINT64_FORMAT ", flags: %d) %p", GST_DEBUG_PAD_NAME (trans->sinkpad), GST_BUFFER_SIZE (inbuf), GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (inbuf)), GST_TIME_ARGS (GST_BUFFER_DURATION (inbuf)), GST_BUFFER_OFFSET (inbuf), GST_BUFFER_OFFSET_END (inbuf), GST_BUFFER_FLAGS (inbuf), inbuf); g_object_notify (G_OBJECT (identity), "last-message"); } time = GST_BUFFER_TIMESTAMP (inbuf); if (identity->datarate > 0) { time = identity->offset * GST_SECOND / identity->datarate; GST_BUFFER_TIMESTAMP (inbuf) = time; GST_BUFFER_DURATION (inbuf) = GST_BUFFER_SIZE (inbuf) * GST_SECOND / identity->datarate; } g_signal_emit (G_OBJECT (identity), gst_identity_signals[SIGNAL_HANDOFF], 0, inbuf); if (i > 1) gst_buffer_ref (inbuf); if (identity->sync) { if (GST_ELEMENT (identity)->clock) { /* gst_element_wait (GST_ELEMENT (identity), time); */ } } identity->offset += GST_BUFFER_SIZE (inbuf); if (identity->sleep_time) g_usleep (identity->sleep_time); gst_buffer_ref (inbuf); *outbuf = inbuf; } return ret; } static void gst_identity_set_property (GObject * object, guint prop_id, const GValue * value, GParamSpec * pspec) { GstIdentity *identity; identity = GST_IDENTITY (object); switch (prop_id) { case PROP_SLEEP_TIME: identity->sleep_time = g_value_get_uint (value); break; case PROP_SILENT: identity->silent = g_value_get_boolean (value); break; case PROP_DUPLICATE: identity->duplicate = g_value_get_uint (value); break; case PROP_DUMP: identity->dump = g_value_get_boolean (value); break; case PROP_ERROR_AFTER: identity->error_after = g_value_get_int (value); break; case PROP_DROP_PROBABILITY: identity->drop_probability = g_value_get_float (value); break; case PROP_DATARATE: identity->datarate = g_value_get_int (value); break; case PROP_SYNC: identity->sync = g_value_get_boolean (value); break; case PROP_CHECK_PERFECT: identity->check_perfect = g_value_get_boolean (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void gst_identity_get_property (GObject * object, guint prop_id, GValue * value, GParamSpec * pspec) { GstIdentity *identity; identity = GST_IDENTITY (object); switch (prop_id) { case PROP_SLEEP_TIME: g_value_set_uint (value, identity->sleep_time); break; case PROP_DUPLICATE: g_value_set_uint (value, identity->duplicate); break; case PROP_ERROR_AFTER: g_value_set_int (value, identity->error_after); break; case PROP_DROP_PROBABILITY: g_value_set_float (value, identity->drop_probability); break; case PROP_DATARATE: g_value_set_int (value, identity->datarate); break; case PROP_SILENT: g_value_set_boolean (value, identity->silent); break; case PROP_DUMP: g_value_set_boolean (value, identity->dump); break; case PROP_LAST_MESSAGE: g_value_set_string (value, identity->last_message); break; case PROP_SYNC: g_value_set_boolean (value, identity->sync); break; case PROP_CHECK_PERFECT: g_value_set_boolean (value, identity->check_perfect); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static gboolean gst_identity_start (GstBaseTransform * trans) { GstIdentity *identity; identity = GST_IDENTITY (trans); identity->offset = 0; identity->prev_timestamp = GST_CLOCK_TIME_NONE; identity->prev_duration = GST_CLOCK_TIME_NONE; identity->prev_offset_end = -1; return TRUE; } static gboolean gst_identity_stop (GstBaseTransform * trans) { GstIdentity *identity; identity = GST_IDENTITY (trans); g_free (identity->last_message); identity->last_message = NULL; return TRUE; }