From dd72f88a8c56126a0b74e98b35151c6d124b9ade Mon Sep 17 00:00:00 2001 From: Wim Taymans Date: Tue, 14 Aug 2007 17:38:05 +0000 Subject: [PATCH] gst-libs/gst/app/gstappsink.*: Make love to appsink. Original commit message from CVS: * gst-libs/gst/app/gstappsink.c: (gst_app_sink_base_init), (gst_app_sink_class_init), (gst_app_sink_dispose), (gst_app_sink_flush_unlocked), (gst_app_sink_start), (gst_app_sink_stop), (gst_app_sink_event), (gst_app_sink_preroll), (gst_app_sink_render), (gst_app_sink_get_caps), (gst_app_sink_set_caps), (gst_app_sink_end_of_stream), (gst_app_sink_pull_preroll), (gst_app_sink_pull_buffer): * gst-libs/gst/app/gstappsink.h: Make love to appsink. Make it support pulling of the preroll buffer. Add docs and debug statements. Fix some races wrt to EOS handling and stopping. Implement getcaps. Implement FLUSHING. API: gst_app_sink_pull_preroll() --- gst-libs/gst/app/gstappsink.c | 236 +++++++++++++++++++++++++++++++++++++++--- gst-libs/gst/app/gstappsink.h | 11 +- 2 files changed, 229 insertions(+), 18 deletions(-) diff --git a/gst-libs/gst/app/gstappsink.c b/gst-libs/gst/app/gstappsink.c index 0d94a62..e7baf63 100644 --- a/gst-libs/gst/app/gstappsink.c +++ b/gst-libs/gst/app/gstappsink.c @@ -34,9 +34,9 @@ GST_DEBUG_CATEGORY (app_sink_debug); static const GstElementDetails app_sink_details = GST_ELEMENT_DETAILS ("AppSink", - "FIXME", - "FIXME", - "autogenerated by makefilter"); + "Generic/Sink", + "Allow the application to get access to raw buffer", + "David Schleef , Wim Taymans start = gst_app_sink_start; basesink_class->stop = gst_app_sink_stop; basesink_class->event = gst_app_sink_event; + basesink_class->render = gst_app_sink_preroll; basesink_class->render = gst_app_sink_render; + basesink_class->get_caps = gst_app_sink_get_caps; } static void @@ -103,6 +105,10 @@ gst_app_sink_dispose (GObject * obj) gst_caps_unref (appsink->caps); appsink->caps = NULL; } + if (appsink->preroll) { + gst_buffer_unref (appsink->preroll); + appsink->preroll = NULL; + } if (appsink->mutex) { g_mutex_free (appsink->mutex); appsink->mutex = NULL; @@ -112,6 +118,7 @@ gst_app_sink_dispose (GObject * obj) appsink->cond = NULL; } if (appsink->queue) { + g_queue_foreach (appsink->queue, (GFunc) gst_mini_object_unref, NULL); g_queue_free (appsink->queue); appsink->queue = NULL; } @@ -157,12 +164,27 @@ gst_app_sink_get_property (GObject * object, guint prop_id, GValue * value, GST_OBJECT_UNLOCK (appsink); } +static void +gst_app_sink_flush_unlocked (GstAppSink * appsink) +{ + GST_DEBUG_OBJECT (appsink, "flushing appsink"); + appsink->end_of_stream = FALSE; + gst_buffer_replace (&appsink->preroll, NULL); + g_queue_foreach (appsink->queue, (GFunc) gst_mini_object_unref, NULL); + g_queue_clear (appsink->queue); + g_cond_signal (appsink->cond); +} + static gboolean gst_app_sink_start (GstBaseSink * psink) { GstAppSink *appsink = GST_APP_SINK (psink); + g_mutex_lock (appsink->mutex); appsink->end_of_stream = FALSE; + appsink->started = TRUE; + GST_DEBUG_OBJECT (appsink, "starting"); + g_mutex_unlock (appsink->mutex); return TRUE; } @@ -170,7 +192,13 @@ gst_app_sink_start (GstBaseSink * psink) static gboolean gst_app_sink_stop (GstBaseSink * psink) { - //GstAppSink *appsink = GST_APP_SINK(psink); + GstAppSink *appsink = GST_APP_SINK (psink); + + g_mutex_lock (appsink->mutex); + GST_DEBUG_OBJECT (appsink, "stopping"); + appsink->started = FALSE; + gst_app_sink_flush_unlocked (appsink); + g_mutex_unlock (appsink->mutex); return TRUE; } @@ -182,14 +210,38 @@ gst_app_sink_event (GstBaseSink * sink, GstEvent * event) switch (event->type) { case GST_EVENT_EOS: + g_mutex_lock (appsink->mutex); + GST_DEBUG_OBJECT (appsink, "receiving EOS"); appsink->end_of_stream = TRUE; + g_cond_signal (appsink->cond); + g_mutex_unlock (appsink->mutex); + break; + case GST_EVENT_FLUSH_START: + break; + case GST_EVENT_FLUSH_STOP: + g_mutex_lock (appsink->mutex); + GST_DEBUG_OBJECT (appsink, "received FLUSH_STOP"); + gst_app_sink_flush_unlocked (appsink); + g_mutex_unlock (appsink->mutex); break; default: break; } + return TRUE; +} - gst_object_unref (sink); - return FALSE; +static GstFlowReturn +gst_app_sink_preroll (GstBaseSink * psink, GstBuffer * buffer) +{ + GstAppSink *appsink = GST_APP_SINK (psink); + + g_mutex_lock (appsink->mutex); + GST_DEBUG_OBJECT (appsink, "setting preroll buffer %p", buffer); + gst_buffer_replace (&appsink->preroll, buffer); + g_cond_signal (appsink->cond); + g_mutex_unlock (appsink->mutex); + + return GST_FLOW_OK; } static GstFlowReturn @@ -198,6 +250,7 @@ gst_app_sink_render (GstBaseSink * psink, GstBuffer * buffer) GstAppSink *appsink = GST_APP_SINK (psink); g_mutex_lock (appsink->mutex); + GST_DEBUG_OBJECT (appsink, "pushing render buffer %p on queue", buffer); g_queue_push_tail (appsink->queue, gst_buffer_ref (buffer)); g_cond_signal (appsink->cond); g_mutex_unlock (appsink->mutex); @@ -205,18 +258,33 @@ gst_app_sink_render (GstBaseSink * psink, GstBuffer * buffer) return GST_FLOW_OK; } +static GstCaps * +gst_app_sink_get_caps (GstBaseSink * psink) +{ + GstCaps *caps; + + GstAppSink *appsink = GST_APP_SINK (psink); + GST_OBJECT_LOCK (appsink); + if ((caps = appsink->caps)) + gst_caps_ref (caps); + GST_DEBUG_OBJECT (appsink, "got caps " GST_PTR_FORMAT, caps); + GST_OBJECT_UNLOCK (appsink); + return caps; +} /* external API */ /** * gst_app_sink_set_caps: - * @appsink: - * @caps: + * @appsink: a #GstAppSink + * @caps: caps to set * * Set the capabilities on the appsink element. This function takes - * ownership of the caps structure. + * a ref of the caps structure. After calling this method, the sink will only + * accept caps that match @caps. If @caps is non-fixed, you must check the caps + * on the buffers to get the actual used caps. */ void gst_app_sink_set_caps (GstAppSink * appsink, GstCaps * caps) @@ -224,9 +292,24 @@ gst_app_sink_set_caps (GstAppSink * appsink, GstCaps * caps) g_return_if_fail (appsink != NULL); g_return_if_fail (GST_IS_APP_SINK (appsink)); + GST_OBJECT_LOCK (appsink); + GST_DEBUG_OBJECT (appsink, "setting caps to " GST_PTR_FORMAT, caps); gst_caps_replace (&appsink->caps, caps); + GST_OBJECT_UNLOCK (appsink); } +/** + * gst_app_sink_end_of_stream: + * @appsink: a #GstAppSink + * + * Check if @appsink is EOS, which is when no more buffers can be pulled because + * an EOS event was received. + * + * This function also returns %TRUE when the appsink is not in the PAUSED or + * PLAYING state. + * + * Returns: %TRUE if no more buffers can be pulled and the appsink is EOS. + */ gboolean gst_app_sink_end_of_stream (GstAppSink * appsink) { @@ -236,16 +319,115 @@ gst_app_sink_end_of_stream (GstAppSink * appsink) g_return_val_if_fail (GST_IS_APP_SINK (appsink), FALSE); g_mutex_lock (appsink->mutex); + if (!appsink->started) + goto not_started; + if (appsink->end_of_stream && g_queue_is_empty (appsink->queue)) { + GST_DEBUG_OBJECT (appsink, "we are EOS and the queue is empty"); ret = TRUE; } else { + GST_DEBUG_OBJECT (appsink, "we are not yet EOS"); ret = FALSE; } g_mutex_unlock (appsink->mutex); return ret; + +not_started: + { + GST_DEBUG_OBJECT (appsink, "we are stopped, return TRUE"); + g_mutex_unlock (appsink->mutex); + return TRUE; + } } +/** + * gst_app_sink_pull_preroll: + * @appsink: a #GstAppSink + * + * Get the last preroll buffer in @appsink. This was the buffer that caused the + * appsink to preroll in the PAUSED state. This buffer can be pulled many times + * and remains available to the application even after EOS. + * + * This function is typically used when dealing with a pipeline in the PAUSED + * state. Calling this function after doing a seek will give the buffer right + * after the seek position. + * + * Note that the preroll buffer will also be returned as the first buffer + * when calling gst_app_sink_pull_buffer(). + * + * If an EOS event was received before any buffers, this function also returns + * %NULL. + * + * This function blocks until a preroll buffer or EOS is received or the appsink + * element is set to the READY/NULL state. + * + * Returns: a #GstBuffer or NULL when the appsink is stopped or EOS. + */ +GstBuffer * +gst_app_sink_pull_preroll (GstAppSink * appsink) +{ + GstBuffer *buf = NULL; + + g_return_val_if_fail (appsink != NULL, NULL); + g_return_val_if_fail (GST_IS_APP_SINK (appsink), NULL); + + g_mutex_lock (appsink->mutex); + + while (TRUE) { + GST_DEBUG_OBJECT (appsink, "trying to grab a buffer"); + if (!appsink->started) + goto not_started; + + if (appsink->preroll != NULL) + break; + + if (appsink->end_of_stream) + goto eos; + + /* nothing to return, wait */ + GST_DEBUG_OBJECT (appsink, "waiting for the preroll buffer"); + g_cond_wait (appsink->cond, appsink->mutex); + } + buf = gst_buffer_ref (appsink->preroll); + GST_DEBUG_OBJECT (appsink, "we have the preroll buffer %p", buf); + g_mutex_unlock (appsink->mutex); + + return buf; + + /* special conditions */ +eos: + { + GST_DEBUG_OBJECT (appsink, "we are EOS, return NULL"); + g_mutex_unlock (appsink->mutex); + return NULL; + } +not_started: + { + GST_DEBUG_OBJECT (appsink, "we are stopped, return NULL"); + g_mutex_unlock (appsink->mutex); + return NULL; + } +} + +/** + * gst_app_sink_pull_buffer: + * @appsink: a #GstAppSink + * + * This function blocks until a buffer or EOS becomes available or the appsink + * element is set to the READY/NULL state. + * + * This function will only return buffers when the appsink is in the PLAYING + * state. All rendered buffers will be put in a queue so that the application + * can pull buffers at its own rate. Note that when the application does not + * pull buffers fast enough, the queued buffers could consume a lot of memory, + * especially when dealing with raw video frames. + * + * If an EOS event was received before any buffers, this function returns + * %NULL. + * + * Returns: a #GstBuffer or NULL when the appsink is stopped or EOS. + */ GstBuffer * gst_app_sink_pull_buffer (GstAppSink * appsink) { @@ -255,15 +437,39 @@ gst_app_sink_pull_buffer (GstAppSink * appsink) g_return_val_if_fail (GST_IS_APP_SINK (appsink), NULL); g_mutex_lock (appsink->mutex); - while (g_queue_is_empty (appsink->queue)) { + + while (TRUE) { + GST_DEBUG_OBJECT (appsink, "trying to grab a buffer"); + if (!appsink->started) + goto not_started; + + if (!g_queue_is_empty (appsink->queue)) + break; + if (appsink->end_of_stream) - goto out; + goto eos; + /* nothing to return, wait */ + GST_DEBUG_OBJECT (appsink, "waiting for a buffer"); g_cond_wait (appsink->cond, appsink->mutex); } buf = g_queue_pop_head (appsink->queue); -out: + GST_DEBUG_OBJECT (appsink, "we have a buffer %p", buf); g_mutex_unlock (appsink->mutex); return buf; + + /* special conditions */ +eos: + { + GST_DEBUG_OBJECT (appsink, "we are EOS, return NULL"); + g_mutex_unlock (appsink->mutex); + return NULL; + } +not_started: + { + GST_DEBUG_OBJECT (appsink, "we are stopped, return NULL"); + g_mutex_unlock (appsink->mutex); + return NULL; + } } diff --git a/gst-libs/gst/app/gstappsink.h b/gst-libs/gst/app/gstappsink.h index 2873e36..92a017b 100644 --- a/gst-libs/gst/app/gstappsink.h +++ b/gst-libs/gst/app/gstappsink.h @@ -49,6 +49,8 @@ struct _GstAppSink GCond *cond; GMutex *mutex; GQueue *queue; + GstBuffer *preroll; + gboolean started; gboolean end_of_stream; }; @@ -61,9 +63,12 @@ GType gst_app_sink_get_type(void); GST_DEBUG_CATEGORY_EXTERN (app_sink_debug); -void gst_app_sink_set_caps (GstAppSink *appsink, GstCaps *caps); -gboolean gst_app_sink_end_of_stream (GstAppSink *appsink); -GstBuffer *gst_app_sink_pull_buffer (GstAppSink *appsink); +void gst_app_sink_set_caps (GstAppSink *appsink, GstCaps *caps); + +gboolean gst_app_sink_end_of_stream (GstAppSink *appsink); + +GstBuffer * gst_app_sink_pull_preroll (GstAppSink *appsink); +GstBuffer * gst_app_sink_pull_buffer (GstAppSink *appsink); G_END_DECLS -- 2.7.4