From 8beb49164ffeffa867d52c9da89c6dbb6e754e4a Mon Sep 17 00:00:00 2001 From: Neil Roberts Date: Thu, 8 Jul 2010 15:15:22 +0100 Subject: [PATCH] cogl-texture: Share the common code in the set_region virtual There was a lot of common code that was copied to all of the backends to convert the data to a suitable format and wrap it into a CoglBitmap so that it can be passed to _cogl_texture_driver_upload_subregion_to_gl. This patch moves the common code to cogl-texture.c so that the virtual just takes a CoglBitmap that is already in the right format. --- clutter/cogl/cogl/cogl-atlas-texture.c | 81 ++++--------------- clutter/cogl/cogl/cogl-sub-texture.c | 21 ++--- clutter/cogl/cogl/cogl-texture-2d-sliced.c | 57 +++---------- clutter/cogl/cogl/cogl-texture-2d.c | 58 +++---------- clutter/cogl/cogl/cogl-texture-private.h | 19 +++-- clutter/cogl/cogl/cogl-texture-rectangle.c | 58 +++---------- clutter/cogl/cogl/cogl-texture.c | 74 ++++++++++++++--- .../cogl/winsys/cogl-texture-pixmap-x11.c | 6 +- 8 files changed, 134 insertions(+), 240 deletions(-) diff --git a/clutter/cogl/cogl/cogl-atlas-texture.c b/clutter/cogl/cogl/cogl-atlas-texture.c index 54e87e7ea..a5fc933b7 100644 --- a/clutter/cogl/cogl/cogl-atlas-texture.c +++ b/clutter/cogl/cogl/cogl-atlas-texture.c @@ -520,79 +520,32 @@ _cogl_atlas_texture_set_region (CoglTexture *tex, int dst_y, unsigned int dst_width, unsigned int dst_height, - int width, - int height, - CoglPixelFormat format, - unsigned int rowstride, - const guint8 *data) + CoglBitmap *bmp) { CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex); /* If the texture is in the atlas then we need to copy the edge pixels to the border */ if (atlas_tex->in_atlas) - { - int bpp; - CoglBitmap source_bmp; - CoglBitmap tmp_bmp; - gboolean tmp_bmp_owner = FALSE; - gboolean success; + /* Upload the data ignoring the premult bit */ + return _cogl_atlas_texture_set_region_with_border (atlas_tex, + src_x, src_y, + dst_x, dst_y, + dst_width, dst_height, + bmp->width, + bmp->height, + bmp->format & + ~COGL_PREMULT_BIT, + bmp->rowstride, + bmp->data); - /* Check for valid format */ - if (format == COGL_PIXEL_FORMAT_ANY) - return FALSE; - - /* Shortcut out early if the image is empty */ - if (width == 0 || height == 0) - return TRUE; - - /* Init source bitmap */ - source_bmp.width = width; - source_bmp.height = height; - source_bmp.format = format; - source_bmp.data = (guint8 *)data; - - /* Rowstride from width if none specified */ - bpp = _cogl_get_format_bpp (format); - source_bmp.rowstride = (rowstride == 0) ? width * bpp : rowstride; - - /* Prepare the bitmap so that it will do the premultiplication - conversion */ - _cogl_texture_prepare_for_upload (&source_bmp, - atlas_tex->format, - NULL, - &tmp_bmp, - &tmp_bmp_owner, - NULL, NULL, NULL); - - /* Upload the data ignoring the premult bit */ - success = - _cogl_atlas_texture_set_region_with_border (atlas_tex, - src_x, src_y, - dst_x, dst_y, - dst_width, dst_height, - tmp_bmp.width, - tmp_bmp.height, - tmp_bmp.format & - ~COGL_PREMULT_BIT, - tmp_bmp.rowstride, - tmp_bmp.data); - - /* Free data if owner */ - if (tmp_bmp_owner) - g_free (tmp_bmp.data); - - return success; - } else /* Otherwise we can just forward on to the sub texture */ - return cogl_texture_set_region (atlas_tex->sub_texture, - src_x, src_y, - dst_x, dst_y, - dst_width, dst_height, - width, height, - format, rowstride, - data); + return _cogl_texture_set_region_from_bitmap (atlas_tex->sub_texture, + src_x, src_y, + dst_x, dst_y, + dst_width, dst_height, + bmp); } static gboolean diff --git a/clutter/cogl/cogl/cogl-sub-texture.c b/clutter/cogl/cogl/cogl-sub-texture.c index bfae32bc7..5d37e8499 100644 --- a/clutter/cogl/cogl/cogl-sub-texture.c +++ b/clutter/cogl/cogl/cogl-sub-texture.c @@ -406,23 +406,16 @@ _cogl_sub_texture_set_region (CoglTexture *tex, int dst_y, unsigned int dst_width, unsigned int dst_height, - int width, - int height, - CoglPixelFormat format, - unsigned int rowstride, - const guint8 *data) + CoglBitmap *bmp) { CoglSubTexture *sub_tex = COGL_SUB_TEXTURE (tex); - return cogl_texture_set_region (sub_tex->full_texture, - src_x, src_y, - dst_x + sub_tex->sub_x, - dst_y + sub_tex->sub_y, - dst_width, dst_height, - width, height, - format, - rowstride, - data); + return _cogl_texture_set_region_from_bitmap (sub_tex->full_texture, + src_x, src_y, + dst_x + sub_tex->sub_x, + dst_y + sub_tex->sub_y, + dst_width, dst_height, + bmp); } static void diff --git a/clutter/cogl/cogl/cogl-texture-2d-sliced.c b/clutter/cogl/cogl/cogl-texture-2d-sliced.c index 3f95eef51..1ad6dce34 100644 --- a/clutter/cogl/cogl/cogl-texture-2d-sliced.c +++ b/clutter/cogl/cogl/cogl-texture-2d-sliced.c @@ -1461,62 +1461,25 @@ _cogl_texture_2d_sliced_set_region (CoglTexture *tex, int dst_y, unsigned int dst_width, unsigned int dst_height, - int width, - int height, - CoglPixelFormat format, - unsigned int rowstride, - const guint8 *data) + CoglBitmap *bmp) { CoglTexture2DSliced *tex_2ds = COGL_TEXTURE_2D_SLICED (tex); - int bpp; - CoglBitmap source_bmp; - CoglBitmap tmp_bmp; - gboolean tmp_bmp_owner = FALSE; - GLenum closest_gl_format; - GLenum closest_gl_type; - - /* Check for valid format */ - if (format == COGL_PIXEL_FORMAT_ANY) - return FALSE; - - /* Shortcut out early if the image is empty */ - if (width == 0 || height == 0) - return TRUE; - - /* Init source bitmap */ - source_bmp.width = width; - source_bmp.height = height; - source_bmp.format = format; - source_bmp.data = (guint8 *)data; - - /* Rowstride from width if none specified */ - bpp = _cogl_get_format_bpp (format); - source_bmp.rowstride = (rowstride == 0) ? width * bpp : rowstride; - - /* Prepare the bitmap so that it will do the premultiplication - conversion */ - _cogl_texture_prepare_for_upload (&source_bmp, - tex_2ds->format, - NULL, - &tmp_bmp, - &tmp_bmp_owner, - NULL, - &closest_gl_format, - &closest_gl_type); + GLenum gl_format; + GLenum gl_type; + _cogl_pixel_format_to_gl (bmp->format, + NULL, /* internal format */ + &gl_format, + &gl_type); /* Send data to GL */ _cogl_texture_2d_sliced_upload_subregion_to_gl (tex_2ds, src_x, src_y, dst_x, dst_y, dst_width, dst_height, - &tmp_bmp, - closest_gl_format, - closest_gl_type); - - /* Free data if owner */ - if (tmp_bmp_owner) - g_free (tmp_bmp.data); + bmp, + gl_format, + gl_type); return TRUE; } diff --git a/clutter/cogl/cogl/cogl-texture-2d.c b/clutter/cogl/cogl/cogl-texture-2d.c index 32eccb143..e5e04f90e 100644 --- a/clutter/cogl/cogl/cogl-texture-2d.c +++ b/clutter/cogl/cogl/cogl-texture-2d.c @@ -450,48 +450,16 @@ _cogl_texture_2d_set_region (CoglTexture *tex, int dst_y, unsigned int dst_width, unsigned int dst_height, - int width, - int height, - CoglPixelFormat format, - unsigned int rowstride, - const guint8 *data) + CoglBitmap *bmp) { - CoglTexture2D *tex_2d = COGL_TEXTURE_2D (tex); - int bpp; - CoglBitmap source_bmp; - CoglBitmap tmp_bmp; - gboolean tmp_bmp_owner = FALSE; - GLenum closest_gl_format; - GLenum closest_gl_type; + CoglTexture2D *tex_2d = COGL_TEXTURE_2D (tex); + GLenum gl_format; + GLenum gl_type; - /* Check for valid format */ - if (format == COGL_PIXEL_FORMAT_ANY) - return FALSE; - - /* Shortcut out early if the image is empty */ - if (width == 0 || height == 0) - return TRUE; - - /* Init source bitmap */ - source_bmp.width = width; - source_bmp.height = height; - source_bmp.format = format; - source_bmp.data = (guint8 *)data; - - /* Rowstride from width if none specified */ - bpp = _cogl_get_format_bpp (format); - source_bmp.rowstride = (rowstride == 0) ? width * bpp : rowstride; - - /* Prepare the bitmap so that it will do the premultiplication - conversion */ - _cogl_texture_prepare_for_upload (&source_bmp, - tex_2d->format, - NULL, - &tmp_bmp, - &tmp_bmp_owner, - NULL, - &closest_gl_format, - &closest_gl_type); + _cogl_pixel_format_to_gl (bmp->format, + NULL, /* internal format */ + &gl_format, + &gl_type); /* Send data to GL */ _cogl_texture_driver_upload_subregion_to_gl (GL_TEXTURE_2D, @@ -500,13 +468,9 @@ _cogl_texture_2d_set_region (CoglTexture *tex, src_x, src_y, dst_x, dst_y, dst_width, dst_height, - &tmp_bmp, - closest_gl_format, - closest_gl_type); - - /* Free data if owner */ - if (tmp_bmp_owner) - g_free (tmp_bmp.data); + bmp, + gl_format, + gl_type); tex_2d->mipmaps_dirty = TRUE; diff --git a/clutter/cogl/cogl/cogl-texture-private.h b/clutter/cogl/cogl/cogl-texture-private.h index 78330c618..949df576e 100644 --- a/clutter/cogl/cogl/cogl-texture-private.h +++ b/clutter/cogl/cogl/cogl-texture-private.h @@ -68,6 +68,9 @@ struct _CoglTextureVtable /* Virtual functions that must be implemented for a texture backend */ + /* This should update the specified sub region of the texture with a + sub region of the given bitmap. The bitmap will have first been + converted to a suitable format for uploading if neccessary. */ gboolean (* set_region) (CoglTexture *tex, int src_x, int src_y, @@ -75,11 +78,7 @@ struct _CoglTextureVtable int dst_y, unsigned int dst_width, unsigned int dst_height, - int width, - int height, - CoglPixelFormat format, - unsigned int rowstride, - const guint8 *data); + CoglBitmap *bitmap); /* This should copy the image data of the texture into @data. The requested format will have been first passed through @@ -258,4 +257,14 @@ _cogl_texture_draw_and_read (CoglHandle handle, gboolean _cogl_texture_is_foreign (CoglHandle handle); +gboolean +_cogl_texture_set_region_from_bitmap (CoglHandle handle, + int src_x, + int src_y, + int dst_x, + int dst_y, + unsigned int dst_width, + unsigned int dst_height, + CoglBitmap *bmp); + #endif /* __COGL_TEXTURE_PRIVATE_H */ diff --git a/clutter/cogl/cogl/cogl-texture-rectangle.c b/clutter/cogl/cogl/cogl-texture-rectangle.c index 2a61e78d3..ca25a409a 100644 --- a/clutter/cogl/cogl/cogl-texture-rectangle.c +++ b/clutter/cogl/cogl/cogl-texture-rectangle.c @@ -446,48 +446,16 @@ _cogl_texture_rectangle_set_region (CoglTexture *tex, int dst_y, unsigned int dst_width, unsigned int dst_height, - int width, - int height, - CoglPixelFormat format, - unsigned int rowstride, - const guint8 *data) + CoglBitmap *bmp) { - CoglTextureRectangle *tex_rect = COGL_TEXTURE_RECTANGLE (tex); - int bpp; - CoglBitmap source_bmp; - CoglBitmap tmp_bmp; - gboolean tmp_bmp_owner = FALSE; - GLenum closest_gl_format; - GLenum closest_gl_type; + CoglTextureRectangle *tex_rect = COGL_TEXTURE_RECTANGLE (tex); + GLenum gl_format; + GLenum gl_type; - /* Check for valid format */ - if (format == COGL_PIXEL_FORMAT_ANY) - return FALSE; - - /* Shortcut out early if the image is empty */ - if (width == 0 || height == 0) - return TRUE; - - /* Init source bitmap */ - source_bmp.width = width; - source_bmp.height = height; - source_bmp.format = format; - source_bmp.data = (guint8 *)data; - - /* Rowstride from width if none specified */ - bpp = _cogl_get_format_bpp (format); - source_bmp.rowstride = (rowstride == 0) ? width * bpp : rowstride; - - /* Prepare the bitmap so that it will do the premultiplication - conversion */ - _cogl_texture_prepare_for_upload (&source_bmp, - tex_rect->format, - NULL, - &tmp_bmp, - &tmp_bmp_owner, - NULL, - &closest_gl_format, - &closest_gl_type); + _cogl_pixel_format_to_gl (bmp->format, + NULL, /* internal format */ + &gl_format, + &gl_type); /* Send data to GL */ _cogl_texture_driver_upload_subregion_to_gl (GL_TEXTURE_RECTANGLE_ARB, @@ -496,13 +464,9 @@ _cogl_texture_rectangle_set_region (CoglTexture *tex, src_x, src_y, dst_x, dst_y, dst_width, dst_height, - &tmp_bmp, - closest_gl_format, - closest_gl_type); - - /* Free data if owner */ - if (tmp_bmp_owner) - g_free (tmp_bmp.data); + bmp, + gl_format, + gl_type); return TRUE; } diff --git a/clutter/cogl/cogl/cogl-texture.c b/clutter/cogl/cogl/cogl-texture.c index c7be4b26b..337898389 100644 --- a/clutter/cogl/cogl/cogl-texture.c +++ b/clutter/cogl/cogl/cogl-texture.c @@ -812,6 +812,51 @@ _cogl_texture_ensure_non_quad_rendering (CoglHandle handle) tex->vtable->ensure_non_quad_rendering (tex); } +gboolean +_cogl_texture_set_region_from_bitmap (CoglHandle handle, + int src_x, + int src_y, + int dst_x, + int dst_y, + unsigned int dst_width, + unsigned int dst_height, + CoglBitmap *bmp) +{ + CoglTexture *tex = COGL_TEXTURE (handle); + CoglBitmap tmp_bmp; + gboolean tmp_bmp_owner = FALSE; + GLenum closest_gl_format; + GLenum closest_gl_type; + gboolean ret; + + /* Shortcut out early if the image is empty */ + if (dst_width == 0 || dst_height == 0) + return TRUE; + + /* Prepare the bitmap so that it will do the premultiplication + conversion */ + _cogl_texture_prepare_for_upload (bmp, + cogl_texture_get_format (handle), + NULL, + &tmp_bmp, + &tmp_bmp_owner, + NULL, + &closest_gl_format, + &closest_gl_type); + + ret = tex->vtable->set_region (handle, + src_x, src_y, + dst_x, dst_y, + dst_width, dst_height, + &tmp_bmp); + + /* Free data if owner */ + if (tmp_bmp_owner) + g_free (tmp_bmp.data); + + return ret; +} + gboolean cogl_texture_set_region (CoglHandle handle, int src_x, @@ -826,21 +871,28 @@ cogl_texture_set_region (CoglHandle handle, unsigned int rowstride, const guint8 *data) { - CoglTexture *tex; + int bpp; + CoglBitmap source_bmp; - if (!cogl_is_texture (handle)) + /* Check for valid format */ + if (format == COGL_PIXEL_FORMAT_ANY) return FALSE; - tex = COGL_TEXTURE (handle); + /* Init source bitmap */ + source_bmp.width = width; + source_bmp.height = height; + source_bmp.format = format; + source_bmp.data = (guint8 *) data; - return tex->vtable->set_region (tex, - src_x, src_y, - dst_x, dst_y, - dst_width, dst_height, - width, height, - format, - rowstride, - data); + /* Rowstride from width if none specified */ + bpp = _cogl_get_format_bpp (format); + source_bmp.rowstride = (rowstride == 0) ? width * bpp : rowstride; + + return _cogl_texture_set_region_from_bitmap (handle, + src_x, src_y, + dst_x, dst_y, + dst_width, dst_height, + &source_bmp); } /* Reads back the contents of a texture by rendering it to the framebuffer diff --git a/clutter/cogl/cogl/winsys/cogl-texture-pixmap-x11.c b/clutter/cogl/cogl/winsys/cogl-texture-pixmap-x11.c index c8ee42fc1..4af893c7b 100644 --- a/clutter/cogl/cogl/winsys/cogl-texture-pixmap-x11.c +++ b/clutter/cogl/cogl/winsys/cogl-texture-pixmap-x11.c @@ -1118,11 +1118,7 @@ _cogl_texture_pixmap_x11_set_region (CoglTexture *tex, int dst_y, unsigned int dst_width, unsigned int dst_height, - int width, - int height, - CoglPixelFormat format, - unsigned int rowstride, - const guint8 *data) + CoglBitmap *bmp) { /* This doesn't make much sense for texture from pixmap so it's not supported */