mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-04 14:38:48 +00:00
gl: add a new glbasefilter class below glfilter
It deals with propagating the gl display/contexts throughout the application/pipeline
This commit is contained in:
parent
41e3b32866
commit
ecdc5568c4
7 changed files with 687 additions and 295 deletions
|
@ -14,6 +14,7 @@ libgstgl_@GST_API_VERSION@_la_SOURCES = \
|
||||||
gstglmemory.c \
|
gstglmemory.c \
|
||||||
gstglbufferpool.c \
|
gstglbufferpool.c \
|
||||||
gstglfilter.c \
|
gstglfilter.c \
|
||||||
|
gstglbasefilter.c \
|
||||||
gstglshader.c \
|
gstglshader.c \
|
||||||
gstglshadervariables.c \
|
gstglshadervariables.c \
|
||||||
gstglcolorconvert.c \
|
gstglcolorconvert.c \
|
||||||
|
@ -35,6 +36,7 @@ libgstgl_@GST_API_VERSION@include_HEADERS = \
|
||||||
gstglmemory.h \
|
gstglmemory.h \
|
||||||
gstglbufferpool.h \
|
gstglbufferpool.h \
|
||||||
gstglfilter.h \
|
gstglfilter.h \
|
||||||
|
gstglbasefilter.h \
|
||||||
gstglshadervariables.h \
|
gstglshadervariables.h \
|
||||||
gstglshader.h \
|
gstglshader.h \
|
||||||
gstglcolorconvert.h \
|
gstglcolorconvert.h \
|
||||||
|
|
|
@ -42,6 +42,7 @@
|
||||||
#include <gst/gl/gstglmemory.h>
|
#include <gst/gl/gstglmemory.h>
|
||||||
#include <gst/gl/gstglbufferpool.h>
|
#include <gst/gl/gstglbufferpool.h>
|
||||||
#include <gst/gl/gstglframebuffer.h>
|
#include <gst/gl/gstglframebuffer.h>
|
||||||
|
#include <gst/gl/gstglbasefilter.h>
|
||||||
#include <gst/gl/gstglfilter.h>
|
#include <gst/gl/gstglfilter.h>
|
||||||
#include <gst/gl/gstglshadervariables.h>
|
#include <gst/gl/gstglshadervariables.h>
|
||||||
#include <gst/gl/gstglsyncmeta.h>
|
#include <gst/gl/gstglsyncmeta.h>
|
||||||
|
|
|
@ -63,6 +63,13 @@ typedef struct _GstGLColorConvert GstGLColorConvert;
|
||||||
typedef struct _GstGLColorConvertClass GstGLColorConvertClass;
|
typedef struct _GstGLColorConvertClass GstGLColorConvertClass;
|
||||||
typedef struct _GstGLColorConvertPrivate GstGLColorConvertPrivate;
|
typedef struct _GstGLColorConvertPrivate GstGLColorConvertPrivate;
|
||||||
|
|
||||||
|
typedef struct _GstGLBaseFilter GstGLBaseFilter;
|
||||||
|
typedef struct _GstGLBaseFilterClass GstGLBaseFilterClass;
|
||||||
|
typedef struct _GstGLBaseFilterPrivate GstGLBaseFilterPrivate;
|
||||||
|
|
||||||
|
typedef struct _GstGLFilter GstGLFilter;
|
||||||
|
typedef struct _GstGLFilterClass GstGLFilterClass;
|
||||||
|
|
||||||
G_END_DECLS
|
G_END_DECLS
|
||||||
|
|
||||||
#endif /* __GST_GL_FWD_H__ */
|
#endif /* __GST_GL_FWD_H__ */
|
||||||
|
|
445
gst-libs/gst/gl/gstglbasefilter.c
Normal file
445
gst-libs/gst/gl/gstglbasefilter.c
Normal file
|
@ -0,0 +1,445 @@
|
||||||
|
/*
|
||||||
|
* GStreamer
|
||||||
|
* Copyright (C) 2015 Matthew Waters <matthew@centricular.com>
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include "config.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <gst/video/gstvideometa.h>
|
||||||
|
|
||||||
|
#include <gst/gl/gl.h>
|
||||||
|
|
||||||
|
#define GST_CAT_DEFAULT gst_gl_base_filter_debug
|
||||||
|
GST_DEBUG_CATEGORY_STATIC (GST_CAT_DEFAULT);
|
||||||
|
|
||||||
|
#define GST_GL_BASE_FILTER_GET_PRIVATE(o) \
|
||||||
|
(G_TYPE_INSTANCE_GET_PRIVATE((o), GST_TYPE_GL_BASE_FILTER, GstGLBaseFilterPrivate))
|
||||||
|
|
||||||
|
struct _GstGLBaseFilterPrivate
|
||||||
|
{
|
||||||
|
GstGLContext *other_context;
|
||||||
|
|
||||||
|
gboolean gl_result;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Properties */
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
PROP_0,
|
||||||
|
PROP_CONTEXT
|
||||||
|
};
|
||||||
|
|
||||||
|
#define gst_gl_base_filter_parent_class parent_class
|
||||||
|
G_DEFINE_TYPE_WITH_CODE (GstGLBaseFilter, gst_gl_base_filter,
|
||||||
|
GST_TYPE_BASE_TRANSFORM, GST_DEBUG_CATEGORY_INIT (gst_gl_base_filter_debug,
|
||||||
|
"glbasefilter", 0, "glbasefilter element");
|
||||||
|
);
|
||||||
|
|
||||||
|
static void gst_gl_base_filter_set_property (GObject * object, guint prop_id,
|
||||||
|
const GValue * value, GParamSpec * pspec);
|
||||||
|
static void gst_gl_base_filter_get_property (GObject * object, guint prop_id,
|
||||||
|
GValue * value, GParamSpec * pspec);
|
||||||
|
|
||||||
|
static void gst_gl_base_filter_set_context (GstElement * element,
|
||||||
|
GstContext * context);
|
||||||
|
static gboolean gst_gl_base_filter_query (GstBaseTransform * trans,
|
||||||
|
GstPadDirection direction, GstQuery * query);
|
||||||
|
static void gst_gl_base_filter_reset (GstGLBaseFilter * filter);
|
||||||
|
static gboolean gst_gl_base_filter_start (GstBaseTransform * bt);
|
||||||
|
static gboolean gst_gl_base_filter_stop (GstBaseTransform * bt);
|
||||||
|
static gboolean gst_gl_base_filter_decide_allocation (GstBaseTransform * trans,
|
||||||
|
GstQuery * query);
|
||||||
|
static gboolean gst_gl_base_filter_propose_allocation (GstBaseTransform * trans,
|
||||||
|
GstQuery * decide_query, GstQuery * query);
|
||||||
|
|
||||||
|
/* GstGLContextThreadFunc */
|
||||||
|
static void gst_gl_base_filter_gl_start (GstGLContext * context, gpointer data);
|
||||||
|
static void gst_gl_base_filter_gl_stop (GstGLContext * context, gpointer data);
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_gl_base_filter_class_init (GstGLBaseFilterClass * klass)
|
||||||
|
{
|
||||||
|
GObjectClass *gobject_class;
|
||||||
|
GstElementClass *element_class;
|
||||||
|
|
||||||
|
g_type_class_add_private (klass, sizeof (GstGLBaseFilterPrivate));
|
||||||
|
|
||||||
|
gobject_class = (GObjectClass *) klass;
|
||||||
|
element_class = GST_ELEMENT_CLASS (klass);
|
||||||
|
|
||||||
|
gobject_class->set_property = gst_gl_base_filter_set_property;
|
||||||
|
gobject_class->get_property = gst_gl_base_filter_get_property;
|
||||||
|
|
||||||
|
GST_BASE_TRANSFORM_CLASS (klass)->query = gst_gl_base_filter_query;
|
||||||
|
GST_BASE_TRANSFORM_CLASS (klass)->start = gst_gl_base_filter_start;
|
||||||
|
GST_BASE_TRANSFORM_CLASS (klass)->stop = gst_gl_base_filter_stop;
|
||||||
|
GST_BASE_TRANSFORM_CLASS (klass)->decide_allocation =
|
||||||
|
gst_gl_base_filter_decide_allocation;
|
||||||
|
GST_BASE_TRANSFORM_CLASS (klass)->propose_allocation =
|
||||||
|
gst_gl_base_filter_propose_allocation;
|
||||||
|
|
||||||
|
element_class->set_context = gst_gl_base_filter_set_context;
|
||||||
|
|
||||||
|
g_object_class_install_property (gobject_class, PROP_CONTEXT,
|
||||||
|
g_param_spec_object ("context",
|
||||||
|
"OpenGL context",
|
||||||
|
"Get OpenGL context",
|
||||||
|
GST_GL_TYPE_CONTEXT, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
||||||
|
|
||||||
|
klass->supported_gl_api = GST_GL_API_ANY;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_gl_base_filter_init (GstGLBaseFilter * filter)
|
||||||
|
{
|
||||||
|
filter->priv = GST_GL_BASE_FILTER_GET_PRIVATE (filter);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_gl_base_filter_set_property (GObject * object, guint prop_id,
|
||||||
|
const GValue * value, GParamSpec * pspec)
|
||||||
|
{
|
||||||
|
switch (prop_id) {
|
||||||
|
default:
|
||||||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_gl_base_filter_get_property (GObject * object, guint prop_id,
|
||||||
|
GValue * value, GParamSpec * pspec)
|
||||||
|
{
|
||||||
|
GstGLBaseFilter *filter = GST_GL_BASE_FILTER (object);
|
||||||
|
|
||||||
|
switch (prop_id) {
|
||||||
|
case PROP_CONTEXT:
|
||||||
|
g_value_set_object (value, filter->context);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_gl_base_filter_set_context (GstElement * element, GstContext * context)
|
||||||
|
{
|
||||||
|
GstGLBaseFilter *filter = GST_GL_BASE_FILTER (element);
|
||||||
|
GstGLBaseFilterClass *filter_class = GST_GL_BASE_FILTER_GET_CLASS (filter);
|
||||||
|
|
||||||
|
gst_gl_handle_set_context (element, context, &filter->display,
|
||||||
|
&filter->priv->other_context);
|
||||||
|
if (filter->display)
|
||||||
|
gst_gl_display_filter_gl_api (filter->display,
|
||||||
|
filter_class->supported_gl_api);
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
_find_local_gl_context (GstGLBaseFilter * filter)
|
||||||
|
{
|
||||||
|
GstQuery *query;
|
||||||
|
GstContext *context;
|
||||||
|
const GstStructure *s;
|
||||||
|
|
||||||
|
if (filter->context)
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
query = gst_query_new_context ("gst.gl.local_context");
|
||||||
|
if (!filter->context
|
||||||
|
&& gst_gl_run_query (GST_ELEMENT (filter), query, GST_PAD_SRC)) {
|
||||||
|
gst_query_parse_context (query, &context);
|
||||||
|
if (context) {
|
||||||
|
s = gst_context_get_structure (context);
|
||||||
|
gst_structure_get (s, "context", GST_GL_TYPE_CONTEXT, &filter->context,
|
||||||
|
NULL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!filter->context
|
||||||
|
&& gst_gl_run_query (GST_ELEMENT (filter), query, GST_PAD_SINK)) {
|
||||||
|
gst_query_parse_context (query, &context);
|
||||||
|
if (context) {
|
||||||
|
s = gst_context_get_structure (context);
|
||||||
|
gst_structure_get (s, "context", GST_GL_TYPE_CONTEXT, &filter->context,
|
||||||
|
NULL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
GST_DEBUG_OBJECT (filter, "found local context %p", filter->context);
|
||||||
|
|
||||||
|
gst_query_unref (query);
|
||||||
|
|
||||||
|
if (filter->context)
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
_ensure_gl_setup (GstGLBaseFilter * filter)
|
||||||
|
{
|
||||||
|
GstGLBaseFilterClass *filter_class = GST_GL_BASE_FILTER_GET_CLASS (filter);
|
||||||
|
|
||||||
|
if (!gst_gl_ensure_element_data (filter, &filter->display,
|
||||||
|
&filter->priv->other_context)) {
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_gl_display_filter_gl_api (filter->display,
|
||||||
|
filter_class->supported_gl_api);
|
||||||
|
|
||||||
|
_find_local_gl_context (filter);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_gl_base_filter_query (GstBaseTransform * trans, GstPadDirection direction,
|
||||||
|
GstQuery * query)
|
||||||
|
{
|
||||||
|
GstGLBaseFilter *filter = GST_GL_BASE_FILTER (trans);
|
||||||
|
GstGLBaseFilterClass *filter_class = GST_GL_BASE_FILTER_GET_CLASS (filter);
|
||||||
|
|
||||||
|
switch (GST_QUERY_TYPE (query)) {
|
||||||
|
case GST_QUERY_ALLOCATION:
|
||||||
|
{
|
||||||
|
if (direction == GST_PAD_SINK
|
||||||
|
&& gst_base_transform_is_passthrough (trans)) {
|
||||||
|
if (!_ensure_gl_setup (filter))
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
return gst_pad_peer_query (GST_BASE_TRANSFORM_SRC_PAD (trans), query);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case GST_QUERY_CONTEXT:
|
||||||
|
{
|
||||||
|
const gchar *context_type;
|
||||||
|
GstContext *context, *old_context;
|
||||||
|
gboolean ret;
|
||||||
|
|
||||||
|
ret = gst_gl_handle_context_query ((GstElement *) filter, query,
|
||||||
|
&filter->display, &filter->priv->other_context);
|
||||||
|
if (filter->display)
|
||||||
|
gst_gl_display_filter_gl_api (filter->display,
|
||||||
|
filter_class->supported_gl_api);
|
||||||
|
|
||||||
|
gst_query_parse_context_type (query, &context_type);
|
||||||
|
|
||||||
|
if (g_strcmp0 (context_type, "gst.gl.local_context") == 0) {
|
||||||
|
GstStructure *s;
|
||||||
|
|
||||||
|
gst_query_parse_context (query, &old_context);
|
||||||
|
|
||||||
|
if (old_context)
|
||||||
|
context = gst_context_copy (old_context);
|
||||||
|
else
|
||||||
|
context = gst_context_new ("gst.gl.local_context", FALSE);
|
||||||
|
|
||||||
|
s = gst_context_writable_structure (context);
|
||||||
|
gst_structure_set (s, "context", GST_GL_TYPE_CONTEXT, filter->context,
|
||||||
|
NULL);
|
||||||
|
gst_query_set_context (query, context);
|
||||||
|
gst_context_unref (context);
|
||||||
|
|
||||||
|
ret = filter->context != NULL;
|
||||||
|
}
|
||||||
|
GST_LOG_OBJECT (filter, "context query of type %s %i", context_type, ret);
|
||||||
|
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return GST_BASE_TRANSFORM_CLASS (parent_class)->query (trans, direction,
|
||||||
|
query);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_gl_base_filter_reset (GstGLBaseFilter * filter)
|
||||||
|
{
|
||||||
|
GstGLBaseFilterClass *filter_class = GST_GL_BASE_FILTER_GET_CLASS (filter);
|
||||||
|
|
||||||
|
if (filter->context) {
|
||||||
|
if (filter_class->gl_stop != NULL) {
|
||||||
|
gst_gl_context_thread_add (filter->context, gst_gl_base_filter_gl_stop,
|
||||||
|
filter);
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_object_unref (filter->context);
|
||||||
|
filter->context = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (filter->display) {
|
||||||
|
gst_object_unref (filter->display);
|
||||||
|
filter->display = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (filter->priv->other_context) {
|
||||||
|
gst_object_unref (filter->priv->other_context);
|
||||||
|
filter->priv->other_context = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_gl_base_filter_start (GstBaseTransform * bt)
|
||||||
|
{
|
||||||
|
GstGLBaseFilter *filter = GST_GL_BASE_FILTER (bt);
|
||||||
|
GstGLBaseFilterClass *filter_class = GST_GL_BASE_FILTER_GET_CLASS (filter);
|
||||||
|
|
||||||
|
if (!gst_gl_ensure_element_data (filter, &filter->display,
|
||||||
|
&filter->priv->other_context))
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
gst_gl_display_filter_gl_api (filter->display,
|
||||||
|
filter_class->supported_gl_api);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_gl_base_filter_stop (GstBaseTransform * bt)
|
||||||
|
{
|
||||||
|
GstGLBaseFilter *filter = GST_GL_BASE_FILTER (bt);
|
||||||
|
|
||||||
|
gst_gl_base_filter_reset (filter);
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_gl_base_filter_gl_start (GstGLContext * context, gpointer data)
|
||||||
|
{
|
||||||
|
GstGLBaseFilter *filter = GST_GL_BASE_FILTER (data);
|
||||||
|
GstGLBaseFilterClass *filter_class = GST_GL_BASE_FILTER_GET_CLASS (filter);
|
||||||
|
|
||||||
|
if (filter_class->gl_start) {
|
||||||
|
filter->priv->gl_result = filter_class->gl_start (filter);
|
||||||
|
} else {
|
||||||
|
filter->priv->gl_result = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_gl_base_filter_gl_stop (GstGLContext * context, gpointer data)
|
||||||
|
{
|
||||||
|
GstGLBaseFilter *filter = GST_GL_BASE_FILTER (data);
|
||||||
|
GstGLBaseFilterClass *filter_class = GST_GL_BASE_FILTER_GET_CLASS (filter);
|
||||||
|
|
||||||
|
if (filter_class->gl_stop)
|
||||||
|
filter_class->gl_stop (filter);
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_gl_base_filter_decide_allocation (GstBaseTransform * trans,
|
||||||
|
GstQuery * query)
|
||||||
|
{
|
||||||
|
GstGLBaseFilter *filter = GST_GL_BASE_FILTER (trans);
|
||||||
|
GError *error = NULL;
|
||||||
|
GstGLContext *other_context = NULL;
|
||||||
|
guint idx;
|
||||||
|
|
||||||
|
if (!_ensure_gl_setup (filter))
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
if (!filter->context && gst_query_find_allocation_meta (query,
|
||||||
|
GST_VIDEO_GL_TEXTURE_UPLOAD_META_API_TYPE, &idx)) {
|
||||||
|
GstGLContext *context;
|
||||||
|
const GstStructure *upload_meta_params;
|
||||||
|
gpointer handle;
|
||||||
|
gchar *type;
|
||||||
|
gchar *apis;
|
||||||
|
|
||||||
|
gst_query_parse_nth_allocation_meta (query, idx, &upload_meta_params);
|
||||||
|
if (upload_meta_params) {
|
||||||
|
if (gst_structure_get (upload_meta_params, "gst.gl.GstGLContext",
|
||||||
|
GST_GL_TYPE_CONTEXT, &context, NULL) && context) {
|
||||||
|
GstGLContext *old = filter->context;
|
||||||
|
|
||||||
|
filter->context = context;
|
||||||
|
if (old)
|
||||||
|
gst_object_unref (old);
|
||||||
|
} else if (gst_structure_get (upload_meta_params, "gst.gl.context.handle",
|
||||||
|
G_TYPE_POINTER, &handle, "gst.gl.context.type", G_TYPE_STRING,
|
||||||
|
&type, "gst.gl.context.apis", G_TYPE_STRING, &apis, NULL)
|
||||||
|
&& handle) {
|
||||||
|
GstGLPlatform platform = GST_GL_PLATFORM_NONE;
|
||||||
|
GstGLAPI gl_apis;
|
||||||
|
|
||||||
|
GST_DEBUG ("got GL context handle 0x%p with type %s and apis %s",
|
||||||
|
handle, type, apis);
|
||||||
|
|
||||||
|
platform = gst_gl_platform_from_string (type);
|
||||||
|
gl_apis = gst_gl_api_from_string (apis);
|
||||||
|
|
||||||
|
if (gl_apis && platform)
|
||||||
|
other_context =
|
||||||
|
gst_gl_context_new_wrapped (filter->display, (guintptr) handle,
|
||||||
|
platform, gl_apis);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (filter->priv->other_context) {
|
||||||
|
if (!other_context) {
|
||||||
|
other_context = filter->priv->other_context;
|
||||||
|
} else {
|
||||||
|
GST_ELEMENT_WARNING (filter, LIBRARY, SETTINGS,
|
||||||
|
("%s", "Cannot share with more than one GL context"),
|
||||||
|
("%s", "Cannot share with more than one GL context"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!filter->context) {
|
||||||
|
filter->context = gst_gl_context_new (filter->display);
|
||||||
|
if (!gst_gl_context_create (filter->context, other_context, &error))
|
||||||
|
goto context_error;
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_gl_context_thread_add (filter->context, gst_gl_base_filter_gl_start,
|
||||||
|
filter);
|
||||||
|
if (!filter->priv->gl_result)
|
||||||
|
goto error;
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
|
||||||
|
context_error:
|
||||||
|
{
|
||||||
|
GST_ELEMENT_ERROR (trans, RESOURCE, NOT_FOUND, ("%s", error->message),
|
||||||
|
(NULL));
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
error:
|
||||||
|
{
|
||||||
|
GST_ELEMENT_ERROR (trans, LIBRARY, INIT,
|
||||||
|
("Subclass failed to initialize."), (NULL));
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_gl_base_filter_propose_allocation (GstBaseTransform * trans,
|
||||||
|
GstQuery * decide_query, GstQuery * query)
|
||||||
|
{
|
||||||
|
return FALSE;
|
||||||
|
}
|
83
gst-libs/gst/gl/gstglbasefilter.h
Normal file
83
gst-libs/gst/gl/gstglbasefilter.h
Normal file
|
@ -0,0 +1,83 @@
|
||||||
|
/*
|
||||||
|
* GStreamer
|
||||||
|
* Copyright (C) 2007 David Schleef <ds@schleef.org>
|
||||||
|
* Copyright (C) 2008 Julien Isorce <julien.isorce@gmail.com>
|
||||||
|
* Copyright (C) 2008 Filippo Argiolas <filippo.argiolas@gmail.com>
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef _GST_GL_BASE_FILTER_H_
|
||||||
|
#define _GST_GL_BASE_FILTER_H_
|
||||||
|
|
||||||
|
#include <gst/gst.h>
|
||||||
|
#include <gst/base/gstbasetransform.h>
|
||||||
|
#include <gst/video/video.h>
|
||||||
|
|
||||||
|
#include <gst/gl/gl.h>
|
||||||
|
|
||||||
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
|
GType gst_gl_base_filter_get_type(void);
|
||||||
|
#define GST_TYPE_GL_BASE_FILTER (gst_gl_base_filter_get_type())
|
||||||
|
#define GST_GL_BASE_FILTER(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_GL_BASE_FILTER,GstGLBaseFilter))
|
||||||
|
#define GST_IS_GL_BASE_FILTER(obj) (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_GL_BASE_FILTER))
|
||||||
|
#define GST_GL_BASE_FILTER_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST((klass) ,GST_TYPE_GL_BASE_FILTER,GstGLBaseFilterClass))
|
||||||
|
#define GST_IS_GL_BASE_FILTER_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass) ,GST_TYPE_GL_BASE_FILTER))
|
||||||
|
#define GST_GL_BASE_FILTER_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS((obj) ,GST_TYPE_GL_BASE_FILTER,GstGLBaseFilterClass))
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GstGLBaseFilter:
|
||||||
|
* @base_transform: parent #GstBaseTransform
|
||||||
|
* @display: the currently configured #GstGLDisplay
|
||||||
|
* @context: the currently configured #GstGLContext
|
||||||
|
*
|
||||||
|
* #GstGLBaseFilter is a base class that provides the logic of getting the
|
||||||
|
* GL context from the pipeline.
|
||||||
|
*/
|
||||||
|
struct _GstGLBaseFilter
|
||||||
|
{
|
||||||
|
GstBaseTransform parent;
|
||||||
|
|
||||||
|
GstGLDisplay *display;
|
||||||
|
GstGLContext *context;
|
||||||
|
|
||||||
|
/* <private> */
|
||||||
|
gpointer _padding[GST_PADDING];
|
||||||
|
|
||||||
|
GstGLBaseFilterPrivate *priv;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GstGLBaseFilterClass:
|
||||||
|
* @base_transform_class: parent class
|
||||||
|
* @gl_start: called in the GL thread to setup the element GL state.
|
||||||
|
* @gl_stop: called in the GL thread to setup the element GL state.
|
||||||
|
*/
|
||||||
|
struct _GstGLBaseFilterClass
|
||||||
|
{
|
||||||
|
GstBaseTransformClass parent_class;
|
||||||
|
GstGLAPI supported_gl_api;
|
||||||
|
|
||||||
|
gboolean (*gl_start) (GstGLBaseFilter *filter);
|
||||||
|
void (*gl_stop) (GstGLBaseFilter *filter);
|
||||||
|
|
||||||
|
gpointer _padding[GST_PADDING];
|
||||||
|
};
|
||||||
|
|
||||||
|
G_END_DECLS
|
||||||
|
|
||||||
|
#endif /* _GST_GL_BASE_FILTER_H_ */
|
|
@ -72,22 +72,19 @@ static GstStaticPadTemplate gst_gl_filter_sink_pad_template =
|
||||||
enum
|
enum
|
||||||
{
|
{
|
||||||
PROP_0,
|
PROP_0,
|
||||||
PROP_CONTEXT
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#define DEBUG_INIT \
|
|
||||||
GST_DEBUG_CATEGORY_INIT (gst_gl_filter_debug, "glfilter", 0, "glfilter element");
|
|
||||||
#define gst_gl_filter_parent_class parent_class
|
#define gst_gl_filter_parent_class parent_class
|
||||||
G_DEFINE_TYPE_WITH_CODE (GstGLFilter, gst_gl_filter, GST_TYPE_BASE_TRANSFORM,
|
G_DEFINE_TYPE_WITH_CODE (GstGLFilter, gst_gl_filter, GST_TYPE_GL_BASE_FILTER,
|
||||||
DEBUG_INIT);
|
GST_DEBUG_CATEGORY_INIT (gst_gl_filter_debug, "glfilter", 0,
|
||||||
|
"glfilter element");
|
||||||
|
);
|
||||||
|
|
||||||
static void gst_gl_filter_set_property (GObject * object, guint prop_id,
|
static void gst_gl_filter_set_property (GObject * object, guint prop_id,
|
||||||
const GValue * value, GParamSpec * pspec);
|
const GValue * value, GParamSpec * pspec);
|
||||||
static void gst_gl_filter_get_property (GObject * object, guint prop_id,
|
static void gst_gl_filter_get_property (GObject * object, guint prop_id,
|
||||||
GValue * value, GParamSpec * pspec);
|
GValue * value, GParamSpec * pspec);
|
||||||
|
|
||||||
static void gst_gl_filter_set_context (GstElement * element,
|
|
||||||
GstContext * context);
|
|
||||||
static gboolean gst_gl_filter_query (GstBaseTransform * trans,
|
static gboolean gst_gl_filter_query (GstBaseTransform * trans,
|
||||||
GstPadDirection direction, GstQuery * query);
|
GstPadDirection direction, GstQuery * query);
|
||||||
static GstCaps *gst_gl_filter_transform_caps (GstBaseTransform * bt,
|
static GstCaps *gst_gl_filter_transform_caps (GstBaseTransform * bt,
|
||||||
|
@ -107,10 +104,8 @@ static gboolean gst_gl_filter_decide_allocation (GstBaseTransform * trans,
|
||||||
GstQuery * query);
|
GstQuery * query);
|
||||||
static gboolean gst_gl_filter_set_caps (GstBaseTransform * bt, GstCaps * incaps,
|
static gboolean gst_gl_filter_set_caps (GstBaseTransform * bt, GstCaps * incaps,
|
||||||
GstCaps * outcaps);
|
GstCaps * outcaps);
|
||||||
|
static gboolean gst_gl_filter_gl_start (GstGLBaseFilter * filter);
|
||||||
/* GstGLContextThreadFunc */
|
static void gst_gl_filter_gl_stop (GstGLBaseFilter * filter);
|
||||||
static void gst_gl_filter_start_gl (GstGLContext * context, gpointer data);
|
|
||||||
static void gst_gl_filter_stop_gl (GstGLContext * context, gpointer data);
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_gl_filter_class_init (GstGLFilterClass * klass)
|
gst_gl_filter_class_init (GstGLFilterClass * klass)
|
||||||
|
@ -138,35 +133,20 @@ gst_gl_filter_class_init (GstGLFilterClass * klass)
|
||||||
gst_gl_filter_decide_allocation;
|
gst_gl_filter_decide_allocation;
|
||||||
GST_BASE_TRANSFORM_CLASS (klass)->get_unit_size = gst_gl_filter_get_unit_size;
|
GST_BASE_TRANSFORM_CLASS (klass)->get_unit_size = gst_gl_filter_get_unit_size;
|
||||||
|
|
||||||
element_class->set_context = gst_gl_filter_set_context;
|
GST_GL_BASE_FILTER_CLASS (klass)->gl_start = gst_gl_filter_gl_start;
|
||||||
|
GST_GL_BASE_FILTER_CLASS (klass)->gl_stop = gst_gl_filter_gl_stop;
|
||||||
|
|
||||||
gst_element_class_add_pad_template (element_class,
|
gst_element_class_add_pad_template (element_class,
|
||||||
gst_static_pad_template_get (&gst_gl_filter_src_pad_template));
|
gst_static_pad_template_get (&gst_gl_filter_src_pad_template));
|
||||||
gst_element_class_add_pad_template (element_class,
|
gst_element_class_add_pad_template (element_class,
|
||||||
gst_static_pad_template_get (&gst_gl_filter_sink_pad_template));
|
gst_static_pad_template_get (&gst_gl_filter_sink_pad_template));
|
||||||
|
|
||||||
g_object_class_install_property (gobject_class, PROP_CONTEXT,
|
|
||||||
g_param_spec_object ("context",
|
|
||||||
"OpenGL context",
|
|
||||||
"Get OpenGL context",
|
|
||||||
GST_GL_TYPE_CONTEXT, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
||||||
|
|
||||||
klass->supported_gl_api = GST_GL_API_ANY;
|
|
||||||
klass->set_caps = NULL;
|
|
||||||
klass->filter = NULL;
|
|
||||||
klass->display_init_cb = NULL;
|
|
||||||
klass->display_reset_cb = NULL;
|
|
||||||
klass->onInitFBO = NULL;
|
|
||||||
klass->onStart = NULL;
|
|
||||||
klass->onStop = NULL;
|
|
||||||
klass->onReset = NULL;
|
|
||||||
klass->filter_texture = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_gl_filter_init (GstGLFilter * filter)
|
gst_gl_filter_init (GstGLFilter * filter)
|
||||||
{
|
{
|
||||||
gst_gl_filter_reset (filter);
|
filter->draw_attr_position_loc = -1;
|
||||||
|
filter->draw_attr_texture_loc = -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -184,38 +164,17 @@ static void
|
||||||
gst_gl_filter_get_property (GObject * object, guint prop_id,
|
gst_gl_filter_get_property (GObject * object, guint prop_id,
|
||||||
GValue * value, GParamSpec * pspec)
|
GValue * value, GParamSpec * pspec)
|
||||||
{
|
{
|
||||||
GstGLFilter *filter = GST_GL_FILTER (object);
|
|
||||||
|
|
||||||
switch (prop_id) {
|
switch (prop_id) {
|
||||||
case PROP_CONTEXT:
|
|
||||||
g_value_set_object (value, filter->context);
|
|
||||||
break;
|
|
||||||
default:
|
default:
|
||||||
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
gst_gl_filter_set_context (GstElement * element, GstContext * context)
|
|
||||||
{
|
|
||||||
GstGLFilter *filter = GST_GL_FILTER (element);
|
|
||||||
GstGLFilterClass *filter_class = GST_GL_FILTER_GET_CLASS (filter);
|
|
||||||
|
|
||||||
gst_gl_handle_set_context (element, context, &filter->display,
|
|
||||||
&filter->other_context);
|
|
||||||
if (filter->display)
|
|
||||||
gst_gl_display_filter_gl_api (filter->display,
|
|
||||||
filter_class->supported_gl_api);
|
|
||||||
}
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
gst_gl_filter_query (GstBaseTransform * trans, GstPadDirection direction,
|
gst_gl_filter_query (GstBaseTransform * trans, GstPadDirection direction,
|
||||||
GstQuery * query)
|
GstQuery * query)
|
||||||
{
|
{
|
||||||
GstGLFilter *filter = GST_GL_FILTER (trans);
|
|
||||||
GstGLFilterClass *filter_class = GST_GL_FILTER_GET_CLASS (filter);
|
|
||||||
|
|
||||||
switch (GST_QUERY_TYPE (query)) {
|
switch (GST_QUERY_TYPE (query)) {
|
||||||
case GST_QUERY_ALLOCATION:
|
case GST_QUERY_ALLOCATION:
|
||||||
{
|
{
|
||||||
|
@ -224,15 +183,6 @@ gst_gl_filter_query (GstBaseTransform * trans, GstPadDirection direction,
|
||||||
return gst_pad_peer_query (GST_BASE_TRANSFORM_SRC_PAD (trans), query);
|
return gst_pad_peer_query (GST_BASE_TRANSFORM_SRC_PAD (trans), query);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case GST_QUERY_CONTEXT:
|
|
||||||
{
|
|
||||||
gboolean ret = gst_gl_handle_context_query ((GstElement *) filter, query,
|
|
||||||
&filter->display, &filter->other_context);
|
|
||||||
if (filter->display)
|
|
||||||
gst_gl_display_filter_gl_api (filter->display,
|
|
||||||
filter_class->supported_gl_api);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -244,8 +194,6 @@ gst_gl_filter_query (GstBaseTransform * trans, GstPadDirection direction,
|
||||||
static void
|
static void
|
||||||
gst_gl_filter_reset (GstGLFilter * filter)
|
gst_gl_filter_reset (GstGLFilter * filter)
|
||||||
{
|
{
|
||||||
GstGLFilterClass *filter_class = GST_GL_FILTER_GET_CLASS (filter);
|
|
||||||
|
|
||||||
gst_caps_replace (&filter->out_caps, NULL);
|
gst_caps_replace (&filter->out_caps, NULL);
|
||||||
|
|
||||||
if (filter->upload) {
|
if (filter->upload) {
|
||||||
|
@ -273,48 +221,6 @@ gst_gl_filter_reset (GstGLFilter * filter)
|
||||||
filter->pool = NULL;
|
filter->pool = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (filter->context) {
|
|
||||||
if (filter_class->onReset)
|
|
||||||
filter_class->onReset (filter);
|
|
||||||
|
|
||||||
if (filter_class->display_reset_cb != NULL) {
|
|
||||||
gst_gl_context_thread_add (filter->context, gst_gl_filter_stop_gl,
|
|
||||||
filter);
|
|
||||||
}
|
|
||||||
/* blocking call, delete the FBO */
|
|
||||||
if (filter->fbo != 0) {
|
|
||||||
gst_gl_context_del_fbo (filter->context, filter->fbo,
|
|
||||||
filter->depthbuffer);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (filter->in_tex_id) {
|
|
||||||
gst_gl_context_del_texture (filter->context, &filter->in_tex_id);
|
|
||||||
filter->in_tex_id = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (filter->out_tex_id) {
|
|
||||||
gst_gl_context_del_texture (filter->context, &filter->out_tex_id);
|
|
||||||
filter->out_tex_id = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
gst_object_unref (filter->context);
|
|
||||||
filter->context = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (filter->display) {
|
|
||||||
gst_object_unref (filter->display);
|
|
||||||
filter->display = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
filter->fbo = 0;
|
|
||||||
filter->depthbuffer = 0;
|
|
||||||
filter->default_shader = NULL;
|
|
||||||
filter->draw_attr_position_loc = -1;
|
|
||||||
filter->draw_attr_texture_loc = -1;
|
|
||||||
if (filter->other_context)
|
|
||||||
gst_object_unref (filter->other_context);
|
|
||||||
filter->other_context = NULL;
|
|
||||||
|
|
||||||
if (filter->in_converted_caps) {
|
if (filter->in_converted_caps) {
|
||||||
gst_caps_unref (filter->in_converted_caps);
|
gst_caps_unref (filter->in_converted_caps);
|
||||||
filter->in_converted_caps = NULL;
|
filter->in_converted_caps = NULL;
|
||||||
|
@ -326,18 +232,15 @@ gst_gl_filter_start (GstBaseTransform * bt)
|
||||||
{
|
{
|
||||||
GstGLFilter *filter = GST_GL_FILTER (bt);
|
GstGLFilter *filter = GST_GL_FILTER (bt);
|
||||||
GstGLFilterClass *filter_class = GST_GL_FILTER_GET_CLASS (filter);
|
GstGLFilterClass *filter_class = GST_GL_FILTER_GET_CLASS (filter);
|
||||||
|
GstGLDisplay *display = GST_GL_BASE_FILTER (bt)->display;
|
||||||
|
|
||||||
if (!gst_gl_ensure_element_data (filter, &filter->display,
|
if (display)
|
||||||
&filter->other_context))
|
gst_gl_display_filter_gl_api (display, filter_class->supported_gl_api);
|
||||||
return FALSE;
|
|
||||||
|
|
||||||
gst_gl_display_filter_gl_api (filter->display,
|
|
||||||
filter_class->supported_gl_api);
|
|
||||||
|
|
||||||
if (filter_class->onStart)
|
if (filter_class->onStart)
|
||||||
filter_class->onStart (filter);
|
filter_class->onStart (filter);
|
||||||
|
|
||||||
return TRUE;
|
return GST_BASE_TRANSFORM_CLASS (parent_class)->start (bt);
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
|
@ -351,26 +254,85 @@ gst_gl_filter_stop (GstBaseTransform * bt)
|
||||||
|
|
||||||
gst_gl_filter_reset (filter);
|
gst_gl_filter_reset (filter);
|
||||||
|
|
||||||
|
return GST_BASE_TRANSFORM_CLASS (parent_class)->stop (bt);
|
||||||
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
gst_gl_filter_gl_start (GstGLBaseFilter * base_filter)
|
||||||
|
{
|
||||||
|
GstGLFilter *filter = GST_GL_FILTER (base_filter);
|
||||||
|
GstGLFilterClass *filter_class = GST_GL_FILTER_GET_CLASS (filter);
|
||||||
|
GstGLContext *context = GST_GL_BASE_FILTER (filter)->context;
|
||||||
|
gint in_width, in_height, out_width, out_height;
|
||||||
|
GError *error = NULL;
|
||||||
|
|
||||||
|
filter->context = base_filter->context;
|
||||||
|
|
||||||
|
in_width = GST_VIDEO_INFO_WIDTH (&filter->in_info);
|
||||||
|
in_height = GST_VIDEO_INFO_HEIGHT (&filter->in_info);
|
||||||
|
out_width = GST_VIDEO_INFO_WIDTH (&filter->out_info);
|
||||||
|
out_height = GST_VIDEO_INFO_HEIGHT (&filter->out_info);
|
||||||
|
|
||||||
|
if (filter->fbo) {
|
||||||
|
gst_gl_context_del_fbo (context, filter->fbo, filter->depthbuffer);
|
||||||
|
filter->fbo = 0;
|
||||||
|
filter->depthbuffer = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (filter->in_tex_id) {
|
||||||
|
gst_gl_context_del_texture (context, &filter->in_tex_id);
|
||||||
|
filter->in_tex_id = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (filter->out_tex_id) {
|
||||||
|
gst_gl_context_del_texture (context, &filter->out_tex_id);
|
||||||
|
filter->out_tex_id = 0;
|
||||||
|
}
|
||||||
|
//blocking call, generate a FBO
|
||||||
|
if (!gst_gl_context_gen_fbo (context, out_width, out_height,
|
||||||
|
&filter->fbo, &filter->depthbuffer))
|
||||||
|
goto context_error;
|
||||||
|
|
||||||
|
gst_gl_context_gen_texture (context, &filter->in_tex_id,
|
||||||
|
GST_VIDEO_FORMAT_RGBA, in_width, in_height);
|
||||||
|
|
||||||
|
gst_gl_context_gen_texture (context, &filter->out_tex_id,
|
||||||
|
GST_VIDEO_FORMAT_RGBA, out_width, out_height);
|
||||||
|
|
||||||
|
if (filter_class->display_init_cb)
|
||||||
|
filter_class->display_init_cb (filter);
|
||||||
|
|
||||||
|
if (filter_class->onInitFBO) {
|
||||||
|
if (!filter_class->onInitFBO (filter))
|
||||||
|
goto error;
|
||||||
|
}
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
|
||||||
|
context_error:
|
||||||
|
{
|
||||||
|
GST_ELEMENT_ERROR (filter, RESOURCE, NOT_FOUND, ("%s", error->message),
|
||||||
|
(NULL));
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
error:
|
||||||
|
{
|
||||||
|
GST_ELEMENT_ERROR (filter, LIBRARY, INIT,
|
||||||
|
("Subclass failed to initialize."), (NULL));
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gst_gl_filter_start_gl (GstGLContext * context, gpointer data)
|
gst_gl_filter_gl_stop (GstGLBaseFilter * base_filter)
|
||||||
{
|
{
|
||||||
GstGLFilter *filter = GST_GL_FILTER (data);
|
GstGLFilter *filter = GST_GL_FILTER (base_filter);
|
||||||
GstGLFilterClass *filter_class = GST_GL_FILTER_GET_CLASS (filter);
|
GstGLFilterClass *filter_class = GST_GL_FILTER_GET_CLASS (filter);
|
||||||
|
GstGLContext *context = GST_GL_BASE_FILTER (filter)->context;
|
||||||
filter_class->display_init_cb (filter);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
gst_gl_filter_stop_gl (GstGLContext * context, gpointer data)
|
|
||||||
{
|
|
||||||
const GstGLFuncs *gl = context->gl_vtable;
|
const GstGLFuncs *gl = context->gl_vtable;
|
||||||
GstGLFilter *filter = GST_GL_FILTER (data);
|
|
||||||
GstGLFilterClass *filter_class = GST_GL_FILTER_GET_CLASS (filter);
|
|
||||||
|
|
||||||
filter_class->display_reset_cb (filter);
|
if (filter_class->display_reset_cb)
|
||||||
|
filter_class->display_reset_cb (filter);
|
||||||
|
|
||||||
if (filter->vao) {
|
if (filter->vao) {
|
||||||
gl->DeleteVertexArrays (1, &filter->vao);
|
gl->DeleteVertexArrays (1, &filter->vao);
|
||||||
|
@ -381,6 +343,26 @@ gst_gl_filter_stop_gl (GstGLContext * context, gpointer data)
|
||||||
gl->DeleteBuffers (1, &filter->vertex_buffer);
|
gl->DeleteBuffers (1, &filter->vertex_buffer);
|
||||||
filter->vertex_buffer = 0;
|
filter->vertex_buffer = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (filter->fbo != 0) {
|
||||||
|
gst_gl_context_del_fbo (context, filter->fbo, filter->depthbuffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (filter->in_tex_id) {
|
||||||
|
gst_gl_context_del_texture (context, &filter->in_tex_id);
|
||||||
|
filter->in_tex_id = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (filter->out_tex_id) {
|
||||||
|
gst_gl_context_del_texture (context, &filter->out_tex_id);
|
||||||
|
filter->out_tex_id = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
filter->fbo = 0;
|
||||||
|
filter->depthbuffer = 0;
|
||||||
|
filter->default_shader = NULL;
|
||||||
|
filter->draw_attr_position_loc = -1;
|
||||||
|
filter->draw_attr_texture_loc = -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static GstCaps *
|
static GstCaps *
|
||||||
|
@ -751,6 +733,7 @@ gst_gl_filter_transform_caps (GstBaseTransform * bt,
|
||||||
GstPadDirection direction, GstCaps * caps, GstCaps * filter_caps)
|
GstPadDirection direction, GstCaps * caps, GstCaps * filter_caps)
|
||||||
{
|
{
|
||||||
GstGLFilter *filter = GST_GL_FILTER (bt);
|
GstGLFilter *filter = GST_GL_FILTER (bt);
|
||||||
|
GstGLContext *context = GST_GL_BASE_FILTER (filter)->context;
|
||||||
GstCaps *tmp = NULL;
|
GstCaps *tmp = NULL;
|
||||||
GstCaps *result = NULL;
|
GstCaps *result = NULL;
|
||||||
|
|
||||||
|
@ -762,13 +745,11 @@ gst_gl_filter_transform_caps (GstBaseTransform * bt,
|
||||||
* (convert <-> download/output) <-> srcpad
|
* (convert <-> download/output) <-> srcpad
|
||||||
*/
|
*/
|
||||||
if (direction == GST_PAD_SINK) {
|
if (direction == GST_PAD_SINK) {
|
||||||
result =
|
result = gst_gl_upload_transform_caps (context, direction, caps, NULL);
|
||||||
gst_gl_upload_transform_caps (filter->context, direction, caps, NULL);
|
|
||||||
tmp = result;
|
tmp = result;
|
||||||
|
|
||||||
result =
|
result =
|
||||||
gst_gl_color_convert_transform_caps (filter->context, direction, tmp,
|
gst_gl_color_convert_transform_caps (context, direction, tmp, NULL);
|
||||||
NULL);
|
|
||||||
gst_caps_unref (tmp);
|
gst_caps_unref (tmp);
|
||||||
} else {
|
} else {
|
||||||
GstCaps *gl_caps = gst_caps_merge (gst_gl_filter_set_caps_features (caps,
|
GstCaps *gl_caps = gst_caps_merge (gst_gl_filter_set_caps_features (caps,
|
||||||
|
@ -776,8 +757,7 @@ gst_gl_filter_transform_caps (GstBaseTransform * bt,
|
||||||
gst_gl_filter_set_caps_features (caps,
|
gst_gl_filter_set_caps_features (caps,
|
||||||
GST_CAPS_FEATURE_META_GST_VIDEO_GL_TEXTURE_UPLOAD_META));
|
GST_CAPS_FEATURE_META_GST_VIDEO_GL_TEXTURE_UPLOAD_META));
|
||||||
|
|
||||||
result =
|
result = gst_gl_download_transform_caps (context, direction, caps, NULL);
|
||||||
gst_gl_download_transform_caps (filter->context, direction, caps, NULL);
|
|
||||||
|
|
||||||
result = gst_caps_merge (gl_caps, result);
|
result = gst_caps_merge (gl_caps, result);
|
||||||
}
|
}
|
||||||
|
@ -790,20 +770,17 @@ gst_gl_filter_transform_caps (GstBaseTransform * bt,
|
||||||
|
|
||||||
if (direction == GST_PAD_SRC) {
|
if (direction == GST_PAD_SRC) {
|
||||||
result =
|
result =
|
||||||
gst_gl_color_convert_transform_caps (filter->context, direction, tmp,
|
gst_gl_color_convert_transform_caps (context, direction, tmp, NULL);
|
||||||
NULL);
|
|
||||||
gst_caps_unref (tmp);
|
gst_caps_unref (tmp);
|
||||||
tmp = result;
|
tmp = result;
|
||||||
result =
|
result = gst_gl_upload_transform_caps (context, direction, tmp, NULL);
|
||||||
gst_gl_upload_transform_caps (filter->context, direction, tmp, NULL);
|
|
||||||
} else {
|
} else {
|
||||||
GstCaps *gl_caps = gst_caps_merge (gst_gl_filter_set_caps_features (tmp,
|
GstCaps *gl_caps = gst_caps_merge (gst_gl_filter_set_caps_features (tmp,
|
||||||
GST_CAPS_FEATURE_MEMORY_GL_MEMORY),
|
GST_CAPS_FEATURE_MEMORY_GL_MEMORY),
|
||||||
gst_gl_filter_set_caps_features (tmp,
|
gst_gl_filter_set_caps_features (tmp,
|
||||||
GST_CAPS_FEATURE_META_GST_VIDEO_GL_TEXTURE_UPLOAD_META));
|
GST_CAPS_FEATURE_META_GST_VIDEO_GL_TEXTURE_UPLOAD_META));
|
||||||
|
|
||||||
result =
|
result = gst_gl_download_transform_caps (context, direction, tmp, NULL);
|
||||||
gst_gl_download_transform_caps (filter->context, direction, tmp, NULL);
|
|
||||||
|
|
||||||
result = gst_caps_merge (gl_caps, result);
|
result = gst_caps_merge (gl_caps, result);
|
||||||
}
|
}
|
||||||
|
@ -824,7 +801,6 @@ gst_gl_filter_transform_caps (GstBaseTransform * bt,
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
gst_gl_filter_get_unit_size (GstBaseTransform * trans, GstCaps * caps,
|
gst_gl_filter_get_unit_size (GstBaseTransform * trans, GstCaps * caps,
|
||||||
gsize * size)
|
gsize * size)
|
||||||
|
@ -889,6 +865,7 @@ static gboolean
|
||||||
_ensure_input_chain (GstGLFilter * filter)
|
_ensure_input_chain (GstGLFilter * filter)
|
||||||
{
|
{
|
||||||
GstBaseTransform *bt = GST_BASE_TRANSFORM (filter);
|
GstBaseTransform *bt = GST_BASE_TRANSFORM (filter);
|
||||||
|
GstGLContext *context = GST_GL_BASE_FILTER (filter)->context;
|
||||||
|
|
||||||
if (!filter->upload) {
|
if (!filter->upload) {
|
||||||
GstCapsFeatures *uploaded_features;
|
GstCapsFeatures *uploaded_features;
|
||||||
|
@ -897,7 +874,7 @@ _ensure_input_chain (GstGLFilter * filter)
|
||||||
GstVideoInfo converted_info;
|
GstVideoInfo converted_info;
|
||||||
GstCaps *in_caps = gst_pad_get_current_caps (bt->sinkpad);
|
GstCaps *in_caps = gst_pad_get_current_caps (bt->sinkpad);
|
||||||
|
|
||||||
filter->upload = gst_gl_upload_new (filter->context);
|
filter->upload = gst_gl_upload_new (context);
|
||||||
|
|
||||||
uploaded_caps = gst_caps_copy (in_caps);
|
uploaded_caps = gst_caps_copy (in_caps);
|
||||||
uploaded_features =
|
uploaded_features =
|
||||||
|
@ -912,7 +889,7 @@ _ensure_input_chain (GstGLFilter * filter)
|
||||||
gst_caps_unref (in_caps);
|
gst_caps_unref (in_caps);
|
||||||
|
|
||||||
if (!filter->in_convert) {
|
if (!filter->in_convert) {
|
||||||
filter->in_convert = gst_gl_color_convert_new (filter->context);
|
filter->in_convert = gst_gl_color_convert_new (context);
|
||||||
}
|
}
|
||||||
|
|
||||||
gst_video_info_set_format (&converted_info, GST_VIDEO_FORMAT_RGBA,
|
gst_video_info_set_format (&converted_info, GST_VIDEO_FORMAT_RGBA,
|
||||||
|
@ -942,6 +919,7 @@ gst_gl_filter_propose_allocation (GstBaseTransform * trans,
|
||||||
GstQuery * decide_query, GstQuery * query)
|
GstQuery * decide_query, GstQuery * query)
|
||||||
{
|
{
|
||||||
GstGLFilter *filter = GST_GL_FILTER (trans);
|
GstGLFilter *filter = GST_GL_FILTER (trans);
|
||||||
|
GstGLContext *context = GST_GL_BASE_FILTER (filter)->context;
|
||||||
GstStructure *config;
|
GstStructure *config;
|
||||||
GstCaps *caps;
|
GstCaps *caps;
|
||||||
guint size;
|
guint size;
|
||||||
|
@ -980,7 +958,7 @@ gst_gl_filter_propose_allocation (GstBaseTransform * trans,
|
||||||
size = info.size;
|
size = info.size;
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (filter, "create new pool");
|
GST_DEBUG_OBJECT (filter, "create new pool");
|
||||||
filter->pool = gst_gl_buffer_pool_new (filter->context);
|
filter->pool = gst_gl_buffer_pool_new (context);
|
||||||
|
|
||||||
config = gst_buffer_pool_get_config (filter->pool);
|
config = gst_buffer_pool_get_config (filter->pool);
|
||||||
gst_buffer_pool_config_set_params (config, caps, size, 0, 0);
|
gst_buffer_pool_config_set_params (config, caps, size, 0, 0);
|
||||||
|
@ -997,7 +975,7 @@ gst_gl_filter_propose_allocation (GstBaseTransform * trans,
|
||||||
|
|
||||||
gst_gl_upload_propose_allocation (filter->upload, decide_query, query);
|
gst_gl_upload_propose_allocation (filter->upload, decide_query, query);
|
||||||
|
|
||||||
if (filter->context->gl_vtable->FenceSync)
|
if (context->gl_vtable->FenceSync)
|
||||||
gst_query_add_allocation_meta (query, GST_GL_SYNC_META_API_TYPE, 0);
|
gst_query_add_allocation_meta (query, GST_GL_SYNC_META_API_TYPE, 0);
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
@ -1023,125 +1001,23 @@ config_failed:
|
||||||
static gboolean
|
static gboolean
|
||||||
gst_gl_filter_decide_allocation (GstBaseTransform * trans, GstQuery * query)
|
gst_gl_filter_decide_allocation (GstBaseTransform * trans, GstQuery * query)
|
||||||
{
|
{
|
||||||
GstGLFilter *filter = GST_GL_FILTER (trans);
|
GstGLContext *context;
|
||||||
GstGLFilterClass *filter_class = GST_GL_FILTER_GET_CLASS (filter);
|
|
||||||
GstBufferPool *pool = NULL;
|
GstBufferPool *pool = NULL;
|
||||||
GstStructure *config;
|
GstStructure *config;
|
||||||
GstCaps *caps;
|
GstCaps *caps;
|
||||||
guint min, max, size;
|
guint min, max, size;
|
||||||
gboolean update_pool;
|
gboolean update_pool;
|
||||||
guint idx;
|
|
||||||
GError *error = NULL;
|
|
||||||
guint in_width, in_height, out_width, out_height;
|
|
||||||
GstGLContext *other_context = NULL;
|
|
||||||
gboolean same_downstream_gl_context = FALSE;
|
|
||||||
|
|
||||||
gst_query_parse_allocation (query, &caps, NULL);
|
gst_query_parse_allocation (query, &caps, NULL);
|
||||||
if (!caps)
|
if (!caps)
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
if (!gst_gl_ensure_element_data (filter, &filter->display,
|
/* get gl context */
|
||||||
&filter->other_context)) {
|
if (!GST_BASE_TRANSFORM_CLASS (parent_class)->decide_allocation (trans,
|
||||||
|
query))
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
|
||||||
|
|
||||||
gst_gl_display_filter_gl_api (filter->display,
|
context = GST_GL_BASE_FILTER (trans)->context;
|
||||||
filter_class->supported_gl_api);
|
|
||||||
|
|
||||||
if (gst_query_find_allocation_meta (query,
|
|
||||||
GST_VIDEO_GL_TEXTURE_UPLOAD_META_API_TYPE, &idx)) {
|
|
||||||
GstGLContext *context;
|
|
||||||
const GstStructure *upload_meta_params;
|
|
||||||
gpointer handle;
|
|
||||||
gchar *type;
|
|
||||||
gchar *apis;
|
|
||||||
|
|
||||||
gst_query_parse_nth_allocation_meta (query, idx, &upload_meta_params);
|
|
||||||
if (upload_meta_params) {
|
|
||||||
if (gst_structure_get (upload_meta_params, "gst.gl.GstGLContext",
|
|
||||||
GST_GL_TYPE_CONTEXT, &context, NULL) && context) {
|
|
||||||
GstGLContext *old = filter->context;
|
|
||||||
|
|
||||||
filter->context = context;
|
|
||||||
if (old)
|
|
||||||
gst_object_unref (old);
|
|
||||||
same_downstream_gl_context = TRUE;
|
|
||||||
} else if (gst_structure_get (upload_meta_params, "gst.gl.context.handle",
|
|
||||||
G_TYPE_POINTER, &handle, "gst.gl.context.type", G_TYPE_STRING,
|
|
||||||
&type, "gst.gl.context.apis", G_TYPE_STRING, &apis, NULL)
|
|
||||||
&& handle) {
|
|
||||||
GstGLPlatform platform = GST_GL_PLATFORM_NONE;
|
|
||||||
GstGLAPI gl_apis;
|
|
||||||
|
|
||||||
GST_DEBUG ("got GL context handle 0x%p with type %s and apis %s",
|
|
||||||
handle, type, apis);
|
|
||||||
|
|
||||||
platform = gst_gl_platform_from_string (type);
|
|
||||||
gl_apis = gst_gl_api_from_string (apis);
|
|
||||||
|
|
||||||
if (gl_apis && platform)
|
|
||||||
other_context =
|
|
||||||
gst_gl_context_new_wrapped (filter->display, (guintptr) handle,
|
|
||||||
platform, gl_apis);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (filter->other_context) {
|
|
||||||
if (!other_context) {
|
|
||||||
other_context = filter->other_context;
|
|
||||||
} else {
|
|
||||||
GST_ELEMENT_WARNING (filter, LIBRARY, SETTINGS,
|
|
||||||
("%s", "Cannot share with more than one GL context"),
|
|
||||||
("%s", "Cannot share with more than one GL context"));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!filter->context) {
|
|
||||||
filter->context = gst_gl_context_new (filter->display);
|
|
||||||
if (!gst_gl_context_create (filter->context, other_context, &error))
|
|
||||||
goto context_error;
|
|
||||||
}
|
|
||||||
|
|
||||||
in_width = GST_VIDEO_INFO_WIDTH (&filter->in_info);
|
|
||||||
in_height = GST_VIDEO_INFO_HEIGHT (&filter->in_info);
|
|
||||||
out_width = GST_VIDEO_INFO_WIDTH (&filter->out_info);
|
|
||||||
out_height = GST_VIDEO_INFO_HEIGHT (&filter->out_info);
|
|
||||||
|
|
||||||
if (filter->fbo) {
|
|
||||||
gst_gl_context_del_fbo (filter->context, filter->fbo, filter->depthbuffer);
|
|
||||||
filter->fbo = 0;
|
|
||||||
filter->depthbuffer = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (filter->in_tex_id) {
|
|
||||||
gst_gl_context_del_texture (filter->context, &filter->in_tex_id);
|
|
||||||
filter->in_tex_id = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (filter->out_tex_id) {
|
|
||||||
gst_gl_context_del_texture (filter->context, &filter->out_tex_id);
|
|
||||||
filter->out_tex_id = 0;
|
|
||||||
}
|
|
||||||
//blocking call, generate a FBO
|
|
||||||
if (!gst_gl_context_gen_fbo (filter->context, out_width, out_height,
|
|
||||||
&filter->fbo, &filter->depthbuffer))
|
|
||||||
goto context_error;
|
|
||||||
|
|
||||||
gst_gl_context_gen_texture (filter->context, &filter->in_tex_id,
|
|
||||||
GST_VIDEO_FORMAT_RGBA, in_width, in_height);
|
|
||||||
|
|
||||||
gst_gl_context_gen_texture (filter->context, &filter->out_tex_id,
|
|
||||||
GST_VIDEO_FORMAT_RGBA, out_width, out_height);
|
|
||||||
|
|
||||||
if (filter_class->display_init_cb != NULL) {
|
|
||||||
gst_gl_context_thread_add (filter->context, gst_gl_filter_start_gl, filter);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (filter_class->onInitFBO) {
|
|
||||||
if (!filter_class->onInitFBO (filter))
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (gst_query_get_n_allocation_pools (query) > 0) {
|
if (gst_query_get_n_allocation_pools (query) > 0) {
|
||||||
gst_query_parse_nth_allocation_pool (query, 0, &pool, &size, &min, &max);
|
gst_query_parse_nth_allocation_pool (query, 0, &pool, &size, &min, &max);
|
||||||
|
@ -1157,16 +1033,9 @@ gst_gl_filter_decide_allocation (GstBaseTransform * trans, GstQuery * query)
|
||||||
update_pool = FALSE;
|
update_pool = FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!pool || (!same_downstream_gl_context
|
if (!pool)
|
||||||
&& gst_query_find_allocation_meta (query, GST_GL_SYNC_META_API_TYPE,
|
pool = gst_gl_buffer_pool_new (context);
|
||||||
NULL)
|
|
||||||
&& !gst_buffer_pool_has_option (pool,
|
|
||||||
GST_BUFFER_POOL_OPTION_GL_SYNC_META))) {
|
|
||||||
/* can't use this pool */
|
|
||||||
if (pool)
|
|
||||||
gst_object_unref (pool);
|
|
||||||
pool = gst_gl_buffer_pool_new (filter->context);
|
|
||||||
}
|
|
||||||
config = gst_buffer_pool_get_config (pool);
|
config = gst_buffer_pool_get_config (pool);
|
||||||
|
|
||||||
gst_buffer_pool_config_set_params (config, caps, size, min, max);
|
gst_buffer_pool_config_set_params (config, caps, size, min, max);
|
||||||
|
@ -1187,19 +1056,6 @@ gst_gl_filter_decide_allocation (GstBaseTransform * trans, GstQuery * query)
|
||||||
gst_object_unref (pool);
|
gst_object_unref (pool);
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
|
||||||
context_error:
|
|
||||||
{
|
|
||||||
GST_ELEMENT_ERROR (trans, RESOURCE, NOT_FOUND, ("%s", error->message),
|
|
||||||
(NULL));
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
error:
|
|
||||||
{
|
|
||||||
GST_ELEMENT_ERROR (trans, LIBRARY, INIT,
|
|
||||||
("Subclass failed to initialize."), (NULL));
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1218,6 +1074,7 @@ gst_gl_filter_filter_texture (GstGLFilter * filter, GstBuffer * inbuf,
|
||||||
GstBuffer * outbuf)
|
GstBuffer * outbuf)
|
||||||
{
|
{
|
||||||
GstGLFilterClass *filter_class;
|
GstGLFilterClass *filter_class;
|
||||||
|
GstGLContext *context = GST_GL_BASE_FILTER (filter)->context;
|
||||||
guint in_tex, out_tex, out_tex_target;
|
guint in_tex, out_tex, out_tex_target;
|
||||||
GstVideoFrame gl_frame, out_frame;
|
GstVideoFrame gl_frame, out_frame;
|
||||||
GstVideoInfo gl_info;
|
GstVideoInfo gl_info;
|
||||||
|
@ -1277,7 +1134,7 @@ gst_gl_filter_filter_texture (GstGLFilter * filter, GstBuffer * inbuf,
|
||||||
"attempting to wrap for download");
|
"attempting to wrap for download");
|
||||||
|
|
||||||
if (!filter->download)
|
if (!filter->download)
|
||||||
filter->download = gst_gl_download_new (filter->context);
|
filter->download = gst_gl_download_new (context);
|
||||||
|
|
||||||
gst_gl_download_set_format (filter->download, &out_frame.info);
|
gst_gl_download_set_format (filter->download, &out_frame.info);
|
||||||
|
|
||||||
|
@ -1314,14 +1171,13 @@ static GstFlowReturn
|
||||||
gst_gl_filter_transform (GstBaseTransform * bt, GstBuffer * inbuf,
|
gst_gl_filter_transform (GstBaseTransform * bt, GstBuffer * inbuf,
|
||||||
GstBuffer * outbuf)
|
GstBuffer * outbuf)
|
||||||
{
|
{
|
||||||
GstGLFilter *filter;
|
GstGLFilter *filter = GST_GL_FILTER (bt);
|
||||||
GstGLFilterClass *filter_class;
|
GstGLFilterClass *filter_class = GST_GL_FILTER_GET_CLASS (bt);
|
||||||
|
GstGLDisplay *display = GST_GL_BASE_FILTER (bt)->display;
|
||||||
|
GstGLContext *context = GST_GL_BASE_FILTER (bt)->context;
|
||||||
GstGLSyncMeta *out_sync_meta, *in_sync_meta;
|
GstGLSyncMeta *out_sync_meta, *in_sync_meta;
|
||||||
|
|
||||||
filter = GST_GL_FILTER (bt);
|
if (!display)
|
||||||
filter_class = GST_GL_FILTER_GET_CLASS (bt);
|
|
||||||
|
|
||||||
if (!filter->display)
|
|
||||||
return GST_FLOW_NOT_NEGOTIATED;
|
return GST_FLOW_NOT_NEGOTIATED;
|
||||||
|
|
||||||
g_assert (filter_class->filter || filter_class->filter_texture);
|
g_assert (filter_class->filter || filter_class->filter_texture);
|
||||||
|
@ -1340,7 +1196,7 @@ gst_gl_filter_transform (GstBaseTransform * bt, GstBuffer * inbuf,
|
||||||
|
|
||||||
out_sync_meta = gst_buffer_get_gl_sync_meta (outbuf);
|
out_sync_meta = gst_buffer_get_gl_sync_meta (outbuf);
|
||||||
if (out_sync_meta)
|
if (out_sync_meta)
|
||||||
gst_gl_sync_meta_set_sync_point (out_sync_meta, filter->context);
|
gst_gl_sync_meta_set_sync_point (out_sync_meta, context);
|
||||||
|
|
||||||
return GST_FLOW_OK;
|
return GST_FLOW_OK;
|
||||||
}
|
}
|
||||||
|
@ -1380,6 +1236,7 @@ void
|
||||||
gst_gl_filter_render_to_target (GstGLFilter * filter, gboolean resize,
|
gst_gl_filter_render_to_target (GstGLFilter * filter, gboolean resize,
|
||||||
GLuint input, GLuint target, GLCB func, gpointer data)
|
GLuint input, GLuint target, GLCB func, gpointer data)
|
||||||
{
|
{
|
||||||
|
GstGLContext *context = GST_GL_BASE_FILTER (filter)->context;
|
||||||
guint in_width, in_height, out_width, out_height;
|
guint in_width, in_height, out_width, out_height;
|
||||||
struct glcb2 cb;
|
struct glcb2 cb;
|
||||||
|
|
||||||
|
@ -1402,7 +1259,7 @@ gst_gl_filter_render_to_target (GstGLFilter * filter, gboolean resize,
|
||||||
cb.width = in_width;
|
cb.width = in_width;
|
||||||
cb.height = in_height;
|
cb.height = in_height;
|
||||||
|
|
||||||
gst_gl_context_use_fbo_v2 (filter->context, out_width, out_height,
|
gst_gl_context_use_fbo_v2 (context, out_width, out_height,
|
||||||
filter->fbo, filter->depthbuffer, target, _glcb2, &cb);
|
filter->fbo, filter->depthbuffer, target, _glcb2, &cb);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1410,10 +1267,11 @@ static void
|
||||||
_draw_with_shader_cb (gint width, gint height, guint texture, gpointer stuff)
|
_draw_with_shader_cb (gint width, gint height, guint texture, gpointer stuff)
|
||||||
{
|
{
|
||||||
GstGLFilter *filter = GST_GL_FILTER (stuff);
|
GstGLFilter *filter = GST_GL_FILTER (stuff);
|
||||||
GstGLFuncs *gl = filter->context->gl_vtable;
|
GstGLContext *context = GST_GL_BASE_FILTER (filter)->context;
|
||||||
|
GstGLFuncs *gl = context->gl_vtable;
|
||||||
|
|
||||||
#if GST_GL_HAVE_OPENGL
|
#if GST_GL_HAVE_OPENGL
|
||||||
if (gst_gl_context_get_gl_api (filter->context) & GST_GL_API_OPENGL) {
|
if (gst_gl_context_get_gl_api (context) & GST_GL_API_OPENGL) {
|
||||||
gl->MatrixMode (GL_PROJECTION);
|
gl->MatrixMode (GL_PROJECTION);
|
||||||
gl->LoadIdentity ();
|
gl->LoadIdentity ();
|
||||||
}
|
}
|
||||||
|
@ -1487,7 +1345,8 @@ static const GLfloat vertices[] = {
|
||||||
static void
|
static void
|
||||||
_bind_buffer (GstGLFilter * filter)
|
_bind_buffer (GstGLFilter * filter)
|
||||||
{
|
{
|
||||||
const GstGLFuncs *gl = filter->context->gl_vtable;
|
GstGLContext *context = GST_GL_BASE_FILTER (filter)->context;
|
||||||
|
const GstGLFuncs *gl = context->gl_vtable;
|
||||||
|
|
||||||
gl->BindBuffer (GL_ARRAY_BUFFER, filter->vertex_buffer);
|
gl->BindBuffer (GL_ARRAY_BUFFER, filter->vertex_buffer);
|
||||||
|
|
||||||
|
@ -1508,7 +1367,8 @@ _bind_buffer (GstGLFilter * filter)
|
||||||
static void
|
static void
|
||||||
_unbind_buffer (GstGLFilter * filter)
|
_unbind_buffer (GstGLFilter * filter)
|
||||||
{
|
{
|
||||||
const GstGLFuncs *gl = filter->context->gl_vtable;
|
GstGLContext *context = GST_GL_BASE_FILTER (filter)->context;
|
||||||
|
const GstGLFuncs *gl = context->gl_vtable;
|
||||||
|
|
||||||
gl->BindBuffer (GL_ARRAY_BUFFER, 0);
|
gl->BindBuffer (GL_ARRAY_BUFFER, 0);
|
||||||
|
|
||||||
|
@ -1529,7 +1389,7 @@ void
|
||||||
gst_gl_filter_draw_texture (GstGLFilter * filter, GLuint texture,
|
gst_gl_filter_draw_texture (GstGLFilter * filter, GLuint texture,
|
||||||
guint width, guint height)
|
guint width, guint height)
|
||||||
{
|
{
|
||||||
GstGLContext *context = filter->context;
|
GstGLContext *context = GST_GL_BASE_FILTER (filter)->context;
|
||||||
GstGLFuncs *gl = context->gl_vtable;
|
GstGLFuncs *gl = context->gl_vtable;
|
||||||
|
|
||||||
GST_DEBUG ("drawing texture:%u dimensions:%ux%u", texture, width, height);
|
GST_DEBUG ("drawing texture:%u dimensions:%ux%u", texture, width, height);
|
||||||
|
|
|
@ -24,7 +24,6 @@
|
||||||
#define _GST_GL_FILTER_H_
|
#define _GST_GL_FILTER_H_
|
||||||
|
|
||||||
#include <gst/gst.h>
|
#include <gst/gst.h>
|
||||||
#include <gst/base/gstbasetransform.h>
|
|
||||||
#include <gst/video/video.h>
|
#include <gst/video/video.h>
|
||||||
|
|
||||||
#include <gst/gl/gl.h>
|
#include <gst/gl/gl.h>
|
||||||
|
@ -39,9 +38,6 @@ GType gst_gl_filter_get_type(void);
|
||||||
#define GST_IS_GL_FILTER_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass) ,GST_TYPE_GL_FILTER))
|
#define GST_IS_GL_FILTER_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass) ,GST_TYPE_GL_FILTER))
|
||||||
#define GST_GL_FILTER_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS((obj) ,GST_TYPE_GL_FILTER,GstGLFilterClass))
|
#define GST_GL_FILTER_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS((obj) ,GST_TYPE_GL_FILTER,GstGLFilterClass))
|
||||||
|
|
||||||
typedef struct _GstGLFilter GstGLFilter;
|
|
||||||
typedef struct _GstGLFilterClass GstGLFilterClass;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* GstGLFilter:
|
* GstGLFilter:
|
||||||
* @base_transform: parent #GstBaseTransform
|
* @base_transform: parent #GstBaseTransform
|
||||||
|
@ -60,12 +56,13 @@ typedef struct _GstGLFilterClass GstGLFilterClass;
|
||||||
*/
|
*/
|
||||||
struct _GstGLFilter
|
struct _GstGLFilter
|
||||||
{
|
{
|
||||||
GstBaseTransform base_transform;
|
GstGLBaseFilter parent;
|
||||||
|
|
||||||
|
/* FIXME remove */
|
||||||
|
GstGLContext *context;
|
||||||
|
|
||||||
GstBufferPool *pool;
|
GstBufferPool *pool;
|
||||||
|
|
||||||
GstGLDisplay *display;
|
|
||||||
|
|
||||||
GstVideoInfo in_info;
|
GstVideoInfo in_info;
|
||||||
GstVideoInfo out_info;
|
GstVideoInfo out_info;
|
||||||
|
|
||||||
|
@ -88,9 +85,6 @@ struct _GstGLFilter
|
||||||
|
|
||||||
GstGLShader *default_shader;
|
GstGLShader *default_shader;
|
||||||
|
|
||||||
GstGLContext *context;
|
|
||||||
GstGLContext *other_context;
|
|
||||||
|
|
||||||
GLuint vao;
|
GLuint vao;
|
||||||
GLuint vertex_buffer;
|
GLuint vertex_buffer;
|
||||||
GLint draw_attr_position_loc;
|
GLint draw_attr_position_loc;
|
||||||
|
@ -116,7 +110,7 @@ struct _GstGLFilter
|
||||||
*/
|
*/
|
||||||
struct _GstGLFilterClass
|
struct _GstGLFilterClass
|
||||||
{
|
{
|
||||||
GstBaseTransformClass base_transform_class;
|
GstGLBaseFilterClass parent_class;
|
||||||
GstGLAPI supported_gl_api;
|
GstGLAPI supported_gl_api;
|
||||||
|
|
||||||
gboolean (*set_caps) (GstGLFilter* filter, GstCaps* incaps, GstCaps* outcaps);
|
gboolean (*set_caps) (GstGLFilter* filter, GstCaps* incaps, GstCaps* outcaps);
|
||||||
|
|
Loading…
Reference in a new issue