2009-12-04 08:06:32 -05:00
|
|
|
/*
|
|
|
|
* Cogl
|
|
|
|
*
|
|
|
|
* An object oriented GL/GLES Abstraction/Utility Layer
|
|
|
|
*
|
2010-02-12 10:33:56 -05:00
|
|
|
* Copyright (C) 2009,2010 Intel Corporation.
|
2009-12-04 08:06:32 -05:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser 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
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
2010-03-01 07:56:10 -05:00
|
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*
|
2009-12-04 08:06:32 -05:00
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* Neil Roberts <neil@linux.intel.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "cogl.h"
|
2010-07-25 16:36:41 -04:00
|
|
|
#include "cogl-debug.h"
|
2009-12-04 08:06:32 -05:00
|
|
|
#include "cogl-internal.h"
|
|
|
|
#include "cogl-util.h"
|
|
|
|
#include "cogl-texture-private.h"
|
|
|
|
#include "cogl-atlas-texture-private.h"
|
|
|
|
#include "cogl-texture-2d-private.h"
|
2010-01-18 04:22:04 -05:00
|
|
|
#include "cogl-sub-texture-private.h"
|
2009-12-04 08:06:32 -05:00
|
|
|
#include "cogl-context.h"
|
|
|
|
#include "cogl-handle.h"
|
|
|
|
#include "cogl-texture-driver.h"
|
2010-08-02 10:24:03 -04:00
|
|
|
#include "cogl-rectangle-map.h"
|
2010-02-12 10:33:56 -05:00
|
|
|
#include "cogl-journal-private.h"
|
2010-07-09 21:53:38 -04:00
|
|
|
#include "cogl-material-opengl-private.h"
|
2010-08-02 11:29:10 -04:00
|
|
|
#include "cogl-atlas.h"
|
2009-12-04 08:06:32 -05:00
|
|
|
|
2009-12-04 13:24:15 -05:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
2009-12-04 08:06:32 -05:00
|
|
|
static void _cogl_atlas_texture_free (CoglAtlasTexture *sub_tex);
|
|
|
|
|
2010-07-09 13:46:31 -04:00
|
|
|
COGL_TEXTURE_INTERNAL_DEFINE (AtlasTexture, atlas_texture);
|
2009-12-04 08:06:32 -05:00
|
|
|
|
|
|
|
static const CoglTextureVtable cogl_atlas_texture_vtable;
|
|
|
|
|
2010-08-02 11:29:10 -04:00
|
|
|
static CoglHandle
|
|
|
|
_cogl_atlas_texture_create_sub_texture (CoglHandle full_texture,
|
|
|
|
const CoglRectangleMapEntry *rectangle)
|
2009-12-05 08:24:01 -05:00
|
|
|
{
|
2010-08-02 11:29:10 -04:00
|
|
|
/* Create a subtexture for the given rectangle not including the
|
|
|
|
1-pixel border */
|
|
|
|
return _cogl_sub_texture_new (full_texture,
|
|
|
|
rectangle->x + 1,
|
|
|
|
rectangle->y + 1,
|
|
|
|
rectangle->width - 2,
|
|
|
|
rectangle->height - 2);
|
|
|
|
}
|
2009-12-05 08:24:01 -05:00
|
|
|
|
2010-08-02 11:29:10 -04:00
|
|
|
static void
|
|
|
|
_cogl_atlas_texture_update_position_cb (gpointer user_data,
|
|
|
|
CoglHandle new_texture,
|
|
|
|
const CoglRectangleMapEntry *rectangle)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = user_data;
|
2009-12-05 08:24:01 -05:00
|
|
|
|
2010-08-02 11:29:10 -04:00
|
|
|
/* Update the sub texture */
|
|
|
|
if (atlas_tex->sub_texture)
|
|
|
|
cogl_handle_unref (atlas_tex->sub_texture);
|
|
|
|
atlas_tex->sub_texture =
|
|
|
|
_cogl_atlas_texture_create_sub_texture (new_texture, rectangle);
|
2009-12-05 08:24:01 -05:00
|
|
|
|
2010-08-02 11:29:10 -04:00
|
|
|
/* Update the position */
|
|
|
|
atlas_tex->rectangle = *rectangle;
|
|
|
|
}
|
2009-12-05 08:24:01 -05:00
|
|
|
|
|
|
|
static void
|
2010-08-02 11:29:10 -04:00
|
|
|
_cogl_atlas_texture_reorganize_foreach_cb (const CoglRectangleMapEntry *entry,
|
|
|
|
void *rectangle_data,
|
|
|
|
void *user_data)
|
2009-12-05 08:24:01 -05:00
|
|
|
{
|
2010-08-02 11:29:10 -04:00
|
|
|
/* Notify cogl-material.c that the texture's underlying GL texture
|
|
|
|
* storage is changing so it knows it may need to bind a new texture
|
|
|
|
* if the CoglTexture is reused with the same texture unit. */
|
|
|
|
_cogl_material_texture_storage_change_notify (rectangle_data);
|
2009-12-05 08:24:01 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-08-02 11:29:10 -04:00
|
|
|
_cogl_atlas_texture_reorganize_cb (void *data)
|
2009-12-05 08:24:01 -05:00
|
|
|
{
|
2010-08-02 11:29:10 -04:00
|
|
|
CoglAtlas *atlas;
|
|
|
|
/* We don't know if any materials may currently be referenced in
|
|
|
|
* the journal that depend on the current underlying GL texture
|
|
|
|
* storage so we flush the journal before migrating.
|
|
|
|
*
|
|
|
|
* We are assuming that texture atlas migration never happens
|
|
|
|
* during a flush so we don't have to consider recursion here.
|
|
|
|
*/
|
|
|
|
_cogl_journal_flush ();
|
|
|
|
|
|
|
|
atlas = _cogl_atlas_texture_get_atlas ();
|
|
|
|
|
|
|
|
if (atlas->map)
|
|
|
|
_cogl_rectangle_map_foreach (atlas->map,
|
|
|
|
_cogl_atlas_texture_reorganize_foreach_cb,
|
|
|
|
NULL);
|
2009-12-05 08:24:01 -05:00
|
|
|
}
|
|
|
|
|
2010-08-02 11:29:10 -04:00
|
|
|
CoglAtlas *
|
|
|
|
_cogl_atlas_texture_get_atlas (void)
|
2009-12-05 08:24:01 -05:00
|
|
|
{
|
2010-08-02 11:29:10 -04:00
|
|
|
_COGL_GET_CONTEXT (ctx, COGL_INVALID_HANDLE);
|
2009-12-05 08:24:01 -05:00
|
|
|
|
2010-08-02 11:29:10 -04:00
|
|
|
if (ctx->atlas == COGL_INVALID_HANDLE)
|
2009-12-05 08:24:01 -05:00
|
|
|
{
|
2010-08-05 08:24:02 -04:00
|
|
|
ctx->atlas = _cogl_atlas_new (COGL_PIXEL_FORMAT_RGBA_8888,
|
|
|
|
0,
|
|
|
|
_cogl_atlas_texture_update_position_cb);
|
2010-08-02 11:29:10 -04:00
|
|
|
|
|
|
|
_cogl_atlas_add_reorganize_callback (ctx->atlas,
|
|
|
|
_cogl_atlas_texture_reorganize_cb,
|
|
|
|
NULL);
|
2009-12-05 08:24:01 -05:00
|
|
|
}
|
2010-08-02 11:29:10 -04:00
|
|
|
|
|
|
|
return ctx->atlas;
|
2009-12-05 08:24:01 -05:00
|
|
|
}
|
|
|
|
|
2009-12-04 08:06:32 -05:00
|
|
|
static void
|
|
|
|
_cogl_atlas_texture_foreach_sub_texture_in_region (
|
|
|
|
CoglTexture *tex,
|
|
|
|
float virtual_tx_1,
|
|
|
|
float virtual_ty_1,
|
|
|
|
float virtual_tx_2,
|
|
|
|
float virtual_ty_2,
|
|
|
|
CoglTextureSliceCallback callback,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
|
|
|
_cogl_texture_foreach_sub_texture_in_region (atlas_tex->sub_texture,
|
|
|
|
virtual_tx_1,
|
|
|
|
virtual_ty_1,
|
|
|
|
virtual_tx_2,
|
|
|
|
virtual_ty_2,
|
|
|
|
callback,
|
|
|
|
user_data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-03-25 13:29:22 -04:00
|
|
|
_cogl_atlas_texture_set_wrap_mode_parameters (CoglTexture *tex,
|
|
|
|
GLenum wrap_mode_s,
|
|
|
|
GLenum wrap_mode_t,
|
2010-07-12 12:01:32 -04:00
|
|
|
GLenum wrap_mode_p)
|
2009-12-04 08:06:32 -05:00
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
2010-03-25 13:29:22 -04:00
|
|
|
_cogl_texture_set_wrap_mode_parameters (atlas_tex->sub_texture,
|
|
|
|
wrap_mode_s,
|
|
|
|
wrap_mode_t,
|
2010-07-12 12:01:32 -04:00
|
|
|
wrap_mode_p);
|
2009-12-04 08:06:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2009-12-05 08:48:03 -05:00
|
|
|
_cogl_atlas_texture_remove_from_atlas (CoglAtlasTexture *atlas_tex)
|
2009-12-04 08:06:32 -05:00
|
|
|
{
|
|
|
|
if (atlas_tex->in_atlas)
|
2009-12-04 13:55:53 -05:00
|
|
|
{
|
2010-08-02 11:29:10 -04:00
|
|
|
_cogl_atlas_remove (_cogl_atlas_texture_get_atlas (),
|
|
|
|
&atlas_tex->rectangle);
|
2009-12-05 08:48:03 -05:00
|
|
|
|
|
|
|
atlas_tex->in_atlas = FALSE;
|
2009-12-04 13:55:53 -05:00
|
|
|
}
|
2009-12-05 08:48:03 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_cogl_atlas_texture_free (CoglAtlasTexture *atlas_tex)
|
|
|
|
{
|
|
|
|
_cogl_atlas_texture_remove_from_atlas (atlas_tex);
|
2009-12-04 08:06:32 -05:00
|
|
|
|
|
|
|
cogl_handle_unref (atlas_tex->sub_texture);
|
2010-04-13 12:26:03 -04:00
|
|
|
|
2010-04-26 05:01:43 -04:00
|
|
|
/* Chain up */
|
|
|
|
_cogl_texture_free (COGL_TEXTURE (atlas_tex));
|
2009-12-04 08:06:32 -05:00
|
|
|
}
|
|
|
|
|
cogl: improves header and coding style consistency
We've had complaints that our Cogl code/headers are a bit "special" so
this is a first pass at tidying things up by giving them some
consistency. These changes are all consistent with how new code in Cogl
is being written, but the style isn't consistently applied across all
code yet.
There are two parts to this patch; but since each one required a large
amount of effort to maintain tidy indenting it made sense to combine the
changes to reduce the time spent re indenting the same lines.
The first change is to use a consistent style for declaring function
prototypes in headers. Cogl headers now consistently use this style for
prototypes:
return_type
cogl_function_name (CoglType arg0,
CoglType arg1);
Not everyone likes this style, but it seems that most of the currently
active Cogl developers agree on it.
The second change is to constrain the use of redundant glib data types
in Cogl. Uses of gint, guint, gfloat, glong, gulong and gchar have all
been replaced with int, unsigned int, float, long, unsigned long and char
respectively. When talking about pixel data; use of guchar has been
replaced with guint8, otherwise unsigned char can be used.
The glib types that we continue to use for portability are gboolean,
gint{8,16,32,64}, guint{8,16,32,64} and gsize.
The general intention is that Cogl should look palatable to the widest
range of C programmers including those outside the Gnome community so
- especially for the public API - we want to minimize the number of
foreign looking typedefs.
2010-02-09 20:57:32 -05:00
|
|
|
static int
|
2009-12-04 08:06:32 -05:00
|
|
|
_cogl_atlas_texture_get_max_waste (CoglTexture *tex)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
|
|
|
return cogl_texture_get_max_waste (atlas_tex->sub_texture);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
_cogl_atlas_texture_is_sliced (CoglTexture *tex)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
|
|
|
return cogl_texture_is_sliced (atlas_tex->sub_texture);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
_cogl_atlas_texture_can_hardware_repeat (CoglTexture *tex)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
|
|
|
return _cogl_texture_can_hardware_repeat (atlas_tex->sub_texture);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_cogl_atlas_texture_transform_coords_to_gl (CoglTexture *tex,
|
|
|
|
float *s,
|
|
|
|
float *t)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
|
|
|
_cogl_texture_transform_coords_to_gl (atlas_tex->sub_texture, s, t);
|
|
|
|
}
|
|
|
|
|
2010-03-01 16:49:04 -05:00
|
|
|
static CoglTransformResult
|
2010-01-18 04:22:04 -05:00
|
|
|
_cogl_atlas_texture_transform_quad_coords_to_gl (CoglTexture *tex,
|
|
|
|
float *coords)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
|
|
|
return _cogl_texture_transform_quad_coords_to_gl (atlas_tex->sub_texture,
|
|
|
|
coords);
|
|
|
|
}
|
|
|
|
|
2009-12-04 08:06:32 -05:00
|
|
|
static gboolean
|
|
|
|
_cogl_atlas_texture_get_gl_texture (CoglTexture *tex,
|
|
|
|
GLuint *out_gl_handle,
|
|
|
|
GLenum *out_gl_target)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
|
|
|
return cogl_texture_get_gl_texture (atlas_tex->sub_texture,
|
|
|
|
out_gl_handle,
|
|
|
|
out_gl_target);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_cogl_atlas_texture_set_filters (CoglTexture *tex,
|
|
|
|
GLenum min_filter,
|
|
|
|
GLenum mag_filter)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
|
|
|
_cogl_texture_set_filters (atlas_tex->sub_texture, min_filter, mag_filter);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-01-18 04:22:04 -05:00
|
|
|
_cogl_atlas_texture_migrate_out_of_atlas (CoglAtlasTexture *atlas_tex)
|
2009-12-04 08:06:32 -05:00
|
|
|
{
|
2010-01-18 04:22:04 -05:00
|
|
|
/* Make sure this texture is not in the atlas */
|
2009-12-05 08:48:03 -05:00
|
|
|
if (atlas_tex->in_atlas)
|
|
|
|
{
|
|
|
|
COGL_NOTE (ATLAS, "Migrating texture out of the atlas");
|
|
|
|
|
2010-06-04 11:42:30 -04:00
|
|
|
/* We don't know if any materials may currently be referenced in
|
|
|
|
* the journal that depend on the current underlying GL texture
|
|
|
|
* storage so we flush the journal before migrating.
|
|
|
|
*
|
|
|
|
* We are assuming that texture atlas migration never happens
|
|
|
|
* during a flush so we don't have to consider recursion here.
|
|
|
|
*/
|
|
|
|
_cogl_journal_flush ();
|
2010-02-12 10:33:56 -05:00
|
|
|
|
2010-04-08 07:21:04 -04:00
|
|
|
/* Notify cogl-material.c that the texture's underlying GL texture
|
|
|
|
* storage is changing so it knows it may need to bind a new texture
|
|
|
|
* if the CoglTexture is reused with the same texture unit. */
|
|
|
|
_cogl_material_texture_storage_change_notify (atlas_tex);
|
|
|
|
|
2009-12-05 08:48:03 -05:00
|
|
|
cogl_handle_unref (atlas_tex->sub_texture);
|
|
|
|
|
|
|
|
atlas_tex->sub_texture =
|
2010-08-02 11:29:10 -04:00
|
|
|
_cogl_atlas_copy_rectangle (_cogl_atlas_texture_get_atlas (),
|
|
|
|
atlas_tex->rectangle.x + 1,
|
|
|
|
atlas_tex->rectangle.y + 1,
|
|
|
|
atlas_tex->rectangle.width - 2,
|
2009-12-05 08:48:03 -05:00
|
|
|
atlas_tex->rectangle.height - 2,
|
|
|
|
COGL_TEXTURE_NO_ATLAS,
|
|
|
|
atlas_tex->format);
|
|
|
|
|
|
|
|
_cogl_atlas_texture_remove_from_atlas (atlas_tex);
|
|
|
|
}
|
2010-01-18 04:22:04 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-06-09 12:39:59 -04:00
|
|
|
_cogl_atlas_texture_pre_paint (CoglTexture *tex, CoglTexturePrePaintFlags flags)
|
2010-01-18 04:22:04 -05:00
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
2010-06-09 12:39:59 -04:00
|
|
|
if ((flags & COGL_TEXTURE_NEEDS_MIPMAP))
|
|
|
|
/* Mipmaps do not work well with the current atlas so instead
|
|
|
|
we'll just migrate the texture out and use a regular texture */
|
|
|
|
_cogl_atlas_texture_migrate_out_of_atlas (atlas_tex);
|
2009-12-04 08:06:32 -05:00
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
2010-06-09 12:39:59 -04:00
|
|
|
_cogl_texture_pre_paint (atlas_tex->sub_texture, flags);
|
2009-12-04 08:06:32 -05:00
|
|
|
}
|
|
|
|
|
2010-01-18 04:22:04 -05:00
|
|
|
static void
|
|
|
|
_cogl_atlas_texture_ensure_non_quad_rendering (CoglTexture *tex)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* Sub textures can't support non-quad rendering so we'll just
|
|
|
|
migrate the texture out */
|
|
|
|
_cogl_atlas_texture_migrate_out_of_atlas (atlas_tex);
|
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
|
|
|
_cogl_texture_ensure_non_quad_rendering (atlas_tex->sub_texture);
|
|
|
|
}
|
|
|
|
|
2010-01-18 05:53:00 -05:00
|
|
|
static gboolean
|
|
|
|
_cogl_atlas_texture_set_region_with_border (CoglAtlasTexture *atlas_tex,
|
|
|
|
int src_x,
|
|
|
|
int src_y,
|
|
|
|
int dst_x,
|
|
|
|
int dst_y,
|
|
|
|
unsigned int dst_width,
|
|
|
|
unsigned int dst_height,
|
2010-07-07 13:44:16 -04:00
|
|
|
CoglBitmap *bmp)
|
2010-01-18 05:53:00 -05:00
|
|
|
{
|
2010-08-02 11:29:10 -04:00
|
|
|
CoglAtlas *atlas = _cogl_atlas_texture_get_atlas ();
|
2010-01-18 05:53:00 -05:00
|
|
|
|
|
|
|
/* Copy the central data */
|
2010-08-02 11:29:10 -04:00
|
|
|
if (!_cogl_texture_set_region_from_bitmap (atlas->texture,
|
2010-07-07 13:44:16 -04:00
|
|
|
src_x, src_y,
|
|
|
|
dst_x + atlas_tex->rectangle.x + 1,
|
|
|
|
dst_y + atlas_tex->rectangle.y + 1,
|
|
|
|
dst_width,
|
|
|
|
dst_height,
|
|
|
|
bmp))
|
2010-01-18 05:53:00 -05:00
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
/* Update the left edge pixels */
|
|
|
|
if (dst_x == 0 &&
|
2010-08-02 11:29:10 -04:00
|
|
|
!_cogl_texture_set_region_from_bitmap (atlas->texture,
|
2010-07-07 13:44:16 -04:00
|
|
|
src_x, src_y,
|
|
|
|
atlas_tex->rectangle.x,
|
|
|
|
dst_y + atlas_tex->rectangle.y + 1,
|
|
|
|
1, dst_height,
|
|
|
|
bmp))
|
2010-01-18 05:53:00 -05:00
|
|
|
return FALSE;
|
|
|
|
/* Update the right edge pixels */
|
|
|
|
if (dst_x + dst_width == atlas_tex->rectangle.width - 2 &&
|
2010-08-02 11:29:10 -04:00
|
|
|
!_cogl_texture_set_region_from_bitmap (atlas->texture,
|
2010-07-07 13:44:16 -04:00
|
|
|
src_x + dst_width - 1, src_y,
|
|
|
|
atlas_tex->rectangle.x +
|
|
|
|
atlas_tex->rectangle.width - 1,
|
|
|
|
dst_y + atlas_tex->rectangle.y + 1,
|
|
|
|
1, dst_height,
|
|
|
|
bmp))
|
2010-01-18 05:53:00 -05:00
|
|
|
return FALSE;
|
|
|
|
/* Update the top edge pixels */
|
|
|
|
if (dst_y == 0 &&
|
2010-08-02 11:29:10 -04:00
|
|
|
!_cogl_texture_set_region_from_bitmap (atlas->texture,
|
2010-07-07 13:44:16 -04:00
|
|
|
src_x, src_y,
|
|
|
|
dst_x + atlas_tex->rectangle.x + 1,
|
|
|
|
atlas_tex->rectangle.y,
|
|
|
|
dst_width, 1,
|
|
|
|
bmp))
|
2010-01-18 05:53:00 -05:00
|
|
|
return FALSE;
|
|
|
|
/* Update the bottom edge pixels */
|
|
|
|
if (dst_y + dst_height == atlas_tex->rectangle.height - 2 &&
|
2010-08-02 11:29:10 -04:00
|
|
|
!_cogl_texture_set_region_from_bitmap (atlas->texture,
|
2010-07-07 13:44:16 -04:00
|
|
|
src_x, src_y + dst_height - 1,
|
|
|
|
dst_x + atlas_tex->rectangle.x + 1,
|
|
|
|
atlas_tex->rectangle.y +
|
|
|
|
atlas_tex->rectangle.height - 1,
|
|
|
|
dst_width, 1,
|
|
|
|
bmp))
|
2010-01-18 05:53:00 -05:00
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2009-12-04 08:06:32 -05:00
|
|
|
static gboolean
|
|
|
|
_cogl_atlas_texture_set_region (CoglTexture *tex,
|
|
|
|
int src_x,
|
|
|
|
int src_y,
|
|
|
|
int dst_x,
|
|
|
|
int dst_y,
|
|
|
|
unsigned int dst_width,
|
|
|
|
unsigned int dst_height,
|
2010-07-08 10:15:22 -04:00
|
|
|
CoglBitmap *bmp)
|
2009-12-04 08:06:32 -05:00
|
|
|
{
|
|
|
|
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)
|
2010-07-07 13:44:16 -04:00
|
|
|
{
|
|
|
|
gboolean ret;
|
|
|
|
|
|
|
|
bmp = _cogl_bitmap_new_shared (bmp,
|
|
|
|
_cogl_bitmap_get_format (bmp) &
|
|
|
|
~COGL_PREMULT_BIT,
|
|
|
|
_cogl_bitmap_get_width (bmp),
|
|
|
|
_cogl_bitmap_get_height (bmp),
|
|
|
|
_cogl_bitmap_get_rowstride (bmp));
|
|
|
|
|
|
|
|
/* Upload the data ignoring the premult bit */
|
|
|
|
ret = _cogl_atlas_texture_set_region_with_border (atlas_tex,
|
|
|
|
src_x, src_y,
|
|
|
|
dst_x, dst_y,
|
|
|
|
dst_width, dst_height,
|
|
|
|
bmp);
|
2010-07-08 10:15:22 -04:00
|
|
|
|
2010-07-07 13:44:16 -04:00
|
|
|
cogl_object_unref (bmp);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2009-12-04 08:06:32 -05:00
|
|
|
else
|
|
|
|
/* Otherwise we can just forward on to the sub texture */
|
2010-07-08 10:15:22 -04:00
|
|
|
return _cogl_texture_set_region_from_bitmap (atlas_tex->sub_texture,
|
|
|
|
src_x, src_y,
|
|
|
|
dst_x, dst_y,
|
|
|
|
dst_width, dst_height,
|
|
|
|
bmp);
|
2009-12-04 08:06:32 -05:00
|
|
|
}
|
|
|
|
|
2010-07-08 08:54:37 -04:00
|
|
|
static gboolean
|
2009-12-04 08:06:32 -05:00
|
|
|
_cogl_atlas_texture_get_data (CoglTexture *tex,
|
|
|
|
CoglPixelFormat format,
|
|
|
|
unsigned int rowstride,
|
|
|
|
guint8 *data)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
|
|
|
return cogl_texture_get_data (atlas_tex->sub_texture,
|
|
|
|
format,
|
|
|
|
rowstride,
|
|
|
|
data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static CoglPixelFormat
|
|
|
|
_cogl_atlas_texture_get_format (CoglTexture *tex)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* We don't want to forward this on the sub-texture because it isn't
|
|
|
|
the necessarily the same format. This will happen if the texture
|
|
|
|
isn't pre-multiplied */
|
|
|
|
return atlas_tex->format;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GLenum
|
|
|
|
_cogl_atlas_texture_get_gl_format (CoglTexture *tex)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
|
|
|
return _cogl_texture_get_gl_format (atlas_tex->sub_texture);
|
|
|
|
}
|
|
|
|
|
cogl: improves header and coding style consistency
We've had complaints that our Cogl code/headers are a bit "special" so
this is a first pass at tidying things up by giving them some
consistency. These changes are all consistent with how new code in Cogl
is being written, but the style isn't consistently applied across all
code yet.
There are two parts to this patch; but since each one required a large
amount of effort to maintain tidy indenting it made sense to combine the
changes to reduce the time spent re indenting the same lines.
The first change is to use a consistent style for declaring function
prototypes in headers. Cogl headers now consistently use this style for
prototypes:
return_type
cogl_function_name (CoglType arg0,
CoglType arg1);
Not everyone likes this style, but it seems that most of the currently
active Cogl developers agree on it.
The second change is to constrain the use of redundant glib data types
in Cogl. Uses of gint, guint, gfloat, glong, gulong and gchar have all
been replaced with int, unsigned int, float, long, unsigned long and char
respectively. When talking about pixel data; use of guchar has been
replaced with guint8, otherwise unsigned char can be used.
The glib types that we continue to use for portability are gboolean,
gint{8,16,32,64}, guint{8,16,32,64} and gsize.
The general intention is that Cogl should look palatable to the widest
range of C programmers including those outside the Gnome community so
- especially for the public API - we want to minimize the number of
foreign looking typedefs.
2010-02-09 20:57:32 -05:00
|
|
|
static int
|
2009-12-04 08:06:32 -05:00
|
|
|
_cogl_atlas_texture_get_width (CoglTexture *tex)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
|
|
|
return cogl_texture_get_width (atlas_tex->sub_texture);
|
|
|
|
}
|
|
|
|
|
cogl: improves header and coding style consistency
We've had complaints that our Cogl code/headers are a bit "special" so
this is a first pass at tidying things up by giving them some
consistency. These changes are all consistent with how new code in Cogl
is being written, but the style isn't consistently applied across all
code yet.
There are two parts to this patch; but since each one required a large
amount of effort to maintain tidy indenting it made sense to combine the
changes to reduce the time spent re indenting the same lines.
The first change is to use a consistent style for declaring function
prototypes in headers. Cogl headers now consistently use this style for
prototypes:
return_type
cogl_function_name (CoglType arg0,
CoglType arg1);
Not everyone likes this style, but it seems that most of the currently
active Cogl developers agree on it.
The second change is to constrain the use of redundant glib data types
in Cogl. Uses of gint, guint, gfloat, glong, gulong and gchar have all
been replaced with int, unsigned int, float, long, unsigned long and char
respectively. When talking about pixel data; use of guchar has been
replaced with guint8, otherwise unsigned char can be used.
The glib types that we continue to use for portability are gboolean,
gint{8,16,32,64}, guint{8,16,32,64} and gsize.
The general intention is that Cogl should look palatable to the widest
range of C programmers including those outside the Gnome community so
- especially for the public API - we want to minimize the number of
foreign looking typedefs.
2010-02-09 20:57:32 -05:00
|
|
|
static int
|
2009-12-04 08:06:32 -05:00
|
|
|
_cogl_atlas_texture_get_height (CoglTexture *tex)
|
|
|
|
{
|
|
|
|
CoglAtlasTexture *atlas_tex = COGL_ATLAS_TEXTURE (tex);
|
|
|
|
|
|
|
|
/* Forward on to the sub texture */
|
|
|
|
return cogl_texture_get_height (atlas_tex->sub_texture);
|
|
|
|
}
|
|
|
|
|
2010-05-05 11:45:21 -04:00
|
|
|
static gboolean
|
|
|
|
_cogl_atlas_texture_can_use_format (CoglPixelFormat format)
|
|
|
|
{
|
|
|
|
/* We don't care about the ordering or the premult status and we can
|
|
|
|
accept RGBA or RGB textures. Although we could also accept
|
|
|
|
luminance and alpha only textures or 16-bit formats it seems that
|
|
|
|
if the application is explicitly using these formats then they've
|
|
|
|
got a reason to want the lower memory requirements so putting
|
|
|
|
them in the atlas might not be a good idea */
|
|
|
|
format &= ~(COGL_PREMULT_BIT | COGL_BGR_BIT | COGL_AFIRST_BIT);
|
|
|
|
return (format == COGL_PIXEL_FORMAT_RGB_888 ||
|
|
|
|
format == COGL_PIXEL_FORMAT_RGBA_8888);
|
|
|
|
}
|
|
|
|
|
2009-12-04 08:06:32 -05:00
|
|
|
CoglHandle
|
2010-07-07 13:44:16 -04:00
|
|
|
_cogl_atlas_texture_new_from_bitmap (CoglBitmap *bmp,
|
2009-12-04 08:06:32 -05:00
|
|
|
CoglTextureFlags flags,
|
|
|
|
CoglPixelFormat internal_format)
|
|
|
|
{
|
2010-02-01 07:11:58 -05:00
|
|
|
CoglAtlasTexture *atlas_tex;
|
2010-07-07 13:44:16 -04:00
|
|
|
CoglBitmap *dst_bmp;
|
|
|
|
CoglBitmap *override_bmp;
|
2010-02-01 07:11:58 -05:00
|
|
|
GLenum gl_intformat;
|
|
|
|
GLenum gl_format;
|
|
|
|
GLenum gl_type;
|
2010-07-07 13:44:16 -04:00
|
|
|
int bmp_width;
|
|
|
|
int bmp_height;
|
|
|
|
CoglPixelFormat bmp_format;
|
2009-12-04 08:06:32 -05:00
|
|
|
|
|
|
|
_COGL_GET_CONTEXT (ctx, COGL_INVALID_HANDLE);
|
|
|
|
|
2010-07-07 13:44:16 -04:00
|
|
|
g_return_val_if_fail (cogl_is_bitmap (bmp), COGL_INVALID_HANDLE);
|
2009-12-04 08:06:32 -05:00
|
|
|
|
2010-01-21 10:34:19 -05:00
|
|
|
/* Don't put textures in the atlas if the user has explicitly
|
|
|
|
requested to disable it */
|
|
|
|
if (G_UNLIKELY (cogl_debug_flags & COGL_DEBUG_DISABLE_ATLAS))
|
|
|
|
return COGL_INVALID_HANDLE;
|
|
|
|
|
2009-12-04 08:06:32 -05:00
|
|
|
/* We can't put the texture in the atlas if there are any special
|
|
|
|
flags. This precludes textures with COGL_TEXTURE_NO_ATLAS and
|
|
|
|
COGL_TEXTURE_NO_SLICING from being atlased */
|
|
|
|
if (flags)
|
|
|
|
return COGL_INVALID_HANDLE;
|
|
|
|
|
2010-07-07 13:44:16 -04:00
|
|
|
bmp_width = _cogl_bitmap_get_width (bmp);
|
|
|
|
bmp_height = _cogl_bitmap_get_height (bmp);
|
|
|
|
bmp_format = _cogl_bitmap_get_format (bmp);
|
|
|
|
|
2009-12-04 08:06:32 -05:00
|
|
|
/* We can't atlas zero-sized textures because it breaks the atlas
|
|
|
|
data structure */
|
2010-07-07 13:44:16 -04:00
|
|
|
if (bmp_width < 1 || bmp_height < 1)
|
2009-12-04 08:06:32 -05:00
|
|
|
return COGL_INVALID_HANDLE;
|
|
|
|
|
2010-02-16 07:01:18 -05:00
|
|
|
/* If we can't use FBOs or we can't read back texture data then it
|
|
|
|
will be too slow to migrate textures and we shouldn't use the
|
|
|
|
atlas */
|
|
|
|
if (!cogl_features_available (COGL_FEATURE_TEXTURE_READ_PIXELS) ||
|
|
|
|
!cogl_features_available (COGL_FEATURE_OFFSCREEN))
|
2009-12-04 08:06:32 -05:00
|
|
|
return COGL_INVALID_HANDLE;
|
|
|
|
|
2010-07-07 13:44:16 -04:00
|
|
|
COGL_NOTE (ATLAS, "Adding texture of size %ix%i", bmp_width, bmp_height);
|
2009-12-04 13:55:53 -05:00
|
|
|
|
2010-07-07 13:44:16 -04:00
|
|
|
internal_format = _cogl_texture_determine_internal_format (bmp_format,
|
2010-02-03 17:54:44 -05:00
|
|
|
internal_format);
|
2010-02-01 07:11:58 -05:00
|
|
|
|
2010-05-05 11:45:21 -04:00
|
|
|
/* If the texture is in a strange format then we won't use it */
|
|
|
|
if (!_cogl_atlas_texture_can_use_format (internal_format))
|
2009-12-04 08:06:32 -05:00
|
|
|
{
|
2009-12-04 13:55:53 -05:00
|
|
|
COGL_NOTE (ATLAS, "Texture can not be added because the "
|
|
|
|
"format is unsupported");
|
|
|
|
|
2009-12-04 08:06:32 -05:00
|
|
|
return COGL_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We need to allocate the texture now because we need the pointer
|
|
|
|
to set as the data for the rectangle in the atlas */
|
|
|
|
atlas_tex = g_new (CoglAtlasTexture, 1);
|
2010-08-02 11:29:10 -04:00
|
|
|
|
|
|
|
atlas_tex->sub_texture = COGL_INVALID_HANDLE;
|
2009-12-04 08:06:32 -05:00
|
|
|
|
|
|
|
/* Try to make some space in the atlas for the texture */
|
2010-08-02 11:29:10 -04:00
|
|
|
if (!_cogl_atlas_reserve_space (_cogl_atlas_texture_get_atlas (),
|
|
|
|
/* Add two pixels for the border */
|
|
|
|
bmp_width + 2, bmp_height + 2,
|
|
|
|
atlas_tex))
|
2009-12-04 08:06:32 -05:00
|
|
|
{
|
|
|
|
g_free (atlas_tex);
|
|
|
|
return COGL_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
2010-07-07 13:44:16 -04:00
|
|
|
dst_bmp = _cogl_texture_prepare_for_upload (bmp,
|
|
|
|
internal_format,
|
|
|
|
&internal_format,
|
|
|
|
&gl_intformat,
|
|
|
|
&gl_format,
|
|
|
|
&gl_type);
|
|
|
|
|
|
|
|
if (dst_bmp == NULL)
|
2009-12-04 08:06:32 -05:00
|
|
|
{
|
2010-08-02 11:29:10 -04:00
|
|
|
_cogl_atlas_remove (_cogl_atlas_texture_get_atlas (),
|
|
|
|
&atlas_tex->rectangle);
|
2009-12-04 08:06:32 -05:00
|
|
|
g_free (atlas_tex);
|
|
|
|
return COGL_INVALID_HANDLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
atlas_tex->_parent.vtable = &cogl_atlas_texture_vtable;
|
|
|
|
atlas_tex->format = internal_format;
|
|
|
|
atlas_tex->in_atlas = TRUE;
|
|
|
|
|
2010-07-07 13:44:16 -04:00
|
|
|
/* Make another bitmap so that we can override the format */
|
|
|
|
override_bmp = _cogl_bitmap_new_shared (dst_bmp,
|
|
|
|
_cogl_bitmap_get_format (dst_bmp) &
|
|
|
|
~COGL_PREMULT_BIT,
|
|
|
|
_cogl_bitmap_get_width (dst_bmp),
|
|
|
|
_cogl_bitmap_get_height (dst_bmp),
|
|
|
|
_cogl_bitmap_get_rowstride (dst_bmp));
|
|
|
|
cogl_object_unref (dst_bmp);
|
|
|
|
|
2009-12-04 08:06:32 -05:00
|
|
|
/* Defer to set_region so that we can share the code for copying the
|
2010-01-18 05:53:00 -05:00
|
|
|
edge pixels to the border. We don't want to pass the actual
|
|
|
|
format of the converted texture because otherwise it will get
|
|
|
|
unpremultiplied. */
|
|
|
|
_cogl_atlas_texture_set_region_with_border (atlas_tex,
|
2010-07-07 13:44:16 -04:00
|
|
|
0, /* src_x */
|
|
|
|
0, /* src_y */
|
|
|
|
0, /* dst_x */
|
|
|
|
0, /* dst_y */
|
|
|
|
bmp_width, /* dst_width */
|
|
|
|
bmp_height, /* dst_height */
|
|
|
|
override_bmp);
|
|
|
|
|
|
|
|
cogl_object_unref (override_bmp);
|
2009-12-04 08:06:32 -05:00
|
|
|
|
|
|
|
return _cogl_atlas_texture_handle_new (atlas_tex);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const CoglTextureVtable
|
|
|
|
cogl_atlas_texture_vtable =
|
|
|
|
{
|
|
|
|
_cogl_atlas_texture_set_region,
|
|
|
|
_cogl_atlas_texture_get_data,
|
|
|
|
_cogl_atlas_texture_foreach_sub_texture_in_region,
|
|
|
|
_cogl_atlas_texture_get_max_waste,
|
|
|
|
_cogl_atlas_texture_is_sliced,
|
|
|
|
_cogl_atlas_texture_can_hardware_repeat,
|
|
|
|
_cogl_atlas_texture_transform_coords_to_gl,
|
2010-01-18 04:22:04 -05:00
|
|
|
_cogl_atlas_texture_transform_quad_coords_to_gl,
|
2009-12-04 08:06:32 -05:00
|
|
|
_cogl_atlas_texture_get_gl_texture,
|
|
|
|
_cogl_atlas_texture_set_filters,
|
2010-06-09 12:39:59 -04:00
|
|
|
_cogl_atlas_texture_pre_paint,
|
2010-01-18 04:22:04 -05:00
|
|
|
_cogl_atlas_texture_ensure_non_quad_rendering,
|
2010-03-25 13:29:22 -04:00
|
|
|
_cogl_atlas_texture_set_wrap_mode_parameters,
|
2009-12-04 08:06:32 -05:00
|
|
|
_cogl_atlas_texture_get_format,
|
|
|
|
_cogl_atlas_texture_get_gl_format,
|
|
|
|
_cogl_atlas_texture_get_width,
|
2010-04-26 05:01:43 -04:00
|
|
|
_cogl_atlas_texture_get_height,
|
|
|
|
NULL /* is_foreign */
|
2009-12-04 08:06:32 -05:00
|
|
|
};
|