mirror of
https://github.com/brl/mutter.git
synced 2025-01-13 13:12:28 +00:00
d6ca75fbec
All of the texture driver virtual functions now take an explicit CoglContext parameter as a step towards removing the global context.
3397 lines
112 KiB
C
3397 lines
112 KiB
C
/*
|
|
* Cogl
|
|
*
|
|
* An object oriented GL/GLES Abstraction/Utility Layer
|
|
*
|
|
* Copyright (C) 2007,2008,2009 Intel Corporation.
|
|
*
|
|
* 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
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <string.h>
|
|
|
|
#include "cogl-debug.h"
|
|
#include "cogl-internal.h"
|
|
#include "cogl-context-private.h"
|
|
#include "cogl-display-private.h"
|
|
#include "cogl-renderer-private.h"
|
|
#include "cogl-object-private.h"
|
|
#include "cogl-util.h"
|
|
#include "cogl-texture-private.h"
|
|
#include "cogl-framebuffer-private.h"
|
|
#include "cogl-onscreen-template-private.h"
|
|
#include "cogl-clip-stack.h"
|
|
#include "cogl-journal-private.h"
|
|
#include "cogl-winsys-private.h"
|
|
#include "cogl-pipeline-state-private.h"
|
|
#include "cogl-matrix-private.h"
|
|
#include "cogl-primitive-private.h"
|
|
#include "cogl-offscreen.h"
|
|
#include "cogl1-context.h"
|
|
#include "cogl-private.h"
|
|
#include "cogl-primitives-private.h"
|
|
|
|
#ifndef GL_FRAMEBUFFER
|
|
#define GL_FRAMEBUFFER 0x8D40
|
|
#endif
|
|
#ifndef GL_RENDERBUFFER
|
|
#define GL_RENDERBUFFER 0x8D41
|
|
#endif
|
|
#ifndef GL_STENCIL_ATTACHMENT
|
|
#define GL_STENCIL_ATTACHMENT 0x8D00
|
|
#endif
|
|
#ifndef GL_COLOR_ATTACHMENT0
|
|
#define GL_COLOR_ATTACHMENT0 0x8CE0
|
|
#endif
|
|
#ifndef GL_FRAMEBUFFER_COMPLETE
|
|
#define GL_FRAMEBUFFER_COMPLETE 0x8CD5
|
|
#endif
|
|
#ifndef GL_STENCIL_INDEX8
|
|
#define GL_STENCIL_INDEX8 0x8D48
|
|
#endif
|
|
#ifndef GL_DEPTH_STENCIL
|
|
#define GL_DEPTH_STENCIL 0x84F9
|
|
#endif
|
|
#ifndef GL_DEPTH24_STENCIL8
|
|
#define GL_DEPTH24_STENCIL8 0x88F0
|
|
#endif
|
|
#ifndef GL_DEPTH_ATTACHMENT
|
|
#define GL_DEPTH_ATTACHMENT 0x8D00
|
|
#endif
|
|
#ifndef GL_DEPTH_COMPONENT16
|
|
#define GL_DEPTH_COMPONENT16 0x81A5
|
|
#endif
|
|
#ifndef GL_FRAMEBUFFER_ATTACHMENT_RED_SIZE
|
|
#define GL_FRAMEBUFFER_ATTACHMENT_RED_SIZE 0x8212
|
|
#endif
|
|
#ifndef GL_FRAMEBUFFER_ATTACHMENT_GREEN_SIZE
|
|
#define GL_FRAMEBUFFER_ATTACHMENT_GREEN_SIZE 0x8213
|
|
#endif
|
|
#ifndef GL_FRAMEBUFFER_ATTACHMENT_BLUE_SIZE
|
|
#define GL_FRAMEBUFFER_ATTACHMENT_BLUE_SIZE 0x8214
|
|
#endif
|
|
#ifndef GL_FRAMEBUFFER_ATTACHMENT_ALPHA_SIZE
|
|
#define GL_FRAMEBUFFER_ATTACHMENT_ALPHA_SIZE 0x8215
|
|
#endif
|
|
#ifndef GL_FRAMEBUFFER_ATTACHMENT_DEPTH_SIZE
|
|
#define GL_FRAMEBUFFER_ATTACHMENT_DEPTH_SIZE 0x8216
|
|
#endif
|
|
#ifndef GL_FRAMEBUFFER_ATTACHMENT_STENCIL_SIZE
|
|
#define GL_FRAMEBUFFER_ATTACHMENT_STENCIL_SIZE 0x8217
|
|
#endif
|
|
#ifndef GL_READ_FRAMEBUFFER
|
|
#define GL_READ_FRAMEBUFFER 0x8CA8
|
|
#endif
|
|
#ifndef GL_DRAW_FRAMEBUFFER
|
|
#define GL_DRAW_FRAMEBUFFER 0x8CA9
|
|
#endif
|
|
#ifndef GL_TEXTURE_SAMPLES_IMG
|
|
#define GL_TEXTURE_SAMPLES_IMG 0x9136
|
|
#endif
|
|
#ifndef GL_PACK_INVERT_MESA
|
|
#define GL_PACK_INVERT_MESA 0x8758
|
|
#endif
|
|
|
|
|
|
typedef enum {
|
|
_TRY_DEPTH_STENCIL = 1L<<0,
|
|
_TRY_DEPTH24_STENCIL8 = 1L<<1,
|
|
_TRY_DEPTH = 1L<<2,
|
|
_TRY_STENCIL = 1L<<3
|
|
} TryFBOFlags;
|
|
|
|
typedef struct _CoglFramebufferStackEntry
|
|
{
|
|
CoglFramebuffer *draw_buffer;
|
|
CoglFramebuffer *read_buffer;
|
|
} CoglFramebufferStackEntry;
|
|
|
|
extern CoglObjectClass _cogl_onscreen_class;
|
|
|
|
static CoglUserDataKey wire_pipeline_key;
|
|
|
|
static void _cogl_offscreen_free (CoglOffscreen *offscreen);
|
|
|
|
COGL_OBJECT_DEFINE_WITH_CODE (Offscreen, offscreen,
|
|
_cogl_offscreen_class.virt_unref =
|
|
_cogl_framebuffer_unref);
|
|
COGL_OBJECT_DEFINE_DEPRECATED_REF_COUNTING (offscreen);
|
|
|
|
/* XXX:
|
|
* The CoglObject macros don't support any form of inheritance, so for
|
|
* now we implement the CoglObject support for the CoglFramebuffer
|
|
* abstract class manually.
|
|
*/
|
|
|
|
GQuark
|
|
cogl_framebuffer_error_quark (void)
|
|
{
|
|
return g_quark_from_static_string ("cogl-framebuffer-error-quark");
|
|
}
|
|
|
|
gboolean
|
|
cogl_is_framebuffer (void *object)
|
|
{
|
|
CoglObject *obj = object;
|
|
|
|
if (obj == NULL)
|
|
return FALSE;
|
|
|
|
return (obj->klass == &_cogl_onscreen_class ||
|
|
obj->klass == &_cogl_offscreen_class);
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_init (CoglFramebuffer *framebuffer,
|
|
CoglContext *ctx,
|
|
CoglFramebufferType type,
|
|
CoglPixelFormat format,
|
|
int width,
|
|
int height)
|
|
{
|
|
framebuffer->context = cogl_object_ref (ctx);
|
|
|
|
framebuffer->type = type;
|
|
framebuffer->width = width;
|
|
framebuffer->height = height;
|
|
framebuffer->format = format;
|
|
framebuffer->viewport_x = 0;
|
|
framebuffer->viewport_y = 0;
|
|
framebuffer->viewport_width = width;
|
|
framebuffer->viewport_height = height;
|
|
framebuffer->dither_enabled = TRUE;
|
|
|
|
framebuffer->modelview_stack = _cogl_matrix_stack_new ();
|
|
framebuffer->projection_stack = _cogl_matrix_stack_new ();
|
|
|
|
framebuffer->dirty_bitmasks = TRUE;
|
|
|
|
framebuffer->color_mask = COGL_COLOR_MASK_ALL;
|
|
|
|
framebuffer->samples_per_pixel = 0;
|
|
|
|
/* Initialise the clip stack */
|
|
_cogl_clip_state_init (&framebuffer->clip_state);
|
|
|
|
framebuffer->journal = _cogl_journal_new (framebuffer);
|
|
|
|
/* Ensure we know the framebuffer->clear_color* members can't be
|
|
* referenced for our fast-path read-pixel optimization (see
|
|
* _cogl_journal_try_read_pixel()) until some region of the
|
|
* framebuffer is initialized.
|
|
*/
|
|
framebuffer->clear_clip_dirty = TRUE;
|
|
|
|
/* XXX: We have to maintain a central list of all framebuffers
|
|
* because at times we need to be able to flush all known journals.
|
|
*
|
|
* Examples where we need to flush all journals are:
|
|
* - because journal entries can reference OpenGL texture
|
|
* coordinates that may not survive texture-atlas reorganization
|
|
* so we need the ability to flush those entries.
|
|
* - because although we generally advise against modifying
|
|
* pipelines after construction we have to handle that possibility
|
|
* and since pipelines may be referenced in journal entries we
|
|
* need to be able to flush them before allowing the pipelines to
|
|
* be changed.
|
|
*
|
|
* Note we don't maintain a list of journals and associate
|
|
* framebuffers with journals by e.g. having a journal->framebuffer
|
|
* reference since that would introduce a circular reference.
|
|
*
|
|
* Note: As a future change to try and remove the need to index all
|
|
* journals it might be possible to defer resolving of OpenGL
|
|
* texture coordinates for rectangle primitives until we come to
|
|
* flush a journal. This would mean for instance that a single
|
|
* rectangle entry in a journal could later be expanded into
|
|
* multiple quad primitives to handle sliced textures but would mean
|
|
* we don't have to worry about retaining references to OpenGL
|
|
* texture coordinates that may later become invalid.
|
|
*/
|
|
ctx->framebuffers = g_list_prepend (ctx->framebuffers, framebuffer);
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_free (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglContext *ctx = framebuffer->context;
|
|
|
|
_cogl_clip_state_destroy (&framebuffer->clip_state);
|
|
|
|
cogl_object_unref (framebuffer->modelview_stack);
|
|
framebuffer->modelview_stack = NULL;
|
|
|
|
cogl_object_unref (framebuffer->projection_stack);
|
|
framebuffer->projection_stack = NULL;
|
|
|
|
cogl_object_unref (framebuffer->journal);
|
|
|
|
ctx->framebuffers = g_list_remove (ctx->framebuffers, framebuffer);
|
|
cogl_object_unref (ctx);
|
|
|
|
if (ctx->current_draw_buffer == framebuffer)
|
|
ctx->current_draw_buffer = NULL;
|
|
if (ctx->current_read_buffer == framebuffer)
|
|
ctx->current_read_buffer = NULL;
|
|
}
|
|
|
|
const CoglWinsysVtable *
|
|
_cogl_framebuffer_get_winsys (CoglFramebuffer *framebuffer)
|
|
{
|
|
return framebuffer->context->display->renderer->winsys_vtable;
|
|
}
|
|
|
|
/* This version of cogl_clear can be used internally as an alternative
|
|
* to avoid flushing the journal or the framebuffer state. This is
|
|
* needed when doing operations that may be called whiling flushing
|
|
* the journal */
|
|
void
|
|
_cogl_framebuffer_clear_without_flush4f (CoglFramebuffer *framebuffer,
|
|
unsigned long buffers,
|
|
float red,
|
|
float green,
|
|
float blue,
|
|
float alpha)
|
|
{
|
|
GLbitfield gl_buffers = 0;
|
|
|
|
_COGL_GET_CONTEXT (ctx, NO_RETVAL);
|
|
|
|
if (buffers & COGL_BUFFER_BIT_COLOR)
|
|
{
|
|
GE( ctx, glClearColor (red, green, blue, alpha) );
|
|
gl_buffers |= GL_COLOR_BUFFER_BIT;
|
|
|
|
if (ctx->current_gl_color_mask != framebuffer->color_mask)
|
|
{
|
|
CoglColorMask color_mask = framebuffer->color_mask;
|
|
GE( ctx, glColorMask (!!(color_mask & COGL_COLOR_MASK_RED),
|
|
!!(color_mask & COGL_COLOR_MASK_GREEN),
|
|
!!(color_mask & COGL_COLOR_MASK_BLUE),
|
|
!!(color_mask & COGL_COLOR_MASK_ALPHA)));
|
|
ctx->current_gl_color_mask = color_mask;
|
|
/* Make sure the ColorMask is updated when the next primitive is drawn */
|
|
ctx->current_pipeline_changes_since_flush |=
|
|
COGL_PIPELINE_STATE_LOGIC_OPS;
|
|
ctx->current_pipeline_age--;
|
|
}
|
|
}
|
|
|
|
if (buffers & COGL_BUFFER_BIT_DEPTH)
|
|
gl_buffers |= GL_DEPTH_BUFFER_BIT;
|
|
|
|
if (buffers & COGL_BUFFER_BIT_STENCIL)
|
|
gl_buffers |= GL_STENCIL_BUFFER_BIT;
|
|
|
|
if (!gl_buffers)
|
|
{
|
|
static gboolean shown = FALSE;
|
|
|
|
if (!shown)
|
|
{
|
|
g_warning ("You should specify at least one auxiliary buffer "
|
|
"when calling cogl_clear");
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
GE (ctx, glClear (gl_buffers));
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_dirty (CoglFramebuffer *framebuffer)
|
|
{
|
|
framebuffer->clear_clip_dirty = TRUE;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_clear4f (CoglFramebuffer *framebuffer,
|
|
unsigned long buffers,
|
|
float red,
|
|
float green,
|
|
float blue,
|
|
float alpha)
|
|
{
|
|
CoglClipStack *clip_stack = _cogl_framebuffer_get_clip_stack (framebuffer);
|
|
int scissor_x0;
|
|
int scissor_y0;
|
|
int scissor_x1;
|
|
int scissor_y1;
|
|
|
|
_cogl_clip_stack_get_bounds (clip_stack,
|
|
&scissor_x0, &scissor_y0,
|
|
&scissor_x1, &scissor_y1);
|
|
|
|
/* NB: the previous clear could have had an arbitrary clip.
|
|
* NB: everything for the last frame might still be in the journal
|
|
* but we can't assume anything about how each entry was
|
|
* clipped.
|
|
* NB: Clutter will scissor its pick renders which would mean all
|
|
* journal entries have a common ClipStack entry, but without
|
|
* a layering violation Cogl has to explicitly walk the journal
|
|
* entries to determine if this is the case.
|
|
* NB: We have a software only read-pixel optimization in the
|
|
* journal that determines the color at a given framebuffer
|
|
* coordinate for simple scenes without rendering with the GPU.
|
|
* When Clutter is hitting this fast-path we can expect to
|
|
* receive calls to clear the framebuffer with an un-flushed
|
|
* journal.
|
|
* NB: To fully support software based picking for Clutter we
|
|
* need to be able to reliably detect when the contents of a
|
|
* journal can be discarded and when we can skip the call to
|
|
* glClear because it matches the previous clear request.
|
|
*/
|
|
|
|
/* Note: we don't check for the stencil buffer being cleared here
|
|
* since there isn't any public cogl api to manipulate the stencil
|
|
* buffer.
|
|
*
|
|
* Note: we check for an exact clip match here because
|
|
* 1) a smaller clip could mean existing journal entries may
|
|
* need to contribute to regions outside the new clear-clip
|
|
* 2) a larger clip would mean we need to issue a real
|
|
* glClear and we only care about cases avoiding a
|
|
* glClear.
|
|
*
|
|
* Note: Comparing without an epsilon is considered
|
|
* appropriate here.
|
|
*/
|
|
if (buffers & COGL_BUFFER_BIT_COLOR &&
|
|
buffers & COGL_BUFFER_BIT_DEPTH &&
|
|
!framebuffer->clear_clip_dirty &&
|
|
framebuffer->clear_color_red == red &&
|
|
framebuffer->clear_color_green == green &&
|
|
framebuffer->clear_color_blue == blue &&
|
|
framebuffer->clear_color_alpha == alpha &&
|
|
scissor_x0 == framebuffer->clear_clip_x0 &&
|
|
scissor_y0 == framebuffer->clear_clip_y0 &&
|
|
scissor_x1 == framebuffer->clear_clip_x1 &&
|
|
scissor_y1 == framebuffer->clear_clip_y1)
|
|
{
|
|
/* NB: We only have to consider the clip state of journal
|
|
* entries if the current clear is clipped since otherwise we
|
|
* know every pixel of the framebuffer is affected by the clear
|
|
* and so all journal entries become redundant and can simply be
|
|
* discarded.
|
|
*/
|
|
if (clip_stack)
|
|
{
|
|
/*
|
|
* Note: the function for checking the journal entries is
|
|
* quite strict. It avoids detailed checking of all entry
|
|
* clip_stacks by only checking the details of the first
|
|
* entry and then it only verifies that the remaining
|
|
* entries share the same clip_stack ancestry. This means
|
|
* it's possible for some false negatives here but that will
|
|
* just result in us falling back to a real clear.
|
|
*/
|
|
if (_cogl_journal_all_entries_within_bounds (framebuffer->journal,
|
|
scissor_x0, scissor_y0,
|
|
scissor_x1, scissor_y1))
|
|
{
|
|
_cogl_journal_discard (framebuffer->journal);
|
|
goto cleared;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
_cogl_journal_discard (framebuffer->journal);
|
|
goto cleared;
|
|
}
|
|
}
|
|
|
|
COGL_NOTE (DRAW, "Clear begin");
|
|
|
|
_cogl_framebuffer_flush_journal (framebuffer);
|
|
|
|
/* NB: _cogl_framebuffer_flush_state may disrupt various state (such
|
|
* as the pipeline state) when flushing the clip stack, so should
|
|
* always be done first when preparing to draw. */
|
|
_cogl_framebuffer_flush_state (framebuffer, framebuffer,
|
|
COGL_FRAMEBUFFER_STATE_ALL);
|
|
|
|
_cogl_framebuffer_clear_without_flush4f (framebuffer, buffers,
|
|
red, green, blue, alpha);
|
|
|
|
/* This is a debugging variable used to visually display the quad
|
|
* batches from the journal. It is reset here to increase the
|
|
* chances of getting the same colours for each frame during an
|
|
* animation */
|
|
if (G_UNLIKELY (COGL_DEBUG_ENABLED (COGL_DEBUG_RECTANGLES)) &&
|
|
buffers & COGL_BUFFER_BIT_COLOR)
|
|
{
|
|
framebuffer->context->journal_rectangles_color = 1;
|
|
}
|
|
|
|
COGL_NOTE (DRAW, "Clear end");
|
|
|
|
cleared:
|
|
|
|
if (buffers & COGL_BUFFER_BIT_COLOR && buffers & COGL_BUFFER_BIT_DEPTH)
|
|
{
|
|
/* For our fast-path for reading back a single pixel of simple
|
|
* scenes where the whole frame is in the journal we need to
|
|
* track the cleared color of the framebuffer in case the point
|
|
* read doesn't intersect any of the journal rectangles. */
|
|
framebuffer->clear_clip_dirty = FALSE;
|
|
framebuffer->clear_color_red = red;
|
|
framebuffer->clear_color_green = green;
|
|
framebuffer->clear_color_blue = blue;
|
|
framebuffer->clear_color_alpha = alpha;
|
|
|
|
/* NB: A clear may be scissored so we need to track the extents
|
|
* that the clear is applicable too... */
|
|
if (clip_stack)
|
|
{
|
|
_cogl_clip_stack_get_bounds (clip_stack,
|
|
&framebuffer->clear_clip_x0,
|
|
&framebuffer->clear_clip_y0,
|
|
&framebuffer->clear_clip_x1,
|
|
&framebuffer->clear_clip_y1);
|
|
}
|
|
else
|
|
{
|
|
/* FIXME: set degenerate clip */
|
|
}
|
|
}
|
|
else
|
|
_cogl_framebuffer_dirty (framebuffer);
|
|
}
|
|
|
|
/* Note: the 'buffers' and 'color' arguments were switched around on
|
|
* purpose compared to the original cogl_clear API since it was odd
|
|
* that you would be expected to specify a color before even
|
|
* necessarily choosing to clear the color buffer.
|
|
*/
|
|
void
|
|
cogl_framebuffer_clear (CoglFramebuffer *framebuffer,
|
|
unsigned long buffers,
|
|
const CoglColor *color)
|
|
{
|
|
cogl_framebuffer_clear4f (framebuffer, buffers,
|
|
cogl_color_get_red_float (color),
|
|
cogl_color_get_green_float (color),
|
|
cogl_color_get_blue_float (color),
|
|
cogl_color_get_alpha_float (color));
|
|
}
|
|
|
|
int
|
|
cogl_framebuffer_get_width (CoglFramebuffer *framebuffer)
|
|
{
|
|
return framebuffer->width;
|
|
}
|
|
|
|
int
|
|
cogl_framebuffer_get_height (CoglFramebuffer *framebuffer)
|
|
{
|
|
return framebuffer->height;
|
|
}
|
|
|
|
CoglClipState *
|
|
_cogl_framebuffer_get_clip_state (CoglFramebuffer *framebuffer)
|
|
{
|
|
return &framebuffer->clip_state;
|
|
}
|
|
|
|
CoglClipStack *
|
|
_cogl_framebuffer_get_clip_stack (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglClipState *clip_state = _cogl_framebuffer_get_clip_state (framebuffer);
|
|
|
|
return _cogl_clip_state_get_stack (clip_state);
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_set_clip_stack (CoglFramebuffer *framebuffer,
|
|
CoglClipStack *stack)
|
|
{
|
|
CoglClipState *clip_state = _cogl_framebuffer_get_clip_state (framebuffer);
|
|
|
|
_cogl_clip_state_set_stack (clip_state, stack);
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_set_viewport (CoglFramebuffer *framebuffer,
|
|
float x,
|
|
float y,
|
|
float width,
|
|
float height)
|
|
{
|
|
_COGL_RETURN_IF_FAIL (width > 0 && height > 0);
|
|
|
|
if (framebuffer->viewport_x == x &&
|
|
framebuffer->viewport_y == y &&
|
|
framebuffer->viewport_width == width &&
|
|
framebuffer->viewport_height == height)
|
|
return;
|
|
|
|
_cogl_framebuffer_flush_journal (framebuffer);
|
|
|
|
framebuffer->viewport_x = x;
|
|
framebuffer->viewport_y = y;
|
|
framebuffer->viewport_width = width;
|
|
framebuffer->viewport_height = height;
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_VIEWPORT;
|
|
}
|
|
|
|
float
|
|
cogl_framebuffer_get_viewport_x (CoglFramebuffer *framebuffer)
|
|
{
|
|
return framebuffer->viewport_x;
|
|
}
|
|
|
|
float
|
|
cogl_framebuffer_get_viewport_y (CoglFramebuffer *framebuffer)
|
|
{
|
|
return framebuffer->viewport_y;
|
|
}
|
|
|
|
float
|
|
cogl_framebuffer_get_viewport_width (CoglFramebuffer *framebuffer)
|
|
{
|
|
return framebuffer->viewport_width;
|
|
}
|
|
|
|
float
|
|
cogl_framebuffer_get_viewport_height (CoglFramebuffer *framebuffer)
|
|
{
|
|
return framebuffer->viewport_height;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_get_viewport4fv (CoglFramebuffer *framebuffer,
|
|
float *viewport)
|
|
{
|
|
viewport[0] = framebuffer->viewport_x;
|
|
viewport[1] = framebuffer->viewport_y;
|
|
viewport[2] = framebuffer->viewport_width;
|
|
viewport[3] = framebuffer->viewport_height;
|
|
}
|
|
|
|
CoglMatrixStack *
|
|
_cogl_framebuffer_get_modelview_stack (CoglFramebuffer *framebuffer)
|
|
{
|
|
return framebuffer->modelview_stack;
|
|
}
|
|
|
|
CoglMatrixStack *
|
|
_cogl_framebuffer_get_projection_stack (CoglFramebuffer *framebuffer)
|
|
{
|
|
return framebuffer->projection_stack;
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_add_dependency (CoglFramebuffer *framebuffer,
|
|
CoglFramebuffer *dependency)
|
|
{
|
|
GList *l;
|
|
|
|
for (l = framebuffer->deps; l; l = l->next)
|
|
{
|
|
CoglFramebuffer *existing_dep = l->data;
|
|
if (existing_dep == dependency)
|
|
return;
|
|
}
|
|
|
|
/* TODO: generalize the primed-array type structure we e.g. use for
|
|
* cogl_object_set_user_data or for pipeline children as a way to
|
|
* avoid quite a lot of mid-scene micro allocations here... */
|
|
framebuffer->deps =
|
|
g_list_prepend (framebuffer->deps, cogl_object_ref (dependency));
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_remove_all_dependencies (CoglFramebuffer *framebuffer)
|
|
{
|
|
GList *l;
|
|
for (l = framebuffer->deps; l; l = l->next)
|
|
cogl_object_unref (l->data);
|
|
g_list_free (framebuffer->deps);
|
|
framebuffer->deps = NULL;
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_flush_journal (CoglFramebuffer *framebuffer)
|
|
{
|
|
_cogl_journal_flush (framebuffer->journal);
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_flush_dependency_journals (CoglFramebuffer *framebuffer)
|
|
{
|
|
GList *l;
|
|
for (l = framebuffer->deps; l; l = l->next)
|
|
_cogl_framebuffer_flush_journal (l->data);
|
|
_cogl_framebuffer_remove_all_dependencies (framebuffer);
|
|
}
|
|
|
|
static inline void
|
|
_cogl_framebuffer_init_bits (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglContext *ctx = framebuffer->context;
|
|
|
|
cogl_framebuffer_allocate (framebuffer, NULL);
|
|
|
|
if (G_LIKELY (!framebuffer->dirty_bitmasks))
|
|
return;
|
|
|
|
#ifdef HAVE_COGL_GL
|
|
if (ctx->driver == COGL_DRIVER_GL &&
|
|
cogl_has_feature (ctx, COGL_FEATURE_ID_OFFSCREEN) &&
|
|
framebuffer->type == COGL_FRAMEBUFFER_TYPE_OFFSCREEN)
|
|
{
|
|
GLenum attachment, pname;
|
|
|
|
attachment = GL_COLOR_ATTACHMENT0;
|
|
|
|
pname = GL_FRAMEBUFFER_ATTACHMENT_RED_SIZE;
|
|
GE( ctx, glGetFramebufferAttachmentParameteriv (GL_FRAMEBUFFER,
|
|
attachment,
|
|
pname,
|
|
&framebuffer->red_bits) );
|
|
|
|
pname = GL_FRAMEBUFFER_ATTACHMENT_GREEN_SIZE;
|
|
GE( ctx, glGetFramebufferAttachmentParameteriv (GL_FRAMEBUFFER,
|
|
attachment,
|
|
pname,
|
|
&framebuffer->green_bits)
|
|
);
|
|
|
|
pname = GL_FRAMEBUFFER_ATTACHMENT_BLUE_SIZE;
|
|
GE( ctx, glGetFramebufferAttachmentParameteriv (GL_FRAMEBUFFER,
|
|
attachment,
|
|
pname,
|
|
&framebuffer->blue_bits)
|
|
);
|
|
|
|
pname = GL_FRAMEBUFFER_ATTACHMENT_ALPHA_SIZE;
|
|
GE( ctx, glGetFramebufferAttachmentParameteriv (GL_FRAMEBUFFER,
|
|
attachment,
|
|
pname,
|
|
&framebuffer->alpha_bits)
|
|
);
|
|
}
|
|
else
|
|
#endif /* HAVE_COGL_GL */
|
|
{
|
|
GE( ctx, glGetIntegerv (GL_RED_BITS, &framebuffer->red_bits) );
|
|
GE( ctx, glGetIntegerv (GL_GREEN_BITS, &framebuffer->green_bits) );
|
|
GE( ctx, glGetIntegerv (GL_BLUE_BITS, &framebuffer->blue_bits) );
|
|
GE( ctx, glGetIntegerv (GL_ALPHA_BITS, &framebuffer->alpha_bits) );
|
|
}
|
|
|
|
|
|
COGL_NOTE (OFFSCREEN,
|
|
"RGBA Bits for framebuffer[%p, %s]: %d, %d, %d, %d",
|
|
framebuffer,
|
|
framebuffer->type == COGL_FRAMEBUFFER_TYPE_OFFSCREEN
|
|
? "offscreen"
|
|
: "onscreen",
|
|
framebuffer->red_bits,
|
|
framebuffer->blue_bits,
|
|
framebuffer->green_bits,
|
|
framebuffer->alpha_bits);
|
|
|
|
framebuffer->dirty_bitmasks = FALSE;
|
|
}
|
|
|
|
CoglOffscreen *
|
|
_cogl_offscreen_new_to_texture_full (CoglTexture *texture,
|
|
CoglOffscreenFlags create_flags,
|
|
unsigned int level)
|
|
{
|
|
CoglOffscreen *offscreen;
|
|
CoglFramebuffer *fb;
|
|
int level_width;
|
|
int level_height;
|
|
int i;
|
|
CoglHandle ret;
|
|
|
|
_COGL_GET_CONTEXT (ctx, COGL_INVALID_HANDLE);
|
|
|
|
if (!cogl_has_feature (ctx, COGL_FEATURE_ID_OFFSCREEN))
|
|
return COGL_INVALID_HANDLE;
|
|
|
|
/* Make texture is a valid texture object */
|
|
if (!cogl_is_texture (texture))
|
|
return COGL_INVALID_HANDLE;
|
|
|
|
/* The texture must not be sliced */
|
|
if (cogl_texture_is_sliced (texture))
|
|
return COGL_INVALID_HANDLE;
|
|
|
|
/* Calculate the size of the texture at this mipmap level to ensure
|
|
that it's a valid level */
|
|
level_width = cogl_texture_get_width (texture);
|
|
level_height = cogl_texture_get_height (texture);
|
|
|
|
for (i = 0; i < level; i++)
|
|
{
|
|
/* If neither dimension can be further divided then the level is
|
|
invalid */
|
|
if (level_width == 1 && level_height == 1)
|
|
{
|
|
g_warning ("Invalid texture level passed to "
|
|
"_cogl_offscreen_new_to_texture_full");
|
|
return COGL_INVALID_HANDLE;
|
|
}
|
|
|
|
if (level_width > 1)
|
|
level_width >>= 1;
|
|
if (level_height > 1)
|
|
level_height >>= 1;
|
|
}
|
|
|
|
offscreen = g_new0 (CoglOffscreen, 1);
|
|
offscreen->texture = cogl_object_ref (texture);
|
|
offscreen->texture_level = level;
|
|
offscreen->texture_level_width = level_width;
|
|
offscreen->texture_level_height = level_height;
|
|
offscreen->create_flags = create_flags;
|
|
|
|
fb = COGL_FRAMEBUFFER (offscreen);
|
|
|
|
_cogl_framebuffer_init (fb,
|
|
ctx,
|
|
COGL_FRAMEBUFFER_TYPE_OFFSCREEN,
|
|
cogl_texture_get_format (texture),
|
|
level_width,
|
|
level_height);
|
|
|
|
ret = _cogl_offscreen_object_new (offscreen);
|
|
|
|
_cogl_texture_associate_framebuffer (texture, fb);
|
|
|
|
return ret;
|
|
}
|
|
|
|
CoglOffscreen *
|
|
cogl_offscreen_new_to_texture (CoglTexture *texture)
|
|
{
|
|
return _cogl_offscreen_new_to_texture_full (texture, 0, 0);
|
|
}
|
|
|
|
static void
|
|
_cogl_offscreen_free (CoglOffscreen *offscreen)
|
|
{
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (offscreen);
|
|
CoglContext *ctx = framebuffer->context;
|
|
GSList *l;
|
|
|
|
/* Chain up to parent */
|
|
_cogl_framebuffer_free (framebuffer);
|
|
|
|
for (l = offscreen->renderbuffers; l; l = l->next)
|
|
{
|
|
GLuint renderbuffer = GPOINTER_TO_UINT (l->data);
|
|
GE (ctx, glDeleteRenderbuffers (1, &renderbuffer));
|
|
}
|
|
g_slist_free (offscreen->renderbuffers);
|
|
|
|
GE (ctx, glDeleteFramebuffers (1, &offscreen->fbo_handle));
|
|
|
|
if (offscreen->texture != COGL_INVALID_HANDLE)
|
|
cogl_object_unref (offscreen->texture);
|
|
|
|
g_free (offscreen);
|
|
}
|
|
|
|
static gboolean
|
|
try_creating_fbo (CoglOffscreen *offscreen,
|
|
TryFBOFlags flags)
|
|
{
|
|
CoglFramebuffer *fb = COGL_FRAMEBUFFER (offscreen);
|
|
CoglContext *ctx = fb->context;
|
|
GLuint gl_depth_stencil_handle;
|
|
GLuint gl_depth_handle;
|
|
GLuint gl_stencil_handle;
|
|
GLuint tex_gl_handle;
|
|
GLenum tex_gl_target;
|
|
GLuint fbo_gl_handle;
|
|
GLenum status;
|
|
int n_samples;
|
|
int height;
|
|
int width;
|
|
|
|
if (!cogl_texture_get_gl_texture (offscreen->texture,
|
|
&tex_gl_handle, &tex_gl_target))
|
|
return FALSE;
|
|
|
|
if (tex_gl_target != GL_TEXTURE_2D
|
|
#ifdef HAVE_COGL_GL
|
|
&& tex_gl_target != GL_TEXTURE_RECTANGLE_ARB
|
|
#endif
|
|
)
|
|
return FALSE;
|
|
|
|
if (fb->config.samples_per_pixel)
|
|
{
|
|
if (!ctx->glFramebufferTexture2DMultisampleIMG)
|
|
return FALSE;
|
|
n_samples = fb->config.samples_per_pixel;
|
|
}
|
|
else
|
|
n_samples = 0;
|
|
|
|
width = offscreen->texture_level_width;
|
|
height = offscreen->texture_level_height;
|
|
|
|
/* We are about to generate and bind a new fbo, so we pretend to
|
|
* change framebuffer state so that the old framebuffer will be
|
|
* rebound again before drawing. */
|
|
ctx->current_draw_buffer_changes |= COGL_FRAMEBUFFER_STATE_BIND;
|
|
|
|
/* Generate framebuffer */
|
|
ctx->glGenFramebuffers (1, &fbo_gl_handle);
|
|
GE (ctx, glBindFramebuffer (GL_FRAMEBUFFER, fbo_gl_handle));
|
|
offscreen->fbo_handle = fbo_gl_handle;
|
|
|
|
if (n_samples)
|
|
{
|
|
GE (ctx, glFramebufferTexture2DMultisampleIMG (GL_FRAMEBUFFER,
|
|
GL_COLOR_ATTACHMENT0,
|
|
tex_gl_target, tex_gl_handle,
|
|
n_samples,
|
|
offscreen->texture_level));
|
|
}
|
|
else
|
|
GE (ctx, glFramebufferTexture2D (GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0,
|
|
tex_gl_target, tex_gl_handle,
|
|
offscreen->texture_level));
|
|
|
|
if (flags & (_TRY_DEPTH_STENCIL | _TRY_DEPTH24_STENCIL8))
|
|
{
|
|
GLenum format = ((flags & _TRY_DEPTH_STENCIL) ?
|
|
GL_DEPTH_STENCIL : GL_DEPTH24_STENCIL8);
|
|
|
|
/* Create a renderbuffer for depth and stenciling */
|
|
GE (ctx, glGenRenderbuffers (1, &gl_depth_stencil_handle));
|
|
GE (ctx, glBindRenderbuffer (GL_RENDERBUFFER, gl_depth_stencil_handle));
|
|
if (n_samples)
|
|
GE (ctx, glRenderbufferStorageMultisampleIMG (GL_RENDERBUFFER,
|
|
n_samples,
|
|
format,
|
|
width, height));
|
|
else
|
|
GE (ctx, glRenderbufferStorage (GL_RENDERBUFFER, format,
|
|
width, height));
|
|
GE (ctx, glBindRenderbuffer (GL_RENDERBUFFER, 0));
|
|
GE (ctx, glFramebufferRenderbuffer (GL_FRAMEBUFFER,
|
|
GL_STENCIL_ATTACHMENT,
|
|
GL_RENDERBUFFER,
|
|
gl_depth_stencil_handle));
|
|
GE (ctx, glFramebufferRenderbuffer (GL_FRAMEBUFFER,
|
|
GL_DEPTH_ATTACHMENT,
|
|
GL_RENDERBUFFER,
|
|
gl_depth_stencil_handle));
|
|
offscreen->renderbuffers =
|
|
g_slist_prepend (offscreen->renderbuffers,
|
|
GUINT_TO_POINTER (gl_depth_stencil_handle));
|
|
}
|
|
|
|
if (flags & _TRY_DEPTH)
|
|
{
|
|
GE (ctx, glGenRenderbuffers (1, &gl_depth_handle));
|
|
GE (ctx, glBindRenderbuffer (GL_RENDERBUFFER, gl_depth_handle));
|
|
/* For now we just ask for GL_DEPTH_COMPONENT16 since this is all that's
|
|
* available under GLES */
|
|
if (n_samples)
|
|
GE (ctx, glRenderbufferStorageMultisampleIMG (GL_RENDERBUFFER,
|
|
n_samples,
|
|
GL_DEPTH_COMPONENT16,
|
|
width, height));
|
|
else
|
|
GE (ctx, glRenderbufferStorage (GL_RENDERBUFFER, GL_DEPTH_COMPONENT16,
|
|
width, height));
|
|
GE (ctx, glBindRenderbuffer (GL_RENDERBUFFER, 0));
|
|
GE (ctx, glFramebufferRenderbuffer (GL_FRAMEBUFFER,
|
|
GL_DEPTH_ATTACHMENT,
|
|
GL_RENDERBUFFER, gl_depth_handle));
|
|
offscreen->renderbuffers =
|
|
g_slist_prepend (offscreen->renderbuffers,
|
|
GUINT_TO_POINTER (gl_depth_handle));
|
|
}
|
|
|
|
if (flags & _TRY_STENCIL)
|
|
{
|
|
GE (ctx, glGenRenderbuffers (1, &gl_stencil_handle));
|
|
GE (ctx, glBindRenderbuffer (GL_RENDERBUFFER, gl_stencil_handle));
|
|
if (n_samples)
|
|
GE (ctx, glRenderbufferStorageMultisampleIMG (GL_RENDERBUFFER,
|
|
n_samples,
|
|
GL_STENCIL_INDEX8,
|
|
width, height));
|
|
else
|
|
GE (ctx, glRenderbufferStorage (GL_RENDERBUFFER, GL_STENCIL_INDEX8,
|
|
width, height));
|
|
GE (ctx, glBindRenderbuffer (GL_RENDERBUFFER, 0));
|
|
GE (ctx, glFramebufferRenderbuffer (GL_FRAMEBUFFER,
|
|
GL_STENCIL_ATTACHMENT,
|
|
GL_RENDERBUFFER, gl_stencil_handle));
|
|
offscreen->renderbuffers =
|
|
g_slist_prepend (offscreen->renderbuffers,
|
|
GUINT_TO_POINTER (gl_stencil_handle));
|
|
}
|
|
|
|
/* Make sure it's complete */
|
|
status = ctx->glCheckFramebufferStatus (GL_FRAMEBUFFER);
|
|
|
|
if (status != GL_FRAMEBUFFER_COMPLETE)
|
|
{
|
|
GSList *l;
|
|
|
|
GE (ctx, glDeleteFramebuffers (1, &fbo_gl_handle));
|
|
|
|
for (l = offscreen->renderbuffers; l; l = l->next)
|
|
{
|
|
GLuint renderbuffer = GPOINTER_TO_UINT (l->data);
|
|
GE (ctx, glDeleteRenderbuffers (1, &renderbuffer));
|
|
}
|
|
|
|
g_slist_free (offscreen->renderbuffers);
|
|
offscreen->renderbuffers = NULL;
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/* Update the real number of samples_per_pixel now that we have a
|
|
* complete framebuffer */
|
|
if (n_samples)
|
|
{
|
|
GLenum attachment = GL_COLOR_ATTACHMENT0;
|
|
GLenum pname = GL_TEXTURE_SAMPLES_IMG;
|
|
int texture_samples;
|
|
|
|
GE( ctx, glGetFramebufferAttachmentParameteriv (GL_FRAMEBUFFER,
|
|
attachment,
|
|
pname,
|
|
&texture_samples) );
|
|
fb->samples_per_pixel = texture_samples;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
_cogl_offscreen_allocate (CoglOffscreen *offscreen,
|
|
GError **error)
|
|
{
|
|
CoglFramebuffer *fb = COGL_FRAMEBUFFER (offscreen);
|
|
CoglContext *ctx = fb->context;
|
|
static TryFBOFlags flags;
|
|
static gboolean have_working_flags = FALSE;
|
|
gboolean fbo_created;
|
|
|
|
/* XXX: The framebuffer_object spec isn't clear in defining whether attaching
|
|
* a texture as a renderbuffer with mipmap filtering enabled while the
|
|
* mipmaps have not been uploaded should result in an incomplete framebuffer
|
|
* object. (different drivers make different decisions)
|
|
*
|
|
* To avoid an error with drivers that do consider this a problem we
|
|
* explicitly set non mipmapped filters here. These will later be reset when
|
|
* the texture is actually used for rendering according to the filters set on
|
|
* the corresponding CoglPipeline.
|
|
*/
|
|
_cogl_texture_set_filters (offscreen->texture, GL_NEAREST, GL_NEAREST);
|
|
|
|
if ((offscreen->create_flags & COGL_OFFSCREEN_DISABLE_DEPTH_AND_STENCIL))
|
|
fbo_created = try_creating_fbo (offscreen, 0);
|
|
else
|
|
{
|
|
if ((have_working_flags &&
|
|
try_creating_fbo (offscreen, flags)) ||
|
|
((ctx->private_feature_flags &
|
|
COGL_PRIVATE_FEATURE_EXT_PACKED_DEPTH_STENCIL) &&
|
|
try_creating_fbo (offscreen, flags = _TRY_DEPTH_STENCIL)) ||
|
|
((ctx->private_feature_flags &
|
|
COGL_PRIVATE_FEATURE_OES_PACKED_DEPTH_STENCIL) &&
|
|
try_creating_fbo (offscreen, flags = _TRY_DEPTH24_STENCIL8)) ||
|
|
try_creating_fbo (offscreen, flags = _TRY_DEPTH | _TRY_STENCIL) ||
|
|
try_creating_fbo (offscreen, flags = _TRY_STENCIL) ||
|
|
try_creating_fbo (offscreen, flags = _TRY_DEPTH) ||
|
|
try_creating_fbo (offscreen, flags = 0))
|
|
{
|
|
/* Record that the last set of flags succeeded so that we can
|
|
try that set first next time */
|
|
have_working_flags = TRUE;
|
|
fbo_created = TRUE;
|
|
}
|
|
else
|
|
fbo_created = FALSE;
|
|
}
|
|
|
|
if (!fbo_created)
|
|
{
|
|
g_set_error (error, COGL_FRAMEBUFFER_ERROR,
|
|
COGL_FRAMEBUFFER_ERROR_ALLOCATE,
|
|
"Failed to create an OpenGL framebuffer object");
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
cogl_framebuffer_allocate (CoglFramebuffer *framebuffer,
|
|
GError **error)
|
|
{
|
|
CoglOnscreen *onscreen = COGL_ONSCREEN (framebuffer);
|
|
const CoglWinsysVtable *winsys = _cogl_framebuffer_get_winsys (framebuffer);
|
|
|
|
if (framebuffer->allocated)
|
|
return TRUE;
|
|
|
|
if (framebuffer->type == COGL_FRAMEBUFFER_TYPE_ONSCREEN)
|
|
{
|
|
if (!winsys->onscreen_init (onscreen, error))
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
if (!_cogl_offscreen_allocate (COGL_OFFSCREEN (framebuffer), error))
|
|
return FALSE;
|
|
}
|
|
|
|
framebuffer->allocated = TRUE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static CoglFramebufferStackEntry *
|
|
create_stack_entry (CoglFramebuffer *draw_buffer,
|
|
CoglFramebuffer *read_buffer)
|
|
{
|
|
CoglFramebufferStackEntry *entry = g_slice_new (CoglFramebufferStackEntry);
|
|
|
|
entry->draw_buffer = draw_buffer;
|
|
entry->read_buffer = read_buffer;
|
|
|
|
return entry;
|
|
}
|
|
|
|
GSList *
|
|
_cogl_create_framebuffer_stack (void)
|
|
{
|
|
CoglFramebufferStackEntry *entry;
|
|
GSList *stack = NULL;
|
|
|
|
entry = create_stack_entry (COGL_INVALID_HANDLE, COGL_INVALID_HANDLE);
|
|
|
|
return g_slist_prepend (stack, entry);
|
|
}
|
|
|
|
void
|
|
_cogl_free_framebuffer_stack (GSList *stack)
|
|
{
|
|
GSList *l;
|
|
|
|
for (l = stack; l != NULL; l = l->next)
|
|
{
|
|
CoglFramebufferStackEntry *entry = l->data;
|
|
|
|
if (entry->draw_buffer)
|
|
cogl_object_unref (entry->draw_buffer);
|
|
|
|
if (entry->read_buffer)
|
|
cogl_object_unref (entry->draw_buffer);
|
|
|
|
g_slice_free (CoglFramebufferStackEntry, entry);
|
|
}
|
|
g_slist_free (stack);
|
|
}
|
|
|
|
static void
|
|
notify_buffers_changed (CoglFramebuffer *old_draw_buffer,
|
|
CoglFramebuffer *new_draw_buffer,
|
|
CoglFramebuffer *old_read_buffer,
|
|
CoglFramebuffer *new_read_buffer)
|
|
{
|
|
/* XXX: To support the deprecated cogl_set_draw_buffer API we keep
|
|
* track of the last onscreen framebuffer that was set so that it
|
|
* can be restored if the COGL_WINDOW_BUFFER enum is used. A
|
|
* reference isn't taken to the framebuffer because otherwise we
|
|
* would have a circular reference between the context and the
|
|
* framebuffer. Instead the pointer is set to NULL in
|
|
* _cogl_onscreen_free as a kind of a cheap weak reference */
|
|
if (new_draw_buffer &&
|
|
new_draw_buffer->type == COGL_FRAMEBUFFER_TYPE_ONSCREEN)
|
|
new_draw_buffer->context->window_buffer = new_draw_buffer;
|
|
}
|
|
|
|
/* Set the current framebuffer without checking if it's already the
|
|
* current framebuffer. This is used by cogl_pop_framebuffer while
|
|
* the top of the stack is currently not up to date. */
|
|
static void
|
|
_cogl_set_framebuffers_real (CoglFramebuffer *draw_buffer,
|
|
CoglFramebuffer *read_buffer)
|
|
{
|
|
CoglFramebufferStackEntry *entry;
|
|
|
|
_COGL_GET_CONTEXT (ctx, NO_RETVAL);
|
|
|
|
_COGL_RETURN_IF_FAIL (ctx != NULL);
|
|
_COGL_RETURN_IF_FAIL (draw_buffer && read_buffer ?
|
|
draw_buffer->context == read_buffer->context : TRUE);
|
|
|
|
entry = ctx->framebuffer_stack->data;
|
|
|
|
notify_buffers_changed (entry->draw_buffer,
|
|
draw_buffer,
|
|
entry->read_buffer,
|
|
read_buffer);
|
|
|
|
if (draw_buffer)
|
|
cogl_object_ref (draw_buffer);
|
|
if (entry->draw_buffer)
|
|
cogl_object_unref (entry->draw_buffer);
|
|
|
|
if (read_buffer)
|
|
cogl_object_ref (read_buffer);
|
|
if (entry->read_buffer)
|
|
cogl_object_unref (entry->read_buffer);
|
|
|
|
entry->draw_buffer = draw_buffer;
|
|
entry->read_buffer = read_buffer;
|
|
}
|
|
|
|
static void
|
|
_cogl_set_framebuffers (CoglFramebuffer *draw_buffer,
|
|
CoglFramebuffer *read_buffer)
|
|
{
|
|
CoglFramebuffer *current_draw_buffer;
|
|
CoglFramebuffer *current_read_buffer;
|
|
|
|
_COGL_RETURN_IF_FAIL (cogl_is_framebuffer (draw_buffer));
|
|
_COGL_RETURN_IF_FAIL (cogl_is_framebuffer (read_buffer));
|
|
|
|
current_draw_buffer = cogl_get_draw_framebuffer ();
|
|
current_read_buffer = _cogl_get_read_framebuffer ();
|
|
|
|
if (current_draw_buffer != draw_buffer ||
|
|
current_read_buffer != read_buffer)
|
|
_cogl_set_framebuffers_real (draw_buffer, read_buffer);
|
|
}
|
|
|
|
void
|
|
cogl_set_framebuffer (CoglFramebuffer *framebuffer)
|
|
{
|
|
_cogl_set_framebuffers (framebuffer, framebuffer);
|
|
}
|
|
|
|
/* XXX: deprecated API */
|
|
void
|
|
cogl_set_draw_buffer (CoglBufferTarget target, CoglHandle handle)
|
|
{
|
|
_COGL_GET_CONTEXT (ctx, NO_RETVAL);
|
|
|
|
if (target == COGL_WINDOW_BUFFER)
|
|
handle = ctx->window_buffer;
|
|
|
|
/* This is deprecated public API. The public API doesn't currently
|
|
really expose the concept of separate draw and read buffers so
|
|
for the time being this actually just sets both buffers */
|
|
cogl_set_framebuffer (handle);
|
|
}
|
|
|
|
CoglFramebuffer *
|
|
cogl_get_draw_framebuffer (void)
|
|
{
|
|
CoglFramebufferStackEntry *entry;
|
|
|
|
_COGL_GET_CONTEXT (ctx, NULL);
|
|
|
|
g_assert (ctx->framebuffer_stack);
|
|
|
|
entry = ctx->framebuffer_stack->data;
|
|
|
|
return entry->draw_buffer;
|
|
}
|
|
|
|
CoglFramebuffer *
|
|
_cogl_get_read_framebuffer (void)
|
|
{
|
|
CoglFramebufferStackEntry *entry;
|
|
|
|
_COGL_GET_CONTEXT (ctx, NULL);
|
|
|
|
g_assert (ctx->framebuffer_stack);
|
|
|
|
entry = ctx->framebuffer_stack->data;
|
|
|
|
return entry->read_buffer;
|
|
}
|
|
|
|
void
|
|
_cogl_push_framebuffers (CoglFramebuffer *draw_buffer,
|
|
CoglFramebuffer *read_buffer)
|
|
{
|
|
CoglContext *ctx;
|
|
CoglFramebuffer *old_draw_buffer, *old_read_buffer;
|
|
|
|
_COGL_RETURN_IF_FAIL (cogl_is_framebuffer (draw_buffer));
|
|
_COGL_RETURN_IF_FAIL (cogl_is_framebuffer (read_buffer));
|
|
|
|
ctx = draw_buffer->context;
|
|
_COGL_RETURN_IF_FAIL (ctx != NULL);
|
|
_COGL_RETURN_IF_FAIL (draw_buffer->context == read_buffer->context);
|
|
|
|
_COGL_RETURN_IF_FAIL (ctx->framebuffer_stack != NULL);
|
|
|
|
/* Copy the top of the stack so that when we call cogl_set_framebuffer
|
|
it will still know what the old framebuffer was */
|
|
old_draw_buffer = cogl_get_draw_framebuffer ();
|
|
if (old_draw_buffer)
|
|
cogl_object_ref (old_draw_buffer);
|
|
old_read_buffer = _cogl_get_read_framebuffer ();
|
|
if (old_read_buffer)
|
|
cogl_object_ref (old_read_buffer);
|
|
ctx->framebuffer_stack =
|
|
g_slist_prepend (ctx->framebuffer_stack,
|
|
create_stack_entry (old_draw_buffer,
|
|
old_read_buffer));
|
|
|
|
_cogl_set_framebuffers (draw_buffer, read_buffer);
|
|
}
|
|
|
|
void
|
|
cogl_push_framebuffer (CoglFramebuffer *buffer)
|
|
{
|
|
_cogl_push_framebuffers (buffer, buffer);
|
|
}
|
|
|
|
/* XXX: deprecated API */
|
|
void
|
|
cogl_push_draw_buffer (void)
|
|
{
|
|
cogl_push_framebuffer (cogl_get_draw_framebuffer ());
|
|
}
|
|
|
|
void
|
|
cogl_pop_framebuffer (void)
|
|
{
|
|
CoglFramebufferStackEntry *to_pop;
|
|
CoglFramebufferStackEntry *to_restore;
|
|
|
|
_COGL_GET_CONTEXT (ctx, NO_RETVAL);
|
|
|
|
g_assert (ctx->framebuffer_stack != NULL);
|
|
g_assert (ctx->framebuffer_stack->next != NULL);
|
|
|
|
to_pop = ctx->framebuffer_stack->data;
|
|
to_restore = ctx->framebuffer_stack->next->data;
|
|
|
|
if (to_pop->draw_buffer != to_restore->draw_buffer ||
|
|
to_pop->read_buffer != to_restore->read_buffer)
|
|
notify_buffers_changed (to_pop->draw_buffer,
|
|
to_restore->draw_buffer,
|
|
to_pop->read_buffer,
|
|
to_restore->read_buffer);
|
|
|
|
cogl_object_unref (to_pop->draw_buffer);
|
|
cogl_object_unref (to_pop->read_buffer);
|
|
g_slice_free (CoglFramebufferStackEntry, to_pop);
|
|
|
|
ctx->framebuffer_stack =
|
|
g_slist_delete_link (ctx->framebuffer_stack,
|
|
ctx->framebuffer_stack);
|
|
}
|
|
|
|
/* XXX: deprecated API */
|
|
void
|
|
cogl_pop_draw_buffer (void)
|
|
{
|
|
cogl_pop_framebuffer ();
|
|
}
|
|
|
|
static void
|
|
bind_gl_framebuffer (CoglContext *ctx,
|
|
GLenum target,
|
|
CoglFramebuffer *framebuffer)
|
|
{
|
|
if (framebuffer->type == COGL_FRAMEBUFFER_TYPE_OFFSCREEN)
|
|
GE (ctx, glBindFramebuffer (target,
|
|
COGL_OFFSCREEN (framebuffer)->fbo_handle));
|
|
else
|
|
{
|
|
const CoglWinsysVtable *winsys =
|
|
_cogl_framebuffer_get_winsys (framebuffer);
|
|
winsys->onscreen_bind (COGL_ONSCREEN (framebuffer));
|
|
/* glBindFramebuffer is an an extension with OpenGL ES 1.1 */
|
|
if (cogl_has_feature (ctx, COGL_FEATURE_ID_OFFSCREEN))
|
|
GE (ctx, glBindFramebuffer (target, 0));
|
|
}
|
|
}
|
|
|
|
static unsigned long
|
|
_cogl_framebuffer_compare_viewport_state (CoglFramebuffer *a,
|
|
CoglFramebuffer *b)
|
|
{
|
|
if (a->viewport_x != b->viewport_x ||
|
|
a->viewport_y != b->viewport_y ||
|
|
a->viewport_width != b->viewport_width ||
|
|
a->viewport_height != b->viewport_height ||
|
|
/* NB: we render upside down to offscreen framebuffers and that
|
|
* can affect how we setup the GL viewport... */
|
|
a->type != b->type)
|
|
return COGL_FRAMEBUFFER_STATE_VIEWPORT;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
static unsigned long
|
|
_cogl_framebuffer_compare_clip_state (CoglFramebuffer *a,
|
|
CoglFramebuffer *b)
|
|
{
|
|
if (((a->clip_state.stacks == NULL || b->clip_state.stacks == NULL) &&
|
|
a->clip_state.stacks != b->clip_state.stacks)
|
|
||
|
|
a->clip_state.stacks->data != b->clip_state.stacks->data)
|
|
return COGL_FRAMEBUFFER_STATE_CLIP;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
static unsigned long
|
|
_cogl_framebuffer_compare_dither_state (CoglFramebuffer *a,
|
|
CoglFramebuffer *b)
|
|
{
|
|
return a->dither_enabled != b->dither_enabled ?
|
|
COGL_FRAMEBUFFER_STATE_DITHER : 0;
|
|
}
|
|
|
|
static unsigned long
|
|
_cogl_framebuffer_compare_modelview_state (CoglFramebuffer *a,
|
|
CoglFramebuffer *b)
|
|
{
|
|
/* We always want to flush the modelview state. All this does is set
|
|
the current modelview stack on the context to the framebuffer's
|
|
stack. */
|
|
return COGL_FRAMEBUFFER_STATE_MODELVIEW;
|
|
}
|
|
|
|
static unsigned long
|
|
_cogl_framebuffer_compare_projection_state (CoglFramebuffer *a,
|
|
CoglFramebuffer *b)
|
|
{
|
|
/* We always want to flush the projection state. All this does is
|
|
set the current projection stack on the context to the
|
|
framebuffer's stack. */
|
|
return COGL_FRAMEBUFFER_STATE_PROJECTION;
|
|
}
|
|
|
|
static unsigned long
|
|
_cogl_framebuffer_compare_color_mask_state (CoglFramebuffer *a,
|
|
CoglFramebuffer *b)
|
|
{
|
|
if (cogl_framebuffer_get_color_mask (a) !=
|
|
cogl_framebuffer_get_color_mask (b))
|
|
return COGL_FRAMEBUFFER_STATE_COLOR_MASK;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
static unsigned long
|
|
_cogl_framebuffer_compare_front_face_winding_state (CoglFramebuffer *a,
|
|
CoglFramebuffer *b)
|
|
{
|
|
if (a->type != b->type)
|
|
return COGL_FRAMEBUFFER_STATE_FRONT_FACE_WINDING;
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
static unsigned long
|
|
_cogl_framebuffer_compare (CoglFramebuffer *a,
|
|
CoglFramebuffer *b,
|
|
unsigned long state)
|
|
{
|
|
unsigned long differences = 0;
|
|
int bit;
|
|
|
|
if (state & COGL_FRAMEBUFFER_STATE_BIND)
|
|
{
|
|
differences |= COGL_FRAMEBUFFER_STATE_BIND;
|
|
state &= ~COGL_FRAMEBUFFER_STATE_BIND;
|
|
}
|
|
|
|
COGL_FLAGS_FOREACH_START (&state, 1, bit)
|
|
{
|
|
/* XXX: We considered having an array of callbacks for each state index
|
|
* that we'd call here but decided that this way the compiler is more
|
|
* likely going to be able to in-line the comparison functions and use
|
|
* the index to jump straight to the required code. */
|
|
switch (bit)
|
|
{
|
|
case COGL_FRAMEBUFFER_STATE_INDEX_VIEWPORT:
|
|
differences |=
|
|
_cogl_framebuffer_compare_viewport_state (a, b);
|
|
break;
|
|
case COGL_FRAMEBUFFER_STATE_INDEX_CLIP:
|
|
differences |= _cogl_framebuffer_compare_clip_state (a, b);
|
|
break;
|
|
case COGL_FRAMEBUFFER_STATE_INDEX_DITHER:
|
|
differences |= _cogl_framebuffer_compare_dither_state (a, b);
|
|
break;
|
|
case COGL_FRAMEBUFFER_STATE_INDEX_MODELVIEW:
|
|
differences |=
|
|
_cogl_framebuffer_compare_modelview_state (a, b);
|
|
break;
|
|
case COGL_FRAMEBUFFER_STATE_INDEX_PROJECTION:
|
|
differences |=
|
|
_cogl_framebuffer_compare_projection_state (a, b);
|
|
break;
|
|
case COGL_FRAMEBUFFER_STATE_INDEX_COLOR_MASK:
|
|
differences |=
|
|
_cogl_framebuffer_compare_color_mask_state (a, b);
|
|
break;
|
|
case COGL_FRAMEBUFFER_STATE_INDEX_FRONT_FACE_WINDING:
|
|
differences |=
|
|
_cogl_framebuffer_compare_front_face_winding_state (a, b);
|
|
break;
|
|
default:
|
|
g_warn_if_reached ();
|
|
}
|
|
}
|
|
COGL_FLAGS_FOREACH_END;
|
|
|
|
return differences;
|
|
}
|
|
|
|
static void
|
|
_cogl_framebuffer_flush_viewport_state (CoglFramebuffer *framebuffer)
|
|
{
|
|
float gl_viewport_y;
|
|
|
|
g_assert (framebuffer->viewport_width >=0 &&
|
|
framebuffer->viewport_height >=0);
|
|
|
|
/* Convert the Cogl viewport y offset to an OpenGL viewport y offset
|
|
* NB: OpenGL defines its window and viewport origins to be bottom
|
|
* left, while Cogl defines them to be top left.
|
|
* NB: We render upside down to offscreen framebuffers so we don't
|
|
* need to convert the y offset in this case. */
|
|
if (cogl_is_offscreen (framebuffer))
|
|
gl_viewport_y = framebuffer->viewport_y;
|
|
else
|
|
gl_viewport_y = framebuffer->height -
|
|
(framebuffer->viewport_y + framebuffer->viewport_height);
|
|
|
|
COGL_NOTE (OPENGL, "Calling glViewport(%f, %f, %f, %f)",
|
|
framebuffer->viewport_x,
|
|
gl_viewport_y,
|
|
framebuffer->viewport_width,
|
|
framebuffer->viewport_height);
|
|
|
|
GE (framebuffer->context,
|
|
glViewport (framebuffer->viewport_x,
|
|
gl_viewport_y,
|
|
framebuffer->viewport_width,
|
|
framebuffer->viewport_height));
|
|
}
|
|
|
|
static void
|
|
_cogl_framebuffer_flush_clip_state (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglClipStack *stack = _cogl_clip_state_get_stack (&framebuffer->clip_state);
|
|
_cogl_clip_stack_flush (stack, framebuffer);
|
|
}
|
|
|
|
static void
|
|
_cogl_framebuffer_flush_dither_state (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglContext *ctx = framebuffer->context;
|
|
|
|
if (ctx->current_gl_dither_enabled != framebuffer->dither_enabled)
|
|
{
|
|
if (framebuffer->dither_enabled)
|
|
GE (ctx, glEnable (GL_DITHER));
|
|
else
|
|
GE (ctx, glDisable (GL_DITHER));
|
|
ctx->current_gl_dither_enabled = framebuffer->dither_enabled;
|
|
}
|
|
}
|
|
|
|
static void
|
|
_cogl_framebuffer_flush_modelview_state (CoglFramebuffer *framebuffer)
|
|
{
|
|
_cogl_context_set_current_modelview (framebuffer->context,
|
|
framebuffer->modelview_stack);
|
|
}
|
|
|
|
static void
|
|
_cogl_framebuffer_flush_projection_state (CoglFramebuffer *framebuffer)
|
|
{
|
|
_cogl_context_set_current_projection (framebuffer->context,
|
|
framebuffer->projection_stack);
|
|
}
|
|
|
|
static void
|
|
_cogl_framebuffer_flush_color_mask_state (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglContext *context = framebuffer->context;
|
|
|
|
/* The color mask state is really owned by a CoglPipeline so to
|
|
* ensure the color mask is updated the next time we draw something
|
|
* we need to make sure the logic ops for the pipeline are
|
|
* re-flushed... */
|
|
context->current_pipeline_changes_since_flush |=
|
|
COGL_PIPELINE_STATE_LOGIC_OPS;
|
|
context->current_pipeline_age--;
|
|
}
|
|
|
|
static void
|
|
_cogl_framebuffer_flush_front_face_winding_state (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglContext *context = framebuffer->context;
|
|
CoglPipelineCullFaceMode mode;
|
|
|
|
/* NB: The face winding state is actually owned by the current
|
|
* CoglPipeline.
|
|
*
|
|
* If we don't have a current pipeline then we can just assume that
|
|
* when we later do flush a pipeline we will check the current
|
|
* framebuffer to know how to setup the winding */
|
|
if (!context->current_pipeline)
|
|
return;
|
|
|
|
mode = cogl_pipeline_get_cull_face_mode (context->current_pipeline);
|
|
|
|
/* If the current CoglPipeline has a culling mode that doesn't care
|
|
* about the winding we can avoid forcing an update of the state and
|
|
* bail out. */
|
|
if (mode == COGL_PIPELINE_CULL_FACE_MODE_NONE ||
|
|
mode == COGL_PIPELINE_CULL_FACE_MODE_BOTH)
|
|
return;
|
|
|
|
/* Since the winding state is really owned by the current pipeline
|
|
* the way we "flush" an updated winding is to dirty the pipeline
|
|
* state... */
|
|
context->current_pipeline_changes_since_flush |=
|
|
COGL_PIPELINE_STATE_CULL_FACE;
|
|
context->current_pipeline_age--;
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_flush_state (CoglFramebuffer *draw_buffer,
|
|
CoglFramebuffer *read_buffer,
|
|
CoglFramebufferState state)
|
|
{
|
|
CoglContext *ctx = draw_buffer->context;
|
|
unsigned long differences;
|
|
int bit;
|
|
|
|
/* We can assume that any state that has changed for the current
|
|
* framebuffer is different to the currently flushed value. */
|
|
differences = ctx->current_draw_buffer_changes;
|
|
|
|
/* Any state of the current framebuffer that hasn't already been
|
|
* flushed is assumed to be unknown so we will always flush that
|
|
* state if asked. */
|
|
differences |= ~ctx->current_draw_buffer_state_flushed;
|
|
|
|
/* We only need to consider the state we've been asked to flush */
|
|
differences &= state;
|
|
|
|
if (ctx->current_draw_buffer != draw_buffer)
|
|
{
|
|
/* If the previous draw buffer is NULL then we'll assume
|
|
everything has changed. This can happen if a framebuffer is
|
|
destroyed while it is the last flushed draw buffer. In that
|
|
case the framebuffer destructor will set
|
|
ctx->current_draw_buffer to NULL */
|
|
if (ctx->current_draw_buffer == NULL)
|
|
differences |= state;
|
|
else
|
|
/* NB: we only need to compare the state we're being asked to flush
|
|
* and we don't need to compare the state we've already decided
|
|
* we will definitely flush... */
|
|
differences |= _cogl_framebuffer_compare (ctx->current_draw_buffer,
|
|
draw_buffer,
|
|
state & ~differences);
|
|
|
|
/* NB: we don't take a reference here, to avoid a circular
|
|
* reference. */
|
|
ctx->current_draw_buffer = draw_buffer;
|
|
ctx->current_draw_buffer_state_flushed = 0;
|
|
}
|
|
|
|
if (ctx->current_read_buffer != read_buffer &&
|
|
state & COGL_FRAMEBUFFER_STATE_BIND)
|
|
{
|
|
differences |= COGL_FRAMEBUFFER_STATE_BIND;
|
|
/* NB: we don't take a reference here, to avoid a circular
|
|
* reference. */
|
|
ctx->current_read_buffer = read_buffer;
|
|
}
|
|
|
|
if (!differences)
|
|
return;
|
|
|
|
/* Lazily ensure the framebuffers have been allocated */
|
|
if (G_UNLIKELY (!draw_buffer->allocated))
|
|
cogl_framebuffer_allocate (draw_buffer, NULL);
|
|
if (G_UNLIKELY (!read_buffer->allocated))
|
|
cogl_framebuffer_allocate (read_buffer, NULL);
|
|
|
|
/* We handle buffer binding separately since the method depends on whether
|
|
* we are binding the same buffer for read and write or not unlike all
|
|
* other state that only relates to the draw_buffer. */
|
|
if (differences & COGL_FRAMEBUFFER_STATE_BIND)
|
|
{
|
|
if (draw_buffer == read_buffer)
|
|
bind_gl_framebuffer (ctx, GL_FRAMEBUFFER, draw_buffer);
|
|
else
|
|
{
|
|
/* NB: Currently we only take advantage of binding separate
|
|
* read/write buffers for offscreen framebuffer blit
|
|
* purposes. */
|
|
_COGL_RETURN_IF_FAIL (ctx->private_feature_flags &
|
|
COGL_PRIVATE_FEATURE_OFFSCREEN_BLIT);
|
|
_COGL_RETURN_IF_FAIL (draw_buffer->type == COGL_FRAMEBUFFER_TYPE_OFFSCREEN);
|
|
_COGL_RETURN_IF_FAIL (read_buffer->type == COGL_FRAMEBUFFER_TYPE_OFFSCREEN);
|
|
|
|
bind_gl_framebuffer (ctx, GL_DRAW_FRAMEBUFFER, draw_buffer);
|
|
bind_gl_framebuffer (ctx, GL_READ_FRAMEBUFFER, read_buffer);
|
|
}
|
|
|
|
differences &= ~COGL_FRAMEBUFFER_STATE_BIND;
|
|
}
|
|
|
|
COGL_FLAGS_FOREACH_START (&differences, 1, bit)
|
|
{
|
|
/* XXX: We considered having an array of callbacks for each state index
|
|
* that we'd call here but decided that this way the compiler is more
|
|
* likely going to be able to in-line the flush functions and use the
|
|
* index to jump straight to the required code. */
|
|
switch (bit)
|
|
{
|
|
case COGL_FRAMEBUFFER_STATE_INDEX_VIEWPORT:
|
|
_cogl_framebuffer_flush_viewport_state (draw_buffer);
|
|
break;
|
|
case COGL_FRAMEBUFFER_STATE_INDEX_CLIP:
|
|
_cogl_framebuffer_flush_clip_state (draw_buffer);
|
|
break;
|
|
case COGL_FRAMEBUFFER_STATE_INDEX_DITHER:
|
|
_cogl_framebuffer_flush_dither_state (draw_buffer);
|
|
break;
|
|
case COGL_FRAMEBUFFER_STATE_INDEX_MODELVIEW:
|
|
_cogl_framebuffer_flush_modelview_state (draw_buffer);
|
|
break;
|
|
case COGL_FRAMEBUFFER_STATE_INDEX_PROJECTION:
|
|
_cogl_framebuffer_flush_projection_state (draw_buffer);
|
|
break;
|
|
case COGL_FRAMEBUFFER_STATE_INDEX_COLOR_MASK:
|
|
_cogl_framebuffer_flush_color_mask_state (draw_buffer);
|
|
break;
|
|
case COGL_FRAMEBUFFER_STATE_INDEX_FRONT_FACE_WINDING:
|
|
_cogl_framebuffer_flush_front_face_winding_state (draw_buffer);
|
|
break;
|
|
default:
|
|
g_warn_if_reached ();
|
|
}
|
|
}
|
|
COGL_FLAGS_FOREACH_END;
|
|
|
|
ctx->current_draw_buffer_state_flushed |= state;
|
|
ctx->current_draw_buffer_changes &= ~state;
|
|
}
|
|
|
|
int
|
|
cogl_framebuffer_get_red_bits (CoglFramebuffer *framebuffer)
|
|
{
|
|
_cogl_framebuffer_init_bits (framebuffer);
|
|
|
|
return framebuffer->red_bits;
|
|
}
|
|
|
|
int
|
|
cogl_framebuffer_get_green_bits (CoglFramebuffer *framebuffer)
|
|
{
|
|
_cogl_framebuffer_init_bits (framebuffer);
|
|
|
|
return framebuffer->green_bits;
|
|
}
|
|
|
|
int
|
|
cogl_framebuffer_get_blue_bits (CoglFramebuffer *framebuffer)
|
|
{
|
|
_cogl_framebuffer_init_bits (framebuffer);
|
|
|
|
return framebuffer->blue_bits;
|
|
}
|
|
|
|
int
|
|
cogl_framebuffer_get_alpha_bits (CoglFramebuffer *framebuffer)
|
|
{
|
|
_cogl_framebuffer_init_bits (framebuffer);
|
|
|
|
return framebuffer->alpha_bits;
|
|
}
|
|
|
|
CoglColorMask
|
|
cogl_framebuffer_get_color_mask (CoglFramebuffer *framebuffer)
|
|
{
|
|
return framebuffer->color_mask;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_set_color_mask (CoglFramebuffer *framebuffer,
|
|
CoglColorMask color_mask)
|
|
{
|
|
/* XXX: Currently color mask changes don't go through the journal */
|
|
_cogl_framebuffer_flush_journal (framebuffer);
|
|
|
|
framebuffer->color_mask = color_mask;
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_COLOR_MASK;
|
|
}
|
|
|
|
gboolean
|
|
cogl_framebuffer_get_dither_enabled (CoglFramebuffer *framebuffer)
|
|
{
|
|
return framebuffer->dither_enabled;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_set_dither_enabled (CoglFramebuffer *framebuffer,
|
|
gboolean dither_enabled)
|
|
{
|
|
if (framebuffer->dither_enabled == dither_enabled)
|
|
return;
|
|
|
|
cogl_flush (); /* Currently dithering changes aren't tracked in the journal */
|
|
framebuffer->dither_enabled = dither_enabled;
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_DITHER;
|
|
}
|
|
|
|
CoglPixelFormat
|
|
cogl_framebuffer_get_color_format (CoglFramebuffer *framebuffer)
|
|
{
|
|
return framebuffer->format;
|
|
}
|
|
|
|
int
|
|
cogl_framebuffer_get_samples_per_pixel (CoglFramebuffer *framebuffer)
|
|
{
|
|
if (framebuffer->allocated)
|
|
return framebuffer->samples_per_pixel;
|
|
else
|
|
return framebuffer->config.samples_per_pixel;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_set_samples_per_pixel (CoglFramebuffer *framebuffer,
|
|
int samples_per_pixel)
|
|
{
|
|
_COGL_RETURN_IF_FAIL (!framebuffer->allocated);
|
|
|
|
framebuffer->config.samples_per_pixel = samples_per_pixel;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_resolve_samples (CoglFramebuffer *framebuffer)
|
|
{
|
|
cogl_framebuffer_resolve_samples_region (framebuffer,
|
|
0, 0,
|
|
framebuffer->width,
|
|
framebuffer->height);
|
|
|
|
/* TODO: Make this happen implicitly when the resolve texture next gets used
|
|
* as a source, either via cogl_texture_get_data(), via cogl_read_pixels() or
|
|
* if used as a source for rendering. We would also implicitly resolve if
|
|
* necessary before freeing a CoglFramebuffer.
|
|
*
|
|
* This API should still be kept but it is optional, only necessary
|
|
* if the user wants to explicitly control when the resolve happens e.g.
|
|
* to ensure it's done in advance of it being used as a source.
|
|
*
|
|
* Every texture should have a CoglFramebuffer *needs_resolve member
|
|
* internally. When the texture gets validated before being used as a source
|
|
* we should first check the needs_resolve pointer and if set we'll
|
|
* automatically call cogl_framebuffer_resolve_samples ().
|
|
*
|
|
* Calling cogl_framebuffer_resolve_samples() or
|
|
* cogl_framebuffer_resolve_samples_region() should reset the textures
|
|
* needs_resolve pointer to NULL.
|
|
*
|
|
* Rendering anything to a framebuffer will cause the corresponding
|
|
* texture's ->needs_resolve pointer to be set.
|
|
*
|
|
* XXX: Note: we only need to address this TODO item when adding support for
|
|
* EXT_framebuffer_multisample because currently we only support hardware
|
|
* that resolves implicitly anyway.
|
|
*/
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_resolve_samples_region (CoglFramebuffer *framebuffer,
|
|
int x,
|
|
int y,
|
|
int width,
|
|
int height)
|
|
{
|
|
/* NOP for now since we don't support EXT_framebuffer_multisample yet which
|
|
* requires an explicit resolve. */
|
|
}
|
|
|
|
CoglContext *
|
|
cogl_framebuffer_get_context (CoglFramebuffer *framebuffer)
|
|
{
|
|
_COGL_RETURN_VAL_IF_FAIL (framebuffer != NULL, NULL);
|
|
|
|
return framebuffer->context;
|
|
}
|
|
|
|
static gboolean
|
|
_cogl_framebuffer_try_fast_read_pixel (CoglFramebuffer *framebuffer,
|
|
int x,
|
|
int y,
|
|
CoglReadPixelsFlags source,
|
|
CoglBitmap *bitmap)
|
|
{
|
|
gboolean found_intersection;
|
|
CoglPixelFormat format;
|
|
|
|
if (G_UNLIKELY (COGL_DEBUG_ENABLED (COGL_DEBUG_DISABLE_FAST_READ_PIXEL)))
|
|
return FALSE;
|
|
|
|
if (source != COGL_READ_PIXELS_COLOR_BUFFER)
|
|
return FALSE;
|
|
|
|
format = cogl_bitmap_get_format (bitmap);
|
|
|
|
if (format != COGL_PIXEL_FORMAT_RGBA_8888_PRE &&
|
|
format != COGL_PIXEL_FORMAT_RGBA_8888)
|
|
return FALSE;
|
|
|
|
if (!_cogl_journal_try_read_pixel (framebuffer->journal,
|
|
x, y, bitmap,
|
|
&found_intersection))
|
|
return FALSE;
|
|
|
|
/* If we can't determine the color from the primitives in the
|
|
* journal then see if we can use the last recorded clear color
|
|
*/
|
|
|
|
/* If _cogl_journal_try_read_pixel() failed even though there was an
|
|
* intersection of the given point with a primitive in the journal
|
|
* then we can't fallback to the framebuffer's last clear color...
|
|
* */
|
|
if (found_intersection)
|
|
return TRUE;
|
|
|
|
/* If the framebuffer has been rendered too since it was last
|
|
* cleared then we can't return the last known clear color. */
|
|
if (framebuffer->clear_clip_dirty)
|
|
return FALSE;
|
|
|
|
if (x >= framebuffer->clear_clip_x0 &&
|
|
x < framebuffer->clear_clip_x1 &&
|
|
y >= framebuffer->clear_clip_y0 &&
|
|
y < framebuffer->clear_clip_y1)
|
|
{
|
|
guint8 *pixel;
|
|
|
|
/* we currently only care about cases where the premultiplied or
|
|
* unpremultipled colors are equivalent... */
|
|
if (framebuffer->clear_color_alpha != 1.0)
|
|
return FALSE;
|
|
|
|
pixel = _cogl_bitmap_map (bitmap,
|
|
COGL_BUFFER_ACCESS_WRITE,
|
|
COGL_BUFFER_MAP_HINT_DISCARD);
|
|
if (pixel == NULL)
|
|
return FALSE;
|
|
|
|
pixel[0] = framebuffer->clear_color_red * 255.0;
|
|
pixel[1] = framebuffer->clear_color_green * 255.0;
|
|
pixel[2] = framebuffer->clear_color_blue * 255.0;
|
|
pixel[3] = framebuffer->clear_color_alpha * 255.0;
|
|
|
|
_cogl_bitmap_unmap (bitmap);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
gboolean
|
|
cogl_framebuffer_read_pixels_into_bitmap (CoglFramebuffer *framebuffer,
|
|
int x,
|
|
int y,
|
|
CoglReadPixelsFlags source,
|
|
CoglBitmap *bitmap)
|
|
{
|
|
CoglContext *ctx;
|
|
int framebuffer_height;
|
|
CoglPixelFormat format;
|
|
CoglPixelFormat required_format;
|
|
GLenum gl_intformat;
|
|
GLenum gl_format;
|
|
GLenum gl_type;
|
|
gboolean pack_invert_set;
|
|
int width;
|
|
int height;
|
|
|
|
_COGL_RETURN_VAL_IF_FAIL (source == COGL_READ_PIXELS_COLOR_BUFFER, FALSE);
|
|
_COGL_RETURN_VAL_IF_FAIL (cogl_is_framebuffer (framebuffer), FALSE);
|
|
|
|
ctx = cogl_framebuffer_get_context (framebuffer);
|
|
|
|
width = cogl_bitmap_get_width (bitmap);
|
|
height = cogl_bitmap_get_height (bitmap);
|
|
|
|
if (width == 1 && height == 1 && !framebuffer->clear_clip_dirty)
|
|
{
|
|
/* If everything drawn so far for this frame is still in the
|
|
* Journal then if all of the rectangles only have a flat
|
|
* opaque color we have a fast-path for reading a single pixel
|
|
* that avoids the relatively high cost of flushing primitives
|
|
* to be drawn on the GPU (considering how simple the geometry
|
|
* is in this case) and then blocking on the long GPU pipelines
|
|
* for the result.
|
|
*/
|
|
if (_cogl_framebuffer_try_fast_read_pixel (framebuffer,
|
|
x, y, source, bitmap))
|
|
return TRUE;
|
|
}
|
|
|
|
/* make sure any batched primitives get emitted to the GL driver
|
|
* before issuing our read pixels...
|
|
*/
|
|
_cogl_framebuffer_flush_journal (framebuffer);
|
|
|
|
_cogl_framebuffer_flush_state (framebuffer,
|
|
framebuffer,
|
|
COGL_FRAMEBUFFER_STATE_BIND);
|
|
|
|
framebuffer_height = cogl_framebuffer_get_height (framebuffer);
|
|
|
|
/* The y co-ordinate should be given in OpenGL's coordinate system
|
|
* so 0 is the bottom row
|
|
*
|
|
* NB: all offscreen rendering is done upside down so no conversion
|
|
* is necissary in this case.
|
|
*/
|
|
if (!cogl_is_offscreen (framebuffer))
|
|
y = framebuffer_height - y - height;
|
|
|
|
format = cogl_bitmap_get_format (bitmap);
|
|
|
|
required_format = ctx->driver_vtable->pixel_format_to_gl (ctx,
|
|
format,
|
|
&gl_intformat,
|
|
&gl_format,
|
|
&gl_type);
|
|
|
|
/* NB: All offscreen rendering is done upside down so there is no need
|
|
* to flip in this case... */
|
|
if ((ctx->private_feature_flags & COGL_PRIVATE_FEATURE_MESA_PACK_INVERT) &&
|
|
!cogl_is_offscreen (framebuffer))
|
|
{
|
|
GE (ctx, glPixelStorei (GL_PACK_INVERT_MESA, TRUE));
|
|
pack_invert_set = TRUE;
|
|
}
|
|
else
|
|
pack_invert_set = FALSE;
|
|
|
|
/* Under GLES only GL_RGBA with GL_UNSIGNED_BYTE as well as an
|
|
implementation specific format under
|
|
GL_IMPLEMENTATION_COLOR_READ_FORMAT_OES and
|
|
GL_IMPLEMENTATION_COLOR_READ_TYPE_OES is supported. We could try
|
|
to be more clever and check if the requested type matches that
|
|
but we would need some reliable functions to convert from GL
|
|
types to Cogl types. For now, lets just always read in
|
|
GL_RGBA/GL_UNSIGNED_BYTE and convert if necessary. We also need
|
|
to use this intermediate buffer if the rowstride has padding
|
|
because GLES does not support setting GL_ROW_LENGTH */
|
|
if ((ctx->driver != COGL_DRIVER_GL &&
|
|
(gl_format != GL_RGBA || gl_type != GL_UNSIGNED_BYTE ||
|
|
cogl_bitmap_get_rowstride (bitmap) != 4 * width)) ||
|
|
(required_format & ~COGL_PREMULT_BIT) != (format & ~COGL_PREMULT_BIT))
|
|
{
|
|
CoglBitmap *tmp_bmp;
|
|
CoglPixelFormat read_format;
|
|
int bpp, rowstride;
|
|
guint8 *tmp_data;
|
|
int succeeded;
|
|
|
|
if (ctx->driver == COGL_DRIVER_GL)
|
|
read_format = required_format;
|
|
else
|
|
{
|
|
read_format = COGL_PIXEL_FORMAT_RGBA_8888;
|
|
gl_format = GL_RGBA;
|
|
gl_type = GL_UNSIGNED_BYTE;
|
|
}
|
|
|
|
if (COGL_PIXEL_FORMAT_CAN_HAVE_PREMULT (read_format))
|
|
read_format = ((read_format & ~COGL_PREMULT_BIT) |
|
|
(framebuffer->format & COGL_PREMULT_BIT));
|
|
|
|
tmp_bmp = _cogl_bitmap_new_with_malloc_buffer (ctx,
|
|
width, height,
|
|
read_format);
|
|
bpp = _cogl_pixel_format_get_bytes_per_pixel (read_format);
|
|
rowstride = cogl_bitmap_get_rowstride (tmp_bmp);
|
|
|
|
ctx->texture_driver->prep_gl_for_pixels_download (ctx, rowstride, bpp);
|
|
|
|
tmp_data = _cogl_bitmap_bind (tmp_bmp,
|
|
COGL_BUFFER_ACCESS_WRITE,
|
|
COGL_BUFFER_MAP_HINT_DISCARD);
|
|
|
|
GE( ctx, glReadPixels (x, y, width, height,
|
|
gl_format, gl_type,
|
|
tmp_data) );
|
|
|
|
_cogl_bitmap_unbind (tmp_bmp);
|
|
|
|
succeeded = _cogl_bitmap_convert_into_bitmap (tmp_bmp, bitmap);
|
|
|
|
cogl_object_unref (tmp_bmp);
|
|
|
|
if (!succeeded)
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
CoglBitmap *shared_bmp;
|
|
CoglPixelFormat bmp_format;
|
|
int bpp, rowstride;
|
|
gboolean succeeded = FALSE;
|
|
guint8 *pixels;
|
|
|
|
rowstride = cogl_bitmap_get_rowstride (bitmap);
|
|
|
|
/* We match the premultiplied state of the target buffer to the
|
|
* premultiplied state of the framebuffer so that it will get
|
|
* converted to the right format below */
|
|
if (COGL_PIXEL_FORMAT_CAN_HAVE_PREMULT (format))
|
|
bmp_format = ((format & ~COGL_PREMULT_BIT) |
|
|
(framebuffer->format & COGL_PREMULT_BIT));
|
|
else
|
|
bmp_format = format;
|
|
|
|
if (bmp_format != format)
|
|
shared_bmp = _cogl_bitmap_new_shared (bitmap,
|
|
bmp_format,
|
|
width, height,
|
|
rowstride);
|
|
else
|
|
shared_bmp = cogl_object_ref (bitmap);
|
|
|
|
bpp = _cogl_pixel_format_get_bytes_per_pixel (bmp_format);
|
|
|
|
ctx->texture_driver->prep_gl_for_pixels_download (ctx, rowstride, bpp);
|
|
|
|
pixels = _cogl_bitmap_bind (shared_bmp,
|
|
COGL_BUFFER_ACCESS_WRITE,
|
|
0 /* hints */);
|
|
|
|
GE( ctx, glReadPixels (x, y,
|
|
width, height,
|
|
gl_format, gl_type,
|
|
pixels) );
|
|
|
|
_cogl_bitmap_unbind (shared_bmp);
|
|
|
|
/* Convert to the premult format specified by the caller
|
|
in-place. This will do nothing if the premult status is already
|
|
correct. */
|
|
if (_cogl_bitmap_convert_premult_status (shared_bmp, format))
|
|
succeeded = TRUE;
|
|
|
|
cogl_object_unref (shared_bmp);
|
|
|
|
if (!succeeded)
|
|
return FALSE;
|
|
}
|
|
|
|
/* Currently this function owns the pack_invert state and we don't want this
|
|
* to interfere with other Cogl components so all other code can assume that
|
|
* we leave the pack_invert state off. */
|
|
if (pack_invert_set)
|
|
GE (ctx, glPixelStorei (GL_PACK_INVERT_MESA, FALSE));
|
|
|
|
/* NB: All offscreen rendering is done upside down so there is no need
|
|
* to flip in this case... */
|
|
if (!cogl_is_offscreen (framebuffer) && !pack_invert_set)
|
|
{
|
|
guint8 *temprow;
|
|
int rowstride;
|
|
guint8 *pixels;
|
|
|
|
rowstride = cogl_bitmap_get_rowstride (bitmap);
|
|
pixels = _cogl_bitmap_map (bitmap,
|
|
COGL_BUFFER_ACCESS_READ |
|
|
COGL_BUFFER_ACCESS_WRITE,
|
|
0 /* hints */);
|
|
|
|
if (pixels == NULL)
|
|
return FALSE;
|
|
|
|
temprow = g_alloca (rowstride * sizeof (guint8));
|
|
|
|
/* vertically flip the buffer in-place */
|
|
for (y = 0; y < height / 2; y++)
|
|
{
|
|
if (y != height - y - 1) /* skip center row */
|
|
{
|
|
memcpy (temprow,
|
|
pixels + y * rowstride, rowstride);
|
|
memcpy (pixels + y * rowstride,
|
|
pixels + (height - y - 1) * rowstride, rowstride);
|
|
memcpy (pixels + (height - y - 1) * rowstride,
|
|
temprow,
|
|
rowstride);
|
|
}
|
|
}
|
|
|
|
_cogl_bitmap_unmap (bitmap);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
gboolean
|
|
cogl_framebuffer_read_pixels (CoglFramebuffer *framebuffer,
|
|
int x,
|
|
int y,
|
|
int width,
|
|
int height,
|
|
CoglPixelFormat format,
|
|
guint8 *pixels)
|
|
{
|
|
int bpp = _cogl_pixel_format_get_bytes_per_pixel (format);
|
|
CoglBitmap *bitmap;
|
|
gboolean ret;
|
|
|
|
bitmap = cogl_bitmap_new_for_data (framebuffer->context,
|
|
width, height,
|
|
format,
|
|
bpp * width, /* rowstride */
|
|
pixels);
|
|
ret = cogl_framebuffer_read_pixels_into_bitmap (framebuffer,
|
|
x, y,
|
|
COGL_READ_PIXELS_COLOR_BUFFER,
|
|
bitmap);
|
|
cogl_object_unref (bitmap);
|
|
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
_cogl_blit_framebuffer (unsigned int src_x,
|
|
unsigned int src_y,
|
|
unsigned int dst_x,
|
|
unsigned int dst_y,
|
|
unsigned int width,
|
|
unsigned int height)
|
|
{
|
|
CoglFramebuffer *draw_buffer;
|
|
CoglFramebuffer *read_buffer;
|
|
CoglContext *ctx;
|
|
|
|
/* FIXME: this function should take explit src and dst framebuffer
|
|
* arguments. */
|
|
draw_buffer = cogl_get_draw_framebuffer ();
|
|
read_buffer = _cogl_get_read_framebuffer ();
|
|
ctx = draw_buffer->context;
|
|
|
|
_COGL_RETURN_IF_FAIL (ctx->private_feature_flags &
|
|
COGL_PRIVATE_FEATURE_OFFSCREEN_BLIT);
|
|
|
|
/* We can only support blitting between offscreen buffers because
|
|
otherwise we would need to mirror the image and GLES2.0 doesn't
|
|
support this */
|
|
_COGL_RETURN_IF_FAIL (cogl_is_offscreen (draw_buffer));
|
|
_COGL_RETURN_IF_FAIL (cogl_is_offscreen (read_buffer));
|
|
/* The buffers must be the same format */
|
|
_COGL_RETURN_IF_FAIL (draw_buffer->format == read_buffer->format);
|
|
|
|
/* Make sure the current framebuffers are bound. We explicitly avoid
|
|
flushing the clip state so we can bind our own empty state */
|
|
_cogl_framebuffer_flush_state (cogl_get_draw_framebuffer (),
|
|
_cogl_get_read_framebuffer (),
|
|
COGL_FRAMEBUFFER_STATE_ALL &
|
|
~COGL_FRAMEBUFFER_STATE_CLIP);
|
|
|
|
/* Flush any empty clip stack because glBlitFramebuffer is affected
|
|
by the scissor and we want to hide this feature for the Cogl API
|
|
because it's not obvious to an app how the clip state will affect
|
|
the scissor */
|
|
_cogl_clip_stack_flush (NULL, draw_buffer);
|
|
|
|
/* XXX: Because we are manually flushing clip state here we need to
|
|
* make sure that the clip state gets updated the next time we flush
|
|
* framebuffer state by marking the current framebuffer's clip state
|
|
* as changed */
|
|
ctx->current_draw_buffer_changes |= COGL_FRAMEBUFFER_STATE_CLIP;
|
|
|
|
ctx->glBlitFramebuffer (src_x, src_y,
|
|
src_x + width, src_y + height,
|
|
dst_x, dst_y,
|
|
dst_x + width, dst_y + height,
|
|
GL_COLOR_BUFFER_BIT,
|
|
GL_NEAREST);
|
|
}
|
|
|
|
static void
|
|
_cogl_framebuffer_discard_buffers_real (CoglFramebuffer *framebuffer,
|
|
unsigned long buffers)
|
|
{
|
|
#ifdef GL_EXT_discard_framebuffer
|
|
CoglContext *ctx = framebuffer->context;
|
|
|
|
if (ctx->glDiscardFramebuffer)
|
|
{
|
|
GLenum attachments[3];
|
|
int i = 0;
|
|
|
|
if (framebuffer->type == COGL_FRAMEBUFFER_TYPE_ONSCREEN)
|
|
{
|
|
if (buffers & COGL_BUFFER_BIT_COLOR)
|
|
attachments[i++] = GL_COLOR_EXT;
|
|
if (buffers & COGL_BUFFER_BIT_DEPTH)
|
|
attachments[i++] = GL_DEPTH_EXT;
|
|
if (buffers & COGL_BUFFER_BIT_STENCIL)
|
|
attachments[i++] = GL_STENCIL_EXT;
|
|
}
|
|
else
|
|
{
|
|
if (buffers & COGL_BUFFER_BIT_COLOR)
|
|
attachments[i++] = GL_COLOR_ATTACHMENT0;
|
|
if (buffers & COGL_BUFFER_BIT_DEPTH)
|
|
attachments[i++] = GL_DEPTH_ATTACHMENT;
|
|
if (buffers & COGL_BUFFER_BIT_STENCIL)
|
|
attachments[i++] = GL_STENCIL_ATTACHMENT;
|
|
}
|
|
|
|
GE (ctx, glDiscardFramebuffer (GL_FRAMEBUFFER, i, attachments));
|
|
}
|
|
#endif /* GL_EXT_discard_framebuffer */
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_discard_buffers (CoglFramebuffer *framebuffer,
|
|
unsigned long buffers)
|
|
{
|
|
_COGL_RETURN_IF_FAIL (buffers & COGL_BUFFER_BIT_COLOR);
|
|
|
|
_cogl_framebuffer_discard_buffers_real (framebuffer, buffers);
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_finish (CoglFramebuffer *framebuffer)
|
|
{
|
|
_cogl_framebuffer_flush_journal (framebuffer);
|
|
GE (framebuffer->context, glFinish ());
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_push_matrix (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglMatrixStack *modelview_stack =
|
|
_cogl_framebuffer_get_modelview_stack (framebuffer);
|
|
_cogl_matrix_stack_push (modelview_stack);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_MODELVIEW;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_pop_matrix (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglMatrixStack *modelview_stack =
|
|
_cogl_framebuffer_get_modelview_stack (framebuffer);
|
|
_cogl_matrix_stack_pop (modelview_stack);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_MODELVIEW;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_identity_matrix (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglMatrixStack *modelview_stack =
|
|
_cogl_framebuffer_get_modelview_stack (framebuffer);
|
|
_cogl_matrix_stack_load_identity (modelview_stack);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_MODELVIEW;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_scale (CoglFramebuffer *framebuffer,
|
|
float x,
|
|
float y,
|
|
float z)
|
|
{
|
|
CoglMatrixStack *modelview_stack =
|
|
_cogl_framebuffer_get_modelview_stack (framebuffer);
|
|
_cogl_matrix_stack_scale (modelview_stack, x, y, z);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_MODELVIEW;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_translate (CoglFramebuffer *framebuffer,
|
|
float x,
|
|
float y,
|
|
float z)
|
|
{
|
|
CoglMatrixStack *modelview_stack =
|
|
_cogl_framebuffer_get_modelview_stack (framebuffer);
|
|
_cogl_matrix_stack_translate (modelview_stack, x, y, z);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_MODELVIEW;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_rotate (CoglFramebuffer *framebuffer,
|
|
float angle,
|
|
float x,
|
|
float y,
|
|
float z)
|
|
{
|
|
CoglMatrixStack *modelview_stack =
|
|
_cogl_framebuffer_get_modelview_stack (framebuffer);
|
|
_cogl_matrix_stack_rotate (modelview_stack, angle, x, y, z);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_MODELVIEW;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_transform (CoglFramebuffer *framebuffer,
|
|
const CoglMatrix *matrix)
|
|
{
|
|
CoglMatrixStack *modelview_stack =
|
|
_cogl_framebuffer_get_modelview_stack (framebuffer);
|
|
_cogl_matrix_stack_multiply (modelview_stack, matrix);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_MODELVIEW;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_perspective (CoglFramebuffer *framebuffer,
|
|
float fov_y,
|
|
float aspect,
|
|
float z_near,
|
|
float z_far)
|
|
{
|
|
float ymax = z_near * tanf (fov_y * G_PI / 360.0);
|
|
|
|
cogl_framebuffer_frustum (framebuffer,
|
|
-ymax * aspect, /* left */
|
|
ymax * aspect, /* right */
|
|
-ymax, /* bottom */
|
|
ymax, /* top */
|
|
z_near,
|
|
z_far);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_PROJECTION;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_frustum (CoglFramebuffer *framebuffer,
|
|
float left,
|
|
float right,
|
|
float bottom,
|
|
float top,
|
|
float z_near,
|
|
float z_far)
|
|
{
|
|
CoglMatrixStack *projection_stack =
|
|
_cogl_framebuffer_get_projection_stack (framebuffer);
|
|
|
|
/* XXX: The projection matrix isn't currently tracked in the journal
|
|
* so we need to flush all journaled primitives first... */
|
|
_cogl_framebuffer_flush_journal (framebuffer);
|
|
|
|
_cogl_matrix_stack_load_identity (projection_stack);
|
|
|
|
_cogl_matrix_stack_frustum (projection_stack,
|
|
left,
|
|
right,
|
|
bottom,
|
|
top,
|
|
z_near,
|
|
z_far);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_PROJECTION;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_orthographic (CoglFramebuffer *framebuffer,
|
|
float x_1,
|
|
float y_1,
|
|
float x_2,
|
|
float y_2,
|
|
float near,
|
|
float far)
|
|
{
|
|
CoglMatrix ortho;
|
|
CoglMatrixStack *projection_stack =
|
|
_cogl_framebuffer_get_projection_stack (framebuffer);
|
|
|
|
/* XXX: The projection matrix isn't currently tracked in the journal
|
|
* so we need to flush all journaled primitives first... */
|
|
_cogl_framebuffer_flush_journal (framebuffer);
|
|
|
|
cogl_matrix_init_identity (&ortho);
|
|
cogl_matrix_orthographic (&ortho, x_1, y_1, x_2, y_2, near, far);
|
|
_cogl_matrix_stack_set (projection_stack, &ortho);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_PROJECTION;
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_push_projection (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglMatrixStack *projection_stack =
|
|
_cogl_framebuffer_get_projection_stack (framebuffer);
|
|
_cogl_matrix_stack_push (projection_stack);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_PROJECTION;
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_pop_projection (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglMatrixStack *projection_stack =
|
|
_cogl_framebuffer_get_projection_stack (framebuffer);
|
|
_cogl_matrix_stack_pop (projection_stack);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_PROJECTION;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_get_modelview_matrix (CoglFramebuffer *framebuffer,
|
|
CoglMatrix *matrix)
|
|
{
|
|
CoglMatrixStack *modelview_stack =
|
|
_cogl_framebuffer_get_modelview_stack (framebuffer);
|
|
_cogl_matrix_stack_get (modelview_stack, matrix);
|
|
_COGL_MATRIX_DEBUG_PRINT (matrix);
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_set_modelview_matrix (CoglFramebuffer *framebuffer,
|
|
CoglMatrix *matrix)
|
|
{
|
|
CoglMatrixStack *modelview_stack =
|
|
_cogl_framebuffer_get_modelview_stack (framebuffer);
|
|
_cogl_matrix_stack_set (modelview_stack, matrix);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_MODELVIEW;
|
|
|
|
_COGL_MATRIX_DEBUG_PRINT (matrix);
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_get_projection_matrix (CoglFramebuffer *framebuffer,
|
|
CoglMatrix *matrix)
|
|
{
|
|
CoglMatrixStack *projection_stack =
|
|
_cogl_framebuffer_get_projection_stack (framebuffer);
|
|
_cogl_matrix_stack_get (projection_stack, matrix);
|
|
_COGL_MATRIX_DEBUG_PRINT (matrix);
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_set_projection_matrix (CoglFramebuffer *framebuffer,
|
|
CoglMatrix *matrix)
|
|
{
|
|
CoglMatrixStack *projection_stack =
|
|
_cogl_framebuffer_get_projection_stack (framebuffer);
|
|
|
|
/* XXX: The projection matrix isn't currently tracked in the journal
|
|
* so we need to flush all journaled primitives first... */
|
|
_cogl_framebuffer_flush_journal (framebuffer);
|
|
|
|
_cogl_matrix_stack_set (projection_stack, matrix);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_PROJECTION;
|
|
|
|
_COGL_MATRIX_DEBUG_PRINT (matrix);
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_push_scissor_clip (CoglFramebuffer *framebuffer,
|
|
int x,
|
|
int y,
|
|
int width,
|
|
int height)
|
|
{
|
|
CoglClipState *clip_state = _cogl_framebuffer_get_clip_state (framebuffer);
|
|
|
|
clip_state->stacks->data =
|
|
_cogl_clip_stack_push_window_rectangle (clip_state->stacks->data,
|
|
x, y, width, height);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_CLIP;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_push_rectangle_clip (CoglFramebuffer *framebuffer,
|
|
float x_1,
|
|
float y_1,
|
|
float x_2,
|
|
float y_2)
|
|
{
|
|
CoglClipState *clip_state = _cogl_framebuffer_get_clip_state (framebuffer);
|
|
CoglMatrix modelview_matrix;
|
|
|
|
cogl_framebuffer_get_modelview_matrix (framebuffer, &modelview_matrix);
|
|
|
|
clip_state->stacks->data =
|
|
_cogl_clip_stack_push_rectangle (clip_state->stacks->data,
|
|
x_1, y_1, x_2, y_2,
|
|
&modelview_matrix);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_CLIP;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_push_path_clip (CoglFramebuffer *framebuffer,
|
|
CoglPath *path)
|
|
{
|
|
CoglClipState *clip_state = _cogl_framebuffer_get_clip_state (framebuffer);
|
|
CoglMatrix modelview_matrix;
|
|
|
|
cogl_framebuffer_get_modelview_matrix (framebuffer, &modelview_matrix);
|
|
|
|
clip_state->stacks->data =
|
|
_cogl_clip_stack_push_from_path (clip_state->stacks->data,
|
|
path,
|
|
&modelview_matrix);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_CLIP;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_push_primitive_clip (CoglFramebuffer *framebuffer,
|
|
CoglPrimitive *primitive,
|
|
float bounds_x1,
|
|
float bounds_y1,
|
|
float bounds_x2,
|
|
float bounds_y2)
|
|
{
|
|
CoglClipState *clip_state = _cogl_framebuffer_get_clip_state (framebuffer);
|
|
CoglMatrix modelview_matrix;
|
|
|
|
cogl_get_modelview_matrix (&modelview_matrix);
|
|
|
|
clip_state->stacks->data =
|
|
_cogl_clip_stack_push_primitive (clip_state->stacks->data,
|
|
primitive,
|
|
bounds_x1, bounds_y1,
|
|
bounds_x2, bounds_y2,
|
|
&modelview_matrix);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_CLIP;
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_pop_clip (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglClipState *clip_state = _cogl_framebuffer_get_clip_state (framebuffer);
|
|
|
|
clip_state->stacks->data = _cogl_clip_stack_pop (clip_state->stacks->data);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_CLIP;
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_save_clip_stack (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglClipState *clip_state = _cogl_framebuffer_get_clip_state (framebuffer);
|
|
_cogl_clip_state_save_clip_stack (clip_state);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_CLIP;
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_restore_clip_stack (CoglFramebuffer *framebuffer)
|
|
{
|
|
CoglClipState *clip_state = _cogl_framebuffer_get_clip_state (framebuffer);
|
|
_cogl_clip_state_restore_clip_stack (clip_state);
|
|
|
|
if (framebuffer->context->current_draw_buffer == framebuffer)
|
|
framebuffer->context->current_draw_buffer_changes |=
|
|
COGL_FRAMEBUFFER_STATE_CLIP;
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_unref (CoglFramebuffer *framebuffer)
|
|
{
|
|
/* The journal holds a reference to the framebuffer whenever it is
|
|
non-empty. Therefore if the journal is non-empty and we will have
|
|
exactly one reference then we know the journal is the only thing
|
|
keeping the framebuffer alive. In that case we want to flush the
|
|
journal and let the framebuffer die. It is fine at this point if
|
|
flushing the journal causes something else to take a reference to
|
|
it and it comes back to life */
|
|
if (framebuffer->journal->entries->len > 0)
|
|
{
|
|
unsigned int ref_count = ((CoglObject *) framebuffer)->ref_count;
|
|
|
|
/* There should be at least two references - the one we are
|
|
about to drop and the one held by the journal */
|
|
if (ref_count < 2)
|
|
g_warning ("Inconsistent ref count on a framebuffer with journal "
|
|
"entries.");
|
|
|
|
if (ref_count == 2)
|
|
_cogl_framebuffer_flush_journal (framebuffer);
|
|
}
|
|
|
|
/* Chain-up */
|
|
_cogl_object_default_unref (framebuffer);
|
|
}
|
|
|
|
#ifdef COGL_ENABLE_DEBUG
|
|
static int
|
|
get_index (void *indices,
|
|
CoglIndicesType type,
|
|
int _index)
|
|
{
|
|
if (!indices)
|
|
return _index;
|
|
|
|
switch (type)
|
|
{
|
|
case COGL_INDICES_TYPE_UNSIGNED_BYTE:
|
|
return ((guint8 *)indices)[_index];
|
|
case COGL_INDICES_TYPE_UNSIGNED_SHORT:
|
|
return ((guint16 *)indices)[_index];
|
|
case COGL_INDICES_TYPE_UNSIGNED_INT:
|
|
return ((guint32 *)indices)[_index];
|
|
}
|
|
|
|
g_return_val_if_reached (0);
|
|
}
|
|
|
|
static void
|
|
add_line (guint32 *line_indices,
|
|
int base,
|
|
void *user_indices,
|
|
CoglIndicesType user_indices_type,
|
|
int index0,
|
|
int index1,
|
|
int *pos)
|
|
{
|
|
index0 = get_index (user_indices, user_indices_type, index0);
|
|
index1 = get_index (user_indices, user_indices_type, index1);
|
|
|
|
line_indices[(*pos)++] = base + index0;
|
|
line_indices[(*pos)++] = base + index1;
|
|
}
|
|
|
|
static int
|
|
get_line_count (CoglVerticesMode mode, int n_vertices)
|
|
{
|
|
if (mode == COGL_VERTICES_MODE_TRIANGLES &&
|
|
(n_vertices % 3) == 0)
|
|
{
|
|
return n_vertices;
|
|
}
|
|
else if (mode == COGL_VERTICES_MODE_TRIANGLE_FAN &&
|
|
n_vertices >= 3)
|
|
{
|
|
return 2 * n_vertices - 3;
|
|
}
|
|
else if (mode == COGL_VERTICES_MODE_TRIANGLE_STRIP &&
|
|
n_vertices >= 3)
|
|
{
|
|
return 2 * n_vertices - 3;
|
|
}
|
|
/* In the journal we are a bit sneaky and actually use GL_QUADS
|
|
* which isn't actually a valid CoglVerticesMode! */
|
|
#ifdef HAVE_COGL_GL
|
|
else if (mode == GL_QUADS && (n_vertices % 4) == 0)
|
|
{
|
|
return n_vertices;
|
|
}
|
|
#endif
|
|
|
|
g_return_val_if_reached (0);
|
|
}
|
|
|
|
static CoglIndices *
|
|
get_wire_line_indices (CoglContext *ctx,
|
|
CoglVerticesMode mode,
|
|
int first_vertex,
|
|
int n_vertices_in,
|
|
CoglIndices *user_indices,
|
|
int *n_indices)
|
|
{
|
|
int n_lines;
|
|
guint32 *line_indices;
|
|
CoglIndexBuffer *index_buffer;
|
|
void *indices;
|
|
CoglIndicesType indices_type;
|
|
int base = first_vertex;
|
|
int pos;
|
|
int i;
|
|
CoglIndices *ret;
|
|
|
|
if (user_indices)
|
|
{
|
|
index_buffer = cogl_indices_get_buffer (user_indices);
|
|
indices = cogl_buffer_map (COGL_BUFFER (index_buffer),
|
|
COGL_BUFFER_ACCESS_READ, 0);
|
|
indices_type = cogl_indices_get_type (user_indices);
|
|
}
|
|
else
|
|
{
|
|
index_buffer = NULL;
|
|
indices = NULL;
|
|
indices_type = COGL_INDICES_TYPE_UNSIGNED_BYTE;
|
|
}
|
|
|
|
n_lines = get_line_count (mode, n_vertices_in);
|
|
|
|
/* Note: we are using COGL_INDICES_TYPE_UNSIGNED_INT so 4 bytes per index. */
|
|
line_indices = g_malloc (4 * n_lines * 2);
|
|
|
|
pos = 0;
|
|
|
|
if (mode == COGL_VERTICES_MODE_TRIANGLES &&
|
|
(n_vertices_in % 3) == 0)
|
|
{
|
|
for (i = 0; i < n_vertices_in; i += 3)
|
|
{
|
|
add_line (line_indices, base, indices, indices_type, i, i+1, &pos);
|
|
add_line (line_indices, base, indices, indices_type, i+1, i+2, &pos);
|
|
add_line (line_indices, base, indices, indices_type, i+2, i, &pos);
|
|
}
|
|
}
|
|
else if (mode == COGL_VERTICES_MODE_TRIANGLE_FAN &&
|
|
n_vertices_in >= 3)
|
|
{
|
|
add_line (line_indices, base, indices, indices_type, 0, 1, &pos);
|
|
add_line (line_indices, base, indices, indices_type, 1, 2, &pos);
|
|
add_line (line_indices, base, indices, indices_type, 0, 2, &pos);
|
|
|
|
for (i = 3; i < n_vertices_in; i++)
|
|
{
|
|
add_line (line_indices, base, indices, indices_type, i - 1, i, &pos);
|
|
add_line (line_indices, base, indices, indices_type, 0, i, &pos);
|
|
}
|
|
}
|
|
else if (mode == COGL_VERTICES_MODE_TRIANGLE_STRIP &&
|
|
n_vertices_in >= 3)
|
|
{
|
|
add_line (line_indices, base, indices, indices_type, 0, 1, &pos);
|
|
add_line (line_indices, base, indices, indices_type, 1, 2, &pos);
|
|
add_line (line_indices, base, indices, indices_type, 0, 2, &pos);
|
|
|
|
for (i = 3; i < n_vertices_in; i++)
|
|
{
|
|
add_line (line_indices, base, indices, indices_type, i - 1, i, &pos);
|
|
add_line (line_indices, base, indices, indices_type, i - 2, i, &pos);
|
|
}
|
|
}
|
|
/* In the journal we are a bit sneaky and actually use GL_QUADS
|
|
* which isn't actually a valid CoglVerticesMode! */
|
|
#ifdef HAVE_COGL_GL
|
|
else if (mode == GL_QUADS && (n_vertices_in % 4) == 0)
|
|
{
|
|
for (i = 0; i < n_vertices_in; i += 4)
|
|
{
|
|
add_line (line_indices,
|
|
base, indices, indices_type, i, i + 1, &pos);
|
|
add_line (line_indices,
|
|
base, indices, indices_type, i + 1, i + 2, &pos);
|
|
add_line (line_indices,
|
|
base, indices, indices_type, i + 2, i + 3, &pos);
|
|
add_line (line_indices,
|
|
base, indices, indices_type, i + 3, i, &pos);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
if (user_indices)
|
|
cogl_buffer_unmap (COGL_BUFFER (index_buffer));
|
|
|
|
*n_indices = n_lines * 2;
|
|
|
|
ret = cogl_indices_new (ctx,
|
|
COGL_INDICES_TYPE_UNSIGNED_INT,
|
|
line_indices,
|
|
*n_indices);
|
|
|
|
g_free (line_indices);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static gboolean
|
|
remove_layer_cb (CoglPipeline *pipeline,
|
|
int layer_index,
|
|
void *user_data)
|
|
{
|
|
cogl_pipeline_remove_layer (pipeline, layer_index);
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
pipeline_destroyed_cb (CoglPipeline *weak_pipeline, void *user_data)
|
|
{
|
|
CoglPipeline *original_pipeline = user_data;
|
|
|
|
/* XXX: I think we probably need to provide a custom unref function for
|
|
* CoglPipeline because it's possible that we will reach this callback
|
|
* because original_pipeline is being freed which means cogl_object_unref
|
|
* will have already freed any associated user data.
|
|
*
|
|
* Setting more user data here will *probably* succeed but that may allocate
|
|
* a new user-data array which could be leaked.
|
|
*
|
|
* Potentially we could have a _cogl_object_free_user_data function so
|
|
* that a custom unref function could be written that can destroy weak
|
|
* pipeline children before removing user data.
|
|
*/
|
|
cogl_object_set_user_data (COGL_OBJECT (original_pipeline),
|
|
&wire_pipeline_key, NULL, NULL);
|
|
|
|
cogl_object_unref (weak_pipeline);
|
|
}
|
|
|
|
static void
|
|
draw_wireframe (CoglContext *ctx,
|
|
CoglFramebuffer *framebuffer,
|
|
CoglPipeline *pipeline,
|
|
CoglVerticesMode mode,
|
|
int first_vertex,
|
|
int n_vertices,
|
|
CoglAttribute **attributes,
|
|
int n_attributes,
|
|
CoglIndices *indices)
|
|
{
|
|
CoglIndices *wire_indices;
|
|
CoglPipeline *wire_pipeline;
|
|
int n_indices;
|
|
|
|
wire_indices = get_wire_line_indices (ctx,
|
|
mode,
|
|
first_vertex,
|
|
n_vertices,
|
|
indices,
|
|
&n_indices);
|
|
|
|
wire_pipeline = cogl_object_get_user_data (COGL_OBJECT (pipeline),
|
|
&wire_pipeline_key);
|
|
|
|
if (!wire_pipeline)
|
|
{
|
|
wire_pipeline =
|
|
_cogl_pipeline_weak_copy (pipeline, pipeline_destroyed_cb, NULL);
|
|
|
|
cogl_object_set_user_data (COGL_OBJECT (pipeline),
|
|
&wire_pipeline_key, wire_pipeline,
|
|
NULL);
|
|
|
|
/* If we have glsl then the pipeline may have an associated
|
|
* vertex program and since we'd like to see the results of the
|
|
* vertex program in the wireframe we just add a final clobber
|
|
* of the wire color leaving the rest of the state untouched. */
|
|
if (cogl_has_feature (framebuffer->context, COGL_FEATURE_ID_GLSL))
|
|
{
|
|
CoglSnippet *snippet = cogl_snippet_new (COGL_SNIPPET_HOOK_FRAGMENT,
|
|
NULL,
|
|
NULL);
|
|
cogl_snippet_set_replace (snippet,
|
|
"cogl_color_out = "
|
|
"vec4 (0.0, 1.0, 0.0, 1.0);\n");
|
|
|
|
cogl_pipeline_add_snippet (wire_pipeline, snippet);
|
|
cogl_object_unref (snippet);
|
|
}
|
|
else
|
|
{
|
|
cogl_pipeline_foreach_layer (wire_pipeline, remove_layer_cb, NULL);
|
|
cogl_pipeline_set_color4f (wire_pipeline, 0, 1, 0, 1);
|
|
}
|
|
}
|
|
|
|
/* temporarily disable the wireframe to avoid recursion! */
|
|
COGL_DEBUG_CLEAR_FLAG (COGL_DEBUG_WIREFRAME);
|
|
_cogl_framebuffer_draw_indexed_attributes (
|
|
framebuffer,
|
|
wire_pipeline,
|
|
COGL_VERTICES_MODE_LINES,
|
|
0,
|
|
n_indices,
|
|
wire_indices,
|
|
attributes,
|
|
n_attributes,
|
|
COGL_DRAW_SKIP_JOURNAL_FLUSH |
|
|
COGL_DRAW_SKIP_PIPELINE_VALIDATION |
|
|
COGL_DRAW_SKIP_FRAMEBUFFER_FLUSH |
|
|
COGL_DRAW_SKIP_LEGACY_STATE);
|
|
COGL_DEBUG_SET_FLAG (COGL_DEBUG_WIREFRAME);
|
|
|
|
cogl_object_unref (wire_indices);
|
|
}
|
|
#endif
|
|
|
|
/* This can be called directly by the CoglJournal to draw attributes
|
|
* skipping the implicit journal flush, the framebuffer flush and
|
|
* pipeline validation. */
|
|
void
|
|
_cogl_framebuffer_draw_attributes (CoglFramebuffer *framebuffer,
|
|
CoglPipeline *pipeline,
|
|
CoglVerticesMode mode,
|
|
int first_vertex,
|
|
int n_vertices,
|
|
CoglAttribute **attributes,
|
|
int n_attributes,
|
|
CoglDrawFlags flags)
|
|
{
|
|
#ifdef COGL_ENABLE_DEBUG
|
|
if (G_UNLIKELY (COGL_DEBUG_ENABLED (COGL_DEBUG_WIREFRAME)))
|
|
draw_wireframe (framebuffer->context,
|
|
framebuffer, pipeline,
|
|
mode, first_vertex, n_vertices,
|
|
attributes, n_attributes, NULL);
|
|
else
|
|
#endif
|
|
{
|
|
_cogl_flush_attributes_state (framebuffer, pipeline, flags,
|
|
attributes, n_attributes);
|
|
|
|
GE (framebuffer->context,
|
|
glDrawArrays ((GLenum)mode, first_vertex, n_vertices));
|
|
}
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_draw_attributes (CoglFramebuffer *framebuffer,
|
|
CoglPipeline *pipeline,
|
|
CoglVerticesMode mode,
|
|
int first_vertex,
|
|
int n_vertices,
|
|
CoglAttribute **attributes,
|
|
int n_attributes)
|
|
{
|
|
_cogl_framebuffer_draw_attributes (framebuffer,
|
|
pipeline,
|
|
mode,
|
|
first_vertex,
|
|
n_vertices,
|
|
attributes, n_attributes,
|
|
COGL_DRAW_SKIP_LEGACY_STATE);
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_vdraw_attributes (CoglFramebuffer *framebuffer,
|
|
CoglPipeline *pipeline,
|
|
CoglVerticesMode mode,
|
|
int first_vertex,
|
|
int n_vertices,
|
|
...)
|
|
{
|
|
va_list ap;
|
|
int n_attributes;
|
|
CoglAttribute *attribute;
|
|
CoglAttribute **attributes;
|
|
int i;
|
|
|
|
va_start (ap, n_vertices);
|
|
for (n_attributes = 0; va_arg (ap, CoglAttribute *); n_attributes++)
|
|
;
|
|
va_end (ap);
|
|
|
|
attributes = g_alloca (sizeof (CoglAttribute *) * n_attributes);
|
|
|
|
va_start (ap, n_vertices);
|
|
for (i = 0; (attribute = va_arg (ap, CoglAttribute *)); i++)
|
|
attributes[i] = attribute;
|
|
va_end (ap);
|
|
|
|
_cogl_framebuffer_draw_attributes (framebuffer,
|
|
pipeline,
|
|
mode, first_vertex, n_vertices,
|
|
attributes, n_attributes,
|
|
COGL_DRAW_SKIP_LEGACY_STATE);
|
|
}
|
|
|
|
static size_t
|
|
sizeof_index_type (CoglIndicesType type)
|
|
{
|
|
switch (type)
|
|
{
|
|
case COGL_INDICES_TYPE_UNSIGNED_BYTE:
|
|
return 1;
|
|
case COGL_INDICES_TYPE_UNSIGNED_SHORT:
|
|
return 2;
|
|
case COGL_INDICES_TYPE_UNSIGNED_INT:
|
|
return 4;
|
|
}
|
|
g_return_val_if_reached (0);
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_draw_indexed_attributes (CoglFramebuffer *framebuffer,
|
|
CoglPipeline *pipeline,
|
|
CoglVerticesMode mode,
|
|
int first_vertex,
|
|
int n_vertices,
|
|
CoglIndices *indices,
|
|
CoglAttribute **attributes,
|
|
int n_attributes,
|
|
CoglDrawFlags flags)
|
|
{
|
|
#ifdef COGL_ENABLE_DEBUG
|
|
if (G_UNLIKELY (COGL_DEBUG_ENABLED (COGL_DEBUG_WIREFRAME)))
|
|
draw_wireframe (framebuffer->context,
|
|
framebuffer, pipeline,
|
|
mode, first_vertex, n_vertices,
|
|
attributes, n_attributes, indices);
|
|
else
|
|
#endif
|
|
{
|
|
CoglBuffer *buffer;
|
|
guint8 *base;
|
|
size_t buffer_offset;
|
|
size_t index_size;
|
|
GLenum indices_gl_type = 0;
|
|
|
|
_cogl_flush_attributes_state (framebuffer, pipeline, flags,
|
|
attributes, n_attributes);
|
|
|
|
buffer = COGL_BUFFER (cogl_indices_get_buffer (indices));
|
|
base = _cogl_buffer_bind (buffer, COGL_BUFFER_BIND_TARGET_INDEX_BUFFER);
|
|
buffer_offset = cogl_indices_get_offset (indices);
|
|
index_size = sizeof_index_type (cogl_indices_get_type (indices));
|
|
|
|
switch (cogl_indices_get_type (indices))
|
|
{
|
|
case COGL_INDICES_TYPE_UNSIGNED_BYTE:
|
|
indices_gl_type = GL_UNSIGNED_BYTE;
|
|
break;
|
|
case COGL_INDICES_TYPE_UNSIGNED_SHORT:
|
|
indices_gl_type = GL_UNSIGNED_SHORT;
|
|
break;
|
|
case COGL_INDICES_TYPE_UNSIGNED_INT:
|
|
indices_gl_type = GL_UNSIGNED_INT;
|
|
break;
|
|
}
|
|
|
|
GE (framebuffer->context,
|
|
glDrawElements ((GLenum)mode,
|
|
n_vertices,
|
|
indices_gl_type,
|
|
base + buffer_offset + index_size * first_vertex));
|
|
|
|
_cogl_buffer_unbind (buffer);
|
|
}
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_draw_indexed_attributes (CoglFramebuffer *framebuffer,
|
|
CoglPipeline *pipeline,
|
|
CoglVerticesMode mode,
|
|
int first_vertex,
|
|
int n_vertices,
|
|
CoglIndices *indices,
|
|
CoglAttribute **attributes,
|
|
int n_attributes)
|
|
{
|
|
_cogl_framebuffer_draw_indexed_attributes (framebuffer,
|
|
pipeline,
|
|
mode, first_vertex,
|
|
n_vertices, indices,
|
|
attributes, n_attributes,
|
|
COGL_DRAW_SKIP_LEGACY_STATE);
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_vdraw_indexed_attributes (CoglFramebuffer *framebuffer,
|
|
CoglPipeline *pipeline,
|
|
CoglVerticesMode mode,
|
|
int first_vertex,
|
|
int n_vertices,
|
|
CoglIndices *indices,
|
|
...)
|
|
{
|
|
va_list ap;
|
|
int n_attributes;
|
|
CoglAttribute **attributes;
|
|
int i;
|
|
CoglAttribute *attribute;
|
|
|
|
va_start (ap, indices);
|
|
for (n_attributes = 0; va_arg (ap, CoglAttribute *); n_attributes++)
|
|
;
|
|
va_end (ap);
|
|
|
|
attributes = g_alloca (sizeof (CoglAttribute *) * n_attributes);
|
|
|
|
va_start (ap, indices);
|
|
for (i = 0; (attribute = va_arg (ap, CoglAttribute *)); i++)
|
|
attributes[i] = attribute;
|
|
va_end (ap);
|
|
|
|
_cogl_framebuffer_draw_indexed_attributes (framebuffer,
|
|
pipeline,
|
|
mode,
|
|
first_vertex,
|
|
n_vertices,
|
|
indices,
|
|
attributes,
|
|
n_attributes,
|
|
COGL_DRAW_SKIP_LEGACY_STATE);
|
|
}
|
|
|
|
void
|
|
_cogl_framebuffer_draw_primitive (CoglFramebuffer *framebuffer,
|
|
CoglPipeline *pipeline,
|
|
CoglPrimitive *primitive,
|
|
CoglDrawFlags flags)
|
|
{
|
|
if (primitive->indices)
|
|
_cogl_framebuffer_draw_indexed_attributes (framebuffer,
|
|
pipeline,
|
|
primitive->mode,
|
|
primitive->first_vertex,
|
|
primitive->n_vertices,
|
|
primitive->indices,
|
|
primitive->attributes,
|
|
primitive->n_attributes,
|
|
flags);
|
|
else
|
|
_cogl_framebuffer_draw_attributes (framebuffer,
|
|
pipeline,
|
|
primitive->mode,
|
|
primitive->first_vertex,
|
|
primitive->n_vertices,
|
|
primitive->attributes,
|
|
primitive->n_attributes,
|
|
flags);
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_draw_primitive (CoglFramebuffer *framebuffer,
|
|
CoglPipeline *pipeline,
|
|
CoglPrimitive *primitive)
|
|
{
|
|
_cogl_framebuffer_draw_primitive (framebuffer, pipeline, primitive,
|
|
COGL_DRAW_SKIP_LEGACY_STATE);
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_draw_rectangle (CoglFramebuffer *framebuffer,
|
|
CoglPipeline *pipeline,
|
|
float x_1,
|
|
float y_1,
|
|
float x_2,
|
|
float y_2)
|
|
{
|
|
const float position[4] = {x_1, y_1, x_2, y_2};
|
|
CoglMultiTexturedRect rect;
|
|
|
|
/* XXX: All the _*_rectangle* APIs normalize their input into an array of
|
|
* _CoglMultiTexturedRect rectangles and pass these on to our work horse;
|
|
* _cogl_framebuffer_draw_multitextured_rectangles.
|
|
*/
|
|
|
|
rect.position = position;
|
|
rect.tex_coords = NULL;
|
|
rect.tex_coords_len = 0;
|
|
|
|
_cogl_framebuffer_draw_multitextured_rectangles (framebuffer,
|
|
pipeline,
|
|
&rect,
|
|
1,
|
|
TRUE);
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_draw_textured_rectangle (CoglFramebuffer *framebuffer,
|
|
CoglPipeline *pipeline,
|
|
float x_1,
|
|
float y_1,
|
|
float x_2,
|
|
float y_2,
|
|
float s_1,
|
|
float t_1,
|
|
float s_2,
|
|
float t_2)
|
|
{
|
|
const float position[4] = {x_1, y_1, x_2, y_2};
|
|
const float tex_coords[4] = {s_1, t_1, s_2, t_2};
|
|
CoglMultiTexturedRect rect;
|
|
|
|
/* XXX: All the _*_rectangle* APIs normalize their input into an array of
|
|
* CoglMultiTexturedRect rectangles and pass these on to our work horse;
|
|
* _cogl_framebuffer_draw_multitextured_rectangles.
|
|
*/
|
|
|
|
rect.position = position;
|
|
rect.tex_coords = tex_coords;
|
|
rect.tex_coords_len = 4;
|
|
|
|
_cogl_framebuffer_draw_multitextured_rectangles (framebuffer,
|
|
pipeline,
|
|
&rect,
|
|
1,
|
|
TRUE);
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_draw_multitextured_rectangle (CoglFramebuffer *framebuffer,
|
|
CoglPipeline *pipeline,
|
|
float x_1,
|
|
float y_1,
|
|
float x_2,
|
|
float y_2,
|
|
const float *tex_coords,
|
|
int tex_coords_len)
|
|
{
|
|
const float position[4] = {x_1, y_1, x_2, y_2};
|
|
CoglMultiTexturedRect rect;
|
|
|
|
/* XXX: All the _*_rectangle* APIs normalize their input into an array of
|
|
* CoglMultiTexturedRect rectangles and pass these on to our work horse;
|
|
* _cogl_framebuffer_draw_multitextured_rectangles.
|
|
*/
|
|
|
|
rect.position = position;
|
|
rect.tex_coords = tex_coords;
|
|
rect.tex_coords_len = tex_coords_len;
|
|
|
|
_cogl_framebuffer_draw_multitextured_rectangles (framebuffer,
|
|
pipeline,
|
|
&rect,
|
|
1,
|
|
TRUE);
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_draw_rectangles (CoglFramebuffer *framebuffer,
|
|
CoglPipeline *pipeline,
|
|
const float *coordinates,
|
|
unsigned int n_rectangles)
|
|
{
|
|
CoglMultiTexturedRect *rects;
|
|
int i;
|
|
|
|
/* XXX: All the _*_rectangle* APIs normalize their input into an array of
|
|
* CoglMultiTexturedRect rectangles and pass these on to our work horse;
|
|
* _cogl_framebuffer_draw_multitextured_rectangles.
|
|
*/
|
|
|
|
rects = g_alloca (n_rectangles * sizeof (CoglMultiTexturedRect));
|
|
|
|
for (i = 0; i < n_rectangles; i++)
|
|
{
|
|
rects[i].position = &coordinates[i * 4];
|
|
rects[i].tex_coords = NULL;
|
|
rects[i].tex_coords_len = 0;
|
|
}
|
|
|
|
_cogl_framebuffer_draw_multitextured_rectangles (framebuffer,
|
|
pipeline,
|
|
rects,
|
|
n_rectangles,
|
|
TRUE);
|
|
}
|
|
|
|
void
|
|
cogl_framebuffer_draw_textured_rectangles (CoglFramebuffer *framebuffer,
|
|
CoglPipeline *pipeline,
|
|
const float *coordinates,
|
|
unsigned int n_rectangles)
|
|
{
|
|
CoglMultiTexturedRect *rects;
|
|
int i;
|
|
|
|
/* XXX: All the _*_rectangle* APIs normalize their input into an array of
|
|
* _CoglMultiTexturedRect rectangles and pass these on to our work horse;
|
|
* _cogl_framebuffer_draw_multitextured_rectangles.
|
|
*/
|
|
|
|
rects = g_alloca (n_rectangles * sizeof (CoglMultiTexturedRect));
|
|
|
|
for (i = 0; i < n_rectangles; i++)
|
|
{
|
|
rects[i].position = &coordinates[i * 8];
|
|
rects[i].tex_coords = &coordinates[i * 8 + 4];
|
|
rects[i].tex_coords_len = 4;
|
|
}
|
|
|
|
_cogl_framebuffer_draw_multitextured_rectangles (framebuffer,
|
|
pipeline,
|
|
rects,
|
|
n_rectangles,
|
|
TRUE);
|
|
}
|