/* GStreamer * Copyright (c) 2005 Edward Hervey * Copyright (C) 2010 Sebastian Dröge * * 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., 51 Franklin St, Fifth Floor, * Boston, MA 02110-1301, USA. */ /** * SECTION:element-imagefreeze * * The imagefreeze element generates a still frame video stream from * the input. It duplicates the first frame with the framerate requested * by downstream, allows seeking and answers queries. * * * Example launch line * |[ * gst-launch-1.0 -v filesrc location=some.png ! decodebin2 ! imagefreeze ! autovideosink * ]| This pipeline shows a still frame stream of a PNG file. * */ /* This is based on the imagefreeze element from PiTiVi: * http://git.gnome.org/browse/pitivi/tree/pitivi/elements/imagefreeze.py */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include "gstimagefreeze.h" static void gst_image_freeze_finalize (GObject * object); static void gst_image_freeze_reset (GstImageFreeze * self); static GstStateChangeReturn gst_image_freeze_change_state (GstElement * element, GstStateChange transition); static GstFlowReturn gst_image_freeze_sink_chain (GstPad * pad, GstObject * parent, GstBuffer * buffer); static gboolean gst_image_freeze_sink_event (GstPad * pad, GstObject * parent, GstEvent * event); static gboolean gst_image_freeze_sink_setcaps (GstImageFreeze * self, GstCaps * caps); static GstCaps *gst_image_freeze_sink_getcaps (GstImageFreeze * self, GstCaps * filter); static gboolean gst_image_freeze_sink_query (GstPad * pad, GstObject * parent, GstQuery * query); static void gst_image_freeze_src_loop (GstPad * pad); static gboolean gst_image_freeze_src_event (GstPad * pad, GstObject * parent, GstEvent * event); static gboolean gst_image_freeze_src_query (GstPad * pad, GstObject * parent, GstQuery * query); static GstStaticPadTemplate sink_pad_template = GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS, GST_STATIC_CAPS ("video/x-raw")); static GstStaticPadTemplate src_pad_template = GST_STATIC_PAD_TEMPLATE ("src", GST_PAD_SRC, GST_PAD_ALWAYS, GST_STATIC_CAPS ("video/x-raw")); GST_DEBUG_CATEGORY_STATIC (gst_image_freeze_debug); #define GST_CAT_DEFAULT gst_image_freeze_debug #define gst_image_freeze_parent_class parent_class G_DEFINE_TYPE (GstImageFreeze, gst_image_freeze, GST_TYPE_ELEMENT); static void gst_image_freeze_class_init (GstImageFreezeClass * klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); GstElementClass *gstelement_class = GST_ELEMENT_CLASS (klass); gobject_class->finalize = gst_image_freeze_finalize; gstelement_class->change_state = GST_DEBUG_FUNCPTR (gst_image_freeze_change_state); gst_element_class_set_static_metadata (gstelement_class, "Still frame stream generator", "Filter/Video", "Generates a still frame stream from an image", "Sebastian Dröge "); gst_element_class_add_pad_template (gstelement_class, gst_static_pad_template_get (&sink_pad_template)); gst_element_class_add_pad_template (gstelement_class, gst_static_pad_template_get (&src_pad_template)); } static void gst_image_freeze_init (GstImageFreeze * self) { self->sinkpad = gst_pad_new_from_static_template (&sink_pad_template, "sink"); gst_pad_set_chain_function (self->sinkpad, GST_DEBUG_FUNCPTR (gst_image_freeze_sink_chain)); gst_pad_set_event_function (self->sinkpad, GST_DEBUG_FUNCPTR (gst_image_freeze_sink_event)); gst_pad_set_query_function (self->sinkpad, GST_DEBUG_FUNCPTR (gst_image_freeze_sink_query)); GST_PAD_SET_PROXY_ALLOCATION (self->sinkpad); gst_element_add_pad (GST_ELEMENT (self), self->sinkpad); self->srcpad = gst_pad_new_from_static_template (&src_pad_template, "src"); gst_pad_set_event_function (self->srcpad, GST_DEBUG_FUNCPTR (gst_image_freeze_src_event)); gst_pad_set_query_function (self->srcpad, GST_DEBUG_FUNCPTR (gst_image_freeze_src_query)); gst_pad_use_fixed_caps (self->srcpad); gst_element_add_pad (GST_ELEMENT (self), self->srcpad); g_mutex_init (&self->lock); gst_image_freeze_reset (self); } static void gst_image_freeze_finalize (GObject * object) { GstImageFreeze *self = GST_IMAGE_FREEZE (object); gst_image_freeze_reset (self); g_mutex_clear (&self->lock); G_OBJECT_CLASS (parent_class)->finalize (object); } static void gst_image_freeze_reset (GstImageFreeze * self) { GST_DEBUG_OBJECT (self, "Resetting internal state"); g_mutex_lock (&self->lock); gst_buffer_replace (&self->buffer, NULL); gst_segment_init (&self->segment, GST_FORMAT_TIME); self->need_segment = TRUE; self->fps_n = self->fps_d = 0; self->offset = 0; g_mutex_unlock (&self->lock); g_atomic_int_set (&self->seeking, 0); } static gboolean gst_image_freeze_sink_setcaps (GstImageFreeze * self, GstCaps * caps) { gboolean ret = FALSE; GstStructure *s; gint fps_n, fps_d; GstCaps *othercaps, *intersection; guint i, n; GstPad *pad; pad = self->sinkpad; caps = gst_caps_copy (caps); GST_DEBUG_OBJECT (pad, "Setting caps: %" GST_PTR_FORMAT, caps); s = gst_caps_get_structure (caps, 0); /* 1. Remove framerate */ gst_structure_remove_field (s, "framerate"); gst_structure_set (s, "framerate", GST_TYPE_FRACTION_RANGE, 0, 1, G_MAXINT, 1, NULL); /* 2. Intersect with template caps */ othercaps = (GstCaps *) gst_pad_get_pad_template_caps (pad); intersection = gst_caps_intersect (caps, othercaps); GST_DEBUG_OBJECT (pad, "Intersecting: %" GST_PTR_FORMAT, caps); GST_DEBUG_OBJECT (pad, "with: %" GST_PTR_FORMAT, othercaps); GST_DEBUG_OBJECT (pad, "gave: %" GST_PTR_FORMAT, intersection); gst_caps_unref (caps); gst_caps_unref (othercaps); caps = intersection; intersection = othercaps = NULL; /* 3. Intersect with downstream peer caps */ othercaps = gst_pad_peer_query_caps (self->srcpad, caps); GST_DEBUG_OBJECT (pad, "Peer query resulted: %" GST_PTR_FORMAT, othercaps); gst_caps_unref (caps); caps = othercaps; othercaps = NULL; /* 4. For every candidate check if it's accepted downstream * and fixate framerate to nearest 25/1 */ n = gst_caps_get_size (caps); for (i = 0; i < n; i++) { GstCaps *candidate = gst_caps_new_empty (); GstStructure *s = gst_structure_copy (gst_caps_get_structure (caps, i)); gst_caps_append_structure (candidate, s); if (gst_structure_has_field_typed (s, "framerate", GST_TYPE_FRACTION) || gst_structure_fixate_field_nearest_fraction (s, "framerate", 25, 1)) { if (gst_pad_peer_query_accept_caps (self->srcpad, candidate)) { gst_structure_get_fraction (s, "framerate", &fps_n, &fps_d); if (fps_d != 0) { g_mutex_lock (&self->lock); self->fps_n = fps_n; self->fps_d = fps_d; g_mutex_unlock (&self->lock); GST_DEBUG_OBJECT (pad, "Setting caps %" GST_PTR_FORMAT, candidate); gst_pad_set_caps (self->srcpad, candidate); gst_caps_unref (candidate); ret = TRUE; goto done; } else { GST_WARNING_OBJECT (pad, "Invalid caps with framerate %d/%d", fps_n, fps_d); } } } gst_caps_unref (candidate); } done: if (!ret) GST_ERROR_OBJECT (pad, "No usable caps found"); gst_caps_unref (caps); return ret; } /* remove framerate in writable @caps */ static void gst_image_freeze_remove_fps (GstImageFreeze * self, GstCaps * caps) { gint i, n; n = gst_caps_get_size (caps); for (i = 0; i < n; i++) { GstStructure *s = gst_caps_get_structure (caps, i); gst_structure_remove_field (s, "framerate"); gst_structure_set (s, "framerate", GST_TYPE_FRACTION_RANGE, 0, 1, G_MAXINT, 1, NULL); } } static GstCaps * gst_image_freeze_sink_getcaps (GstImageFreeze * self, GstCaps * filter) { GstCaps *ret, *tmp, *templ; GstPad *pad; pad = self->sinkpad; if (gst_pad_has_current_caps (pad)) { ret = gst_pad_get_current_caps (pad); goto done; } if (filter) { filter = gst_caps_copy (filter); gst_image_freeze_remove_fps (self, filter); } templ = gst_pad_get_pad_template_caps (pad); tmp = gst_pad_peer_query_caps (self->srcpad, filter); if (tmp) { GST_LOG_OBJECT (self, "peer caps %" GST_PTR_FORMAT, tmp); ret = gst_caps_intersect (tmp, templ); gst_caps_unref (tmp); } else { GST_LOG_OBJECT (self, "going to copy"); ret = gst_caps_copy (templ); } if (templ) gst_caps_unref (templ); if (filter) gst_caps_unref (filter); ret = gst_caps_make_writable (ret); gst_image_freeze_remove_fps (self, ret); done: GST_LOG_OBJECT (pad, "Returning caps: %" GST_PTR_FORMAT, ret); return ret; } static gboolean gst_image_freeze_sink_query (GstPad * pad, GstObject * parent, GstQuery * query) { GstImageFreeze *self = GST_IMAGE_FREEZE (parent); gboolean ret; GST_LOG_OBJECT (pad, "Handling query of type '%s'", gst_query_type_get_name (GST_QUERY_TYPE (query))); switch (GST_QUERY_TYPE (query)) { case GST_QUERY_CAPS: { GstCaps *caps; gst_query_parse_caps (query, &caps); caps = gst_image_freeze_sink_getcaps (self, caps); gst_query_set_caps_result (query, caps); gst_caps_unref (caps); ret = TRUE; break; } default: ret = gst_pad_query_default (pad, parent, query); } return ret; } static gboolean gst_image_freeze_convert (GstImageFreeze * self, GstFormat src_format, gint64 src_value, GstFormat * dest_format, gint64 * dest_value) { gboolean ret = FALSE; if (src_format == *dest_format) { *dest_value = src_value; return TRUE; } if (src_value == -1) { *dest_value = -1; return TRUE; } switch (src_format) { case GST_FORMAT_DEFAULT:{ switch (*dest_format) { case GST_FORMAT_TIME: g_mutex_lock (&self->lock); if (self->fps_n == 0) *dest_value = -1; else *dest_value = gst_util_uint64_scale (src_value, GST_SECOND * self->fps_d, self->fps_n); g_mutex_unlock (&self->lock); ret = TRUE; break; default: break; } break; } case GST_FORMAT_TIME:{ switch (*dest_format) { case GST_FORMAT_DEFAULT: g_mutex_lock (&self->lock); *dest_value = gst_util_uint64_scale (src_value, self->fps_n, self->fps_d * GST_SECOND); g_mutex_unlock (&self->lock); ret = TRUE; break; default: break; } break; } default: break; } return ret; } static gboolean gst_image_freeze_src_query (GstPad * pad, GstObject * parent, GstQuery * query) { GstImageFreeze *self = GST_IMAGE_FREEZE (parent); gboolean ret = FALSE; GST_LOG_OBJECT (pad, "Handling query of type '%s'", gst_query_type_get_name (GST_QUERY_TYPE (query))); switch (GST_QUERY_TYPE (query)) { case GST_QUERY_CONVERT:{ GstFormat src_format, dest_format; gint64 src_value, dest_value; gst_query_parse_convert (query, &src_format, &src_value, &dest_format, &dest_value); ret = gst_image_freeze_convert (self, src_format, src_value, &dest_format, &dest_value); if (ret) gst_query_set_convert (query, src_format, src_value, dest_format, dest_value); break; } case GST_QUERY_POSITION:{ GstFormat format; gint64 position; gst_query_parse_position (query, &format, NULL); switch (format) { case GST_FORMAT_DEFAULT:{ g_mutex_lock (&self->lock); position = self->offset; g_mutex_unlock (&self->lock); ret = TRUE; break; } case GST_FORMAT_TIME:{ g_mutex_lock (&self->lock); position = self->segment.position; g_mutex_unlock (&self->lock); ret = TRUE; break; } default: break; } if (ret) { gst_query_set_position (query, format, position); GST_DEBUG_OBJECT (pad, "Returning position %" G_GINT64_FORMAT " in format %s", position, gst_format_get_name (format)); } else { GST_DEBUG_OBJECT (pad, "Position query failed"); } break; } case GST_QUERY_DURATION:{ GstFormat format; gint64 duration; gst_query_parse_duration (query, &format, NULL); switch (format) { case GST_FORMAT_TIME:{ g_mutex_lock (&self->lock); duration = self->segment.stop; g_mutex_unlock (&self->lock); ret = TRUE; break; } case GST_FORMAT_DEFAULT:{ g_mutex_lock (&self->lock); duration = self->segment.stop; if (duration != -1) duration = gst_util_uint64_scale (duration, self->fps_n, GST_SECOND * self->fps_d); g_mutex_unlock (&self->lock); ret = TRUE; break; } default: break; } if (ret) { gst_query_set_duration (query, format, duration); GST_DEBUG_OBJECT (pad, "Returning duration %" G_GINT64_FORMAT " in format %s", duration, gst_format_get_name (format)); } else { GST_DEBUG_OBJECT (pad, "Duration query failed"); } break; } case GST_QUERY_SEEKING:{ GstFormat format; gboolean seekable; gst_query_parse_seeking (query, &format, NULL, NULL, NULL); seekable = (format == GST_FORMAT_TIME || format == GST_FORMAT_DEFAULT); gst_query_set_seeking (query, format, seekable, (seekable ? 0 : -1), -1); ret = TRUE; break; } default: ret = FALSE; break; } return ret; } static gboolean gst_image_freeze_sink_event (GstPad * pad, GstObject * parent, GstEvent * event) { GstImageFreeze *self = GST_IMAGE_FREEZE (parent); gboolean ret; GST_LOG_OBJECT (pad, "Got %s event", GST_EVENT_TYPE_NAME (event)); switch (GST_EVENT_TYPE (event)) { case GST_EVENT_CAPS: { GstCaps *caps; gst_event_parse_caps (event, &caps); gst_image_freeze_sink_setcaps (self, caps); gst_event_unref (event); ret = TRUE; break; } case GST_EVENT_EOS: if (!self->buffer) { /* if we receive EOS before a buffer arrives, then let it pass */ GST_DEBUG_OBJECT (self, "EOS without input buffer, passing on"); ret = gst_pad_push_event (self->srcpad, event); break; } /* fall-through */ case GST_EVENT_SEGMENT: GST_DEBUG_OBJECT (pad, "Dropping event"); gst_event_unref (event); ret = TRUE; break; case GST_EVENT_FLUSH_START: gst_image_freeze_reset (self); /* fall through */ default: ret = gst_pad_push_event (self->srcpad, event); break; } return ret; } static gboolean gst_image_freeze_src_event (GstPad * pad, GstObject * parent, GstEvent * event) { GstImageFreeze *self = GST_IMAGE_FREEZE (parent); gboolean ret; GST_LOG_OBJECT (pad, "Got %s event", GST_EVENT_TYPE_NAME (event)); switch (GST_EVENT_TYPE (event)) { case GST_EVENT_NAVIGATION: case GST_EVENT_QOS: case GST_EVENT_LATENCY: case GST_EVENT_STEP: GST_DEBUG_OBJECT (pad, "Dropping event"); gst_event_unref (event); ret = TRUE; break; case GST_EVENT_SEEK:{ gdouble rate; GstFormat format; GstSeekFlags flags; GstSeekType start_type, stop_type; gint64 start, stop; gint64 last_stop; gboolean start_task; gboolean flush; gst_event_parse_seek (event, &rate, &format, &flags, &start_type, &start, &stop_type, &stop); gst_event_unref (event); flush = ! !(flags & GST_SEEK_FLAG_FLUSH); if (format != GST_FORMAT_TIME && format != GST_FORMAT_DEFAULT) { GST_ERROR_OBJECT (pad, "Seek in invalid format: %s", gst_format_get_name (format)); ret = FALSE; break; } if (format == GST_FORMAT_DEFAULT) { format = GST_FORMAT_TIME; if (!gst_image_freeze_convert (self, GST_FORMAT_DEFAULT, start, &format, &start) || !gst_image_freeze_convert (self, GST_FORMAT_DEFAULT, stop, &format, &stop) || start == -1 || stop == -1) { GST_ERROR_OBJECT (pad, "Failed to convert seek from DEFAULT format into TIME format"); ret = FALSE; break; } } if (flush) { GstEvent *e; g_atomic_int_set (&self->seeking, 1); e = gst_event_new_flush_start (); gst_pad_push_event (self->srcpad, e); } else { gst_pad_pause_task (self->srcpad); } GST_PAD_STREAM_LOCK (self->srcpad); g_mutex_lock (&self->lock); gst_segment_do_seek (&self->segment, rate, format, flags, start_type, start, stop_type, stop, NULL); self->need_segment = TRUE; last_stop = self->segment.position; start_task = self->buffer != NULL; g_mutex_unlock (&self->lock); if (flush) { GstEvent *e; e = gst_event_new_flush_stop (TRUE); gst_pad_push_event (self->srcpad, e); g_atomic_int_set (&self->seeking, 0); } if (flags & GST_SEEK_FLAG_SEGMENT) { GstMessage *m; m = gst_message_new_segment_start (GST_OBJECT (self), format, last_stop); gst_element_post_message (GST_ELEMENT (self), m); } GST_PAD_STREAM_UNLOCK (self->srcpad); GST_DEBUG_OBJECT (pad, "Seek successful"); if (start_task) { g_mutex_lock (&self->lock); if (self->buffer != NULL) gst_pad_start_task (self->srcpad, (GstTaskFunction) gst_image_freeze_src_loop, self->srcpad, NULL); g_mutex_unlock (&self->lock); } ret = TRUE; break; } case GST_EVENT_FLUSH_START: gst_image_freeze_reset (self); /* fall through */ default: ret = gst_pad_push_event (self->sinkpad, event); break; } return ret; } static GstFlowReturn gst_image_freeze_sink_chain (GstPad * pad, GstObject * parent, GstBuffer * buffer) { GstImageFreeze *self = GST_IMAGE_FREEZE (parent); g_return_val_if_fail (self->buffer == NULL, GST_FLOW_ERROR); g_mutex_lock (&self->lock); self->buffer = buffer; gst_pad_start_task (self->srcpad, (GstTaskFunction) gst_image_freeze_src_loop, self->srcpad, NULL); g_mutex_unlock (&self->lock); return GST_FLOW_EOS; } static void gst_image_freeze_src_loop (GstPad * pad) { GstImageFreeze *self = GST_IMAGE_FREEZE (GST_PAD_PARENT (pad)); GstBuffer *buffer; guint64 offset; GstClockTime timestamp, timestamp_end; guint64 cstart, cstop; gboolean in_seg, eos; GstFlowReturn flow_ret = GST_FLOW_OK; g_mutex_lock (&self->lock); if (!gst_pad_has_current_caps (self->srcpad)) { GST_ERROR_OBJECT (pad, "Not negotiated yet"); flow_ret = GST_FLOW_NOT_NEGOTIATED; g_mutex_unlock (&self->lock); goto pause_task; } if (!self->buffer) { GST_ERROR_OBJECT (pad, "Have no buffer yet"); flow_ret = GST_FLOW_ERROR; g_mutex_unlock (&self->lock); goto pause_task; } buffer = gst_buffer_copy (self->buffer); g_mutex_unlock (&self->lock); if (self->need_segment) { GstEvent *e; GST_DEBUG_OBJECT (pad, "Pushing SEGMENT event: %" GST_SEGMENT_FORMAT, &self->segment); e = gst_event_new_segment (&self->segment); g_mutex_lock (&self->lock); if (self->segment.rate >= 0) { self->offset = gst_util_uint64_scale (self->segment.start, self->fps_n, self->fps_d * GST_SECOND); } else { self->offset = gst_util_uint64_scale (self->segment.stop, self->fps_n, self->fps_d * GST_SECOND); } g_mutex_unlock (&self->lock); self->need_segment = FALSE; gst_pad_push_event (self->srcpad, e); } g_mutex_lock (&self->lock); offset = self->offset; if (self->fps_n != 0) { timestamp = gst_util_uint64_scale (offset, self->fps_d * GST_SECOND, self->fps_n); timestamp_end = gst_util_uint64_scale (offset + 1, self->fps_d * GST_SECOND, self->fps_n); } else { timestamp = self->segment.start; timestamp_end = GST_CLOCK_TIME_NONE; } eos = (self->fps_n == 0 && offset > 0) || (self->segment.rate >= 0 && self->segment.stop != -1 && timestamp > self->segment.stop) || (self->segment.rate < 0 && offset == 0) || (self->segment.rate < 0 && self->segment.start != -1 && timestamp_end < self->segment.start); if (self->fps_n == 0 && offset > 0) in_seg = FALSE; else in_seg = gst_segment_clip (&self->segment, GST_FORMAT_TIME, timestamp, timestamp_end, &cstart, &cstop); if (in_seg) { self->segment.position = cstart; if (self->segment.rate >= 0) self->segment.position = cstop; } if (self->segment.rate >= 0) self->offset++; else self->offset--; g_mutex_unlock (&self->lock); GST_DEBUG_OBJECT (pad, "Handling buffer with timestamp %" GST_TIME_FORMAT, GST_TIME_ARGS (timestamp)); if (in_seg) { GST_BUFFER_DTS (buffer) = GST_CLOCK_TIME_NONE; GST_BUFFER_PTS (buffer) = cstart; GST_BUFFER_DURATION (buffer) = cstop - cstart; GST_BUFFER_OFFSET (buffer) = offset; GST_BUFFER_OFFSET_END (buffer) = offset + 1; flow_ret = gst_pad_push (self->srcpad, buffer); GST_DEBUG_OBJECT (pad, "Pushing buffer resulted in %s", gst_flow_get_name (flow_ret)); if (flow_ret != GST_FLOW_OK) goto pause_task; } else { gst_buffer_unref (buffer); } if (eos) { flow_ret = GST_FLOW_EOS; goto pause_task; } return; pause_task: { const gchar *reason = gst_flow_get_name (flow_ret); GST_LOG_OBJECT (self, "pausing task, reason %s", reason); gst_pad_pause_task (pad); if (flow_ret == GST_FLOW_EOS) { if ((self->segment.flags & GST_SEEK_FLAG_SEGMENT)) { GstMessage *m; GstEvent *e; GST_DEBUG_OBJECT (pad, "Sending segment done at end of segment"); if (self->segment.rate >= 0) { m = gst_message_new_segment_done (GST_OBJECT_CAST (self), GST_FORMAT_TIME, self->segment.stop); e = gst_event_new_segment_done (GST_FORMAT_TIME, self->segment.stop); } else { m = gst_message_new_segment_done (GST_OBJECT_CAST (self), GST_FORMAT_TIME, self->segment.start); e = gst_event_new_segment_done (GST_FORMAT_TIME, self->segment.start); } gst_element_post_message (GST_ELEMENT_CAST (self), m); gst_pad_push_event (self->srcpad, e); } else { GST_DEBUG_OBJECT (pad, "Sending EOS at end of segment"); gst_pad_push_event (self->srcpad, gst_event_new_eos ()); } } else if (flow_ret == GST_FLOW_NOT_LINKED || flow_ret < GST_FLOW_EOS) { GST_ELEMENT_ERROR (self, STREAM, FAILED, ("Internal data stream error."), ("stream stopped, reason %s", reason)); gst_pad_push_event (self->srcpad, gst_event_new_eos ()); } return; } } static GstStateChangeReturn gst_image_freeze_change_state (GstElement * element, GstStateChange transition) { GstImageFreeze *self = GST_IMAGE_FREEZE (element); GstStateChangeReturn ret = GST_STATE_CHANGE_SUCCESS; switch (transition) { case GST_STATE_CHANGE_READY_TO_PAUSED: gst_image_freeze_reset (self); break; case GST_STATE_CHANGE_PAUSED_TO_READY: gst_pad_stop_task (self->srcpad); gst_image_freeze_reset (self); break; default: break; } if (GST_ELEMENT_CLASS (parent_class)->change_state) ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition); switch (transition) { default: break; } return ret; } static gboolean plugin_init (GstPlugin * plugin) { GST_DEBUG_CATEGORY_INIT (gst_image_freeze_debug, "imagefreeze", 0, "imagefreeze element"); if (!gst_element_register (plugin, "imagefreeze", GST_RANK_NONE, GST_TYPE_IMAGE_FREEZE)) return FALSE; return TRUE; } GST_PLUGIN_DEFINE (GST_VERSION_MAJOR, GST_VERSION_MINOR, imagefreeze, "Still frame stream generator", plugin_init, VERSION, "LGPL", GST_PACKAGE_NAME, GST_PACKAGE_ORIGIN)