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()
This commit is contained in:
Wim Taymans 2007-08-14 17:38:05 +00:00
parent 5ff55c7a30
commit dd72f88a8c
2 changed files with 230 additions and 19 deletions

View file

@ -34,9 +34,9 @@ GST_DEBUG_CATEGORY (app_sink_debug);
static const GstElementDetails app_sink_details = static const GstElementDetails app_sink_details =
GST_ELEMENT_DETAILS ("AppSink", GST_ELEMENT_DETAILS ("AppSink",
"FIXME", "Generic/Sink",
"FIXME", "Allow the application to get access to raw buffer",
"autogenerated by makefilter"); "David Schleef <ds@schleef.org>, Wim Taymans <wim.taymans@gmail.com");
enum enum
{ {
@ -57,8 +57,11 @@ static void gst_app_sink_dispose (GObject * object);
static gboolean gst_app_sink_start (GstBaseSink * psink); static gboolean gst_app_sink_start (GstBaseSink * psink);
static gboolean gst_app_sink_stop (GstBaseSink * psink); static gboolean gst_app_sink_stop (GstBaseSink * psink);
static gboolean gst_app_sink_event (GstBaseSink * sink, GstEvent * event); static gboolean gst_app_sink_event (GstBaseSink * sink, GstEvent * event);
static GstFlowReturn gst_app_sink_preroll (GstBaseSink * psink,
GstBuffer * buffer);
static GstFlowReturn gst_app_sink_render (GstBaseSink * psink, static GstFlowReturn gst_app_sink_render (GstBaseSink * psink,
GstBuffer * buffer); GstBuffer * buffer);
static GstCaps *gst_app_sink_get_caps (GstBaseSink * psink);
GST_BOILERPLATE (GstAppSink, gst_app_sink, GstBaseSink, GST_TYPE_BASE_SINK); GST_BOILERPLATE (GstAppSink, gst_app_sink, GstBaseSink, GST_TYPE_BASE_SINK);
@ -67,15 +70,12 @@ gst_app_sink_base_init (gpointer g_class)
{ {
GstElementClass *element_class = GST_ELEMENT_CLASS (g_class); GstElementClass *element_class = GST_ELEMENT_CLASS (g_class);
//GObjectClass *gobject_class = G_OBJECT_CLASS (g_class);
GST_DEBUG_CATEGORY_INIT (app_sink_debug, "appsink", 0, "appsink element"); GST_DEBUG_CATEGORY_INIT (app_sink_debug, "appsink", 0, "appsink element");
gst_element_class_set_details (element_class, &app_sink_details); gst_element_class_set_details (element_class, &app_sink_details);
gst_element_class_add_pad_template (element_class, gst_element_class_add_pad_template (element_class,
gst_static_pad_template_get (&gst_app_sink_template)); gst_static_pad_template_get (&gst_app_sink_template));
} }
static void static void
@ -91,7 +91,9 @@ gst_app_sink_class_init (GstAppSinkClass * klass)
basesink_class->start = gst_app_sink_start; basesink_class->start = gst_app_sink_start;
basesink_class->stop = gst_app_sink_stop; basesink_class->stop = gst_app_sink_stop;
basesink_class->event = gst_app_sink_event; basesink_class->event = gst_app_sink_event;
basesink_class->render = gst_app_sink_preroll;
basesink_class->render = gst_app_sink_render; basesink_class->render = gst_app_sink_render;
basesink_class->get_caps = gst_app_sink_get_caps;
} }
static void static void
@ -103,6 +105,10 @@ gst_app_sink_dispose (GObject * obj)
gst_caps_unref (appsink->caps); gst_caps_unref (appsink->caps);
appsink->caps = NULL; appsink->caps = NULL;
} }
if (appsink->preroll) {
gst_buffer_unref (appsink->preroll);
appsink->preroll = NULL;
}
if (appsink->mutex) { if (appsink->mutex) {
g_mutex_free (appsink->mutex); g_mutex_free (appsink->mutex);
appsink->mutex = NULL; appsink->mutex = NULL;
@ -112,6 +118,7 @@ gst_app_sink_dispose (GObject * obj)
appsink->cond = NULL; appsink->cond = NULL;
} }
if (appsink->queue) { if (appsink->queue) {
g_queue_foreach (appsink->queue, (GFunc) gst_mini_object_unref, NULL);
g_queue_free (appsink->queue); g_queue_free (appsink->queue);
appsink->queue = NULL; appsink->queue = NULL;
} }
@ -157,12 +164,27 @@ gst_app_sink_get_property (GObject * object, guint prop_id, GValue * value,
GST_OBJECT_UNLOCK (appsink); 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 static gboolean
gst_app_sink_start (GstBaseSink * psink) gst_app_sink_start (GstBaseSink * psink)
{ {
GstAppSink *appsink = GST_APP_SINK (psink); GstAppSink *appsink = GST_APP_SINK (psink);
g_mutex_lock (appsink->mutex);
appsink->end_of_stream = FALSE; appsink->end_of_stream = FALSE;
appsink->started = TRUE;
GST_DEBUG_OBJECT (appsink, "starting");
g_mutex_unlock (appsink->mutex);
return TRUE; return TRUE;
} }
@ -170,7 +192,13 @@ gst_app_sink_start (GstBaseSink * psink)
static gboolean static gboolean
gst_app_sink_stop (GstBaseSink * psink) 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; return TRUE;
} }
@ -182,14 +210,38 @@ gst_app_sink_event (GstBaseSink * sink, GstEvent * event)
switch (event->type) { switch (event->type) {
case GST_EVENT_EOS: case GST_EVENT_EOS:
g_mutex_lock (appsink->mutex);
GST_DEBUG_OBJECT (appsink, "receiving EOS");
appsink->end_of_stream = TRUE; 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; break;
default: default:
break; break;
} }
return TRUE;
}
gst_object_unref (sink); static GstFlowReturn
return FALSE; 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 static GstFlowReturn
@ -198,6 +250,7 @@ gst_app_sink_render (GstBaseSink * psink, GstBuffer * buffer)
GstAppSink *appsink = GST_APP_SINK (psink); GstAppSink *appsink = GST_APP_SINK (psink);
g_mutex_lock (appsink->mutex); 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_queue_push_tail (appsink->queue, gst_buffer_ref (buffer));
g_cond_signal (appsink->cond); g_cond_signal (appsink->cond);
g_mutex_unlock (appsink->mutex); g_mutex_unlock (appsink->mutex);
@ -205,18 +258,33 @@ gst_app_sink_render (GstBaseSink * psink, GstBuffer * buffer)
return GST_FLOW_OK; 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 */ /* external API */
/** /**
* gst_app_sink_set_caps: * gst_app_sink_set_caps:
* @appsink: * @appsink: a #GstAppSink
* @caps: * @caps: caps to set
* *
* Set the capabilities on the appsink element. This function takes * 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 void
gst_app_sink_set_caps (GstAppSink * appsink, GstCaps * caps) 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 (appsink != NULL);
g_return_if_fail (GST_IS_APP_SINK (appsink)); 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_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 gboolean
gst_app_sink_end_of_stream (GstAppSink * appsink) 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_return_val_if_fail (GST_IS_APP_SINK (appsink), FALSE);
g_mutex_lock (appsink->mutex); g_mutex_lock (appsink->mutex);
if (!appsink->started)
goto not_started;
if (appsink->end_of_stream && g_queue_is_empty (appsink->queue)) { 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; ret = TRUE;
} else { } else {
GST_DEBUG_OBJECT (appsink, "we are not yet EOS");
ret = FALSE; ret = FALSE;
} }
g_mutex_unlock (appsink->mutex); g_mutex_unlock (appsink->mutex);
return ret; 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 * GstBuffer *
gst_app_sink_pull_buffer (GstAppSink * appsink) 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_return_val_if_fail (GST_IS_APP_SINK (appsink), NULL);
g_mutex_lock (appsink->mutex); g_mutex_lock (appsink->mutex);
while (g_queue_is_empty (appsink->queue)) {
if (appsink->end_of_stream)
goto out;
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 eos;
/* nothing to return, wait */
GST_DEBUG_OBJECT (appsink, "waiting for a buffer");
g_cond_wait (appsink->cond, appsink->mutex); g_cond_wait (appsink->cond, appsink->mutex);
} }
buf = g_queue_pop_head (appsink->queue); buf = g_queue_pop_head (appsink->queue);
out: GST_DEBUG_OBJECT (appsink, "we have a buffer %p", buf);
g_mutex_unlock (appsink->mutex); g_mutex_unlock (appsink->mutex);
return buf; 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;
}
} }

View file

@ -49,6 +49,8 @@ struct _GstAppSink
GCond *cond; GCond *cond;
GMutex *mutex; GMutex *mutex;
GQueue *queue; GQueue *queue;
GstBuffer *preroll;
gboolean started;
gboolean end_of_stream; gboolean end_of_stream;
}; };
@ -62,7 +64,10 @@ GType gst_app_sink_get_type(void);
GST_DEBUG_CATEGORY_EXTERN (app_sink_debug); GST_DEBUG_CATEGORY_EXTERN (app_sink_debug);
void gst_app_sink_set_caps (GstAppSink *appsink, GstCaps *caps); void gst_app_sink_set_caps (GstAppSink *appsink, GstCaps *caps);
gboolean gst_app_sink_end_of_stream (GstAppSink *appsink); 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); GstBuffer * gst_app_sink_pull_buffer (GstAppSink *appsink);
G_END_DECLS G_END_DECLS