2009-06-24 15:58:35 +00:00
/*
* GStreamer
* Copyright ( C ) 2009 Julien Isorce < julien . isorce @ mail . 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
2012-11-08 11:53:56 +00:00
* Free Software Foundation , Inc . , 51 Franklin St , Fifth Floor ,
* Boston , MA 02110 - 1301 , USA .
2009-06-24 15:58:35 +00:00
*/
/**
* SECTION : element - deinterlace
*
* Deinterlacing using based on fragment shaders .
*
* < refsect2 >
* < title > Examples < / title >
* | [
* gst - launch videotestsrc ! glupload ! gldeinterlace ! glimagesink
* ] |
* FBO ( Frame Buffer Object ) and GLSL ( OpenGL Shading Language ) are required .
* < / refsect2 >
*/
# ifdef HAVE_CONFIG_H
# include "config.h"
# endif
# include "gstgldeinterlace.h"
# define GST_CAT_DEFAULT gst_gl_deinterlace_debug
GST_DEBUG_CATEGORY_STATIC ( GST_CAT_DEFAULT ) ;
enum
{
PROP_0
} ;
2012-05-30 03:46:21 +00:00
# define DEBUG_INIT \
2009-06-24 15:58:35 +00:00
GST_DEBUG_CATEGORY_INIT ( gst_gl_deinterlace_debug , " gldeinterlace " , 0 , " gldeinterlace element " ) ;
2012-05-30 03:46:21 +00:00
G_DEFINE_TYPE_WITH_CODE ( GstGLDeinterlace , gst_gl_deinterlace ,
GST_TYPE_GL_FILTER , DEBUG_INIT ) ;
2009-06-24 15:58:35 +00:00
static void gst_gl_deinterlace_set_property ( GObject * object ,
guint prop_id , const GValue * value , GParamSpec * pspec ) ;
static void gst_gl_deinterlace_get_property ( GObject * object ,
guint prop_id , GValue * value , GParamSpec * pspec ) ;
static void gst_gl_deinterlace_reset ( GstGLFilter * filter ) ;
2011-11-24 15:02:32 +00:00
static gboolean gst_gl_deinterlace_init_shader ( GstGLFilter * filter ) ;
2009-06-24 15:58:35 +00:00
static gboolean gst_gl_deinterlace_filter ( GstGLFilter * filter ,
2012-07-11 04:20:14 +00:00
GstBuffer * inbuf , GstBuffer * outbuf ) ;
2012-09-20 13:06:49 +00:00
static gboolean gst_gl_deinterlace_filter_texture ( GstGLFilter * filter ,
guint in_tex , guint out_tex ) ;
2009-06-24 15:58:35 +00:00
static void gst_gl_deinterlace_callback ( gint width , gint height ,
guint texture , gpointer stuff ) ;
2011-11-24 15:02:32 +00:00
static const gchar * greedyh_fragment_source = " #extension GL_ARB_texture_rectangle : enable \n " " uniform sampler2DRect tex; \n " " uniform sampler2DRect tex_prev; \n " " uniform float max_comb; \n " " uniform float motion_threshold; \n " " uniform float motion_sense; \n " " uniform int width; \n " " uniform int height; \n " " " " void main () { \n " " vec2 texcoord = gl_TexCoord[0].xy; \n " " if (int(mod(texcoord.y, 2.0)) == 0) \n " " gl_FragColor = vec4(texture2DRect(tex_prev, texcoord).rgb, 1.0); \n " " else { \n " " " // cannot have __ in a var so __ is replaced by _a
" vec2 texcoord_L1_a1, texcoord_L3_a1, texcoord_L1, texcoord_L3, texcoord_L1_1, texcoord_L3_1; \n " " vec3 L1_a1, L3_a1, L1, L3, L1_1, L3_1; \n " " " " texcoord_L1 = vec2(texcoord.x, texcoord.y - 1.0); \n " " texcoord_L3 = vec2(texcoord.x, texcoord.y + 1.0); \n " " L1 = texture2DRect(tex_prev, texcoord_L1).rgb; \n " " L3 = texture2DRect(tex_prev, texcoord_L3).rgb; \n " " if (int(ceil(texcoord.x)) == width && int(ceil(texcoord.y)) == height) { \n " " L1_1 = L1; \n " " L3_1 = L3; \n " " } else { \n " " texcoord_L1_1 = vec2(texcoord.x + 1.0, texcoord.y - 1.0); \n " " texcoord_L3_1 = vec2(texcoord.x + 1.0, texcoord.y + 1.0); \n " " L1_1 = texture2DRect(tex_prev, texcoord_L1_1).rgb; \n " " L3_1 = texture2DRect(tex_prev, texcoord_L3_1).rgb; \n " " } \n " " if (int(ceil(texcoord.x + texcoord.y)) == 0) { \n " " L1_a1 = L1; \n " " L3_a1 = L3; \n " " } else { \n " " texcoord_L1_a1 = vec2(texcoord.x - 1.0, texcoord.y - 1.0); \n " " texcoord_L3_a1 = vec2(texcoord.x - 1.0, texcoord.y + 1.0); \n " " L1_a1 = texture2DRect(tex_prev, texcoord_L1_a1).rgb; \n " " L3_a1 = texture2DRect(tex_prev, texcoord_L3_a1).rgb; \n " " } \n " " " " " //STEP 1
" vec3 avg_a1 = (L1_a1 + L3_a1) / 2.0; \n " " vec3 avg = (L1 + L3) / 2.0; \n " " vec3 avg_1 = (L1_1 + L3_1) / 2.0; \n " " " " vec3 avg_s = (avg_a1 + avg_1) / 2.0; \n " " " " vec3 avg_sc = (avg_s + avg) / 2.0; \n " " " " vec3 L2 = texture2DRect(tex, texcoord).rgb; \n " " vec3 LP2 = texture2DRect(tex_prev, texcoord).rgb; \n " " " " vec3 best; \n " " " " if (abs(L2.r - avg_sc.r) < abs(LP2.r - avg_sc.r)) { \n " " best.r = L2.r; \n " " } else { \n " " best.r = LP2.r; \n " " } \n " " " " if (abs(L2.g - avg_sc.g) < abs(LP2.g - avg_sc.g)) { \n " " best.g = L2.g; \n " " } else { \n " " best.g = LP2.g; \n " " } \n " " " " if (abs(L2.b - avg_sc.b) < abs(LP2.b - avg_sc.b)) { \n " " best.b = L2.b; \n " " } else { \n " " best.b = LP2.b; \n " " } \n " " " " " //STEP 2
" vec3 last; \n " " last.r = clamp(best.r, max(min(L1.r, L3.r) - max_comb, 0.0), min(max(L1.r, L3.r) + max_comb, 1.0)); \n " " last.g = clamp(best.g, max(min(L1.g, L3.g) - max_comb, 0.0), min(max(L1.g, L3.g) + max_comb, 1.0)); \n " " last.b = clamp(best.b, max(min(L1.b, L3.b) - max_comb, 0.0), min(max(L1.b, L3.b) + max_comb, 1.0)); \n " " " " " //STEP 3
2009-06-25 19:32:38 +00:00
" const vec3 luma = vec3 (0.299011, 0.586987, 0.114001); "
" float mov = min(max(abs(dot(L2 - LP2, luma)) - motion_threshold, 0.0) * motion_sense, 1.0); \n "
" last = last * (1.0 - mov) + avg_sc * mov; \n "
2011-11-24 15:02:32 +00:00
" " " gl_FragColor = vec4(last, 1.0); \n " " } \n " " } \n " ;
2009-06-24 15:58:35 +00:00
static void
gst_gl_deinterlace_class_init ( GstGLDeinterlaceClass * klass )
{
GObjectClass * gobject_class ;
2012-05-30 03:46:21 +00:00
GstElementClass * element_class ;
2009-06-24 15:58:35 +00:00
gobject_class = ( GObjectClass * ) klass ;
2012-05-30 03:46:21 +00:00
element_class = GST_ELEMENT_CLASS ( klass ) ;
2009-06-24 15:58:35 +00:00
gobject_class - > set_property = gst_gl_deinterlace_set_property ;
gobject_class - > get_property = gst_gl_deinterlace_get_property ;
2012-11-21 02:34:15 +00:00
gst_element_class_set_metadata ( element_class ,
2012-05-30 03:46:21 +00:00
" OpenGL deinterlacing filter " , " Deinterlace " ,
" Deinterlacing based on fragment shaders " ,
" Julien Isorce <julien.isorce@mail.com> " ) ;
2009-06-24 15:58:35 +00:00
GST_GL_FILTER_CLASS ( klass ) - > filter = gst_gl_deinterlace_filter ;
2012-09-20 13:06:49 +00:00
GST_GL_FILTER_CLASS ( klass ) - > filter_texture =
gst_gl_deinterlace_filter_texture ;
2009-06-24 15:58:35 +00:00
GST_GL_FILTER_CLASS ( klass ) - > onInitFBO = gst_gl_deinterlace_init_shader ;
GST_GL_FILTER_CLASS ( klass ) - > onReset = gst_gl_deinterlace_reset ;
}
static void
2012-05-30 03:46:21 +00:00
gst_gl_deinterlace_init ( GstGLDeinterlace * filter )
2009-06-24 15:58:35 +00:00
{
filter - > shader = NULL ;
2012-09-20 13:06:49 +00:00
filter - > prev_buffer = NULL ;
filter - > prev_tex = 0 ;
2009-06-24 15:58:35 +00:00
}
static void
gst_gl_deinterlace_reset ( GstGLFilter * filter )
{
GstGLDeinterlace * deinterlace_filter = GST_GL_DEINTERLACE ( filter ) ;
2012-09-20 13:06:49 +00:00
if ( deinterlace_filter - > prev_buffer ) {
gst_buffer_unref ( deinterlace_filter - > prev_buffer ) ;
deinterlace_filter - > prev_buffer = NULL ;
2009-06-24 15:58:35 +00:00
}
//blocking call, wait the opengl thread has destroyed the shader
gst_gl_display_del_shader ( filter - > display , deinterlace_filter - > shader ) ;
}
static void
gst_gl_deinterlace_set_property ( GObject * object , guint prop_id ,
const GValue * value , GParamSpec * pspec )
{
//GstGLDeinterlace *filter = GST_GL_DEINTERLACE (object);
switch ( prop_id ) {
default :
G_OBJECT_WARN_INVALID_PROPERTY_ID ( object , prop_id , pspec ) ;
break ;
}
}
static void
gst_gl_deinterlace_get_property ( GObject * object , guint prop_id ,
GValue * value , GParamSpec * pspec )
{
//GstGLDeinterlace *filter = GST_GL_DEINTERLACE (object);
switch ( prop_id ) {
default :
G_OBJECT_WARN_INVALID_PROPERTY_ID ( object , prop_id , pspec ) ;
break ;
}
}
2011-11-24 15:02:32 +00:00
static gboolean
2009-06-24 15:58:35 +00:00
gst_gl_deinterlace_init_shader ( GstGLFilter * filter )
{
GstGLDeinterlace * deinterlace_filter = GST_GL_DEINTERLACE ( filter ) ;
//blocking call, wait the opengl thread has compiled the shader
2011-11-24 15:02:32 +00:00
return gst_gl_display_gen_shader ( filter - > display , 0 , greedyh_fragment_source ,
2009-06-24 15:58:35 +00:00
& deinterlace_filter - > shader ) ;
}
2012-09-20 13:06:49 +00:00
static gboolean
gst_gl_deinterlace_filter_texture ( GstGLFilter * filter , guint in_tex ,
guint out_tex )
{
GstGLDeinterlace * deinterlace_filter = GST_GL_DEINTERLACE ( filter ) ;
//blocking call, use a FBO
2012-09-25 08:06:14 +00:00
gst_gl_filter_render_to_target ( filter , TRUE , in_tex , out_tex ,
2012-09-20 13:06:49 +00:00
gst_gl_deinterlace_callback , deinterlace_filter ) ;
return TRUE ;
}
2009-06-24 15:58:35 +00:00
static gboolean
2012-07-11 04:20:14 +00:00
gst_gl_deinterlace_filter ( GstGLFilter * filter , GstBuffer * inbuf ,
GstBuffer * outbuf )
2009-06-24 15:58:35 +00:00
{
GstGLDeinterlace * deinterlace_filter = GST_GL_DEINTERLACE ( filter ) ;
2012-09-20 13:06:49 +00:00
gst_gl_filter_filter_texture ( filter , inbuf , outbuf ) ;
2012-07-11 04:20:14 +00:00
2012-09-20 13:06:49 +00:00
if ( deinterlace_filter - > prev_buffer ) {
gst_buffer_unref ( deinterlace_filter - > prev_buffer ) ;
2012-07-11 04:20:14 +00:00
}
2012-09-20 13:06:49 +00:00
deinterlace_filter - > prev_buffer = gst_buffer_ref ( inbuf ) ;
2009-06-24 15:58:35 +00:00
return TRUE ;
}
//opengl scene, params: input texture (not the output filter->texture)
static void
gst_gl_deinterlace_callback ( gint width , gint height , guint texture ,
gpointer stuff )
{
GstGLDeinterlace * deinterlace_filter = GST_GL_DEINTERLACE ( stuff ) ;
GstGLFilter * filter = GST_GL_FILTER ( stuff ) ;
2012-09-20 13:06:49 +00:00
guint temp ;
2009-06-24 15:58:35 +00:00
glMatrixMode ( GL_PROJECTION ) ;
glLoadIdentity ( ) ;
gst_gl_shader_use ( deinterlace_filter - > shader ) ;
glEnable ( GL_TEXTURE_RECTANGLE_ARB ) ;
2012-09-20 13:06:49 +00:00
if ( G_UNLIKELY ( deinterlace_filter - > prev_tex = = 0 ) ) {
gst_gl_display_gen_texture_thread ( filter - > display ,
& deinterlace_filter - > prev_tex ,
GST_VIDEO_INFO_FORMAT ( & filter - > out_info ) ,
GST_VIDEO_INFO_WIDTH ( & filter - > out_info ) ,
GST_VIDEO_INFO_HEIGHT ( & filter - > out_info ) ) ;
} else {
2009-06-24 15:58:35 +00:00
glActiveTexture ( GL_TEXTURE1_ARB ) ;
gst_gl_shader_set_uniform_1i ( deinterlace_filter - > shader , " tex_prev " , 1 ) ;
2012-09-20 13:06:49 +00:00
glBindTexture ( GL_TEXTURE_RECTANGLE_ARB , deinterlace_filter - > prev_tex ) ;
2009-06-24 15:58:35 +00:00
}
glActiveTexture ( GL_TEXTURE0_ARB ) ;
gst_gl_shader_set_uniform_1i ( deinterlace_filter - > shader , " tex " , 0 ) ;
glBindTexture ( GL_TEXTURE_RECTANGLE_ARB , texture ) ;
2011-11-24 15:02:32 +00:00
gst_gl_shader_set_uniform_1f ( deinterlace_filter - > shader , " max_comb " ,
5.0f / 255.0f ) ;
gst_gl_shader_set_uniform_1f ( deinterlace_filter - > shader , " motion_threshold " ,
25.0f / 255.0f ) ;
gst_gl_shader_set_uniform_1f ( deinterlace_filter - > shader , " motion_sense " ,
30.0f / 255.0f ) ;
gst_gl_shader_set_uniform_1i ( deinterlace_filter - > shader , " width " ,
2012-09-20 13:06:49 +00:00
GST_VIDEO_INFO_WIDTH ( & filter - > out_info ) ) ;
2011-11-24 15:02:32 +00:00
gst_gl_shader_set_uniform_1i ( deinterlace_filter - > shader , " height " ,
2012-09-20 13:06:49 +00:00
GST_VIDEO_INFO_HEIGHT ( & filter - > out_info ) ) ;
2009-06-24 15:58:35 +00:00
glBegin ( GL_QUADS ) ;
glMultiTexCoord2iARB ( GL_TEXTURE0_ARB , 0 , 0 ) ;
glMultiTexCoord2iARB ( GL_TEXTURE1_ARB , 0 , 0 ) ;
glVertex2i ( - 1 , - 1 ) ;
glMultiTexCoord2iARB ( GL_TEXTURE0_ARB , width , 0 ) ;
glMultiTexCoord2iARB ( GL_TEXTURE1_ARB , width , 0 ) ;
glVertex2i ( 1 , - 1 ) ;
glMultiTexCoord2iARB ( GL_TEXTURE0_ARB , width , height ) ;
glMultiTexCoord2iARB ( GL_TEXTURE1_ARB , width , height ) ;
glVertex2i ( 1 , 1 ) ;
glMultiTexCoord2iARB ( GL_TEXTURE0_ARB , 0 , height ) ;
glMultiTexCoord2iARB ( GL_TEXTURE1_ARB , 0 , height ) ;
glVertex2i ( - 1 , 1 ) ;
glEnd ( ) ;
glDisable ( GL_TEXTURE_RECTANGLE_ARB ) ;
2012-09-20 13:06:49 +00:00
if ( texture = = filter - > in_tex_id ) {
temp = filter - > in_tex_id ;
filter - > in_tex_id = deinterlace_filter - > prev_tex ;
deinterlace_filter - > prev_tex = temp ;
} else {
deinterlace_filter - > prev_tex = texture ;
}
2009-06-24 15:58:35 +00:00
}