summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSebastian Dröge <sebastian.droege@collabora.co.uk>2012-10-16 15:40:11 +0200
committerSebastian Dröge <sebastian.droege@collabora.co.uk>2012-10-18 14:22:47 +0200
commitb39c1f45ae53dd75d76aae81aa8ec7b152ca7408 (patch)
tree9050019cb11e2f73dc8a3e6be545967863fdef86
parentbbd16791b8ac064fdb0d786f38d036111ae43889 (diff)
eglglessink: Refactor rendering and all context specific GL things into a separate thread
Also fixes some threading related problems.
-rw-r--r--ext/eglgles/gsteglglessink.c317
1 files changed, 205 insertions, 112 deletions
diff --git a/ext/eglgles/gsteglglessink.c b/ext/eglgles/gsteglglessink.c
index 75372039f..bb7f552c4 100644
--- a/ext/eglgles/gsteglglessink.c
+++ b/ext/eglgles/gsteglglessink.c
@@ -136,8 +136,6 @@
#include <GLES2/gl2.h>
#include <GLES2/gl2ext.h>
-#include <pthread.h>
-
#include "video_platform_wrapper.h"
#include "gsteglglessink.h"
@@ -429,14 +427,18 @@ static gboolean gst_eglglessink_init_egl_surface (GstEglGlesSink * eglglessink);
static void gst_eglglessink_init_egl_exts (GstEglGlesSink * eglglessink);
static gboolean gst_eglglessink_setup_vbo (GstEglGlesSink * eglglessink,
gboolean reset);
+static gboolean
+gst_eglglessink_configure_caps (GstEglGlesSink *eglglessink, GstCaps * caps);
static GstFlowReturn gst_eglglessink_render_and_display (GstEglGlesSink * sink,
GstBuffer * buf);
+static GstFlowReturn gst_eglglessink_queue_buffer (GstEglGlesSink * sink,
+ GstBuffer * buf);
static inline gboolean got_gl_error (const char *wtf);
static inline void show_egl_error (const char *wtf);
static void gst_eglglessink_wipe_fmt (gpointer data);
static inline gboolean egl_init (GstEglGlesSink * eglglessink);
static gboolean gst_eglglessink_context_make_current (GstEglGlesSink *
- eglglessink, gboolean bind, gboolean streaming_thread);
+ eglglessink, gboolean bind);
static GstBufferClass *gsteglglessink_buffer_parent_class = NULL;
#define GST_TYPE_EGLGLESBUFFER (gst_eglglesbuffer_get_type())
@@ -1062,38 +1064,42 @@ HANDLE_ERROR:
return FALSE;
}
-static gboolean
-gst_eglglessink_start (GstEglGlesSink * eglglessink)
+static gpointer
+render_thread_func (GstEglGlesSink * eglglessink)
{
- if (!eglglessink->egl_started) {
- GST_ERROR_OBJECT (eglglessink, "EGL uninitialized. Bailing out");
- goto HANDLE_ERROR;
- }
+ GstDataQueueItem *item = NULL;
- /* Ask for a window to render to */
- if (!eglglessink->have_window)
- gst_x_overlay_prepare_xwindow_id (GST_X_OVERLAY (eglglessink));
+ while (gst_data_queue_pop (eglglessink->queue, &item)) {
+ GstBuffer *buf = NULL;
- if (!eglglessink->have_window && !eglglessink->create_window) {
- GST_ERROR_OBJECT (eglglessink, "Window handle unavailable and we "
- "were instructed not to create an internal one. Bailing out.");
- goto HANDLE_ERROR;
- }
+ GST_DEBUG_OBJECT (eglglessink, "Handling object %" GST_PTR_FORMAT, item->object);
- return TRUE;
+ if (item->object) {
+ GstCaps * caps;
-HANDLE_ERROR:
- GST_ERROR_OBJECT (eglglessink, "Couldn't start");
- return FALSE;
-}
+ buf = GST_BUFFER (item->object);
+ caps = GST_BUFFER_CAPS (buf);
+ if (caps != eglglessink->configured_caps) {
+ if (!gst_eglglessink_configure_caps (eglglessink, caps)) {
+ eglglessink->last_flow = GST_FLOW_NOT_NEGOTIATED;
+ g_cond_broadcast (eglglessink->render_cond);
+ item->destroy (item);
+ break;
+ }
+ }
+ }
-static gboolean
-gst_eglglessink_stop (GstEglGlesSink * eglglessink)
-{
- /* EGL/GLES2 cleanup */
- if (!gst_eglglessink_context_make_current (eglglessink, TRUE, FALSE))
- return FALSE;
+ eglglessink->last_flow = gst_eglglessink_render_and_display (eglglessink, buf);
+ g_cond_broadcast (eglglessink->render_cond);
+ item->destroy (item);
+ if (eglglessink->last_flow != GST_FLOW_OK)
+ break;
+ }
+
+ if (eglglessink->last_flow == GST_FLOW_OK)
+ eglglessink->last_flow = GST_FLOW_WRONG_STATE;
+ /* EGL/GLES cleanup */
if (eglglessink->rendering_path == GST_EGLGLESSINK_RENDER_SLOW) {
glUseProgram (0);
@@ -1123,7 +1129,7 @@ gst_eglglessink_stop (GstEglGlesSink * eglglessink)
}
}
- if (!gst_eglglessink_context_make_current (eglglessink, FALSE, FALSE))
+ if (!gst_eglglessink_context_make_current (eglglessink, FALSE))
return FALSE;
if (eglglessink->eglglesctx.surface) {
@@ -1139,6 +1145,69 @@ gst_eglglessink_stop (GstEglGlesSink * eglglessink)
eglglessink->eglglesctx.eglcontext = NULL;
}
+ if (eglglessink->configured_caps) {
+ gst_caps_unref (eglglessink->configured_caps);
+ eglglessink->configured_caps = NULL;
+ }
+
+ return NULL;
+}
+
+static gboolean
+gst_eglglessink_start (GstEglGlesSink * eglglessink)
+{
+ GError *error = NULL;
+
+ if (!eglglessink->egl_started) {
+ GST_ERROR_OBJECT (eglglessink, "EGL uninitialized. Bailing out");
+ goto HANDLE_ERROR;
+ }
+
+ /* Ask for a window to render to */
+ if (!eglglessink->have_window)
+ gst_x_overlay_prepare_xwindow_id (GST_X_OVERLAY (eglglessink));
+
+ if (!eglglessink->have_window && !eglglessink->create_window) {
+ GST_ERROR_OBJECT (eglglessink, "Window handle unavailable and we "
+ "were instructed not to create an internal one. Bailing out.");
+ goto HANDLE_ERROR;
+ }
+
+ eglglessink->last_flow = GST_FLOW_OK;
+ gst_data_queue_set_flushing (eglglessink->queue, FALSE);
+
+#if !GLIB_CHECK_VERSION (2, 31, 0)
+ eglglessink->thread =
+ g_thread_create ((GThreadFunc) render_thread_func, eglglessink, TRUE,
+ &error);
+#else
+ eglglessink->thread = g_thread_try_new ("eglglessink-render",
+ (GThreadFunc) render_thread_func, eglglessink, &error);
+#endif
+
+ if (!eglglessink->thread || error != NULL)
+ goto HANDLE_ERROR;
+
+ return TRUE;
+
+HANDLE_ERROR:
+ GST_ERROR_OBJECT (eglglessink, "Couldn't start");
+ g_clear_error (&error);
+ return FALSE;
+}
+
+static gboolean
+gst_eglglessink_stop (GstEglGlesSink * eglglessink)
+{
+ gst_data_queue_set_flushing (eglglessink->queue, TRUE);
+ g_cond_broadcast (eglglessink->render_cond);
+
+ if (eglglessink->thread) {
+ g_thread_join (eglglessink->thread);
+ eglglessink->thread = NULL;
+ }
+ eglglessink->last_flow = GST_FLOW_WRONG_STATE;
+
if (eglglessink->using_own_window) {
platform_destroy_native_window (eglglessink->eglglesctx.display,
eglglessink->eglglesctx.used_window);
@@ -1227,7 +1296,7 @@ gst_eglglessink_expose (GstXOverlay * overlay)
GST_DEBUG_OBJECT (eglglessink, "Expose catched, redisplay");
/* Render from last seen buffer */
- ret = gst_eglglessink_render_and_display (eglglessink, NULL);
+ ret = gst_eglglessink_queue_buffer (eglglessink, NULL);
if (ret == GST_FLOW_ERROR)
GST_ERROR_OBJECT (eglglessink, "Redisplay failed");
}
@@ -1495,61 +1564,32 @@ gst_eglglessink_update_surface_dimensions (GstEglGlesSink * eglglessink)
return FALSE;
}
-static pthread_key_t context_key;
-
-static void
-detach_context (void *data)
-{
- GstEglGlesSink *eglglessink = data;
-
- GST_DEBUG_OBJECT (eglglessink,
- "Detaching current context from streaming thread");
- gst_eglglessink_context_make_current (eglglessink, FALSE, TRUE);
- gst_object_unref (eglglessink);
-}
-
static gboolean
gst_eglglessink_context_make_current (GstEglGlesSink * eglglessink,
- gboolean bind, gboolean streaming_thread)
+ gboolean bind)
{
g_assert (eglglessink->eglglesctx.display != NULL);
if (bind && eglglessink->eglglesctx.surface &&
eglglessink->eglglesctx.eglcontext) {
- if (streaming_thread) {
- EGLContext *ctx = eglGetCurrentContext ();
-
- if (ctx == eglglessink->eglglesctx.eglcontext) {
- GST_DEBUG_OBJECT (eglglessink, "Already attached the context");
- return TRUE;
- }
+ EGLContext *ctx = eglGetCurrentContext ();
- GST_DEBUG_OBJECT (eglglessink, "Attaching context to streaming thread");
- if (!eglMakeCurrent (eglglessink->eglglesctx.display,
- eglglessink->eglglesctx.surface,
- eglglessink->eglglesctx.surface,
- eglglessink->eglglesctx.eglcontext)) {
- show_egl_error ("eglMakeCurrent");
- GST_ERROR_OBJECT (eglglessink, "Couldn't bind context");
- return FALSE;
- }
+ if (ctx == eglglessink->eglglesctx.eglcontext) {
+ GST_DEBUG_OBJECT (eglglessink, "Already attached the context to thread %p", g_thread_self ());
+ return TRUE;
+ }
- if (!pthread_getspecific (context_key)) {
- pthread_setspecific (context_key, gst_object_ref (eglglessink));
- }
- } else {
- GST_DEBUG_OBJECT (eglglessink, "Attaching context");
- if (!eglMakeCurrent (eglglessink->eglglesctx.display,
- eglglessink->eglglesctx.surface,
- eglglessink->eglglesctx.surface,
- eglglessink->eglglesctx.eglcontext)) {
- show_egl_error ("eglMakeCurrent");
- GST_ERROR_OBJECT (eglglessink, "Couldn't bind context");
- return FALSE;
- }
+ GST_DEBUG_OBJECT (eglglessink, "Attaching context to thread %p", g_thread_self ());
+ if (!eglMakeCurrent (eglglessink->eglglesctx.display,
+ eglglessink->eglglesctx.surface,
+ eglglessink->eglglesctx.surface,
+ eglglessink->eglglesctx.eglcontext)) {
+ show_egl_error ("eglMakeCurrent");
+ GST_ERROR_OBJECT (eglglessink, "Couldn't bind context");
+ return FALSE;
}
} else {
- GST_DEBUG_OBJECT (eglglessink, "Detaching context");
+ GST_DEBUG_OBJECT (eglglessink, "Detaching context from thread %p", g_thread_self ());
if (!eglMakeCurrent (eglglessink->eglglesctx.display,
EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT)) {
show_egl_error ("eglMakeCurrent");
@@ -1583,7 +1623,7 @@ gst_eglglessink_init_egl_surface (GstEglGlesSink * eglglessink)
goto HANDLE_EGL_ERROR_LOCKED;
}
- if (!gst_eglglessink_context_make_current (eglglessink, TRUE, TRUE))
+ if (!gst_eglglessink_context_make_current (eglglessink, TRUE))
goto HANDLE_EGL_ERROR_LOCKED;
/* Save surface dims */
@@ -1968,6 +2008,40 @@ gst_eglglessink_set_render_rectangle (GstXOverlay * overlay, gint x, gint y,
return;
}
+static void
+queue_item_destroy (GstDataQueueItem * item)
+{
+ gst_mini_object_replace (&item->object, NULL);
+ g_slice_free (GstDataQueueItem, item);
+}
+
+static GstFlowReturn
+gst_eglglessink_queue_buffer (GstEglGlesSink * eglglessink,
+ GstBuffer * buf)
+{
+ GstDataQueueItem *item = g_slice_new0 (GstDataQueueItem);
+
+ item->object = GST_MINI_OBJECT_CAST ((buf ? gst_buffer_ref (buf) : NULL));
+ item->size = GST_BUFFER_SIZE (buf);
+ item->duration = GST_BUFFER_DURATION (buf);
+ item->visible = (buf ? TRUE : FALSE);
+ item->destroy = (GDestroyNotify) queue_item_destroy;
+
+ if (buf)
+ g_mutex_lock (eglglessink->render_lock);
+ if (!gst_data_queue_push (eglglessink->queue, item)) {
+ g_mutex_unlock (eglglessink->render_lock);
+ return GST_FLOW_WRONG_STATE;
+ }
+
+ if (buf) {
+ g_cond_wait (eglglessink->render_cond, eglglessink->render_lock);
+ g_mutex_unlock (eglglessink->render_lock);
+ }
+
+ return (buf ? eglglessink->last_flow : GST_FLOW_OK);
+}
+
/* Rendering and display */
static GstFlowReturn
gst_eglglessink_render_and_display (GstEglGlesSink * eglglessink,
@@ -1984,9 +2058,6 @@ gst_eglglessink_render_and_display (GstEglGlesSink * eglglessink,
};
#endif
- if (!gst_eglglessink_context_make_current (eglglessink, TRUE, TRUE))
- goto HANDLE_EGL_ERROR;
-
w = GST_VIDEO_SINK_WIDTH (eglglessink);
h = GST_VIDEO_SINK_HEIGHT (eglglessink);
@@ -2248,21 +2319,14 @@ gst_eglglessink_show_frame (GstVideoSink * vsink, GstBuffer * buf)
eglglessink = GST_EGLGLESSINK (vsink);
GST_DEBUG_OBJECT (eglglessink, "Got buffer: %p", buf);
- if (!eglglessink->have_window) {
- GST_ERROR_OBJECT (eglglessink, "I don't have a window to render to");
- return GST_FLOW_ERROR;
- }
-
- if (!eglglessink->have_surface) {
- GST_ERROR_OBJECT (eglglessink, "I don't have a surface to render to");
- return GST_FLOW_ERROR;
- }
#ifndef EGL_ANDROID_image_native_buffer
GST_WARNING_OBJECT (eglglessink, "EGL_ANDROID_image_native_buffer not "
"available");
#endif
- return gst_eglglessink_render_and_display (eglglessink, buf);
+ buf = gst_buffer_make_metadata_writable (gst_buffer_ref (buf));
+ gst_buffer_set_caps (buf, eglglessink->current_caps);
+ return gst_eglglessink_queue_buffer (eglglessink, buf);
}
static GstCaps *
@@ -2287,21 +2351,14 @@ gst_eglglessink_getcaps (GstBaseSink * bsink)
}
static gboolean
-gst_eglglessink_setcaps (GstBaseSink * bsink, GstCaps * caps)
+gst_eglglessink_configure_caps (GstEglGlesSink *eglglessink, GstCaps * caps)
{
- GstEglGlesSink *eglglessink;
gboolean ret = TRUE;
gint width, height;
int par_n, par_d;
EGLNativeWindowType window;
GstEglGlesImageFmt *format;
- eglglessink = GST_EGLGLESSINK (bsink);
-
- GST_DEBUG_OBJECT (eglglessink,
- "In setcaps. Possible caps %" GST_PTR_FORMAT ", setting caps %"
- GST_PTR_FORMAT, eglglessink->current_caps, caps);
-
if (!(ret = gst_video_format_parse_caps (caps, &eglglessink->format, &width,
&height))) {
GST_ERROR_OBJECT (eglglessink, "Got weird and/or incomplete caps");
@@ -2330,9 +2387,9 @@ gst_eglglessink_setcaps (GstBaseSink * bsink, GstCaps * caps)
GST_VIDEO_SINK_WIDTH (eglglessink) = width;
GST_VIDEO_SINK_HEIGHT (eglglessink) = height;
- if (eglglessink->current_caps) {
+ if (eglglessink->configured_caps) {
GST_ERROR_OBJECT (eglglessink, "Caps were already set");
- if (gst_caps_can_intersect (caps, eglglessink->current_caps)) {
+ if (gst_caps_can_intersect (caps, eglglessink->configured_caps)) {
GST_INFO_OBJECT (eglglessink, "Caps are compatible anyway");
goto SUCCEED;
}
@@ -2340,10 +2397,6 @@ gst_eglglessink_setcaps (GstBaseSink * bsink, GstCaps * caps)
GST_DEBUG_OBJECT (eglglessink, "Caps are not compatible, reconfiguring");
/* Cleanup */
-
- if (!gst_eglglessink_context_make_current (eglglessink, TRUE, TRUE))
- return FALSE;
-
if (eglglessink->rendering_path == GST_EGLGLESSINK_RENDER_SLOW) {
glUseProgram (0);
@@ -2373,7 +2426,7 @@ gst_eglglessink_setcaps (GstBaseSink * bsink, GstCaps * caps)
}
}
- if (!gst_eglglessink_context_make_current (eglglessink, FALSE, TRUE))
+ if (!gst_eglglessink_context_make_current (eglglessink, FALSE))
return FALSE;
if (eglglessink->eglglesctx.surface) {
@@ -2398,8 +2451,8 @@ gst_eglglessink_setcaps (GstBaseSink * bsink, GstCaps * caps)
eglglessink->display_region.h = 0;
GST_OBJECT_UNLOCK (eglglessink);
- gst_caps_unref (eglglessink->current_caps);
- eglglessink->current_caps = NULL;
+ gst_caps_unref (eglglessink->configured_caps);
+ eglglessink->configured_caps = NULL;
}
if (!gst_eglglessink_choose_config (eglglessink)) {
@@ -2407,7 +2460,7 @@ gst_eglglessink_setcaps (GstBaseSink * bsink, GstCaps * caps)
goto HANDLE_ERROR;
}
- eglglessink->current_caps = gst_caps_ref (caps);
+ gst_caps_replace (&eglglessink->configured_caps, caps);
/* By now the application should have set a window
* if it meant to do so
@@ -2440,14 +2493,30 @@ gst_eglglessink_setcaps (GstBaseSink * bsink, GstCaps * caps)
}
SUCCEED:
- GST_INFO_OBJECT (eglglessink, "Setcaps succeed");
+ GST_INFO_OBJECT (eglglessink, "Configured caps successfully");
return TRUE;
HANDLE_ERROR:
- GST_ERROR_OBJECT (eglglessink, "Setcaps failed");
+ GST_ERROR_OBJECT (eglglessink, "Configuring caps failed");
return FALSE;
}
+static gboolean
+gst_eglglessink_setcaps (GstBaseSink * bsink, GstCaps * caps)
+{
+ GstEglGlesSink *eglglessink;
+
+ eglglessink = GST_EGLGLESSINK (bsink);
+
+ GST_DEBUG_OBJECT (eglglessink,
+ "Current caps %" GST_PTR_FORMAT ", setting caps %"
+ GST_PTR_FORMAT, eglglessink->current_caps, caps);
+
+ gst_caps_replace (&eglglessink->current_caps, caps);
+
+ return TRUE;
+}
+
static void
gst_eglglessink_wipe_fmt (gpointer data)
{
@@ -2469,7 +2538,6 @@ gst_eglglessink_open (GstEglGlesSink * eglglessink)
static gboolean
gst_eglglessink_close (GstEglGlesSink * eglglessink)
{
- g_mutex_lock (eglglessink->flow_lock);
if (eglglessink->eglglesctx.display) {
eglTerminate (eglglessink->eglglesctx.display);
eglglessink->eglglesctx.display = NULL;
@@ -2481,7 +2549,6 @@ gst_eglglessink_close (GstEglGlesSink * eglglessink)
gst_caps_unref (eglglessink->sinkcaps);
eglglessink->sinkcaps = NULL;
eglglessink->egl_started = FALSE;
- g_mutex_unlock (eglglessink->flow_lock);
return TRUE;
}
@@ -2528,6 +2595,7 @@ gst_eglglessink_change_state (GstElement * element, GstStateChange transition)
if (!gst_eglglessink_stop (eglglessink)) {
ret = GST_STATE_CHANGE_FAILURE;
goto done;
+ }
break;
default:
break;
@@ -2540,6 +2608,23 @@ done:
static void
gst_eglglessink_finalize (GObject * object)
{
+ GstEglGlesSink *eglglessink;
+
+ g_return_if_fail (GST_IS_EGLGLESSINK (object));
+
+ eglglessink = GST_EGLGLESSINK (object);
+
+ if (eglglessink->queue)
+ g_object_unref (eglglessink->queue);
+ eglglessink->queue = NULL;
+
+ if (eglglessink->render_cond)
+ g_cond_free (eglglessink->render_cond);
+ eglglessink->render_cond = NULL;
+ if (eglglessink->render_lock);
+ g_mutex_free (eglglessink->render_lock);
+ eglglessink->render_lock = NULL;
+
G_OBJECT_CLASS (parent_class)->finalize (object);
}
@@ -2659,6 +2744,11 @@ gst_eglglessink_class_init (GstEglGlesSinkClass * klass)
TRUE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
}
+static gboolean
+queue_check_full_func (GstDataQueue * queue, guint visible, guint bytes, guint64 time, gpointer checkdata)
+{
+ return visible != 0;
+}
static void
gst_eglglessink_init (GstEglGlesSink * eglglessink,
@@ -2681,6 +2771,11 @@ gst_eglglessink_init (GstEglGlesSink * eglglessink,
eglglessink->par_n = 1;
eglglessink->par_d = 1;
+
+ eglglessink->render_lock = g_mutex_new ();
+ eglglessink->render_cond = g_cond_new ();
+ eglglessink->queue = gst_data_queue_new (queue_check_full_func, NULL);
+ eglglessink->last_flow = GST_FLOW_WRONG_STATE;
}
/* Interface initializations. Used here for initializing the XOverlay
@@ -2714,8 +2809,6 @@ eglglessink_plugin_init (GstPlugin * plugin)
GST_DEBUG_CATEGORY_INIT (gst_eglglessink_debug, "eglglessink",
0, "Simple EGL/GLES Sink");
- pthread_key_create (&context_key, detach_context);
-
return gst_element_register (plugin, "eglglessink", GST_RANK_PRIMARY,
GST_TYPE_EGLGLESSINK);
}