diff options
-rw-r--r-- | ChangeLog | 19 | ||||
-rw-r--r-- | sys/glsink/Makefile.am | 4 | ||||
-rw-r--r-- | sys/glsink/glextensions.c | 57 | ||||
-rw-r--r-- | sys/glsink/glextensions.h | 24 | ||||
-rw-r--r-- | sys/glsink/glimagesink.c | 29 | ||||
-rw-r--r-- | sys/glsink/glimagesink.h | 1 | ||||
-rw-r--r-- | sys/glsink/glvideo.c | 285 | ||||
-rw-r--r-- | sys/glsink/glvideo.h | 62 | ||||
-rw-r--r-- | sys/glsink/gstglbuffer.c | 239 | ||||
-rw-r--r-- | sys/glsink/gstglbuffer.h | 47 | ||||
-rw-r--r-- | sys/glsink/gstgldownload.c | 294 | ||||
-rw-r--r-- | sys/glsink/gstglfilter.c | 416 | ||||
-rw-r--r-- | sys/glsink/gstglupload.c | 112 | ||||
-rw-r--r-- | sys/glsink/gstopengl.c | 12 |
14 files changed, 1413 insertions, 188 deletions
@@ -1,3 +1,22 @@ +2007-12-20 David Schleef <ds@schleef.org> + + * sys/glsink/Makefile.am: + * sys/glsink/glextensions.c: + * sys/glsink/glextensions.h: + * sys/glsink/glimagesink.c: + * sys/glsink/glimagesink.h: + * sys/glsink/glvideo.c: + * sys/glsink/glvideo.h: + * sys/glsink/gstglbuffer.c: + * sys/glsink/gstglbuffer.h: + * sys/glsink/gstgldownload.c: + * sys/glsink/gstglfilter.c: + * sys/glsink/gstglupload.c: + * sys/glsink/gstopengl.c: + Switch to using framebuffer_objects instead of GLXPixmaps, + because that's what my driver supports. Remove GLDrawable, + since GstGLDisplay now has a default drawable and context. + 2007-12-20 Tim-Philipp Müller <tim at centricular dot net> * gst/equalizer/.cvsignore: diff --git a/sys/glsink/Makefile.am b/sys/glsink/Makefile.am index 43e60378..49d8ac4f 100644 --- a/sys/glsink/Makefile.am +++ b/sys/glsink/Makefile.am @@ -7,7 +7,9 @@ libgstglimagesink_la_SOURCES = \ gstopengl.c \ glextensions.c \ gstglbuffer.c \ - gstglupload.c + gstglupload.c \ + gstgldownload.c \ + gstglfilter.c libgstglimagesink_la_CFLAGS = $(GST_CFLAGS) $(X_CFLAGS) $(GST_BASE_CFLAGS) \ $(GST_PLUGINS_BASE_CFLAGS) libgstglimagesink_la_LIBADD = $(X_LIBS) $(XSHM_LIBS) -lGL \ diff --git a/sys/glsink/glextensions.c b/sys/glsink/glextensions.c index 31b862e3..6c48e8b2 100644 --- a/sys/glsink/glextensions.c +++ b/sys/glsink/glextensions.c @@ -122,3 +122,60 @@ DEFINE_FUNC_RET (glXWaitForSbcOML, Bool, DEFINE_FUNC_RET (glXSwapIntervalSGI, int, (int interval), (interval)); DEFINE_FUNC_RET (glXSwapIntervalMESA, int, (unsigned int interval), (interval)); + +#if 0 +DEFINE_FUNC (glBindFramebufferEXT, (int target, uint framebuffer), + (target, framebuffer)); + +DEFINE_FUNC (glDeleteRenderbuffersEXT, (int n, unsigned int *renderbuffers), + (n, renderbuffers)); +DEFINE_FUNC (glGenRenderbuffersEXT, (int n, unsigned int *renderbuffers), + (n, renderbuffers)); +DEFINE_FUNC (glRenderbufferStorageEXT, (int target, int internalformat, + int width, int height), (target, internalformat, width, height)); +DEFINE_FUNC (glBindRenderbufferEXT, (int target, unsigned int renderbuffer), + (target, renderbuffer)); +DEFINE_FUNC (glFramebufferRenderbufferEXT, + (int target, int attachment, int renderbuffertarget, + unsigned int renderbuffer), (target, attachment, renderbuffertarget, + renderbuffer)); +#endif + +/* EXT_framebuffer_object */ +DEFINE_FUNC_RET (glIsRenderbufferEXT, Bool, + (GLuint renderbuffer), (renderbuffer)); +DEFINE_FUNC (glBindRenderbufferEXT, + (GLenum target, GLuint renderbuffer), (target, renderbuffer)); +DEFINE_FUNC (glDeleteRenderbuffersEXT, + (GLsizei n, GLuint * renderbuffers), (n, renderbuffers)); +DEFINE_FUNC (glGenRenderbuffersEXT, + (GLsizei n, GLuint * renderbuffers), (n, renderbuffers)); +DEFINE_FUNC (glRenderbufferStorageEXT, + (GLenum target, GLenum internalformat, GLsizei width, GLsizei height), + (target, internalformat, width, height)); +DEFINE_FUNC (glGetRenderbufferParameterivEXT, + (GLenum target, GLenum pname, GLint * params), (target, pname, params)); +DEFINE_FUNC_RET (glIsFramebufferEXT, Bool, (GLuint framebuffer), (framebuffer)); +DEFINE_FUNC (glBindFramebufferEXT, + (GLenum target, GLuint framebuffer), (target, framebuffer)); +DEFINE_FUNC (glDeleteFramebuffersEXT, + (GLsizei n, GLuint * framebuffers), (n, framebuffers)); +DEFINE_FUNC (glGenFramebuffersEXT, + (GLsizei n, GLuint * framebuffers), (n, framebuffers)); +DEFINE_FUNC_RET (glCheckFramebufferStatusEXT, GLenum, + (GLenum target), (target)); +DEFINE_FUNC (glFramebufferTexture1DEXT, + (GLenum target, GLenum attachment, GLenum textarget, GLuint texture, + GLint level), (target, attachment, textarget, texture, level)); +DEFINE_FUNC (glFramebufferTexture2DEXT, (GLenum target, GLenum attachment, + GLenum textarget, GLuint texture, GLint level), (target, attachment, + textarget, texture, level)); +DEFINE_FUNC (glFramebufferTexture3DEXT, (GLenum target, GLenum attachment, + GLenum textarget, GLuint texture, GLint level, GLint zoffset), (target, + attachment, textarget, texture, level, zoffset)); +DEFINE_FUNC (glFramebufferRenderbufferEXT, (GLenum target, GLenum attachment, + GLenum renderbuffertarget, GLuint renderbuffer), (target, attachment, + renderbuffertarget, renderbuffer)); +DEFINE_FUNC (glGetFramebufferAttachmentParameterivEXT, (GLenum target, + GLenum pname, GLint * params), (target, pname, params)); +DEFINE_FUNC (glGenerateMipmapEXT, (GLenum target), (target)); diff --git a/sys/glsink/glextensions.h b/sys/glsink/glextensions.h index fd3f8ea7..48161c49 100644 --- a/sys/glsink/glextensions.h +++ b/sys/glsink/glextensions.h @@ -29,6 +29,30 @@ Bool glXWaitForSbcOML (Display *, GLXDrawable, int64_t, int64_t *, int64_t *, in int glXSwapIntervalSGI (int); int glXSwapIntervalMESA (unsigned int); +/* EXT_framebuffer_object */ +Bool glIsRenderbufferEXT (GLuint renderbuffer); +void glBindRenderbufferEXT (GLenum target, GLuint renderbuffer); +void glDeleteRenderbuffersEXT (GLsizei n, GLuint *renderbuffers); +void glGenRenderbuffersEXT (GLsizei n, GLuint *renderbuffers); +void glRenderbufferStorageEXT (GLenum target, GLenum internalformat, + GLsizei width, GLsizei height); +void glGetRenderbufferParameterivEXT (GLenum target, GLenum pname, GLint *params); +Bool glIsFramebufferEXT (GLuint framebuffer); +void glBindFramebufferEXT (GLenum target, GLuint framebuffer); +void glDeleteFramebuffersEXT (GLsizei n, GLuint *framebuffers); +void glGenFramebuffersEXT (GLsizei n, GLuint *framebuffers); +GLenum glCheckFramebufferStatusEXT (GLenum target); +void glFramebufferTexture1DEXT (GLenum target, GLenum attachment, + GLenum textarget, GLuint texture, GLint level); +void glFramebufferTexture2DEXT (GLenum target, GLenum attachment, + GLenum textarget, GLuint texture, GLint level); +void glFramebufferTexture3DEXT (GLenum target, GLenum attachment, + GLenum textarget, GLuint texture, GLint level, GLint zoffset); +void glFramebufferRenderbufferEXT (GLenum target, GLenum attachment, + GLenum renderbuffertarget, GLuint renderbuffer); +void glGetFramebufferAttachmentParameterivEXT (GLenum target, GLenum pname, + GLint *params); +void glGenerateMipmapEXT (GLenum target); #endif diff --git a/sys/glsink/glimagesink.c b/sys/glsink/glimagesink.c index 8df11e00..560a523f 100644 --- a/sys/glsink/glimagesink.c +++ b/sys/glsink/glimagesink.c @@ -282,27 +282,22 @@ static gboolean gst_glimage_sink_start (GstBaseSink * bsink) { GstGLImageSink *glimage_sink; + gboolean ret; GST_DEBUG ("start"); glimage_sink = GST_GLIMAGE_SINK (bsink); - glimage_sink->display = gst_gl_display_new (glimage_sink->display_name); - if (glimage_sink->display == NULL) { + glimage_sink->display = gst_gl_display_new (); + ret = gst_gl_display_connect (glimage_sink->display, + glimage_sink->display_name); + if (!ret) { GST_ERROR ("failed to open display"); return FALSE; } if (glimage_sink->window_id) { - glimage_sink->drawable = - gst_gl_drawable_new_from_window (glimage_sink->display, - glimage_sink->window_id); - } else { - glimage_sink->drawable = gst_gl_drawable_new_window (glimage_sink->display); - } - if (glimage_sink->drawable == NULL) { - GST_ERROR ("failed to create window"); - return FALSE; + gst_gl_display_set_window (glimage_sink->display, glimage_sink->window_id); } GST_DEBUG ("start done"); @@ -319,11 +314,9 @@ gst_glimage_sink_stop (GstBaseSink * bsink) glimage_sink = GST_GLIMAGE_SINK (bsink); - gst_gl_drawable_free (glimage_sink->drawable); - gst_gl_display_free (glimage_sink->display); + g_object_unref (glimage_sink->display); glimage_sink->display = NULL; - glimage_sink->drawable = NULL; return TRUE; } @@ -461,7 +454,7 @@ gst_glimage_sink_render (GstBaseSink * bsink, GstBuffer * buf) glimage_sink = GST_GLIMAGE_SINK (bsink); - gst_gl_drawable_draw_image (glimage_sink->drawable, + gst_gl_display_draw_image (glimage_sink->display, glimage_sink->type, GST_BUFFER_DATA (buf), GST_VIDEO_SINK_WIDTH (glimage_sink), GST_VIDEO_SINK_HEIGHT (glimage_sink)); @@ -495,11 +488,7 @@ gst_glimage_sink_set_xwindow_id (GstXOverlay * overlay, XID window_id) return; } glimage_sink->window_id = window_id; - if (glimage_sink->drawable) { - gst_gl_drawable_free (glimage_sink->drawable); - glimage_sink->drawable = - gst_gl_drawable_new_from_window (glimage_sink->display, window_id); - } + gst_gl_display_set_window (glimage_sink->display, glimage_sink->window_id); } static void diff --git a/sys/glsink/glimagesink.h b/sys/glsink/glimagesink.h index 9d8294b1..7eee7f3b 100644 --- a/sys/glsink/glimagesink.h +++ b/sys/glsink/glimagesink.h @@ -58,7 +58,6 @@ struct _GstGLImageSink int par_n, par_d; GstGLDisplay *display; - GstGLDrawable *drawable; GstGLImageType type; XID window_id; diff --git a/sys/glsink/glvideo.c b/sys/glsink/glvideo.c index 5b4e222d..f7328153 100644 --- a/sys/glsink/glvideo.c +++ b/sys/glsink/glvideo.c @@ -27,33 +27,115 @@ #include <string.h> +static void gst_gl_display_finalize (GObject * object); +static void gst_gl_display_init_tmp_window (GstGLDisplay * display); -static gboolean gst_gl_display_check_features (GstGLDisplay * display); +GST_BOILERPLATE (GstGLDisplay, gst_gl_display, GObject, G_TYPE_OBJECT); + +static void +gst_gl_display_base_init (gpointer g_class) +{ + +} + +static void +gst_gl_display_class_init (GstGLDisplayClass * klass) +{ + G_OBJECT_CLASS (klass)->finalize = gst_gl_display_finalize; +} + +static void +gst_gl_display_init (GstGLDisplay * display, GstGLDisplayClass * klass) +{ + + display->lock = g_mutex_new (); +} + +static void +gst_gl_display_finalize (GObject * object) +{ + GstGLDisplay *display = GST_GL_DISPLAY (object); + + if (display->assigned_window == None) { + XDestroyWindow (display->display, display->window); + } + if (display->context) { + glXDestroyContext (display->display, display->context); + } + if (display->visinfo) { + XFree (display->visinfo); + } + if (display->display) { + XCloseDisplay (display->display); + } + + if (display->lock) { + g_mutex_free (display->lock); + } +} + +static gboolean gst_gl_display_check_features (GstGLDisplay * display); GstGLDisplay * -gst_gl_display_new (const char *display_name) +gst_gl_display_new (void) { - GstGLDisplay *display; - gboolean usable; + return g_object_new (GST_TYPE_GL_DISPLAY, NULL); +} - display = g_malloc0 (sizeof (GstGLDisplay)); +#define HANDLE_X_ERRORS +#ifdef HANDLE_X_ERRORS +static int +x_error_handler (Display * display, XErrorEvent * event) +{ + g_assert_not_reached (); +} +#endif + +gboolean +gst_gl_display_connect (GstGLDisplay * display, const char *display_name) +{ + gboolean usable; + XGCValues values; + XPixmapFormatValues *px_formats; + int n_formats; + int i; display->display = XOpenDisplay (display_name); if (display->display == NULL) { - g_free (display); - return NULL; + return FALSE; } +#ifdef HANDLE_X_ERRORS + XSynchronize (display->display, True); + XSetErrorHandler (x_error_handler); +#endif usable = gst_gl_display_check_features (display); if (!usable) { - g_free (display); - return NULL; + return FALSE; } - display->lock = g_mutex_new (); + display->screen = DefaultScreenOfDisplay (display->display); + display->screen_num = DefaultScreen (display->display); + display->visual = DefaultVisual (display->display, display->screen_num); + display->root = DefaultRootWindow (display->display); + display->white = XWhitePixel (display->display, display->screen_num); + display->black = XBlackPixel (display->display, display->screen_num); + display->depth = DefaultDepthOfScreen (display->screen); + + display->gc = XCreateGC (display->display, + DefaultRootWindow (display->display), 0, &values); + + px_formats = XListPixmapFormats (display->display, &n_formats); + for (i = 0; i < n_formats; i++) { + GST_DEBUG ("%d: depth %d bpp %d pad %d", i, + px_formats[i].depth, + px_formats[i].bits_per_pixel, px_formats[i].scanline_pad); + } + + gst_gl_display_init_tmp_window (display); - return display; + return TRUE; } static gboolean @@ -161,56 +243,29 @@ gst_gl_display_can_handle_type (GstGLDisplay * display, GstGLImageType type) } void -gst_gl_display_free (GstGLDisplay * display) -{ - /* sure hope nobody is using it as it's being freed */ - g_mutex_lock (display->lock); - g_mutex_unlock (display->lock); - - if (display->context) { - glXDestroyContext (display->display, display->context); - } - if (display->visinfo) { - XFree (display->visinfo); - } - if (display->display) { - XCloseDisplay (display->display); - } - - g_mutex_free (display->lock); - - g_free (display); -} - -void gst_gl_display_lock (GstGLDisplay * display) { g_mutex_lock (display->lock); + glXMakeCurrent (display->display, display->window, display->context); } void gst_gl_display_unlock (GstGLDisplay * display) { + glXMakeCurrent (display->display, None, NULL); g_mutex_unlock (display->lock); } - -/* drawable */ - -GstGLDrawable * -gst_gl_drawable_new_window (GstGLDisplay * display) +static void +gst_gl_display_init_tmp_window (GstGLDisplay * display) { - GstGLDrawable *drawable; XSetWindowAttributes attr = { 0 }; int scrnum; int mask; Window root; Screen *screen; - drawable = g_malloc0 (sizeof (GstGLDrawable)); - - g_mutex_lock (display->lock); - drawable->display = display; + GST_ERROR ("creating temp window"); screen = XDefaultScreenOfDisplay (display->display); scrnum = XScreenNumberOfScreen (screen); @@ -229,114 +284,72 @@ gst_gl_drawable_new_window (GstGLDisplay * display) mask = CWBackPixel | CWBorderPixel | CWColormap | CWOverrideRedirect; - drawable->window = XCreateWindow (display->display, + display->window = XCreateWindow (display->display, root, 0, 0, 100, 100, 0, display->visinfo->depth, InputOutput, display->visinfo->visual, mask, &attr); - XMapWindow (display->display, drawable->window); - drawable->destroy_on_free = TRUE; - - g_mutex_unlock (display->lock); - - return drawable; -} - -GstGLDrawable * -gst_gl_drawable_new_root_window (GstGLDisplay * display) -{ - GstGLDrawable *drawable; - int scrnum; - Screen *screen; - - drawable = g_malloc0 (sizeof (GstGLDrawable)); - - g_mutex_lock (display->lock); - drawable->display = display; - - screen = XDefaultScreenOfDisplay (display->display); - scrnum = XScreenNumberOfScreen (screen); - - drawable->window = XRootWindow (display->display, scrnum); - drawable->destroy_on_free = FALSE; - g_mutex_unlock (display->lock); - - return drawable; + XMapWindow (display->display, display->window); + XSync (display->display, FALSE); } -GstGLDrawable * -gst_gl_drawable_new_from_window (GstGLDisplay * display, Window window) +static void +gst_gl_display_destroy_tmp_window (GstGLDisplay * display) { - GstGLDrawable *drawable; - - drawable = g_malloc0 (sizeof (GstGLDrawable)); - - g_mutex_lock (display->lock); - drawable->display = display; - - drawable->window = window; - drawable->destroy_on_free = FALSE; - - g_mutex_unlock (display->lock); - return drawable; + XDestroyWindow (display->display, display->window); } void -gst_gl_drawable_free (GstGLDrawable * drawable) +gst_gl_display_set_window (GstGLDisplay * display, Window window) { + g_mutex_lock (display->lock); - g_mutex_lock (drawable->display->lock); - if (drawable->destroy_on_free) { - XDestroyWindow (drawable->display->display, drawable->window); + if (window != display->assigned_window) { + if (display->assigned_window == None) { + gst_gl_display_destroy_tmp_window (display); + } + display->assigned_window = window; + if (display->assigned_window == None) { + gst_gl_display_init_tmp_window (display); + } else { + display->window = window; + } } - g_mutex_unlock (drawable->display->lock); - - g_free (drawable); -} - -void -gst_gl_drawable_lock (GstGLDrawable * drawable) -{ - g_mutex_lock (drawable->display->lock); - glXMakeCurrent (drawable->display->display, drawable->window, - drawable->display->context); -} -void -gst_gl_drawable_unlock (GstGLDrawable * drawable) -{ - glXMakeCurrent (drawable->display->display, None, NULL); - g_mutex_unlock (drawable->display->lock); + g_mutex_unlock (display->lock); } void -gst_gl_drawable_update_attributes (GstGLDrawable * drawable) +gst_gl_display_update_attributes (GstGLDisplay * display) { XWindowAttributes attr; - XGetWindowAttributes (drawable->display->display, drawable->window, &attr); - drawable->win_width = attr.width; - drawable->win_height = attr.height; - + if (display->window != None) { + XGetWindowAttributes (display->display, display->window, &attr); + display->win_width = attr.width; + display->win_height = attr.height; + } else { + display->win_width = 0; + display->win_height = 0; + } } void -gst_gl_drawable_clear (GstGLDrawable * drawable) +gst_gl_display_clear (GstGLDisplay * display) { - - gst_gl_drawable_lock (drawable); + gst_gl_display_lock (display); glDepthFunc (GL_LESS); glEnable (GL_DEPTH_TEST); glClearColor (0.2, 0.2, 0.2, 1.0); - glViewport (0, 0, drawable->win_width, drawable->win_height); + glViewport (0, 0, display->win_width, display->win_height); - gst_gl_drawable_unlock (drawable); + gst_gl_display_unlock (display); } static void -draw_rect_texture (GstGLDrawable * drawable, GstGLImageType type, +draw_rect_texture (GstGLDisplay * display, GstGLImageType type, void *data, int width, int height) { GLuint texture; @@ -437,7 +450,7 @@ draw_rect_texture (GstGLDrawable * drawable, GstGLImageType type, } static void -draw_pow2_texture (GstGLDrawable * drawable, GstGLImageType type, +draw_pow2_texture (GstGLDisplay * display, GstGLImageType type, void *data, int width, int height) { int pow2_width; @@ -544,14 +557,14 @@ draw_pow2_texture (GstGLDrawable * drawable, GstGLImageType type, } void -gst_gl_drawable_draw_image (GstGLDrawable * drawable, GstGLImageType type, +gst_gl_display_draw_image (GstGLDisplay * display, GstGLImageType type, void *data, int width, int height) { g_return_if_fail (data != NULL); g_return_if_fail (width > 0); g_return_if_fail (height > 0); - gst_gl_drawable_lock (drawable); + gst_gl_display_lock (display); #if 0 /* Doesn't work */ @@ -561,9 +574,9 @@ gst_gl_drawable_draw_image (GstGLDrawable * drawable, GstGLImageType type, int64_t sbc = 1234; gboolean ret; - ret = glXGetSyncValuesOML (drawable->display->display, drawable->window, + ret = glXGetSyncValuesOML (display->display, display->window, &ust, &mst, &sbc); - GST_ERROR ("sync values %d %lld %lld %lld", ret, ust, mst, sbc); + GST_DEBUG ("sync values %d %lld %lld %lld", ret, ust, mst, sbc); } #endif @@ -574,16 +587,15 @@ gst_gl_drawable_draw_image (GstGLDrawable * drawable, GstGLImageType type, int32_t den = 1234; gboolean ret; - ret = glXGetMscRateOML (drawable->display->display, drawable->window, - &num, &den); - GST_ERROR ("rate %d %d %d", ret, num, den); + ret = glXGetMscRateOML (display->display, display->window, &num, &den); + GST_DEBUG ("rate %d %d %d", ret, num, den); } #endif - gst_gl_drawable_update_attributes (drawable); + gst_gl_display_update_attributes (display); - glXSwapIntervalSGI (1); - glViewport (0, 0, drawable->win_width, drawable->win_height); + //glXSwapIntervalSGI (1); + glViewport (0, 0, display->win_width, display->win_height); glClear (GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); @@ -598,23 +610,22 @@ gst_gl_drawable_draw_image (GstGLDrawable * drawable, GstGLImageType type, glColor4f (1, 1, 1, 1); - if (drawable->display->have_texture_rectangle) { - draw_rect_texture (drawable, type, data, width, height); + if (display->have_texture_rectangle) { + draw_rect_texture (display, type, data, width, height); } else { - draw_pow2_texture (drawable, type, data, width, height); + draw_pow2_texture (display, type, data, width, height); } - glXSwapBuffers (drawable->display->display, drawable->window); + glXSwapBuffers (display->display, display->window); #if 0 /* Doesn't work */ { - ret = glXSwapBuffersMscOML (drawable->display->display, drawable->window, - 0, 1, 0); + ret = glXSwapBuffersMscOML (display->display, display->window, 0, 1, 0); if (ret == 0) { - GST_ERROR ("glXSwapBuffersMscOML failed"); + GST_DEBUG ("glXSwapBuffersMscOML failed"); } } #endif - gst_gl_drawable_unlock (drawable); + gst_gl_display_unlock (display); } diff --git a/sys/glsink/glvideo.h b/sys/glsink/glvideo.h index e1bdfe92..aef064e3 100644 --- a/sys/glsink/glvideo.h +++ b/sys/glsink/glvideo.h @@ -6,9 +6,6 @@ #include <GL/gl.h> #include <gst/gst.h> -typedef struct _GstGLDisplay GstGLDisplay; -typedef struct _GstGLDrawable GstGLDrawable; - typedef enum { GST_GL_IMAGE_TYPE_RGBx, GST_GL_IMAGE_TYPE_BGRx, @@ -19,54 +16,71 @@ typedef enum { GST_GL_IMAGE_TYPE_AYUV, } GstGLImageType; +typedef struct _GstGLDisplay GstGLDisplay; +typedef struct _GstGLDisplayClass GstGLDisplayClass; + +#define GST_TYPE_GL_DISPLAY \ + (gst_gl_display_get_type()) +#define GST_GL_DISPLAY(obj) \ + (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_GL_DISPLAY,GstGLDisplay)) +#define GST_GL_DISPLAY_CLASS(klass) \ + (G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_GL_DISPLAY,GstGLDisplayClass)) +#define GST_IS_GL_DISPLAY(obj) \ + (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_GL_DISPLAY)) +#define GST_IS_GL_DISPLAY_CLASS(klass) \ + (G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_GL_DISPLAY)) + struct _GstGLDisplay { + GObject object; + Display *display; + GC gc; XVisualInfo *visinfo; GLXContext context; GMutex *lock; Screen *screen; - int scrnum; + int screen_num; + Visual *visual; Window root; + guint32 white; + guint32 black; + int depth; int max_texture_size; gboolean have_ycbcr_texture; gboolean have_texture_rectangle; gboolean have_color_matrix; -}; - -struct _GstGLDrawable { - GstGLDisplay *display; Window window; - - gboolean destroy_on_free; + Window assigned_window; int win_width; int win_height; + +}; + +struct _GstGLDisplayClass { + GObjectClass object_class; }; +GType gst_gl_display_get_type (void); + -GstGLDisplay *gst_gl_display_new (const char *display_name); +GstGLDisplay *gst_gl_display_new (void); +gboolean gst_gl_display_connect (GstGLDisplay *display, + const char *display_name); gboolean gst_gl_display_can_handle_type (GstGLDisplay *display, GstGLImageType type); -void gst_gl_display_free (GstGLDisplay *display); void gst_gl_display_lock (GstGLDisplay *display); void gst_gl_display_unlock (GstGLDisplay *display); - -/* drawable */ - -GstGLDrawable * gst_gl_drawable_new_window (GstGLDisplay *display); -GstGLDrawable * gst_gl_drawable_new_root_window (GstGLDisplay *display); -GstGLDrawable * gst_gl_drawable_new_from_window (GstGLDisplay *display, Window window); -void gst_gl_drawable_free (GstGLDrawable *drawable); -void gst_gl_drawable_lock (GstGLDrawable *drawable); -void gst_gl_drawable_unlock (GstGLDrawable *drawable); -void gst_gl_drawable_update_attributes (GstGLDrawable *drawable); -void gst_gl_drawable_clear (GstGLDrawable *drawable); -void gst_gl_drawable_draw_image (GstGLDrawable *drawable, GstGLImageType type, void *data, int width, int height); +void gst_gl_display_set_window (GstGLDisplay *display, Window window); +void gst_gl_display_update_attributes (GstGLDisplay *display); +void gst_gl_display_clear (GstGLDisplay *display); +void gst_gl_display_draw_image (GstGLDisplay * display, GstGLImageType type, + void *data, int width, int height); #endif diff --git a/sys/glsink/gstglbuffer.c b/sys/glsink/gstglbuffer.c new file mode 100644 index 00000000..04421ce7 --- /dev/null +++ b/sys/glsink/gstglbuffer.c @@ -0,0 +1,239 @@ + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include <gstglbuffer.h> +#include <glvideo.h> +#include "glextensions.h" + +#include <string.h> + +static GObjectClass *gst_gl_buffer_parent_class; + +static void +gst_gl_buffer_finalize (GstGLBuffer * buffer) +{ + gst_gl_display_lock (buffer->display); + + switch (buffer->type) { + case GST_GL_BUFFER_XIMAGE: + GST_DEBUG ("freeing pixmap %ld", buffer->pixmap); + XFreeGC (buffer->display->display, buffer->gc); + XFreePixmap (buffer->display->display, buffer->pixmap); + break; + case GST_GL_BUFFER_RBO: + glDeleteRenderbuffersEXT (1, &buffer->rbo); + break; + default: + g_assert_not_reached (); + break; + } + gst_gl_display_unlock (buffer->display); + g_object_unref (buffer->display); + + GST_MINI_OBJECT_CLASS (gst_gl_buffer_parent_class)-> + finalize (GST_MINI_OBJECT (buffer)); +} + +static void +gst_gl_buffer_init (GstGLBuffer * buffer, gpointer g_class) +{ + +} + +static void +gst_gl_buffer_class_init (gpointer g_class, gpointer class_data) +{ + GstMiniObjectClass *mini_object_class = GST_MINI_OBJECT_CLASS (g_class); + + gst_gl_buffer_parent_class = g_type_class_peek_parent (g_class); + + mini_object_class->finalize = (GstMiniObjectFinalizeFunction) + gst_gl_buffer_finalize; +} + + +GType +gst_gl_buffer_get_type (void) +{ + static GType _gst_gl_buffer_type; + + if (G_UNLIKELY (_gst_gl_buffer_type == 0)) { + static const GTypeInfo info = { + sizeof (GstBufferClass), + NULL, + NULL, + gst_gl_buffer_class_init, + NULL, + NULL, + sizeof (GstGLBuffer), + 0, + (GInstanceInitFunc) gst_gl_buffer_init, + NULL + }; + _gst_gl_buffer_type = g_type_register_static (GST_TYPE_BUFFER, + "GstGLBuffer", &info, 0); + } + return _gst_gl_buffer_type; +} + + +GstGLBuffer * +gst_gl_buffer_new (GstGLDisplay * display, GstVideoFormat format, + int width, int height) +{ + GstGLBuffer *buffer; + XGCValues values = { 0 }; + + g_return_val_if_fail (format == GST_VIDEO_FORMAT_BGRx, NULL); + g_return_val_if_fail (width > 0, NULL); + g_return_val_if_fail (height > 0, NULL); + + buffer = (GstGLBuffer *) gst_mini_object_new (GST_TYPE_GL_BUFFER); + + buffer->display = g_object_ref (display); + buffer->type = GST_GL_BUFFER_RBO; + + buffer->width = width; + buffer->height = height; + + switch (buffer->type) { + case GST_GL_BUFFER_XIMAGE: + { + buffer->pixmap = XCreatePixmap (display->display, + DefaultRootWindow (display->display), width, height, 32); + XSync (display->display, False); + + buffer->gc = XCreateGC (display->display, buffer->pixmap, 0, &values); + + GST_DEBUG ("new pixmap %dx%d xid %ld", width, height, buffer->pixmap); + break; + } + case GST_GL_BUFFER_RBO: + { + gst_gl_display_lock (buffer->display); + + glGenRenderbuffersEXT (1, &buffer->rbo); + glBindRenderbufferEXT (GL_RENDERBUFFER_EXT, buffer->rbo); + + glFramebufferRenderbufferEXT (GL_FRAMEBUFFER_EXT, + GL_COLOR_ATTACHMENT1_EXT, GL_RENDERBUFFER_EXT, buffer->rbo); + glRenderbufferStorageEXT (GL_RENDERBUFFER_EXT, GL_RGB, + buffer->width, buffer->height); + + + + gst_gl_display_unlock (buffer->display); + break; + } + default: + g_assert_not_reached (); + } + + return buffer; +} + +void +gst_gl_buffer_upload (GstGLBuffer * buffer, void *data) +{ + Display *display = buffer->display->display; + + GST_DEBUG ("uploading %p %dx%d", data, buffer->width, buffer->height); + + gst_gl_display_lock (buffer->display); + + switch (buffer->type) { + case GST_GL_BUFFER_XIMAGE: + { + XImage *image; + Visual *visual; + int depth; + int bpp; + + visual = DefaultVisual (display, 0); + depth = 32; + bpp = 32; + + image = XCreateImage (display, visual, depth, ZPixmap, 0, NULL, + buffer->width, buffer->height, bpp, 0); + GST_DEBUG ("image %p", image); + image->data = data; + + XPutImage (display, buffer->pixmap, buffer->gc, + image, 0, 0, 0, 0, buffer->width, buffer->height); + + XDestroyImage (image); + break; + } + case GST_GL_BUFFER_RBO: + { + unsigned int fbo; + + glGenFramebuffersEXT (1, &fbo); + glBindFramebufferEXT (GL_FRAMEBUFFER_EXT, fbo); + + glFramebufferRenderbufferEXT (GL_FRAMEBUFFER_EXT, + GL_COLOR_ATTACHMENT1_EXT, GL_RENDERBUFFER_EXT, buffer->rbo); + + glDrawBuffer (GL_COLOR_ATTACHMENT1_EXT); + //glWindowPos2iARB(0, 0); + glDrawPixels (buffer->width, buffer->height, GL_RGBA, + GL_UNSIGNED_BYTE, data); + + glDeleteFramebuffersEXT (1, &fbo); + + break; + } + default: + g_assert_not_reached (); + } + + gst_gl_display_unlock (buffer->display); +} + + +void +gst_gl_buffer_download (GstGLBuffer * buffer, void *data) +{ + gst_gl_display_lock (buffer->display); + + GST_DEBUG ("downloading"); + + switch (buffer->type) { + case GST_GL_BUFFER_XIMAGE: + { + XImage *image; + + image = XGetImage (buffer->display->display, buffer->pixmap, + 0, 0, buffer->width, buffer->height, 0xffffffff, ZPixmap); + + memcpy (data, image->data, buffer->width * buffer->height * 4); + + XDestroyImage (image); + break; + } + case GST_GL_BUFFER_RBO: + { + unsigned int fbo; + + glGenFramebuffersEXT (1, &fbo); + glBindFramebufferEXT (GL_FRAMEBUFFER_EXT, fbo); + + glFramebufferRenderbufferEXT (GL_FRAMEBUFFER_EXT, + GL_COLOR_ATTACHMENT1_EXT, GL_RENDERBUFFER_EXT, buffer->rbo); + + glReadBuffer (GL_COLOR_ATTACHMENT1_EXT); + glReadPixels (0, 0, buffer->width, buffer->height, GL_RGBA, + GL_UNSIGNED_BYTE, data); + + glDeleteFramebuffersEXT (1, &fbo); + + break; + } + default: + g_assert_not_reached (); + } + + gst_gl_display_unlock (buffer->display); +} diff --git a/sys/glsink/gstglbuffer.h b/sys/glsink/gstglbuffer.h new file mode 100644 index 00000000..969f2161 --- /dev/null +++ b/sys/glsink/gstglbuffer.h @@ -0,0 +1,47 @@ + +#ifndef _GST_GL_BUFFER_H_ +#define _GST_GL_BUFFER_H_ + +#include <gst/gst.h> +#include <gst/video/video.h> +#include <xcb/xcb.h> +#include <glvideo.h> + +typedef struct _GstGLBuffer GstGLBuffer; + +#define GST_TYPE_GL_BUFFER (gst_gl_buffer_get_type()) + +#define GST_IS_GL_BUFFER(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GST_TYPE_GL_BUFFER)) +#define GST_GL_BUFFER(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GST_TYPE_GL_BUFFER, GstGLBuffer)) + +typedef enum { + GST_GL_BUFFER_UNKNOWN, + GST_GL_BUFFER_XIMAGE, + GST_GL_BUFFER_RBO +} GstGLBufferType; + +struct _GstGLBuffer { + GstBuffer buffer; + + GstGLDisplay *display; + + GstGLBufferType type; + + XID pixmap; + GC gc; + + GLuint rbo; + + int width; + int height; +}; + +GType gst_gl_buffer_get_type (void); + +GstGLBuffer * gst_gl_buffer_new (GstGLDisplay *display, GstVideoFormat format, + int width, int height); +void gst_gl_buffer_upload (GstGLBuffer *buffer, void *data); +void gst_gl_buffer_download (GstGLBuffer *buffer, void *data); + +#endif + diff --git a/sys/glsink/gstgldownload.c b/sys/glsink/gstgldownload.c new file mode 100644 index 00000000..55f9e759 --- /dev/null +++ b/sys/glsink/gstgldownload.c @@ -0,0 +1,294 @@ +/* + * GStreamer + * Copyright (C) 2007 David Schleef <ds@schleef.org> + * + * 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., 59 Temple Place - Suite 330, + * Boston, MA 02111-1307, USA. + */ + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include <gst/gst.h> +#include <gst/video/video.h> +#include <gstglbuffer.h> + +#define GST_CAT_DEFAULT gst_gl_download_debug +GST_DEBUG_CATEGORY_STATIC (GST_CAT_DEFAULT); + +#define GST_TYPE_GL_DOWNLOAD (gst_gl_download_get_type()) +#define GST_GL_DOWNLOAD(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_GL_DOWNLOAD,GstGLDownload)) +#define GST_IS_GL_DOWNLOAD(obj) (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_GL_DOWNLOAD)) +#define GST_GL_DOWNLOAD_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST((klass) ,GST_TYPE_GL_DOWNLOAD,GstGLDownloadClass)) +#define GST_IS_GL_DOWNLOAD_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE((klass) ,GST_TYPE_GL_DOWNLOAD)) +#define GST_GL_DOWNLOAD_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS((obj) ,GST_TYPE_GL_DOWNLOAD,GstGLDownloadClass)) +typedef struct _GstGLDownload GstGLDownload; +typedef struct _GstGLDownloadClass GstGLDownloadClass; + +typedef void (*GstGLDownloadProcessFunc) (GstGLDownload *, guint8 *, guint); + +struct _GstGLDownload +{ + GstElement element; + + GstPad *srcpad; + GstPad *sinkpad; + + /* < private > */ + + GstGLDisplay *display; + GstVideoFormat format; + int width; + int height; +}; + +struct _GstGLDownloadClass +{ + GstElementClass element_class; +}; + +static const GstElementDetails element_details = GST_ELEMENT_DETAILS ("FIXME", + "Filter/Effect", + "FIXME example filter", + "FIXME <fixme@fixme.com>"); + +#define GST_GL_VIDEO_CAPS "video/x-raw-gl" + +static GstStaticPadTemplate gst_gl_download_src_pad_template = +GST_STATIC_PAD_TEMPLATE ("src", + GST_PAD_SRC, + GST_PAD_ALWAYS, + GST_STATIC_CAPS (GST_VIDEO_CAPS_BGRx) + ); + +static GstStaticPadTemplate gst_gl_download_sink_pad_template = +GST_STATIC_PAD_TEMPLATE ("sink", + GST_PAD_SINK, + GST_PAD_ALWAYS, + GST_STATIC_CAPS (GST_GL_VIDEO_CAPS) + ); + +enum +{ + PROP_0 +}; + +#define DEBUG_INIT(bla) \ + GST_DEBUG_CATEGORY_INIT (gst_gl_download_debug, "gldownload", 0, "gldownload element"); + +GST_BOILERPLATE_FULL (GstGLDownload, gst_gl_download, GstElement, + GST_TYPE_ELEMENT, DEBUG_INIT); + +static void gst_gl_download_set_property (GObject * object, guint prop_id, + const GValue * value, GParamSpec * pspec); +static void gst_gl_download_get_property (GObject * object, guint prop_id, + GValue * value, GParamSpec * pspec); + +static GstFlowReturn gst_gl_download_chain (GstPad * pad, GstBuffer * buf); +static void gst_gl_download_reset (GstGLDownload * download); +static GstStateChangeReturn +gst_gl_download_change_state (GstElement * element, GstStateChange transition); +static gboolean gst_gl_download_sink_setcaps (GstPad * pad, GstCaps * caps); + + +static void +gst_gl_download_base_init (gpointer klass) +{ + GstElementClass *element_class = GST_ELEMENT_CLASS (klass); + + gst_element_class_set_details (element_class, &element_details); + + gst_element_class_add_pad_template (element_class, + gst_static_pad_template_get (&gst_gl_download_src_pad_template)); + gst_element_class_add_pad_template (element_class, + gst_static_pad_template_get (&gst_gl_download_sink_pad_template)); +} + +static void +gst_gl_download_class_init (GstGLDownloadClass * klass) +{ + GObjectClass *gobject_class; + + gobject_class = (GObjectClass *) klass; + gobject_class->set_property = gst_gl_download_set_property; + gobject_class->get_property = gst_gl_download_get_property; + + GST_ELEMENT_CLASS (klass)->change_state = gst_gl_download_change_state; +} + +static void +gst_gl_download_init (GstGLDownload * download, GstGLDownloadClass * klass) +{ + gst_element_create_all_pads (GST_ELEMENT (download)); + + download->sinkpad = + gst_element_get_static_pad (GST_ELEMENT (download), "sink"); + download->srcpad = gst_element_get_static_pad (GST_ELEMENT (download), "src"); + + gst_pad_set_setcaps_function (download->sinkpad, + gst_gl_download_sink_setcaps); + gst_pad_set_chain_function (download->sinkpad, gst_gl_download_chain); + + gst_gl_download_reset (download); +} + +static void +gst_gl_download_set_property (GObject * object, guint prop_id, + const GValue * value, GParamSpec * pspec) +{ + //GstGLDownload *download = GST_GL_DOWNLOAD (object); + + switch (prop_id) { + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); + break; + } +} + +static void +gst_gl_download_get_property (GObject * object, guint prop_id, + GValue * value, GParamSpec * pspec) +{ + //GstGLDownload *download = GST_GL_DOWNLOAD (object); + + switch (prop_id) { + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); + break; + } +} + +static void +gst_gl_download_reset (GstGLDownload * download) +{ + if (download->display) { + g_object_unref (download->display); + download->display = NULL; + } + download->format = GST_VIDEO_FORMAT_BGRx; +} + +static gboolean +gst_gl_download_start (GstGLDownload * download) +{ + gboolean ret; + + download->format = GST_VIDEO_FORMAT_BGRx; + download->display = gst_gl_display_new (); + ret = gst_gl_display_connect (download->display, NULL); + + return ret; +} + +static gboolean +gst_gl_download_stop (GstGLDownload * download) +{ + gst_gl_download_reset (download); + + return TRUE; +} + +static gboolean +gst_gl_download_sink_setcaps (GstPad * pad, GstCaps * caps) +{ + GstGLDownload *download; + gboolean ret; + GstStructure *structure; + GstCaps *srccaps; + + download = GST_GL_DOWNLOAD (gst_pad_get_parent (pad)); + + structure = gst_caps_get_structure (caps, 0); + + ret = gst_structure_get_int (structure, "width", &download->width); + ret &= gst_structure_get_int (structure, "height", &download->height); + if (!ret) + return FALSE; + + srccaps = gst_video_format_new_caps (download->format, + download->width, download->height, 30, 1, 1, 1); + ret = gst_pad_set_caps (download->srcpad, srccaps); + gst_caps_unref (srccaps); + + return ret; +} + +static GstFlowReturn +gst_gl_download_chain (GstPad * pad, GstBuffer * buf) +{ + GstGLDownload *download; + GstGLBuffer *inbuf = GST_GL_BUFFER (buf); + GstBuffer *outbuf; + + GST_ERROR ("got here"); + download = GST_GL_DOWNLOAD (gst_pad_get_parent (pad)); + + outbuf = gst_buffer_new_and_alloc (inbuf->width * inbuf->height * 4); + + gst_buffer_copy_metadata (GST_BUFFER (outbuf), buf, + GST_BUFFER_COPY_TIMESTAMPS | GST_BUFFER_COPY_FLAGS); + gst_buffer_set_caps (GST_BUFFER (outbuf), GST_PAD_CAPS (download->srcpad)); + + GST_ERROR ("downloading %p size %d", + GST_BUFFER_DATA (outbuf), GST_BUFFER_SIZE (outbuf)); + gst_gl_buffer_download (inbuf, GST_BUFFER_DATA (outbuf)); + + gst_pad_push (download->srcpad, GST_BUFFER (outbuf)); + + gst_object_unref (download); + return GST_FLOW_OK; +} + +static GstStateChangeReturn +gst_gl_download_change_state (GstElement * element, GstStateChange transition) +{ + GstGLDownload *download; + GstStateChangeReturn ret = GST_STATE_CHANGE_SUCCESS; + + GST_DEBUG ("change state"); + + download = GST_GL_DOWNLOAD (element); + + switch (transition) { + case GST_STATE_CHANGE_NULL_TO_READY: + break; + case GST_STATE_CHANGE_READY_TO_PAUSED: + gst_gl_download_start (download); + break; + case GST_STATE_CHANGE_PAUSED_TO_PLAYING: + break; + default: + break; + } + + ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition); + if (ret == GST_STATE_CHANGE_FAILURE) + return ret; + + switch (transition) { + case GST_STATE_CHANGE_PLAYING_TO_PAUSED: + break; + case GST_STATE_CHANGE_PAUSED_TO_READY: + gst_gl_download_stop (download); + break; + case GST_STATE_CHANGE_READY_TO_NULL: + break; + default: + break; + } + + return ret; +} diff --git a/sys/glsink/gstglfilter.c b/sys/glsink/gstglfilter.c new file mode 100644 index 00000000..c2b91f45 --- /dev/null +++ b/sys/glsink/gstglfilter.c @@ -0,0 +1,416 @@ +/* + * GStreamer + * Copyright (C) 2007 David Schleef <ds@schleef.org> + * + * 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., 59 Temple Place - Suite 330, + * Boston, MA 02111-1307, USA. + */ + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include <gst/gst.h> +#include <gst/video/video.h> +#include <gstglbuffer.h> + +#define GST_CAT_DEFAULT gst_gl_filter_debug +GST_DEBUG_CATEGORY_STATIC (GST_CAT_DEFAULT); + +#define GST_TYPE_GL_FILTER (gst_gl_filter_get_type()) +#define GST_GL_FILTER(obj) (G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_GL_FILTER,GstGLFilter)) +#define GST_IS_GL_FILTER(obj) (G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_GL_FILTER)) +#define GST_GL_FILTER_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST((klass) ,GST_TYPE_GL_FILTER,GstGLFilterClass)) +#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)) +typedef struct _GstGLFilter GstGLFilter; +typedef struct _GstGLFilterClass GstGLFilterClass; + +typedef void (*GstGLFilterProcessFunc) (GstGLFilter *, guint8 *, guint); + +struct _GstGLFilter +{ + GstElement element; + + GstPad *srcpad; + GstPad *sinkpad; + + /* < private > */ + + GstGLDisplay *display; + GstVideoFormat format; + int width; + int height; +}; + +struct _GstGLFilterClass +{ + GstElementClass element_class; +}; + +static const GstElementDetails element_details = GST_ELEMENT_DETAILS ("FIXME", + "Filter/Effect", + "FIXME example filter", + "FIXME <fixme@fixme.com>"); + +#define GST_GL_VIDEO_CAPS "video/x-raw-gl" + +static GstStaticPadTemplate gst_gl_filter_src_pad_template = +GST_STATIC_PAD_TEMPLATE ("src", + GST_PAD_SRC, + GST_PAD_ALWAYS, + GST_STATIC_CAPS (GST_GL_VIDEO_CAPS) + ); + +static GstStaticPadTemplate gst_gl_filter_sink_pad_template = +GST_STATIC_PAD_TEMPLATE ("sink", + GST_PAD_SINK, + GST_PAD_ALWAYS, + GST_STATIC_CAPS (GST_GL_VIDEO_CAPS) + ); + +enum +{ + PROP_0 +}; + +#define DEBUG_INIT(bla) \ + GST_DEBUG_CATEGORY_INIT (gst_gl_filter_debug, "glfilter", 0, "glfilter element"); + +GST_BOILERPLATE_FULL (GstGLFilter, gst_gl_filter, GstElement, + GST_TYPE_ELEMENT, DEBUG_INIT); + +static void gst_gl_filter_set_property (GObject * object, guint prop_id, + const GValue * value, GParamSpec * pspec); +static void gst_gl_filter_get_property (GObject * object, guint prop_id, + GValue * value, GParamSpec * pspec); + +static GstFlowReturn gst_gl_filter_chain (GstPad * pad, GstBuffer * buf); +static void gst_gl_filter_reset (GstGLFilter * filter); +static GstStateChangeReturn +gst_gl_filter_change_state (GstElement * element, GstStateChange transition); +static gboolean gst_gl_filter_sink_setcaps (GstPad * pad, GstCaps * caps); +static gboolean gst_gl_filter_transform (GstGLBuffer * outbuf, + GstGLBuffer * inbuf); + + +static void +gst_gl_filter_base_init (gpointer klass) +{ + GstElementClass *element_class = GST_ELEMENT_CLASS (klass); + + gst_element_class_set_details (element_class, &element_details); + + gst_element_class_add_pad_template (element_class, + gst_static_pad_template_get (&gst_gl_filter_src_pad_template)); + gst_element_class_add_pad_template (element_class, + gst_static_pad_template_get (&gst_gl_filter_sink_pad_template)); +} + +static void +gst_gl_filter_class_init (GstGLFilterClass * klass) +{ + GObjectClass *gobject_class; + + gobject_class = (GObjectClass *) klass; + gobject_class->set_property = gst_gl_filter_set_property; + gobject_class->get_property = gst_gl_filter_get_property; + + GST_ELEMENT_CLASS (klass)->change_state = gst_gl_filter_change_state; +} + +static void +gst_gl_filter_init (GstGLFilter * filter, GstGLFilterClass * klass) +{ + gst_element_create_all_pads (GST_ELEMENT (filter)); + + filter->sinkpad = gst_element_get_static_pad (GST_ELEMENT (filter), "sink"); + filter->srcpad = gst_element_get_static_pad (GST_ELEMENT (filter), "src"); + + gst_pad_set_setcaps_function (filter->sinkpad, gst_gl_filter_sink_setcaps); + gst_pad_set_chain_function (filter->sinkpad, gst_gl_filter_chain); + + gst_gl_filter_reset (filter); +} + +static void +gst_gl_filter_set_property (GObject * object, guint prop_id, + const GValue * value, GParamSpec * pspec) +{ + //GstGLFilter *filter = GST_GL_FILTER (object); + + switch (prop_id) { + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); + break; + } +} + +static void +gst_gl_filter_get_property (GObject * object, guint prop_id, + GValue * value, GParamSpec * pspec) +{ + //GstGLFilter *filter = GST_GL_FILTER (object); + + switch (prop_id) { + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); + break; + } +} + +static void +gst_gl_filter_reset (GstGLFilter * filter) +{ + if (filter->display) { + g_object_unref (filter->display); + filter->display = NULL; + } + filter->format = GST_VIDEO_FORMAT_BGRx; +} + +static gboolean +gst_gl_filter_start (GstGLFilter * filter) +{ + gboolean ret; + + filter->format = GST_VIDEO_FORMAT_BGRx; + filter->display = gst_gl_display_new (); + ret = gst_gl_display_connect (filter->display, NULL); + + return ret; +} + +static gboolean +gst_gl_filter_stop (GstGLFilter * filter) +{ + gst_gl_filter_reset (filter); + + return TRUE; +} + +static gboolean +gst_gl_filter_sink_setcaps (GstPad * pad, GstCaps * caps) +{ + GstGLFilter *filter; + gboolean ret; + GstStructure *structure; + + filter = GST_GL_FILTER (gst_pad_get_parent (pad)); + + structure = gst_caps_get_structure (caps, 0); + + ret = gst_structure_get_int (structure, "width", &filter->width); + ret &= gst_structure_get_int (structure, "height", &filter->height); + if (!ret) + return FALSE; + + GST_ERROR ("setcaps %d %d", filter->width, filter->height); + + ret = gst_pad_set_caps (filter->srcpad, caps); + + return ret; +} + +static GstFlowReturn +gst_gl_filter_chain (GstPad * pad, GstBuffer * buf) +{ + GstGLFilter *filter; + GstGLBuffer *inbuf; + GstGLBuffer *outbuf; + + filter = GST_GL_FILTER (gst_pad_get_parent (pad)); + inbuf = GST_GL_BUFFER (buf); + + outbuf = gst_gl_buffer_new (inbuf->display, filter->format, + filter->width, filter->height); + + gst_buffer_copy_metadata (GST_BUFFER (outbuf), buf, + GST_BUFFER_COPY_TIMESTAMPS | GST_BUFFER_COPY_FLAGS); + gst_buffer_set_caps (GST_BUFFER (outbuf), GST_PAD_CAPS (filter->srcpad)); + + gst_gl_filter_transform (outbuf, inbuf); + + gst_pad_push (filter->srcpad, GST_BUFFER (outbuf)); + + gst_object_unref (filter); + return GST_FLOW_OK; +} + +static GstStateChangeReturn +gst_gl_filter_change_state (GstElement * element, GstStateChange transition) +{ + GstGLFilter *filter; + GstStateChangeReturn ret = GST_STATE_CHANGE_SUCCESS; + + GST_DEBUG ("change state"); + + filter = GST_GL_FILTER (element); + + switch (transition) { + case GST_STATE_CHANGE_NULL_TO_READY: + break; + case GST_STATE_CHANGE_READY_TO_PAUSED: + gst_gl_filter_start (filter); + break; + case GST_STATE_CHANGE_PAUSED_TO_PLAYING: + break; + default: + break; + } + + ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition); + if (ret == GST_STATE_CHANGE_FAILURE) + return ret; + + switch (transition) { + case GST_STATE_CHANGE_PLAYING_TO_PAUSED: + break; + case GST_STATE_CHANGE_PAUSED_TO_READY: + gst_gl_filter_stop (filter); + break; + case GST_STATE_CHANGE_READY_TO_NULL: + break; + default: + break; + } + + return ret; +} + +void +dump_fbconfigs (Display * display) +{ + GLXFBConfig *fbconfigs; + int n; + int i; + int j; + int ret; + int value; + struct + { + int attr; + char *name; + } list[] = { + { + GLX_DRAWABLE_TYPE, "drawable type"}, { + GLX_BIND_TO_TEXTURE_TARGETS_EXT, "bind to texture targets"}, { + GLX_BIND_TO_TEXTURE_RGBA_EXT, "bind to texture rgba"}, { + GLX_MAX_PBUFFER_WIDTH, "max pbuffer width"}, { + GLX_MAX_PBUFFER_HEIGHT, "max pbuffer height"}, { + GLX_MAX_PBUFFER_PIXELS, "max pbuffer pixels"}, { + GLX_RENDER_TYPE, "render type"}, { + 0, 0} + }; + + g_print ("screen count: %d\n", ScreenCount (display)); + + fbconfigs = glXGetFBConfigs (display, 0, &n); + for (i = 0; i < n; i++) { + g_print ("%d:\n", i); + for (j = 0; list[j].attr; j++) { + ret = glXGetFBConfigAttrib (display, fbconfigs[i], list[j].attr, &value); + if (ret != Success) { + g_print ("%s: failed\n", list[j].name); + } else { + g_print ("%s: %d\n", list[j].name, value); + } + } + } + +} + +static gboolean +gst_gl_filter_transform (GstGLBuffer * outbuf, GstGLBuffer * inbuf) +{ + GstGLDisplay *display; + +#if 0 + int pixmapAttribs[] = { + GLX_TEXTURE_TARGET_EXT, GLX_TEXTURE_RECTANGLE_EXT, + GLX_TEXTURE_FORMAT_EXT, GLX_TEXTURE_FORMAT_RGBA_EXT, + None + }; +#endif + GLXFBConfig *fbconfigs; + int n; + int i; + GLXDrawable glxpixmap; + GLXContext context = 0; + int fb_index = 0; + int attrib[] = { GLX_RGBA, GLX_RED_SIZE, 8, + GLX_GREEN_SIZE, 8, GLX_BLUE_SIZE, 8, None + }; + XVisualInfo *visinfo; + + display = outbuf->display; + gst_gl_display_lock (display); + + //context = glXCreateContext (display->display, visinfo, NULL, True); + + dump_fbconfigs (display->display); + + fbconfigs = glXGetFBConfigs (display->display, display->screen_num, &n); + for (i = 0; i < n; i++) { + XVisualInfo *visinfo; + int value; + + GST_DEBUG ("fbconfig %d", i); + + visinfo = glXGetVisualFromFBConfig (display->display, fbconfigs[i]); + GST_DEBUG ("visinfo %p", visinfo); + + glXGetFBConfigAttrib (display->display, fbconfigs[i], + GLX_DRAWABLE_TYPE, &value); + if (!(value & GLX_WINDOW_BIT)) { + GST_DEBUG ("GLX_DRAWABLE_TYPE doesn't have GLX_WINDOW_BIT set"); + continue; + } + + glXGetFBConfigAttrib (display->display, fbconfigs[i], + GLX_BIND_TO_TEXTURE_TARGETS_EXT, &value); + if (!(value & GLX_TEXTURE_2D_BIT_EXT)) { + GST_DEBUG + ("GLX_BIND_TO_TEXTURE_TARGETS_EXT doesn't have GLX_TEXTURE_2D_BIT_EXT set"); + continue; + } + + glXGetFBConfigAttrib (display->display, fbconfigs[i], + GLX_BIND_TO_TEXTURE_RGBA_EXT, &value); + GST_DEBUG ("GLX_BIND_TO_TEXTURE_RGBA_EXT %d", value); + + } + + fb_index = 0; + +#if 0 + { + pb = glXCreatePbuffer (display->display, fbconfigs[fb_index], attribs); + } +#endif + + XSync (display->display, False); + visinfo = glXChooseVisual (display->display, 0, attrib); + glxpixmap = glXCreateGLXPixmap (display->display, visinfo, outbuf->pixmap); + + XSync (display->display, False); + + glXMakeCurrent (display->display, glxpixmap, context); + + glXMakeCurrent (display->display, None, NULL); + gst_gl_display_unlock (display); + + return TRUE; +} diff --git a/sys/glsink/gstglupload.c b/sys/glsink/gstglupload.c index 116d9d44..81d73175 100644 --- a/sys/glsink/gstglupload.c +++ b/sys/glsink/gstglupload.c @@ -78,7 +78,7 @@ static GstStaticPadTemplate gst_gl_upload_sink_pad_template = GST_STATIC_PAD_TEMPLATE ("sink", GST_PAD_SINK, GST_PAD_ALWAYS, - GST_STATIC_CAPS (GST_VIDEO_CAPS_RGBx) + GST_STATIC_CAPS (GST_VIDEO_CAPS_BGRx) ); enum @@ -99,6 +99,9 @@ static void gst_gl_upload_get_property (GObject * object, guint prop_id, static GstFlowReturn gst_gl_upload_chain (GstPad * pad, GstBuffer * buf); static void gst_gl_upload_reset (GstGLUpload * upload); +static GstStateChangeReturn +gst_gl_upload_change_state (GstElement * element, GstStateChange transition); +static gboolean gst_gl_upload_sink_setcaps (GstPad * pad, GstCaps * caps); static void @@ -123,11 +126,18 @@ gst_gl_upload_class_init (GstGLUploadClass * klass) gobject_class->set_property = gst_gl_upload_set_property; gobject_class->get_property = gst_gl_upload_get_property; + GST_ELEMENT_CLASS (klass)->change_state = gst_gl_upload_change_state; } static void gst_gl_upload_init (GstGLUpload * upload, GstGLUploadClass * klass) { + gst_element_create_all_pads (GST_ELEMENT (upload)); + + upload->sinkpad = gst_element_get_static_pad (GST_ELEMENT (upload), "sink"); + upload->srcpad = gst_element_get_static_pad (GST_ELEMENT (upload), "src"); + + gst_pad_set_setcaps_function (upload->sinkpad, gst_gl_upload_sink_setcaps); gst_pad_set_chain_function (upload->sinkpad, gst_gl_upload_chain); gst_gl_upload_reset (upload); @@ -162,6 +172,61 @@ gst_gl_upload_get_property (GObject * object, guint prop_id, static void gst_gl_upload_reset (GstGLUpload * upload) { + if (upload->display) { + g_object_unref (upload->display); + upload->display = NULL; + } + upload->format = GST_VIDEO_FORMAT_BGRx; +} + +static gboolean +gst_gl_upload_start (GstGLUpload * upload) +{ + gboolean ret; + + upload->format = GST_VIDEO_FORMAT_BGRx; + upload->display = gst_gl_display_new (); + ret = gst_gl_display_connect (upload->display, NULL); + + return ret; +} + +static gboolean +gst_gl_upload_stop (GstGLUpload * upload) +{ + gst_gl_upload_reset (upload); + + return TRUE; +} + +static gboolean +gst_gl_upload_sink_setcaps (GstPad * pad, GstCaps * caps) +{ + GstGLUpload *upload; + GstVideoFormat format; + int height; + int width; + gboolean ret; + GstCaps *srccaps; + + upload = GST_GL_UPLOAD (gst_pad_get_parent (pad)); + + ret = gst_video_format_parse_caps (caps, &format, &width, &height); + if (!ret) + return FALSE; + + upload->format = format; + upload->width = width; + upload->height = height; + + GST_ERROR ("setcaps %d %d %d", format, width, height); + + srccaps = gst_caps_new_simple ("video/x-raw-gl", + "width", G_TYPE_INT, width, "height", G_TYPE_INT, height, NULL); + ret = gst_pad_set_caps (upload->srcpad, srccaps); + gst_caps_unref (srccaps); + + return ret; } static GstFlowReturn @@ -177,6 +242,10 @@ gst_gl_upload_chain (GstPad * pad, GstBuffer * buf) gst_buffer_copy_metadata (GST_BUFFER (outbuf), buf, GST_BUFFER_COPY_TIMESTAMPS | GST_BUFFER_COPY_FLAGS); + gst_buffer_set_caps (GST_BUFFER (outbuf), GST_PAD_CAPS (upload->srcpad)); + + GST_DEBUG ("uploading %p size %d", GST_BUFFER_DATA (buf), + GST_BUFFER_SIZE (buf)); gst_gl_buffer_upload (outbuf, GST_BUFFER_DATA (buf)); gst_pad_push (upload->srcpad, GST_BUFFER (outbuf)); @@ -184,3 +253,44 @@ gst_gl_upload_chain (GstPad * pad, GstBuffer * buf) gst_object_unref (upload); return GST_FLOW_OK; } + +static GstStateChangeReturn +gst_gl_upload_change_state (GstElement * element, GstStateChange transition) +{ + GstGLUpload *upload; + GstStateChangeReturn ret = GST_STATE_CHANGE_SUCCESS; + + GST_DEBUG ("change state"); + + upload = GST_GL_UPLOAD (element); + + switch (transition) { + case GST_STATE_CHANGE_NULL_TO_READY: + break; + case GST_STATE_CHANGE_READY_TO_PAUSED: + gst_gl_upload_start (upload); + break; + case GST_STATE_CHANGE_PAUSED_TO_PLAYING: + break; + default: + break; + } + + ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition); + if (ret == GST_STATE_CHANGE_FAILURE) + return ret; + + switch (transition) { + case GST_STATE_CHANGE_PLAYING_TO_PAUSED: + break; + case GST_STATE_CHANGE_PAUSED_TO_READY: + gst_gl_upload_stop (upload); + break; + case GST_STATE_CHANGE_READY_TO_NULL: + break; + default: + break; + } + + return ret; +} diff --git a/sys/glsink/gstopengl.c b/sys/glsink/gstopengl.c index 7116174a..4889e0a6 100644 --- a/sys/glsink/gstopengl.c +++ b/sys/glsink/gstopengl.c @@ -33,6 +33,8 @@ #include <glimagesink.h> GType gst_gl_upload_get_type (void); +GType gst_gl_download_get_type (void); +GType gst_gl_filter_get_type (void); static gboolean @@ -46,15 +48,17 @@ plugin_init (GstPlugin * plugin) return FALSE; } if (!gst_element_register (plugin, "glupload", - GST_RANK_MARGINAL, gst_gl_upload_get_type ())) { + GST_RANK_NONE, gst_gl_upload_get_type ())) { return FALSE; } -#if 0 if (!gst_element_register (plugin, "gldownload", - GST_RANK_MARGINAL, GST_TYPE_GL_DOWNLOAD)) { + GST_RANK_NONE, gst_gl_download_get_type ())) { + return FALSE; + } + if (!gst_element_register (plugin, "glfilter", + GST_RANK_NONE, gst_gl_filter_get_type ())) { return FALSE; } -#endif return TRUE; } |