/* * 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 . * * */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #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" #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->texture_driver->pixel_format_to_gl (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 (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 (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; } 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); }