[cogl/wip/neil/snippets: 23/26] Flush matrices in the progend and flip with a vector



commit 9216fa855b9733a09d81899c319ce976a754a8d5
Author: Neil Roberts <neil linux intel com>
Date:   Tue Nov 29 14:21:07 2011 +0000

    Flush matrices in the progend and flip with a vector
    
    Previously flushing the matrices was performed as part of the
    framebuffer state. When on GLES2 this matrix flushing is actually
    diverted so that it only keeps a reference to the intended matrix
    stack. This is necessary because on GLES2 there are no builtin
    uniforms so it can't actually flush the matrices until the program for
    the pipeline is generated. When the matrices are flushed it would
    store the age of modifications on the matrix stack so that it could
    detect when the matrix hasn't changed and avoid flushing it.
    
    This patch changes it so that the pipeline is responsible for flushing
    the matrices even when we are using the GL builtins. The same
    mechanism for detecting unmodified matrix stacks is used in all
    cases. There is a new CoglMatrixStackCache type which is used to store
    a reference to the intended matrix stack along with its last flushed
    age. There are now two of these attached to the CoglContext to track
    the flushed state for the global matrix builtins and also two for each
    glsl progend program state to track the flushed state for a
    program. The framebuffer matrix flush now just updates the intended
    matrix stacks without actually trying to flush.
    
    When a vertex snippet is attached to the pipeline, the GLSL vertend
    will now avoid using the projection matrix to flip the rendering. This
    is necessary because any vertex snippet may cause the projection
    matrix not to be used. Instead the flip is done as a forced final step
    by multiplying cogl_position_out by a vec4 uniform. This uniform is
    updated as part of the progend pre_paint depending on whether the
    framebuffer is offscreen or not.
    
    The patch also adds the option for a winsys to report which kinds of
    buffers need a flipped rendering. This will be used by the KMS backend
    because that also needs flipped rendering for onscreen buffers. There
    is a virtual winsys function to report a bitmask of types of buffers
    that need to be flipped. The default implementation just reports
    offscreen buffers. None of the winsyss actually implement it yet.

 cogl/Makefile.am                           |    2 +
 cogl/cogl-clip-stack.c                     |   40 ++---
 cogl/cogl-context-private.h                |   23 ++-
 cogl/cogl-context.c                        |   22 +-
 cogl/cogl-framebuffer-private.h            |    3 +
 cogl/cogl-framebuffer.c                    |   45 +++--
 cogl/cogl-journal.c                        |   12 +-
 cogl/cogl-matrix-stack.c                   |  224 +++++++++++--------
 cogl/cogl-matrix-stack.h                   |   37 +++-
 cogl/cogl-pipeline-opengl.c                |    2 +-
 cogl/cogl-pipeline-private.h               |   22 ++-
 cogl/cogl-pipeline-progend-fixed-private.h |   36 +++
 cogl/cogl-pipeline-progend-fixed.c         |   69 ++++++
 cogl/cogl-pipeline-progend-glsl.c          |  330 ++++++++++++++-------------
 cogl/cogl-pipeline-vertend-fixed.c         |    5 +-
 cogl/cogl-pipeline-vertend-glsl.c          |   38 +++-
 cogl/cogl-pipeline.c                       |   14 +-
 cogl/cogl-renderer-private.h               |    3 +
 cogl/cogl-renderer.c                       |   12 +
 cogl/cogl.c                                |    6 +-
 cogl/winsys/cogl-winsys-egl.c              |   11 +-
 cogl/winsys/cogl-winsys-glx.c              |   11 +-
 cogl/winsys/cogl-winsys-private.h          |   15 ++
 23 files changed, 623 insertions(+), 359 deletions(-)
---
diff --git a/cogl/Makefile.am b/cogl/Makefile.am
index a49a65b..777f366 100644
--- a/cogl/Makefile.am
+++ b/cogl/Makefile.am
@@ -271,6 +271,8 @@ cogl_sources_c = \
 	$(srcdir)/cogl-pipeline-vertend-glsl-private.h	\
 	$(srcdir)/cogl-pipeline-vertend-fixed.c		\
 	$(srcdir)/cogl-pipeline-vertend-fixed-private.h	\
+	$(srcdir)/cogl-pipeline-progend-fixed.c		\
+	$(srcdir)/cogl-pipeline-progend-fixed-private.h	\
 	$(srcdir)/cogl-pipeline-progend-glsl.c		\
 	$(srcdir)/cogl-pipeline-progend-glsl-private.h	\
 	$(srcdir)/cogl-pipeline-snippet-private.h	\
diff --git a/cogl/cogl-clip-stack.c b/cogl/cogl-clip-stack.c
index 8600f30..9c44407 100644
--- a/cogl/cogl-clip-stack.c
+++ b/cogl/cogl-clip-stack.c
@@ -109,7 +109,13 @@ set_clip_plane (CoglFramebuffer *framebuffer,
   _cogl_matrix_stack_translate (modelview_stack,
                                 -vertex_a[0], -vertex_a[1], -vertex_a[2]);
 
-  _cogl_matrix_stack_flush_to_gl (ctx, modelview_stack, COGL_MATRIX_MODELVIEW);
+  /* Clip planes can only be used when a fixed function backend is in
+     use so we know we can directly push this matrix to the builtin
+     state */
+  _cogl_matrix_stack_flush_to_gl_builtins (ctx,
+                                           modelview_stack,
+                                           COGL_MATRIX_MODELVIEW,
+                                           FALSE /* don't disable flip */);
 
   planef[0] = 0;
   planef[1] = -1.0;
@@ -218,12 +224,8 @@ add_stencil_clip_rectangle (CoglFramebuffer *framebuffer,
   /* This can be called from the journal code which doesn't flush
      the matrix stacks between calls so we need to ensure they're
      flushed now */
-  _cogl_matrix_stack_flush_to_gl (ctx,
-                                  modelview_stack,
-                                  COGL_MATRIX_MODELVIEW);
-  _cogl_matrix_stack_flush_to_gl (ctx,
-                                  projection_stack,
-                                  COGL_MATRIX_PROJECTION);
+  _cogl_matrix_stack_set_current_projection (ctx, projection_stack);
+  _cogl_matrix_stack_set_current_modelview (ctx, modelview_stack);
 
   if (first)
     {
@@ -258,12 +260,8 @@ add_stencil_clip_rectangle (CoglFramebuffer *framebuffer,
       _cogl_matrix_stack_push (modelview_stack);
       _cogl_matrix_stack_load_identity (modelview_stack);
 
-      _cogl_matrix_stack_flush_to_gl (ctx,
-                                      modelview_stack,
-                                      COGL_MATRIX_MODELVIEW);
-      _cogl_matrix_stack_flush_to_gl (ctx,
-                                      projection_stack,
-                                      COGL_MATRIX_PROJECTION);
+      _cogl_matrix_stack_set_current_projection (ctx, projection_stack);
+      _cogl_matrix_stack_set_current_modelview (ctx, modelview_stack);
 
       _cogl_rectangle_immediate (-1.0, -1.0, 1.0, 1.0);
 
@@ -301,12 +299,8 @@ add_stencil_clip_silhouette (CoglFramebuffer *framebuffer,
   /* This can be called from the clip stack code which doesn't flush
      the matrix stacks between calls so we need to ensure they're
      flushed now */
-  _cogl_matrix_stack_flush_to_gl (ctx,
-                                  modelview_stack,
-                                  COGL_MATRIX_MODELVIEW);
-  _cogl_matrix_stack_flush_to_gl (ctx,
-                                  projection_stack,
-                                  COGL_MATRIX_PROJECTION);
+  _cogl_matrix_stack_set_current_projection (ctx, projection_stack);
+  _cogl_matrix_stack_set_current_modelview (ctx, modelview_stack);
 
   /* Just setup a simple pipeline that doesn't use texturing... */
   _cogl_push_source (ctx->stencil_pipeline, FALSE);
@@ -366,15 +360,9 @@ add_stencil_clip_silhouette (CoglFramebuffer *framebuffer,
 
       _cogl_matrix_stack_push (projection_stack);
       _cogl_matrix_stack_load_identity (projection_stack);
-      _cogl_matrix_stack_flush_to_gl (ctx,
-                                      projection_stack,
-                                      COGL_MATRIX_PROJECTION);
 
       _cogl_matrix_stack_push (modelview_stack);
       _cogl_matrix_stack_load_identity (modelview_stack);
-      _cogl_matrix_stack_flush_to_gl (ctx,
-                                      modelview_stack,
-                                      COGL_MATRIX_MODELVIEW);
 
       _cogl_rectangle_immediate (-1.0, -1.0, 1.0, 1.0);
       _cogl_rectangle_immediate (-1.0, -1.0, 1.0, 1.0);
@@ -873,7 +861,7 @@ _cogl_clip_stack_flush (CoglClipStack *stack,
        * down so in this case no conversion is needed.
        */
 
-      if (cogl_is_offscreen (framebuffer))
+      if (_cogl_framebuffer_is_flipped (framebuffer))
         scissor_y_start = scissor_y0;
       else
         {
diff --git a/cogl/cogl-context-private.h b/cogl/cogl-context-private.h
index f8b9cf0..8336639 100644
--- a/cogl/cogl-context-private.h
+++ b/cogl/cogl-context-private.h
@@ -94,16 +94,18 @@ struct _CoglContext
   CoglMatrix        identity_matrix;
   CoglMatrix        y_flip_matrix;
 
-  /* Client-side matrix stack or NULL if none */
+  /* Value that was last used when calling glMatrixMode to avoid
+     calling it multiple times */
   CoglMatrixMode    flushed_matrix_mode;
 
-  /* On GLES2 we need to track the matrices separately because the are
-     stored in GLSL uniforms rather than using the fixed function
-     API. We keep track of the matrix stack that Cogl is trying to
-     flush so we can flush it later after the program is generated. A
-     reference is taken on the stacks. */
-  CoglMatrixStack  *flushed_modelview_stack;
-  CoglMatrixStack  *flushed_projection_stack;
+  /* The matrix stack that should be used for the next render */
+  CoglMatrixStack  *current_projection_stack;
+  CoglMatrixStack  *current_modelview_stack;
+
+  /* The last matrix stack with age that was flushed to the GL matrix
+     builtins */
+  CoglMatrixStackCache builtin_flushed_projection;
+  CoglMatrixStackCache builtin_flushed_modelview;
 
   GArray           *texture_units;
   int               active_texture_unit;
@@ -275,6 +277,11 @@ struct _CoglContext
   GHashTable *uniform_name_hash;
   int n_uniform_names;
 
+  /* Cache of the result of _cogl_renderer_get_framebuffer_orientation().
+     This will be used every time a matrix is flushed so it's probably
+     worthwhile caching it */
+  CoglRendererFramebufferOrientation framebuffer_orientation;
+
   /* This defines a list of function pointers that Cogl uses from
      either GL or GLES. All functions are accessed indirectly through
      these pointers rather than linking to them directly */
diff --git a/cogl/cogl-context.c b/cogl/cogl-context.c
index 08b527e..37901cd 100644
--- a/cogl/cogl-context.c
+++ b/cogl/cogl-context.c
@@ -371,10 +371,12 @@ cogl_context_new (CoglDisplay *display,
     GE (context, glEnable (GL_ALPHA_TEST));
 #endif
 
-#ifdef HAVE_COGL_GLES2
-  _context->flushed_modelview_stack = NULL;
-  _context->flushed_projection_stack = NULL;
-#endif
+  _context->current_modelview_stack = NULL;
+  _context->current_projection_stack = NULL;
+  _cogl_matrix_stack_init_cache (&_context->builtin_flushed_projection);
+  _cogl_matrix_stack_init_cache (&_context->builtin_flushed_modelview);
+  _context->framebuffer_orientation =
+    _cogl_renderer_get_framebuffer_orientation (display->renderer);
 
   /* Create default textures used for fall backs */
   context->default_gl_texture_2d_tex =
@@ -491,12 +493,12 @@ _cogl_context_free (CoglContext *context)
   g_slist_free (context->texture_types);
   g_slist_free (context->buffer_types);
 
-#ifdef HAVE_COGL_GLES2
-  if (_context->flushed_modelview_stack)
-    cogl_object_unref (_context->flushed_modelview_stack);
-  if (_context->flushed_projection_stack)
-    cogl_object_unref (_context->flushed_projection_stack);
-#endif
+  if (_context->current_modelview_stack)
+    cogl_object_unref (_context->current_modelview_stack);
+  if (_context->current_projection_stack)
+    cogl_object_unref (_context->current_projection_stack);
+  _cogl_matrix_stack_destroy_cache (&context->builtin_flushed_projection);
+  _cogl_matrix_stack_destroy_cache (&context->builtin_flushed_modelview);
 
   cogl_pipeline_cache_free (context->pipeline_cache);
 
diff --git a/cogl/cogl-framebuffer-private.h b/cogl/cogl-framebuffer-private.h
index fce41e6..bdf9e45 100644
--- a/cogl/cogl-framebuffer-private.h
+++ b/cogl/cogl-framebuffer-private.h
@@ -363,4 +363,7 @@ _cogl_framebuffer_save_clip_stack (CoglFramebuffer *framebuffer);
 void
 _cogl_framebuffer_restore_clip_stack (CoglFramebuffer *framebuffer);
 
+gboolean
+_cogl_framebuffer_is_flipped (CoglFramebuffer *framebuffer);
+
 #endif /* __COGL_FRAMEBUFFER_PRIVATE_H */
diff --git a/cogl/cogl-framebuffer.c b/cogl/cogl-framebuffer.c
index eaa0be7..ddd2936 100644
--- a/cogl/cogl-framebuffer.c
+++ b/cogl/cogl-framebuffer.c
@@ -1382,20 +1382,20 @@ static unsigned long
 _cogl_framebuffer_compare_modelview_state (CoglFramebuffer *a,
                                            CoglFramebuffer *b)
 {
-  if (!_cogl_matrix_stack_equal (a->modelview_stack, b->modelview_stack))
-    return COGL_FRAMEBUFFER_STATE_MODELVIEW;
-  else
-    return 0;
+  /* 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)
 {
-  if (!_cogl_matrix_stack_equal (a->projection_stack, b->projection_stack))
-    return COGL_FRAMEBUFFER_STATE_MODELVIEW;
-  else
-    return 0;
+  /* 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
@@ -1489,7 +1489,7 @@ _cogl_framebuffer_flush_viewport_state (CoglFramebuffer *framebuffer)
    * 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))
+  if (_cogl_framebuffer_is_flipped (framebuffer))
     gl_viewport_y = framebuffer->viewport_y;
   else
     gl_viewport_y = framebuffer->height -
@@ -1533,17 +1533,15 @@ _cogl_framebuffer_flush_dither_state (CoglFramebuffer *framebuffer)
 static void
 _cogl_framebuffer_flush_modelview_state (CoglFramebuffer *framebuffer)
 {
-  _cogl_matrix_stack_flush_to_gl (framebuffer->context,
-                                  framebuffer->modelview_stack,
-                                  COGL_MATRIX_MODELVIEW);
+  _cogl_matrix_stack_set_current_modelview (framebuffer->context,
+                                            framebuffer->modelview_stack);
 }
 
 static void
 _cogl_framebuffer_flush_projection_state (CoglFramebuffer *framebuffer)
 {
-  _cogl_matrix_stack_flush_to_gl (framebuffer->context,
-                                  framebuffer->projection_stack,
-                                  COGL_MATRIX_PROJECTION);
+  _cogl_matrix_stack_set_current_projection (framebuffer->context,
+                                             framebuffer->projection_stack);
 }
 
 static void
@@ -2409,3 +2407,20 @@ _cogl_framebuffer_restore_clip_stack (CoglFramebuffer *framebuffer)
     framebuffer->context->current_draw_buffer_changes |=
       COGL_FRAMEBUFFER_STATE_CLIP;
 }
+
+gboolean
+_cogl_framebuffer_is_flipped (CoglFramebuffer *framebuffer)
+{
+  CoglRendererFramebufferOrientation orientation =
+    framebuffer->context->framebuffer_orientation;
+
+  switch (framebuffer->type)
+    {
+    case COGL_FRAMEBUFFER_TYPE_ONSCREEN:
+      return !!(orientation & COGL_RENDERER_FLIP_ONSCREEN);
+    case COGL_FRAMEBUFFER_TYPE_OFFSCREEN:
+      return !!(orientation & COGL_RENDERER_FLIP_OFFSCREEN);
+    }
+
+  g_assert_not_reached ();
+}
diff --git a/cogl/cogl-journal.c b/cogl/cogl-journal.c
index e04bf1c..cfc128e 100644
--- a/cogl/cogl-journal.c
+++ b/cogl/cogl-journal.c
@@ -292,9 +292,7 @@ _cogl_journal_flush_modelview_and_entries (CoglJournalEntry *batch_start,
     {
       _cogl_matrix_stack_set (state->modelview_stack,
                               &batch_start->model_view);
-      _cogl_matrix_stack_flush_to_gl (ctx,
-                                      state->modelview_stack,
-                                      COGL_MATRIX_MODELVIEW);
+      _cogl_matrix_stack_set_current_modelview (ctx, state->modelview_stack);
     }
 
   attributes = (CoglAttribute **)state->attributes->data;
@@ -725,17 +723,13 @@ _cogl_journal_flush_clip_stacks_and_entries (CoglJournalEntry *batch_start,
   if (G_LIKELY (!(COGL_DEBUG_ENABLED (COGL_DEBUG_DISABLE_SOFTWARE_TRANSFORM))))
     {
       _cogl_matrix_stack_load_identity (state->modelview_stack);
-      _cogl_matrix_stack_flush_to_gl (ctx,
-                                      state->modelview_stack,
-                                      COGL_MATRIX_MODELVIEW);
+      _cogl_matrix_stack_set_current_modelview (ctx, state->modelview_stack);
     }
 
   /* Setting up the clip state can sometimes also flush the projection
      matrix so we should flush it again. This will be a no-op if the
      clip code didn't modify the projection */
-  _cogl_matrix_stack_flush_to_gl (ctx,
-                                  state->projection_stack,
-                                  COGL_MATRIX_PROJECTION);
+  _cogl_matrix_stack_set_current_projection (ctx, state->projection_stack);
 
   batch_and_call (batch_start,
                   batch_len,
diff --git a/cogl/cogl-matrix-stack.c b/cogl/cogl-matrix-stack.c
index a6e30dd..160b42d 100644
--- a/cogl/cogl-matrix-stack.c
+++ b/cogl/cogl-matrix-stack.c
@@ -362,124 +362,150 @@ _cogl_matrix_stack_set (CoglMatrixStack  *stack,
   stack->age++;
 }
 
-#if defined (HAVE_COGL_GL) || defined (HAVE_COGL_GLES)
-
-static void
-flush_to_fixed_api_gl (CoglContext *context,
-                       gboolean is_identity,
-                       const CoglMatrix *matrix,
-                       void *user_data)
+void
+_cogl_matrix_stack_set_current_projection (CoglContext *context,
+                                           CoglMatrixStack *stack)
 {
-  if (is_identity)
-    GE (context, glLoadIdentity ());
-  else
-    GE (context, glLoadMatrixf (cogl_matrix_get_array (matrix)) );
+  cogl_object_ref (stack);
+  if (context->current_projection_stack)
+    cogl_object_unref (context->current_projection_stack);
+  context->current_projection_stack = stack;
 }
 
-#endif
-
 void
-_cogl_prepare_matrix_stack_for_flush (CoglContext *context,
-                                      CoglMatrixStack *stack,
-                                      CoglMatrixMode mode,
-                                      CoglMatrixStackFlushFunc callback,
-                                      void *user_data)
+_cogl_matrix_stack_set_current_modelview (CoglContext *context,
+                                          CoglMatrixStack *stack)
 {
-  CoglMatrixState *state = _cogl_matrix_stack_top (stack);
-
-  /* Because Cogl defines texture coordinates to have a top left origin and
-   * because offscreen framebuffers may be used for rendering to textures we
-   * always render upside down to offscreen buffers.
-   */
-  if (mode == COGL_MATRIX_PROJECTION &&
-      cogl_is_offscreen (cogl_get_draw_framebuffer ()))
-    {
-      CoglMatrix flipped_projection;
-      CoglMatrix *projection =
-        state->is_identity ? &context->identity_matrix : &state->matrix;
-
-      cogl_matrix_multiply (&flipped_projection,
-                            &context->y_flip_matrix, projection);
-      callback (context, FALSE, &flipped_projection, user_data);
-    }
-  else
-    {
-      CoglMatrix *modelview =
-        state->is_identity ? &context->identity_matrix : &state->matrix;
-      callback (context, state->is_identity, modelview, user_data);
-    }
+  cogl_object_ref (stack);
+  if (context->current_modelview_stack)
+    cogl_object_unref (context->current_modelview_stack);
+  context->current_modelview_stack = stack;
 }
 
-void
-_cogl_matrix_stack_flush_to_gl (CoglContext *context,
-                                CoglMatrixStack *stack,
-                                CoglMatrixMode mode)
+static void
+_cogl_matrix_stack_flush_matrix_to_gl_builtin (CoglContext *ctx,
+                                               gboolean is_identity,
+                                               CoglMatrix *matrix,
+                                               CoglMatrixMode mode)
 {
-  if (context->driver == COGL_DRIVER_GLES2)
+  g_assert (ctx->driver == COGL_DRIVER_GL ||
+            ctx->driver == COGL_DRIVER_GLES1);
+
+#if defined (HAVE_COGL_GL) || defined (HAVE_COGL_GLES)
+  if (ctx->flushed_matrix_mode != mode)
     {
-      /* Under GLES2 we need to flush the matrices differently because
-         they are stored in uniforms attached to the program instead of
-         the global GL context state. At this point we can't be sure that
-         the right program will be generated so instead we'll just store a
-         reference to the matrix stack that is intended to be flushed and
-         update the uniform once the program is ready. */
+      GLenum gl_mode = 0;
 
       switch (mode)
         {
         case COGL_MATRIX_MODELVIEW:
-          cogl_object_ref (stack);
-          if (context->flushed_modelview_stack)
-            cogl_object_unref (context->flushed_modelview_stack);
-          context->flushed_modelview_stack = stack;
+          gl_mode = GL_MODELVIEW;
           break;
 
         case COGL_MATRIX_PROJECTION:
-          cogl_object_ref (stack);
-          if (context->flushed_projection_stack)
-            cogl_object_unref (context->flushed_projection_stack);
-          context->flushed_projection_stack = stack;
+          gl_mode = GL_PROJECTION;
           break;
 
         case COGL_MATRIX_TEXTURE:
-          /* This shouldn't happen because the texture matrices are
-             handled by the GLSL pipeline backend */
-          g_assert_not_reached ();
+          gl_mode = GL_TEXTURE;
           break;
         }
+
+      GE (ctx, glMatrixMode (gl_mode));
+      ctx->flushed_matrix_mode = mode;
     }
 
-#if defined (HAVE_COGL_GL) || defined (HAVE_COGL_GLES)
+  if (is_identity)
+    GE (ctx, glLoadIdentity ());
   else
-    {
-      if (context->flushed_matrix_mode != mode)
-        {
-          GLenum gl_mode = 0;
-
-          switch (mode)
-            {
-            case COGL_MATRIX_MODELVIEW:
-              gl_mode = GL_MODELVIEW;
-              break;
-
-            case COGL_MATRIX_PROJECTION:
-              gl_mode = GL_PROJECTION;
-              break;
-
-            case COGL_MATRIX_TEXTURE:
-              gl_mode = GL_TEXTURE;
-              break;
-            }
+    GE (ctx, glLoadMatrixf (cogl_matrix_get_array (matrix)));
+#endif
+}
 
-          GE (context, glMatrixMode (gl_mode));
-          context->flushed_matrix_mode = mode;
-        }
+void
+_cogl_matrix_stack_flush_to_gl_builtins (CoglContext *ctx,
+                                         CoglMatrixStack *stack,
+                                         CoglMatrixMode mode,
+                                         gboolean disable_flip)
+{
+  g_assert (ctx->driver == COGL_DRIVER_GL ||
+            ctx->driver == COGL_DRIVER_GLES1);
 
-      _cogl_prepare_matrix_stack_for_flush (context,
-                                            stack,
-                                            mode,
-                                            flush_to_fixed_api_gl,
-                                            stack);
-    }
+#if defined (HAVE_COGL_GL) || defined (HAVE_COGL_GLES)
+  {
+    gboolean is_identity, needs_flip;
+    CoglMatrixState *state;
+    CoglMatrixStackCache *cache;
+
+    state = _cogl_matrix_stack_top (stack);
+
+    if (mode == COGL_MATRIX_PROJECTION)
+      {
+        /* Because Cogl defines texture coordinates to have a top left
+         * origin and because offscreen framebuffers may be used for
+         * rendering to textures we always render upside down to
+         * offscreen buffers. Also for some backends we need to render
+         * onscreen buffers upside-down too.
+         */
+        if (disable_flip)
+          needs_flip = FALSE;
+        else
+          needs_flip =
+            _cogl_framebuffer_is_flipped (cogl_get_draw_framebuffer ());
+
+        cache = &ctx->builtin_flushed_projection;
+      }
+    else
+      {
+        needs_flip = FALSE;
+
+        if (mode == COGL_MATRIX_MODELVIEW)
+          cache = &ctx->builtin_flushed_modelview;
+        else
+          cache = NULL;
+      }
+
+    is_identity = state->is_identity && !needs_flip;
+
+    /* We don't need to do anything if the state is the same */
+    if (cache &&
+        ((is_identity && cache->flushed_identity) ||
+         (stack == cache->stack && stack->age == cache->age &&
+          needs_flip == cache->flipped)))
+      return;
+
+    if (needs_flip)
+      {
+        CoglMatrix flipped_matrix;
+
+        cogl_matrix_multiply (&flipped_matrix,
+                              &ctx->y_flip_matrix,
+                              state->is_identity ?
+                              &ctx->identity_matrix :
+                              &state->matrix);
+
+        _cogl_matrix_stack_flush_matrix_to_gl_builtin (ctx,
+                                                       FALSE, /* not identity */
+                                                       &flipped_matrix,
+                                                       mode);
+      }
+    else
+      _cogl_matrix_stack_flush_matrix_to_gl_builtin (ctx,
+                                                     is_identity,
+                                                     &state->matrix,
+                                                     mode);
+
+    if (cache)
+      {
+        cache->age = stack->age;
+        cogl_object_ref (stack);
+        if (cache->stack)
+          cogl_object_unref (cache->stack);
+        cache->stack = stack;
+        cache->flushed_identity = is_identity;
+        cache->flipped = needs_flip;
+      }
+  }
 #endif
 }
 
@@ -510,3 +536,17 @@ _cogl_matrix_stack_equal (CoglMatrixStack *stack0,
   else
     return cogl_matrix_equal (&state0->matrix, &state1->matrix);
 }
+
+void
+_cogl_matrix_stack_init_cache (CoglMatrixStackCache *cache)
+{
+  cache->stack = NULL;
+  cache->flushed_identity = FALSE;
+}
+
+void
+_cogl_matrix_stack_destroy_cache (CoglMatrixStackCache *cache)
+{
+  if (cache->stack)
+    cogl_object_unref (cache->stack);
+}
diff --git a/cogl/cogl-matrix-stack.h b/cogl/cogl-matrix-stack.h
index e16125a..a3f1346 100644
--- a/cogl/cogl-matrix-stack.h
+++ b/cogl/cogl-matrix-stack.h
@@ -34,6 +34,14 @@
 
 typedef struct _CoglMatrixStack CoglMatrixStack;
 
+typedef struct
+{
+  CoglMatrixStack *stack;
+  unsigned int age;
+  gboolean flushed_identity;
+  gboolean flipped;
+} CoglMatrixStackCache;
+
 typedef enum {
   COGL_MATRIX_MODELVIEW,
   COGL_MATRIX_PROJECTION,
@@ -107,10 +115,20 @@ _cogl_matrix_stack_get (CoglMatrixStack *stack,
 void
 _cogl_matrix_stack_set (CoglMatrixStack *stack,
                         const CoglMatrix *matrix);
+
+void
+_cogl_matrix_stack_set_current_projection (CoglContext *context,
+                                           CoglMatrixStack *stack);
+
+void
+_cogl_matrix_stack_set_current_modelview (CoglContext *context,
+                                          CoglMatrixStack *stack);
+
 void
-_cogl_matrix_stack_flush_to_gl (CoglContext *ctx,
-                                CoglMatrixStack *stack,
-                                CoglMatrixMode mode);
+_cogl_matrix_stack_flush_to_gl_builtins (CoglContext *ctx,
+                                         CoglMatrixStack *stack,
+                                         CoglMatrixMode mode,
+                                         gboolean disable_flip);
 
 unsigned int
 _cogl_matrix_stack_get_age (CoglMatrixStack *stack);
@@ -122,15 +140,14 @@ _cogl_matrix_stack_get_age (CoglMatrixStack *stack);
 gboolean
 _cogl_matrix_stack_has_identity_flag (CoglMatrixStack *stack);
 
-void
-_cogl_prepare_matrix_stack_for_flush (CoglContext *context,
-                                      CoglMatrixStack *stack,
-                                      CoglMatrixMode mode,
-                                      CoglMatrixStackFlushFunc callback,
-                                      void *user_data);
-
 gboolean
 _cogl_matrix_stack_equal (CoglMatrixStack *stack0,
                           CoglMatrixStack *stack1);
 
+void
+_cogl_matrix_stack_init_cache (CoglMatrixStackCache *cache);
+
+void
+_cogl_matrix_stack_destroy_cache (CoglMatrixStackCache *cache);
+
 #endif /* __COGL_MATRIX_STACK_H */
diff --git a/cogl/cogl-pipeline-opengl.c b/cogl/cogl-pipeline-opengl.c
index aaf5ea8..4e5950f 100644
--- a/cogl/cogl-pipeline-opengl.c
+++ b/cogl/cogl-pipeline-opengl.c
@@ -643,7 +643,7 @@ _cogl_pipeline_flush_color_blend_alpha_depth_state (
           /* If we are painting to an offscreen framebuffer then we
              need to invert the winding of the front face because
              everything is painted upside down */
-          invert_winding = cogl_is_offscreen (draw_framebuffer);
+          invert_winding = _cogl_framebuffer_is_flipped (draw_framebuffer);
 
           switch (cull_face_state->front_winding)
             {
diff --git a/cogl/cogl-pipeline-private.h b/cogl/cogl-pipeline-private.h
index 8e7f321..f676b1a 100644
--- a/cogl/cogl-pipeline-private.h
+++ b/cogl/cogl-pipeline-private.h
@@ -137,11 +137,21 @@
 
 /* If we have either of the GLSL backends then we also need a GLSL
    progend to combine the shaders generated into a single
-   program. Currently there is only one progend but if we ever add
-   other languages they would likely need their own progend too. The
-   progends are different from the other backends because there can be
-   more than one in use for each pipeline. All of the progends are
-   invoked whenever a pipeline is flushed. */
+   program. Same goes for the fixed progends which are used to flush
+   the matrices */
+#ifdef COGL_PIPELINE_FRAGEND_FIXED
+
+#define COGL_PIPELINE_PROGEND_FIXED      0
+
+#ifdef COGL_PIPELINE_FRAGEND_GLSL
+#define COGL_PIPELINE_PROGEND_GLSL       1
+#define COGL_PIPELINE_N_PROGENDS         2
+#else
+#define COGL_PIPELINE_N_PROGENDS         1
+#endif
+
+#else /* COGL_PIPELINE_FRAGEND_FIXED */
+
 #ifdef COGL_PIPELINE_FRAGEND_GLSL
 #define COGL_PIPELINE_PROGEND_GLSL       0
 #define COGL_PIPELINE_N_PROGENDS         1
@@ -149,6 +159,8 @@
 #define COGL_PIPELINE_N_PROGENDS         0
 #endif
 
+#endif /* COGL_PIPELINE_FRAGEND_FIXED */
+
 /* XXX: should I rename these as
  * COGL_PIPELINE_STATE_INDEX_XYZ... ?
  */
diff --git a/cogl/cogl-pipeline-progend-fixed-private.h b/cogl/cogl-pipeline-progend-fixed-private.h
new file mode 100644
index 0000000..a256059
--- /dev/null
+++ b/cogl/cogl-pipeline-progend-fixed-private.h
@@ -0,0 +1,36 @@
+/*
+ * Cogl
+ *
+ * An object oriented GL/GLES Abstraction/Utility Layer
+ *
+ * Copyright (C) 2011 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/>.
+ *
+ *
+ *
+ * Authors:
+ *   Neil Roberts <neil linux intel com>
+ */
+
+#ifndef __COGL_PIPELINE_PROGEND_FIXED_PRIVATE_H
+#define __COGL_PIPELINE_PROGEND_FIXED_PRIVATE_H
+
+#include "cogl-pipeline-private.h"
+
+extern const CoglPipelineProgend _cogl_pipeline_fixed_progend;
+
+#endif /* __COGL_PIPELINE_PROGEND_FIXED_PRIVATE_H */
+
diff --git a/cogl/cogl-pipeline-progend-fixed.c b/cogl/cogl-pipeline-progend-fixed.c
new file mode 100644
index 0000000..d715439
--- /dev/null
+++ b/cogl/cogl-pipeline-progend-fixed.c
@@ -0,0 +1,69 @@
+/*
+ * Cogl
+ *
+ * An object oriented GL/GLES Abstraction/Utility Layer
+ *
+ * Copyright (C) 2011 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/>.
+ *
+ *
+ *
+ * Authors:
+ *   Neil Roberts <neil linux intel com>
+ */
+
+#ifdef HAVE_CONFIG_H
+#include "config.h"
+#endif
+
+#include <string.h>
+
+#include "cogl-pipeline-private.h"
+
+#ifdef COGL_PIPELINE_PROGEND_FIXED
+
+#include "cogl-context.h"
+#include "cogl-context-private.h"
+
+static void
+_cogl_pipeline_progend_fixed_pre_paint (CoglPipeline *pipeline)
+{
+  _COGL_GET_CONTEXT (ctx, NO_RETVAL);
+
+  if (pipeline->vertend != COGL_PIPELINE_VERTEND_FIXED)
+    return;
+
+  if (ctx->current_projection_stack)
+    _cogl_matrix_stack_flush_to_gl_builtins (ctx,
+                                             ctx->current_projection_stack,
+                                             COGL_MATRIX_PROJECTION,
+                                             FALSE /* enable flip */);
+  if (ctx->current_modelview_stack)
+    _cogl_matrix_stack_flush_to_gl_builtins (ctx,
+                                             ctx->current_modelview_stack,
+                                             COGL_MATRIX_MODELVIEW,
+                                             FALSE /* enable flip */);
+}
+
+const CoglPipelineProgend _cogl_pipeline_fixed_progend =
+  {
+    NULL, /* end */
+    NULL, /* pre_change_notify */
+    NULL, /* layer_pre_change_notify */
+    _cogl_pipeline_progend_fixed_pre_paint
+  };
+
+#endif /* COGL_PIPELINE_PROGEND_FIXED */
diff --git a/cogl/cogl-pipeline-progend-glsl.c b/cogl/cogl-pipeline-progend-glsl.c
index bc96cb8..7cc9a09 100644
--- a/cogl/cogl-pipeline-progend-glsl.c
+++ b/cogl/cogl-pipeline-progend-glsl.c
@@ -48,6 +48,7 @@
 #include "cogl-pipeline-cache.h"
 #include "cogl-pipeline-state-private.h"
 #include "cogl-attribute-private.h"
+#include "cogl-framebuffer-private.h"
 
 #ifdef HAVE_COGL_GLES2
 
@@ -121,11 +122,8 @@ typedef struct
   GLint projection_uniform;
   GLint mvp_uniform;
 
-  CoglMatrixStack *flushed_modelview_stack;
-  unsigned int flushed_modelview_stack_age;
-  gboolean flushed_modelview_is_identity;
-  CoglMatrixStack *flushed_projection_stack;
-  unsigned int flushed_projection_stack_age;
+  CoglMatrixStackCache projection_cache;
+  CoglMatrixStackCache modelview_cache;
 #endif
 
   /* We need to track the last pipeline that the program was used with
@@ -140,6 +138,13 @@ typedef struct
   /* Array of attribute locations. */
   GArray *attribute_locations;
 
+  /* The 'flip' uniform is used to flip the geometry upside-down when
+     the framebuffer requires it only when there are vertex
+     snippets. Otherwise this is acheived using the projection
+     matrix */
+  GLint flip_uniform;
+  int flushed_flip_state;
+
   UnitState *unit_state;
 } CoglPipelineProgramState;
 
@@ -225,17 +230,10 @@ clear_attribute_cache (CoglPipelineProgramState *program_state)
 static void
 clear_flushed_matrix_stacks (CoglPipelineProgramState *program_state)
 {
-  if (program_state->flushed_modelview_stack)
-    {
-      cogl_object_unref (program_state->flushed_modelview_stack);
-      program_state->flushed_modelview_stack = NULL;
-    }
-  if (program_state->flushed_projection_stack)
-    {
-      cogl_object_unref (program_state->flushed_projection_stack);
-      program_state->flushed_projection_stack = NULL;
-    }
-  program_state->flushed_modelview_is_identity = FALSE;
+  _cogl_matrix_stack_destroy_cache (&program_state->projection_cache);
+  _cogl_matrix_stack_init_cache (&program_state->projection_cache);
+  _cogl_matrix_stack_destroy_cache (&program_state->modelview_cache);
+  _cogl_matrix_stack_init_cache (&program_state->modelview_cache);
 }
 
 #endif /* HAVE_COGL_GLES2 */
@@ -253,9 +251,8 @@ program_state_new (int n_layers)
   program_state->uniform_locations = NULL;
   program_state->attribute_locations = NULL;
 #ifdef HAVE_COGL_GLES2
-  program_state->flushed_modelview_stack = NULL;
-  program_state->flushed_modelview_is_identity = FALSE;
-  program_state->flushed_projection_stack = NULL;
+  _cogl_matrix_stack_init_cache (&program_state->modelview_cache);
+  _cogl_matrix_stack_init_cache (&program_state->projection_cache);
 #endif
 
   return program_state;
@@ -282,7 +279,10 @@ destroy_program_state (void *user_data,
 
 #ifdef HAVE_COGL_GLES2
       if (ctx->driver == COGL_DRIVER_GLES2)
-        clear_flushed_matrix_stacks (program_state);
+        {
+          _cogl_matrix_stack_destroy_cache (&program_state->projection_cache);
+          _cogl_matrix_stack_destroy_cache (&program_state->modelview_cache);
+        }
 #endif
 
       if (program_state->program)
@@ -793,6 +793,10 @@ _cogl_pipeline_progend_glsl_end (CoglPipeline *pipeline,
                                    get_uniform_cb,
                                    &state);
       clear_attribute_cache (program_state);
+
+      GE_RET (program_state->flip_uniform,
+              ctx, glGetUniformLocation (gl_program, "_cogl_flip_vector"));
+      program_state->flushed_flip_state = -1;
     }
 
   state.unit = 0;
@@ -925,172 +929,185 @@ _cogl_pipeline_progend_glsl_layer_pre_change_notify (
 
 #ifdef HAVE_COGL_GLES2
 
-static void
-flush_modelview_cb (CoglContext *ctx,
-                    gboolean is_identity,
-                    const CoglMatrix *matrix,
-                    void *user_data)
-{
-  CoglPipelineProgramState *program_state = user_data;
-
-  GE( ctx, glUniformMatrix4fv (program_state->modelview_uniform, 1, FALSE,
-                               cogl_matrix_get_array (matrix)) );
-}
-
-static void
-flush_projection_cb (CoglContext *ctx,
-                     gboolean is_identity,
-                     const CoglMatrix *matrix,
-                     void *user_data)
+static gboolean
+check_and_update_matrix_cache (CoglMatrixStack *stack,
+                               CoglMatrixStackCache *cache,
+                               gboolean flip)
 {
-  CoglPipelineProgramState *program_state = user_data;
+  gboolean is_identity;
+  gboolean is_dirty;
+  unsigned int age;
 
-  GE( ctx, glUniformMatrix4fv (program_state->projection_uniform, 1, FALSE,
-                               cogl_matrix_get_array (matrix)) );
-}
+  is_identity = !flip && _cogl_matrix_stack_has_identity_flag (stack);
 
-typedef struct
-{
-  CoglPipelineProgramState *program_state;
-  const CoglMatrix *projection_matrix;
-} FlushCombinedData;
+  age = _cogl_matrix_stack_get_age (stack);
 
-static void
-flush_combined_step_two_cb (CoglContext *ctx,
-                            gboolean is_identity,
-                            const CoglMatrix *matrix,
-                            void *user_data)
-{
-  FlushCombinedData *data = user_data;
-  CoglMatrix mvp_matrix;
+  is_dirty = !((is_identity && cache->flushed_identity) ||
+               (stack == cache->stack &&
+                age == cache->age &&
+                flip == cache->flipped));
 
-  /* If the modelview is the identity then we can bypass the matrix
-     multiplication */
-  if (is_identity)
-    {
-      const float *array = cogl_matrix_get_array (data->projection_matrix);
-      GE( ctx, glUniformMatrix4fv (data->program_state->mvp_uniform,
-                                   1, FALSE, array ) );
-    }
-  else
-    {
-      cogl_matrix_multiply (&mvp_matrix,
-                            data->projection_matrix,
-                            matrix);
+  cache->age = age;
+  cogl_object_ref (stack);
+  if (cache->stack)
+    cogl_object_unref (cache->stack);
+  cache->stack = stack;
+  cache->flushed_identity = is_identity;
+  cache->flipped = flip;
 
-      GE( ctx, glUniformMatrix4fv (data->program_state->mvp_uniform, 1, FALSE,
-                                   cogl_matrix_get_array (&mvp_matrix)) );
-    }
+  return is_dirty;
 }
 
-static void
-flush_combined_step_one_cb (CoglContext *ctx,
-                            gboolean is_identity,
-                            const CoglMatrix *matrix,
-                            void *user_data)
-{
-  FlushCombinedData data;
-
-  data.program_state = user_data;
-  data.projection_matrix = matrix;
-
-  _cogl_prepare_matrix_stack_for_flush (ctx,
-                                        ctx->flushed_modelview_stack,
-                                        COGL_MATRIX_MODELVIEW,
-                                        flush_combined_step_two_cb,
-                                        &data);
-}
+#endif /* HAVE_COGL_GLES2 */
 
 static void
 _cogl_pipeline_progend_glsl_pre_paint (CoglPipeline *pipeline)
 {
-  CoglPipelineProgramState *program_state = get_program_state (pipeline);
-  gboolean modelview_changed;
-  gboolean projection_changed;
+  gboolean needs_flip;
+  CoglMatrixStack *projection_stack;
+  CoglMatrixStack *modelview_stack;
+  CoglPipelineProgramState *program_state;
 
   _COGL_GET_CONTEXT (ctx, NO_RETVAL);
 
-  if (ctx->driver != COGL_DRIVER_GLES2)
+  if (pipeline->vertend != COGL_PIPELINE_VERTEND_GLSL)
     return;
 
-  /* We only need to update the matrices if we're using the the GLSL
-     vertend, but this is a requirement on GLES2 anyway */
-  _COGL_RETURN_IF_FAIL (pipeline->vertend == COGL_PIPELINE_VERTEND_GLSL);
-
   program_state = get_program_state (pipeline);
 
+  projection_stack = ctx->current_projection_stack;
+  modelview_stack = ctx->current_modelview_stack;
+
   /* An initial pipeline is flushed while creating the context. At
-     this point there are no matrices flushed so we can't do
+     this point there are no matrices selected so we can't do
      anything */
-  if (ctx->flushed_modelview_stack == NULL ||
-      ctx->flushed_projection_stack == NULL)
+  if (modelview_stack == NULL || projection_stack == NULL)
     return;
 
-  /* When flushing from the journal the modelview matrix is usually
-     the identity matrix so it makes sense to optimise this case by
-     specifically checking whether we already have the identity matrix
-     which will catch a lot of common cases of redundant flushing */
-  if (program_state->flushed_modelview_is_identity &&
-      _cogl_matrix_stack_has_identity_flag (ctx->flushed_modelview_stack))
-    modelview_changed = FALSE;
-  else
-    modelview_changed =
-      program_state->flushed_modelview_stack != ctx->flushed_modelview_stack ||
-      program_state->flushed_modelview_stack_age !=
-      _cogl_matrix_stack_get_age (program_state->flushed_modelview_stack);
+  needs_flip = _cogl_framebuffer_is_flipped (cogl_get_draw_framebuffer ());
+
+#ifdef HAVE_COGL_GLES2
+  if (ctx->driver == COGL_DRIVER_GLES2)
+    {
+      gboolean modelview_changed;
+      gboolean projection_changed;
+      gboolean need_modelview;
+      gboolean need_projection;
+      CoglMatrix modelview, projection;
+
+      projection_changed =
+        check_and_update_matrix_cache (projection_stack,
+                                       &program_state->projection_cache,
+                                       needs_flip &&
+                                       program_state->flip_uniform == -1);
+
+      modelview_changed =
+        check_and_update_matrix_cache (modelview_stack,
+                                       &program_state->modelview_cache,
+                                       FALSE /* never flip modelview */);
+
+      if (modelview_changed || projection_changed)
+        {
+          if (program_state->mvp_uniform != -1)
+            need_modelview = need_projection = TRUE;
+          else
+            {
+              need_projection = (program_state->projection_uniform != -1 &&
+                                 projection_changed);
+              need_modelview = (program_state->modelview_uniform != -1 &&
+                                modelview_changed);
+            }
 
-  projection_changed =
-    program_state->flushed_projection_stack != ctx->flushed_projection_stack ||
-    program_state->flushed_projection_stack_age !=
-    _cogl_matrix_stack_get_age (program_state->flushed_projection_stack);
+          if (need_modelview)
+            _cogl_matrix_stack_get (modelview_stack, &modelview);
+          if (need_projection)
+            {
+              if (needs_flip && program_state->flip_uniform == -1)
+                {
+                  CoglMatrix tmp_matrix;
+                  _cogl_matrix_stack_get (projection_stack, &tmp_matrix);
+                  cogl_matrix_multiply (&projection,
+                                        &ctx->y_flip_matrix,
+                                        &tmp_matrix);
+                }
+              else
+                _cogl_matrix_stack_get (projection_stack, &projection);
+            }
+
+          if (projection_changed && program_state->projection_uniform != -1)
+            GE (ctx, glUniformMatrix4fv (program_state->projection_uniform,
+                                         1, /* count */
+                                         FALSE, /* transpose */
+                                         cogl_matrix_get_array (&projection)));
+
+          if (modelview_changed && program_state->modelview_uniform != -1)
+            GE (ctx, glUniformMatrix4fv (program_state->modelview_uniform,
+                                         1, /* count */
+                                         FALSE, /* transpose */
+                                         cogl_matrix_get_array (&modelview)));
 
-  if (modelview_changed)
+          if (program_state->mvp_uniform != -1)
+            {
+              /* The journal usually uses an identity matrix for the
+                 modelview so we can optimise this common case by
+                 avoiding the matrix multiplication */
+              if (_cogl_matrix_stack_has_identity_flag (modelview_stack))
+                {
+                  GE (ctx,
+                      glUniformMatrix4fv (program_state->mvp_uniform,
+                                          1, /* count */
+                                          FALSE, /* transpose */
+                                          cogl_matrix_get_array (&projection)));
+                }
+              else
+                {
+                  CoglMatrix combined;
+
+                  cogl_matrix_multiply (&combined,
+                                        &projection,
+                                        &modelview);
+                  GE (ctx,
+                      glUniformMatrix4fv (program_state->mvp_uniform,
+                                          1, /* count */
+                                          FALSE, /* transpose */
+                                          cogl_matrix_get_array (&combined)));
+                }
+            }
+        }
+    }
+  else
+#endif
     {
-      cogl_object_ref (ctx->flushed_modelview_stack);
-      if (program_state->flushed_modelview_stack)
-        cogl_object_unref (program_state->flushed_modelview_stack);
-      program_state->flushed_modelview_stack = ctx->flushed_modelview_stack;
-      program_state->flushed_modelview_stack_age =
-        _cogl_matrix_stack_get_age (ctx->flushed_modelview_stack);
-      program_state->flushed_modelview_is_identity =
-        _cogl_matrix_stack_has_identity_flag (ctx->flushed_modelview_stack);
-
-      if (program_state->modelview_uniform != -1)
-        _cogl_prepare_matrix_stack_for_flush (ctx,
-                                              program_state
-                                                ->flushed_modelview_stack,
-                                              COGL_MATRIX_MODELVIEW,
-                                              flush_modelview_cb,
-                                              program_state);
+      gboolean disable_flip;
+
+      /* If there are vertex snippets, then we'll disable flipping the
+         geometry via the matrix and use the flip vertex instead */
+      disable_flip = program_state->flip_uniform != -1;
+
+      _cogl_matrix_stack_flush_to_gl_builtins (ctx,
+                                               projection_stack,
+                                               COGL_MATRIX_PROJECTION,
+                                               disable_flip);
+      _cogl_matrix_stack_flush_to_gl_builtins (ctx,
+                                               modelview_stack,
+                                               COGL_MATRIX_MODELVIEW,
+                                               disable_flip);
     }
 
-  if (projection_changed)
+  if (program_state->flip_uniform != -1
+      && program_state->flushed_flip_state != needs_flip)
     {
-      cogl_object_ref (ctx->flushed_projection_stack);
-      if (program_state->flushed_projection_stack)
-        cogl_object_unref (program_state->flushed_projection_stack);
-      program_state->flushed_projection_stack = ctx->flushed_projection_stack;
-      program_state->flushed_projection_stack_age =
-        _cogl_matrix_stack_get_age (ctx->flushed_projection_stack);
-
-      if (program_state->projection_uniform != -1)
-        _cogl_prepare_matrix_stack_for_flush (ctx,
-                                              program_state
-                                                ->flushed_projection_stack,
-                                              COGL_MATRIX_PROJECTION,
-                                              flush_projection_cb,
-                                              program_state);
+      static const float do_flip[4] = { 1.0f, -1.0f, 1.0f, 1.0f };
+      static const float dont_flip[4] = { 1.0f, 1.0f, 1.0f, 1.0f };
+      GE( ctx, glUniform4fv (program_state->flip_uniform,
+                             1, /* count */
+                             needs_flip ? do_flip : dont_flip) );
+      program_state->flushed_flip_state = needs_flip;
     }
-
-  if (program_state->mvp_uniform != -1 &&
-      (modelview_changed || projection_changed))
-    _cogl_prepare_matrix_stack_for_flush (ctx,
-                                          ctx->flushed_projection_stack,
-                                          COGL_MATRIX_PROJECTION,
-                                          flush_combined_step_one_cb,
-                                          program_state);
 }
 
+#ifdef HAVE_COGL_GLES2
+
 static void
 update_float_uniform (CoglPipeline *pipeline,
                       int uniform_location,
@@ -1112,12 +1129,7 @@ const CoglPipelineProgend _cogl_pipeline_glsl_progend =
     _cogl_pipeline_progend_glsl_end,
     _cogl_pipeline_progend_glsl_pre_change_notify,
     _cogl_pipeline_progend_glsl_layer_pre_change_notify,
-#ifdef HAVE_COGL_GLES2
     _cogl_pipeline_progend_glsl_pre_paint
-#else
-    NULL
-#endif
   };
 
 #endif /* COGL_PIPELINE_PROGEND_GLSL */
-
diff --git a/cogl/cogl-pipeline-vertend-fixed.c b/cogl/cogl-pipeline-vertend-fixed.c
index 3b19016..1d60c83 100644
--- a/cogl/cogl-pipeline-vertend-fixed.c
+++ b/cogl/cogl-pipeline-vertend-fixed.c
@@ -99,8 +99,9 @@ _cogl_pipeline_vertend_fixed_add_layer (CoglPipeline *pipeline,
 
       _cogl_set_active_texture_unit (unit_index);
 
-      _cogl_matrix_stack_flush_to_gl (ctx, unit->matrix_stack,
-                                      COGL_MATRIX_TEXTURE);
+      _cogl_matrix_stack_flush_to_gl_builtins (ctx, unit->matrix_stack,
+                                               COGL_MATRIX_TEXTURE,
+                                               FALSE /* enable flip */);
     }
 
   return TRUE;
diff --git a/cogl/cogl-pipeline-vertend-glsl.c b/cogl/cogl-pipeline-vertend-glsl.c
index bd99557..602cf02 100644
--- a/cogl/cogl-pipeline-vertend-glsl.c
+++ b/cogl/cogl-pipeline-vertend-glsl.c
@@ -43,6 +43,7 @@
 #include "cogl-handle.h"
 #include "cogl-program-private.h"
 #include "cogl-pipeline-vertend-glsl-private.h"
+#include "cogl-pipeline-state-private.h"
 
 const CoglPipelineVertend _cogl_pipeline_glsl_vertend;
 
@@ -328,9 +329,10 @@ _cogl_pipeline_vertend_glsl_add_layer (CoglPipeline *pipeline,
 
           _cogl_set_active_texture_unit (unit_index);
 
-          _cogl_matrix_stack_flush_to_gl (ctx,
-                                          unit->matrix_stack,
-                                          COGL_MATRIX_TEXTURE);
+          _cogl_matrix_stack_flush_to_gl_builtins (ctx,
+                                                   unit->matrix_stack,
+                                                   COGL_MATRIX_TEXTURE,
+                                                   FALSE /* do flip */);
         }
     }
 
@@ -410,6 +412,7 @@ _cogl_pipeline_vertend_glsl_end (CoglPipeline *pipeline,
       GLint compile_status;
       GLuint shader;
       CoglPipelineSnippetData snippet_data;
+      CoglPipelineSnippetList *vertex_snippets;
 
       COGL_STATIC_COUNTER (vertend_glsl_compile_counter,
                            "glsl vertex compile counter",
@@ -432,9 +435,11 @@ _cogl_pipeline_vertend_glsl_end (CoglPipeline *pipeline,
                        "  cogl_color_out = cogl_color_in;\n"
                        "}\n");
 
+      vertex_snippets = get_vertex_snippets (pipeline);
+
       /* Add hooks for the vertex transform part */
       memset (&snippet_data, 0, sizeof (snippet_data));
-      snippet_data.snippets = get_vertex_snippets (pipeline);
+      snippet_data.snippets = vertex_snippets;
       snippet_data.hook = COGL_SNIPPET_HOOK_VERTEX_TRANSFORM;
       snippet_data.chain_function = "cogl_real_vertex_transform";
       snippet_data.final_name = "cogl_vertex_transform";
@@ -444,14 +449,35 @@ _cogl_pipeline_vertend_glsl_end (CoglPipeline *pipeline,
 
       /* Add all of the hooks for vertex processing */
       memset (&snippet_data, 0, sizeof (snippet_data));
-      snippet_data.snippets = get_vertex_snippets (pipeline);
+      snippet_data.snippets = vertex_snippets;
       snippet_data.hook = COGL_SNIPPET_HOOK_VERTEX;
       snippet_data.chain_function = "cogl_generated_source";
-      snippet_data.final_name = "main";
+      snippet_data.final_name = "cogl_vertex_hook";
       snippet_data.function_prefix = "cogl_vertex_hook";
       snippet_data.source_buf = shader_state->source;
       _cogl_pipeline_snippet_generate_code (&snippet_data);
 
+      g_string_append (shader_state->source,
+                       "void\n"
+                       "main ()\n"
+                       "{\n"
+                       "  cogl_vertex_hook ();\n");
+
+      /* If there are any snippets then we can't rely on the
+         projection matrix to flip the rendering for offscreen buffers
+         so we'll need to flip it using an extra statement and a
+         uniform */
+      if (_cogl_pipeline_has_vertex_snippets (pipeline))
+        {
+          g_string_append (shader_state->header,
+                           "uniform vec4 _cogl_flip_vector;\n");
+          g_string_append (shader_state->source,
+                           "  cogl_position_out *= _cogl_flip_vector;\n");
+        }
+
+      g_string_append (shader_state->source,
+                       "}\n");
+
       GE_RET( shader, ctx, glCreateShader (GL_VERTEX_SHADER) );
 
       lengths[0] = shader_state->header->len;
diff --git a/cogl/cogl-pipeline.c b/cogl/cogl-pipeline.c
index 7277782..74ba160 100644
--- a/cogl/cogl-pipeline.c
+++ b/cogl/cogl-pipeline.c
@@ -71,9 +71,6 @@ _cogl_pipeline_progends[MAX (COGL_PIPELINE_N_PROGENDS, 1)];
 #ifdef COGL_PIPELINE_FRAGEND_FIXED
 #include "cogl-pipeline-fragend-fixed-private.h"
 #endif
-#ifdef COGL_PIPELINE_PROGEND_GLSL
-#include "cogl-pipeline-progend-glsl-private.h"
-#endif
 
 #ifdef COGL_PIPELINE_VERTEND_GLSL
 #include "cogl-pipeline-vertend-glsl-private.h"
@@ -82,6 +79,13 @@ _cogl_pipeline_progends[MAX (COGL_PIPELINE_N_PROGENDS, 1)];
 #include "cogl-pipeline-vertend-fixed-private.h"
 #endif
 
+#ifdef COGL_PIPELINE_PROGEND_FIXED
+#include "cogl-pipeline-progend-fixed-private.h"
+#endif
+#ifdef COGL_PIPELINE_PROGEND_GLSL
+#include "cogl-pipeline-progend-glsl-private.h"
+#endif
+
 COGL_OBJECT_DEFINE (Pipeline, pipeline);
 
 GQuark
@@ -128,6 +132,10 @@ _cogl_pipeline_init_default_pipeline (void)
   _cogl_pipeline_fragends[COGL_PIPELINE_FRAGEND_FIXED] =
     &_cogl_pipeline_fixed_fragend;
 #endif
+#ifdef COGL_PIPELINE_PROGEND_FIXED
+  _cogl_pipeline_progends[COGL_PIPELINE_PROGEND_FIXED] =
+    &_cogl_pipeline_fixed_progend;
+#endif
 #ifdef COGL_PIPELINE_PROGEND_GLSL
   _cogl_pipeline_progends[COGL_PIPELINE_PROGEND_GLSL] =
     &_cogl_pipeline_glsl_progend;
diff --git a/cogl/cogl-renderer-private.h b/cogl/cogl-renderer-private.h
index 45222ef..1a354d8 100644
--- a/cogl/cogl-renderer-private.h
+++ b/cogl/cogl-renderer-private.h
@@ -83,4 +83,7 @@ void *
 _cogl_renderer_get_proc_address (CoglRenderer *renderer,
                                  const char *name);
 
+CoglRendererFramebufferOrientation
+_cogl_renderer_get_framebuffer_orientation (CoglRenderer *renderer);
+
 #endif /* __COGL_RENDERER_PRIVATE_H */
diff --git a/cogl/cogl-renderer.c b/cogl/cogl-renderer.c
index a691bda..8557846 100644
--- a/cogl/cogl-renderer.c
+++ b/cogl/cogl-renderer.c
@@ -403,6 +403,18 @@ _cogl_renderer_get_proc_address (CoglRenderer *renderer,
   return winsys->renderer_get_proc_address (renderer, name);
 }
 
+CoglRendererFramebufferOrientation
+_cogl_renderer_get_framebuffer_orientation (CoglRenderer *renderer)
+{
+  const CoglWinsysVtable *winsys = _cogl_renderer_get_winsys (renderer);
+
+  if (winsys->renderer_get_framebuffer_orientation)
+    return winsys->renderer_get_framebuffer_orientation (renderer);
+  else
+    /* Most backends just need to flip the offscreen buffers */
+    return COGL_RENDERER_FLIP_OFFSCREEN;
+}
+
 int
 cogl_renderer_get_n_fragment_texture_units (CoglRenderer *renderer)
 {
diff --git a/cogl/cogl.c b/cogl/cogl.c
index cba3de7..f1d2456 100644
--- a/cogl/cogl.c
+++ b/cogl/cogl.c
@@ -431,7 +431,7 @@ _cogl_read_pixels_with_rowstride (int x,
    * NB: all offscreen rendering is done upside down so no conversion
    * is necissary in this case.
    */
-  if (!cogl_is_offscreen (framebuffer))
+  if (!_cogl_framebuffer_is_flipped (framebuffer))
     y = framebuffer_height - y - height;
 
   /* Initialise the CoglBitmap */
@@ -462,7 +462,7 @@ _cogl_read_pixels_with_rowstride (int x,
   /* 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))
+      !_cogl_framebuffer_is_flipped (framebuffer))
     {
       GE (ctx, glPixelStorei (GL_PACK_INVERT_MESA, TRUE));
       pack_invert_set = TRUE;
@@ -541,7 +541,7 @@ _cogl_read_pixels_with_rowstride (int x,
 
   /* 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)
+  if (!_cogl_framebuffer_is_flipped (framebuffer) && !pack_invert_set)
     {
       guint8 *temprow = g_alloca (rowstride * sizeof (guint8));
 
diff --git a/cogl/winsys/cogl-winsys-egl.c b/cogl/winsys/cogl-winsys-egl.c
index 15d0d7a..676c3c6 100644
--- a/cogl/winsys/cogl-winsys-egl.c
+++ b/cogl/winsys/cogl-winsys-egl.c
@@ -1532,11 +1532,12 @@ _cogl_winsys_onscreen_swap_region (CoglOnscreen *onscreen,
    * but we are given rectangles relative to the top left so we need to flip
    * them... */
   memcpy (rectangles, user_rectangles, sizeof (int) * n_rectangles * 4);
-  for (i = 0; i < n_rectangles; i++)
-    {
-      int *rect = &rectangles[4 * i];
-      rect[1] = framebuffer_height - rect[1] - rect[3];
-    }
+  if (!_cogl_framebuffer_is_flipped (framebuffer))
+    for (i = 0; i < n_rectangles; i++)
+      {
+        int *rect = &rectangles[4 * i];
+        rect[1] = framebuffer_height - rect[1] - rect[3];
+      }
 
   if (egl_renderer->pf_eglSwapBuffersRegion (egl_renderer->edpy,
                                              egl_onscreen->egl_surface,
diff --git a/cogl/winsys/cogl-winsys-glx.c b/cogl/winsys/cogl-winsys-glx.c
index 58640af..4ed7742 100644
--- a/cogl/winsys/cogl-winsys-glx.c
+++ b/cogl/winsys/cogl-winsys-glx.c
@@ -1211,11 +1211,12 @@ _cogl_winsys_onscreen_swap_region (CoglOnscreen *onscreen,
    * we are given rectangles relative to the top left so we need to flip
    * them... */
   memcpy (rectangles, user_rectangles, sizeof (int) * n_rectangles * 4);
-  for (i = 0; i < n_rectangles; i++)
-    {
-      int *rect = &rectangles[4 * i];
-      rect[1] = framebuffer_height - rect[1] - rect[3];
-    }
+  if (!_cogl_framebuffer_is_flipped (framebuffer))
+    for (i = 0; i < n_rectangles; i++)
+      {
+        int *rect = &rectangles[4 * i];
+        rect[1] = framebuffer_height - rect[1] - rect[3];
+      }
 
   _cogl_framebuffer_flush_state (framebuffer,
                                  framebuffer,
diff --git a/cogl/winsys/cogl-winsys-private.h b/cogl/winsys/cogl-winsys-private.h
index 16317b7..4fb1cba 100644
--- a/cogl/winsys/cogl-winsys-private.h
+++ b/cogl/winsys/cogl-winsys-private.h
@@ -54,6 +54,18 @@ typedef enum
   COGL_WINSYS_RECTANGLE_STATE_ENABLE
 } CoglWinsysRectangleState;
 
+/* Flags to specify which types of buffers need to be rendered to
+   upside-down for this particularly renderer. Currently all offscreen
+   buffers need to be rendered upside down in Cogl because the texture
+   coordinate 0,0 is expected to be the topleft whereas in GL it is
+   the bottom-left. Additionally, some backends may require rendering
+   upside-down to onscreen buffers as well */
+typedef enum
+{
+  COGL_RENDERER_FLIP_ONSCREEN = (1 << 0),
+  COGL_RENDERER_FLIP_OFFSCREEN = (1 << 1),
+} CoglRendererFramebufferOrientation;
+
 typedef struct _CoglWinsysVtable
 {
   CoglWinsysID id;
@@ -110,6 +122,9 @@ typedef struct _CoglWinsysVtable
                            const int *rectangles,
                            int n_rectangles);
 
+  CoglRendererFramebufferOrientation
+  (*renderer_get_framebuffer_orientation) (CoglRenderer *renderer);
+
 #ifdef COGL_HAS_EGL_SUPPORT
   EGLDisplay
   (*context_egl_get_egl_display) (CoglContext *context);



[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]