mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-26 00:58:12 +00:00
glshaderelement: implement on-demand create-shader signalling
One may not have an GstGLContext available or current in the thread where one would need to update the shader. Support this by signalling create-shader whenever the one-shot 'update-shader' is set to TRUE.
This commit is contained in:
parent
9650e92bea
commit
3a8fc708ee
2 changed files with 40 additions and 20 deletions
|
@ -47,6 +47,7 @@ enum
|
||||||
PROP_SHADER,
|
PROP_SHADER,
|
||||||
PROP_VERTEX,
|
PROP_VERTEX,
|
||||||
PROP_FRAGMENT,
|
PROP_FRAGMENT,
|
||||||
|
PROP_UPDATE_SHADER,
|
||||||
PROP_LAST,
|
PROP_LAST,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -111,6 +112,20 @@ gst_gl_filtershader_class_init (GstGLFilterShaderClass * klass)
|
||||||
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
||||||
/* FIXME: add other stages */
|
/* FIXME: add other stages */
|
||||||
|
|
||||||
|
g_object_class_install_property (gobject_class, PROP_UPDATE_SHADER,
|
||||||
|
g_param_spec_boolean ("update-shader", "Update Shader",
|
||||||
|
"Emit the \'create-shader\' signal for the next frame",
|
||||||
|
FALSE, G_PARAM_WRITABLE | G_PARAM_STATIC_STRINGS));
|
||||||
|
|
||||||
|
/*
|
||||||
|
* GstGLFilterShader::create-shader:
|
||||||
|
* @object: the #GstGLFilterShader
|
||||||
|
*
|
||||||
|
* Ask's the application for a shader to render with as a result of
|
||||||
|
* inititialization or setting the 'update-shader' property.
|
||||||
|
*
|
||||||
|
* Returns: a new #GstGLShader for use in the rendering pipeline
|
||||||
|
*/
|
||||||
gst_gl_shader_signals[SIGNAL_CREATE_SHADER] =
|
gst_gl_shader_signals[SIGNAL_CREATE_SHADER] =
|
||||||
g_signal_new ("create-shader", G_TYPE_FROM_CLASS (klass),
|
g_signal_new ("create-shader", G_TYPE_FROM_CLASS (klass),
|
||||||
G_SIGNAL_RUN_LAST, 0, NULL, NULL, g_cclosure_marshal_generic,
|
G_SIGNAL_RUN_LAST, 0, NULL, NULL, g_cclosure_marshal_generic,
|
||||||
|
@ -139,13 +154,6 @@ gst_gl_filtershader_init (GstGLFilterShader * filtershader)
|
||||||
static void
|
static void
|
||||||
gst_gl_filtershader_finalize (GObject * object)
|
gst_gl_filtershader_finalize (GObject * object)
|
||||||
{
|
{
|
||||||
GstGLFilterShader *filtershader = GST_GL_FILTERSHADER (object);
|
|
||||||
|
|
||||||
if (filtershader->shader_prop) {
|
|
||||||
gst_object_unref (filtershader->shader_prop);
|
|
||||||
filtershader->shader_prop = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
G_OBJECT_CLASS (gst_gl_filtershader_parent_class)->finalize (object);
|
G_OBJECT_CLASS (gst_gl_filtershader_parent_class)->finalize (object);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -158,9 +166,9 @@ gst_gl_filtershader_set_property (GObject * object, guint prop_id,
|
||||||
switch (prop_id) {
|
switch (prop_id) {
|
||||||
case PROP_SHADER:
|
case PROP_SHADER:
|
||||||
GST_OBJECT_LOCK (filtershader);
|
GST_OBJECT_LOCK (filtershader);
|
||||||
gst_object_replace ((GstObject **) & filtershader->shader_prop,
|
gst_object_replace ((GstObject **) & filtershader->shader,
|
||||||
g_value_dup_object (value));
|
g_value_dup_object (value));
|
||||||
filtershader->new_source = TRUE;
|
filtershader->new_source = FALSE;
|
||||||
GST_OBJECT_UNLOCK (filtershader);
|
GST_OBJECT_UNLOCK (filtershader);
|
||||||
break;
|
break;
|
||||||
case PROP_VERTEX:
|
case PROP_VERTEX:
|
||||||
|
@ -179,6 +187,11 @@ gst_gl_filtershader_set_property (GObject * object, guint prop_id,
|
||||||
filtershader->new_source = TRUE;
|
filtershader->new_source = TRUE;
|
||||||
GST_OBJECT_UNLOCK (filtershader);
|
GST_OBJECT_UNLOCK (filtershader);
|
||||||
break;
|
break;
|
||||||
|
case PROP_UPDATE_SHADER:
|
||||||
|
GST_OBJECT_LOCK (filtershader);
|
||||||
|
filtershader->update_shader = g_value_get_boolean (value);
|
||||||
|
GST_OBJECT_UNLOCK (filtershader);
|
||||||
|
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;
|
||||||
|
@ -194,7 +207,7 @@ gst_gl_filtershader_get_property (GObject * object, guint prop_id,
|
||||||
switch (prop_id) {
|
switch (prop_id) {
|
||||||
case PROP_SHADER:
|
case PROP_SHADER:
|
||||||
GST_OBJECT_LOCK (filtershader);
|
GST_OBJECT_LOCK (filtershader);
|
||||||
g_value_set_object (value, filtershader->shader_prop);
|
g_value_set_object (value, filtershader->shader);
|
||||||
GST_OBJECT_UNLOCK (filtershader);
|
GST_OBJECT_UNLOCK (filtershader);
|
||||||
break;
|
break;
|
||||||
case PROP_VERTEX:
|
case PROP_VERTEX:
|
||||||
|
@ -292,21 +305,28 @@ _maybe_recompile_shader (GstGLFilterShader * filtershader)
|
||||||
|
|
||||||
GST_OBJECT_LOCK (filtershader);
|
GST_OBJECT_LOCK (filtershader);
|
||||||
|
|
||||||
if (filtershader->shader_prop) {
|
if (!filtershader->shader || filtershader->update_shader) {
|
||||||
shader = gst_object_ref (filtershader->shader_prop);
|
filtershader->update_shader = FALSE;
|
||||||
|
GST_OBJECT_UNLOCK (filtershader);
|
||||||
|
g_signal_emit (filtershader, gst_gl_shader_signals[SIGNAL_CREATE_SHADER], 0,
|
||||||
|
&shader);
|
||||||
|
GST_OBJECT_LOCK (filtershader);
|
||||||
|
|
||||||
|
if (shader) {
|
||||||
|
if (filtershader->shader)
|
||||||
|
gst_object_unref (filtershader->shader);
|
||||||
|
filtershader->new_source = FALSE;
|
||||||
|
filtershader->shader = gst_object_ref (shader);
|
||||||
GST_OBJECT_UNLOCK (filtershader);
|
GST_OBJECT_UNLOCK (filtershader);
|
||||||
return shader;
|
return shader;
|
||||||
}
|
}
|
||||||
if (!filtershader->shader) {
|
}
|
||||||
g_signal_emit (filtershader, gst_gl_shader_signals[SIGNAL_CREATE_SHADER], 0,
|
|
||||||
&filtershader->shader);
|
|
||||||
if (filtershader->shader) {
|
if (filtershader->shader) {
|
||||||
filtershader->new_source = FALSE;
|
|
||||||
shader = gst_object_ref (filtershader->shader);
|
shader = gst_object_ref (filtershader->shader);
|
||||||
GST_OBJECT_UNLOCK (filtershader);
|
GST_OBJECT_UNLOCK (filtershader);
|
||||||
return shader;
|
return shader;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (filtershader->new_source) {
|
if (filtershader->new_source) {
|
||||||
GstGLSLStage *stage;
|
GstGLSLStage *stage;
|
||||||
|
|
|
@ -39,12 +39,12 @@ struct _GstGLFilterShader
|
||||||
GstGLFilter filter;
|
GstGLFilter filter;
|
||||||
|
|
||||||
/* properties */
|
/* properties */
|
||||||
GstGLShader *shader_prop;
|
GstGLShader *shader;
|
||||||
gchar *vertex;
|
gchar *vertex;
|
||||||
gchar *fragment;
|
gchar *fragment;
|
||||||
|
gboolean update_shader; /* update the shader on the next draw */
|
||||||
GstStructure *uniforms;
|
GstStructure *uniforms;
|
||||||
|
|
||||||
GstGLShader *shader;
|
|
||||||
gboolean new_source;
|
gboolean new_source;
|
||||||
gdouble time;
|
gdouble time;
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue