2011-05-08 19:27:10 -04:00
|
|
|
/*
|
|
|
|
* Clutter.
|
|
|
|
*
|
|
|
|
* An OpenGL based 'interactive canvas' library.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2007,2008,2009,2010,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:
|
|
|
|
* Matthew Allum
|
|
|
|
* Robert Bragg
|
|
|
|
* Neil Roberts
|
|
|
|
* Emmanuele Bassi
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2007-10-12 04:17:00 -04:00
|
|
|
#ifdef HAVE_CONFIG_H
|
2007-07-06 09:56:01 -04:00
|
|
|
#include "config.h"
|
2007-10-12 04:17:00 -04:00
|
|
|
#endif
|
2007-07-06 09:56:01 -04:00
|
|
|
|
2012-02-21 08:22:17 -05:00
|
|
|
#define CLUTTER_ENABLE_EXPERIMENTAL_API
|
|
|
|
|
Add a new GDK backend
This commit introduces a new flavour for Clutter, that uses GDK
for handling all window system specific interactions (except for
creating the cogl context, as cogl does not know about GDK), including
in particular events. This is not compatible with the X11 (glx)
flavour, and this is reflected by the different soname (libclutter-gdk-1.0.so),
as all X11 specific functions and classes are not available. If you
wish to be compatible, you should check for CLUTTER_WINDOWING_X11.
Other than that, this backend should be on feature parity with X11,
including XInput 2, XSettings and EMWH (with much, much less code)
https://bugzilla.gnome.org/show_bug.cgi?id=657434
2011-08-25 21:09:18 -04:00
|
|
|
#include "clutter-config.h"
|
|
|
|
|
2011-05-08 19:27:10 -04:00
|
|
|
#include "clutter-stage-cogl.h"
|
2007-07-06 09:56:01 -04:00
|
|
|
|
2011-11-04 12:52:44 -04:00
|
|
|
#include "clutter-actor-private.h"
|
|
|
|
#include "clutter-backend-private.h"
|
2010-10-21 06:29:09 -04:00
|
|
|
#include "clutter-debug.h"
|
|
|
|
#include "clutter-event.h"
|
|
|
|
#include "clutter-enum-types.h"
|
|
|
|
#include "clutter-feature.h"
|
|
|
|
#include "clutter-main.h"
|
|
|
|
#include "clutter-private.h"
|
2011-11-18 16:04:31 -05:00
|
|
|
#include "clutter-profile.h"
|
2010-10-21 06:29:09 -04:00
|
|
|
#include "clutter-stage-private.h"
|
2007-07-06 09:56:01 -04:00
|
|
|
|
2008-04-23 13:20:59 -04:00
|
|
|
static void clutter_stage_window_iface_init (ClutterStageWindowIface *iface);
|
|
|
|
|
2011-05-08 19:27:10 -04:00
|
|
|
G_DEFINE_TYPE_WITH_CODE (ClutterStageCogl,
|
|
|
|
_clutter_stage_cogl,
|
2010-02-27 04:42:42 -05:00
|
|
|
G_TYPE_OBJECT,
|
2008-04-23 13:20:59 -04:00
|
|
|
G_IMPLEMENT_INTERFACE (CLUTTER_TYPE_STAGE_WINDOW,
|
|
|
|
clutter_stage_window_iface_init));
|
2007-07-06 09:56:01 -04:00
|
|
|
|
2011-08-26 18:16:12 -04:00
|
|
|
enum {
|
|
|
|
PROP_0,
|
|
|
|
PROP_WRAPPER,
|
|
|
|
PROP_BACKEND,
|
|
|
|
PROP_LAST
|
|
|
|
};
|
|
|
|
|
2007-07-06 09:56:01 -04:00
|
|
|
static void
|
2011-05-08 19:27:10 -04:00
|
|
|
clutter_stage_cogl_unrealize (ClutterStageWindow *stage_window)
|
2010-06-17 17:26:12 -04:00
|
|
|
{
|
2011-05-08 19:27:10 -04:00
|
|
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
2010-06-17 17:26:12 -04:00
|
|
|
|
2011-05-08 19:27:10 -04:00
|
|
|
CLUTTER_NOTE (BACKEND, "Unrealizing Cogl stage [%p]", stage_cogl);
|
2010-06-17 17:26:12 -04:00
|
|
|
|
2011-09-09 08:52:55 -04:00
|
|
|
if (stage_cogl->onscreen != NULL)
|
|
|
|
{
|
|
|
|
cogl_object_unref (stage_cogl->onscreen);
|
|
|
|
stage_cogl->onscreen = NULL;
|
|
|
|
}
|
2011-05-08 19:27:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
handle_swap_complete_cb (CoglFramebuffer *framebuffer,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
ClutterStageCogl *stage_cogl = user_data;
|
|
|
|
|
|
|
|
/* Early versions of the swap_event implementation in Mesa
|
|
|
|
* deliver BufferSwapComplete event when not selected for,
|
|
|
|
* so if we get a swap event we aren't expecting, just ignore it.
|
|
|
|
*
|
|
|
|
* https://bugs.freedesktop.org/show_bug.cgi?id=27962
|
|
|
|
*
|
|
|
|
* FIXME: This issue can be hidden inside Cogl so we shouldn't
|
|
|
|
* need to care about this bug here.
|
|
|
|
*/
|
|
|
|
if (stage_cogl->pending_swaps > 0)
|
|
|
|
stage_cogl->pending_swaps--;
|
2010-06-17 17:26:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2011-05-08 19:27:10 -04:00
|
|
|
clutter_stage_cogl_realize (ClutterStageWindow *stage_window)
|
2007-07-06 09:56:01 -04:00
|
|
|
{
|
2011-05-08 19:27:10 -04:00
|
|
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
2011-02-24 19:31:41 -05:00
|
|
|
ClutterBackend *backend;
|
|
|
|
CoglFramebuffer *framebuffer;
|
|
|
|
GError *error = NULL;
|
|
|
|
gfloat width = 800;
|
|
|
|
gfloat height = 600;
|
2010-06-17 17:26:12 -04:00
|
|
|
|
2011-01-18 09:16:11 -05:00
|
|
|
CLUTTER_NOTE (BACKEND, "Realizing stage '%s' [%p]",
|
2011-05-08 19:27:10 -04:00
|
|
|
G_OBJECT_TYPE_NAME (stage_cogl),
|
|
|
|
stage_cogl);
|
2010-06-17 17:26:12 -04:00
|
|
|
|
2011-06-17 07:03:21 -04:00
|
|
|
backend = clutter_get_default_backend ();
|
2010-06-17 17:26:12 -04:00
|
|
|
|
2011-08-26 18:16:12 -04:00
|
|
|
if (stage_cogl->onscreen == NULL)
|
2011-05-08 20:09:39 -04:00
|
|
|
{
|
2011-08-26 18:16:12 -04:00
|
|
|
stage_cogl->onscreen = cogl_onscreen_new (backend->cogl_context,
|
|
|
|
width, height);
|
2011-05-08 20:09:39 -04:00
|
|
|
}
|
2011-02-24 19:31:41 -05:00
|
|
|
|
2011-11-04 12:39:52 -04:00
|
|
|
cogl_onscreen_set_swap_throttled (stage_cogl->onscreen,
|
|
|
|
_clutter_get_sync_to_vblank ());
|
2010-06-17 17:26:12 -04:00
|
|
|
|
2011-05-08 19:27:10 -04:00
|
|
|
framebuffer = COGL_FRAMEBUFFER (stage_cogl->onscreen);
|
2011-02-24 19:31:41 -05:00
|
|
|
if (!cogl_framebuffer_allocate (framebuffer, &error))
|
2010-06-17 17:26:12 -04:00
|
|
|
{
|
2011-02-24 19:31:41 -05:00
|
|
|
g_warning ("Failed to allocate stage: %s", error->message);
|
|
|
|
g_error_free (error);
|
2011-05-08 19:27:10 -04:00
|
|
|
cogl_object_unref (stage_cogl->onscreen);
|
|
|
|
stage_cogl->onscreen = NULL;
|
2011-02-24 19:31:41 -05:00
|
|
|
return FALSE;
|
2010-06-17 17:26:12 -04:00
|
|
|
}
|
2011-06-17 07:03:21 -04:00
|
|
|
|
2011-08-26 18:16:12 -04:00
|
|
|
/* FIXME: for fullscreen Cogl platforms then the size we gave
|
2011-02-24 19:31:41 -05:00
|
|
|
* will be ignored, so we need to make sure the stage size is
|
|
|
|
* updated to this size. */
|
2010-06-17 17:26:12 -04:00
|
|
|
|
2011-05-08 19:27:10 -04:00
|
|
|
if (cogl_clutter_winsys_has_feature (COGL_WINSYS_FEATURE_SWAP_BUFFERS_EVENT))
|
|
|
|
{
|
|
|
|
stage_cogl->swap_callback_id =
|
2012-02-21 08:37:55 -05:00
|
|
|
cogl_onscreen_add_swap_buffers_callback (stage_cogl->onscreen,
|
|
|
|
handle_swap_complete_cb,
|
|
|
|
stage_cogl);
|
2011-05-08 19:27:10 -04:00
|
|
|
}
|
|
|
|
|
2011-02-24 19:31:41 -05:00
|
|
|
return TRUE;
|
2010-06-17 17:26:12 -04:00
|
|
|
}
|
|
|
|
|
2011-05-08 19:27:10 -04:00
|
|
|
static int
|
|
|
|
clutter_stage_cogl_get_pending_swaps (ClutterStageWindow *stage_window)
|
|
|
|
{
|
|
|
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
|
|
|
|
|
|
|
return stage_cogl->pending_swaps;
|
|
|
|
}
|
|
|
|
|
2010-02-27 04:42:42 -05:00
|
|
|
static ClutterActor *
|
2011-05-08 19:27:10 -04:00
|
|
|
clutter_stage_cogl_get_wrapper (ClutterStageWindow *stage_window)
|
2010-02-27 04:42:42 -05:00
|
|
|
{
|
2011-05-08 19:27:10 -04:00
|
|
|
return CLUTTER_ACTOR (CLUTTER_STAGE_COGL (stage_window)->wrapper);
|
2007-07-06 09:56:01 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-05-08 19:27:10 -04:00
|
|
|
clutter_stage_cogl_show (ClutterStageWindow *stage_window,
|
2011-08-26 18:16:12 -04:00
|
|
|
gboolean do_raise)
|
2007-07-06 09:56:01 -04:00
|
|
|
{
|
2011-05-08 19:27:10 -04:00
|
|
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
2008-06-11 06:19:02 -04:00
|
|
|
|
2011-05-08 19:27:10 -04:00
|
|
|
clutter_actor_map (CLUTTER_ACTOR (stage_cogl->wrapper));
|
2008-06-11 06:19:02 -04:00
|
|
|
}
|
|
|
|
|
2007-07-06 09:56:01 -04:00
|
|
|
static void
|
2011-05-08 19:27:10 -04:00
|
|
|
clutter_stage_cogl_hide (ClutterStageWindow *stage_window)
|
2007-07-06 09:56:01 -04:00
|
|
|
{
|
2011-05-08 19:27:10 -04:00
|
|
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
2007-07-06 09:56:01 -04:00
|
|
|
|
2011-05-08 19:27:10 -04:00
|
|
|
clutter_actor_unmap (CLUTTER_ACTOR (stage_cogl->wrapper));
|
2007-07-06 09:56:01 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-08-16 11:01:22 -04:00
|
|
|
clutter_stage_cogl_get_geometry (ClutterStageWindow *stage_window,
|
|
|
|
cairo_rectangle_int_t *geometry)
|
2007-07-06 09:56:01 -04:00
|
|
|
{
|
2011-05-08 19:27:10 -04:00
|
|
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
2007-07-06 09:56:01 -04:00
|
|
|
|
2010-02-27 04:42:42 -05:00
|
|
|
if (geometry)
|
|
|
|
{
|
2011-05-08 19:27:10 -04:00
|
|
|
if (stage_cogl->onscreen)
|
2011-02-24 19:31:41 -05:00
|
|
|
{
|
|
|
|
CoglFramebuffer *framebuffer =
|
2011-05-08 19:27:10 -04:00
|
|
|
COGL_FRAMEBUFFER (stage_cogl->onscreen);
|
2011-02-24 19:31:41 -05:00
|
|
|
|
|
|
|
geometry->x = geometry->y = 0;
|
2008-04-23 13:20:59 -04:00
|
|
|
|
2011-02-24 19:31:41 -05:00
|
|
|
geometry->width = cogl_framebuffer_get_width (framebuffer);
|
|
|
|
geometry->height = cogl_framebuffer_get_height (framebuffer);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
geometry->x = geometry->y = 0;
|
|
|
|
geometry->width = 800;
|
|
|
|
geometry->height = 600;
|
|
|
|
}
|
2010-02-27 04:42:42 -05:00
|
|
|
}
|
2007-07-06 09:56:01 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-05-08 19:27:10 -04:00
|
|
|
clutter_stage_cogl_resize (ClutterStageWindow *stage_window,
|
2010-02-27 04:42:42 -05:00
|
|
|
gint width,
|
|
|
|
gint height)
|
2007-07-06 09:56:01 -04:00
|
|
|
{
|
2008-04-25 09:37:36 -04:00
|
|
|
}
|
|
|
|
|
2010-11-16 08:54:15 -05:00
|
|
|
static gboolean
|
2011-05-08 19:27:10 -04:00
|
|
|
clutter_stage_cogl_has_redraw_clips (ClutterStageWindow *stage_window)
|
2010-11-16 08:54:15 -05:00
|
|
|
{
|
2011-05-08 19:27:10 -04:00
|
|
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
2010-11-16 08:54:15 -05:00
|
|
|
|
2010-11-23 11:05:44 -05:00
|
|
|
/* NB: at the start of each new frame there is an implied clip that
|
|
|
|
* clips everything (i.e. nothing would be drawn) so we need to make
|
|
|
|
* sure we return True in the un-initialized case here.
|
|
|
|
*
|
|
|
|
* NB: a clip width of 0 means a full stage redraw has been queued
|
|
|
|
* so we effectively don't have any redraw clips in that case.
|
|
|
|
*/
|
2011-05-08 19:27:10 -04:00
|
|
|
if (!stage_cogl->initialized_redraw_clip ||
|
|
|
|
(stage_cogl->initialized_redraw_clip &&
|
|
|
|
stage_cogl->bounding_redraw_clip.width != 0))
|
2010-11-16 08:54:15 -05:00
|
|
|
return TRUE;
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2011-05-08 19:27:10 -04:00
|
|
|
clutter_stage_cogl_ignoring_redraw_clips (ClutterStageWindow *stage_window)
|
2010-11-16 08:54:15 -05:00
|
|
|
{
|
2011-05-08 19:27:10 -04:00
|
|
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
2010-11-16 08:54:15 -05:00
|
|
|
|
2010-11-23 11:05:44 -05:00
|
|
|
/* NB: a clip width of 0 means a full stage redraw is required */
|
2011-05-08 19:27:10 -04:00
|
|
|
if (stage_cogl->initialized_redraw_clip &&
|
|
|
|
stage_cogl->bounding_redraw_clip.width == 0)
|
2010-11-16 08:54:15 -05:00
|
|
|
return TRUE;
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* A redraw clip represents (in stage coordinates) the bounding box of
|
|
|
|
* something that needs to be redraw. Typically they are added to the
|
|
|
|
* StageWindow as a result of clutter_actor_queue_clipped_redraw() by
|
2011-05-08 19:27:10 -04:00
|
|
|
* actors such as ClutterGLXTexturePixmap. All redraw clips are
|
2010-11-16 08:54:15 -05:00
|
|
|
* discarded after the next paint.
|
|
|
|
*
|
|
|
|
* A NULL stage_clip means the whole stage needs to be redrawn.
|
|
|
|
*
|
|
|
|
* What we do with this information:
|
|
|
|
* - we keep track of the bounding box for all redraw clips
|
|
|
|
* - when we come to redraw; we scissor the redraw to that box and use
|
|
|
|
* glBlitFramebuffer to present the redraw to the front
|
|
|
|
* buffer.
|
|
|
|
*/
|
|
|
|
static void
|
2011-08-16 11:01:22 -04:00
|
|
|
clutter_stage_cogl_add_redraw_clip (ClutterStageWindow *stage_window,
|
|
|
|
cairo_rectangle_int_t *stage_clip)
|
2010-11-16 08:54:15 -05:00
|
|
|
{
|
2011-05-08 19:27:10 -04:00
|
|
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
2010-11-16 08:54:15 -05:00
|
|
|
|
|
|
|
/* If we are already forced to do a full stage redraw then bail early */
|
2011-05-08 19:27:10 -04:00
|
|
|
if (clutter_stage_cogl_ignoring_redraw_clips (stage_window))
|
2010-11-16 08:54:15 -05:00
|
|
|
return;
|
|
|
|
|
|
|
|
/* A NULL stage clip means a full stage redraw has been queued and
|
2010-11-23 11:05:44 -05:00
|
|
|
* we keep track of this by setting a zero width
|
2011-05-08 19:27:10 -04:00
|
|
|
* stage_cogl->bounding_redraw_clip */
|
2010-11-16 08:54:15 -05:00
|
|
|
if (stage_clip == NULL)
|
|
|
|
{
|
2011-05-08 19:27:10 -04:00
|
|
|
stage_cogl->bounding_redraw_clip.width = 0;
|
|
|
|
stage_cogl->initialized_redraw_clip = TRUE;
|
2010-11-16 08:54:15 -05:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Ignore requests to add degenerate/empty clip rectangles */
|
|
|
|
if (stage_clip->width == 0 || stage_clip->height == 0)
|
|
|
|
return;
|
|
|
|
|
2011-05-08 19:27:10 -04:00
|
|
|
if (!stage_cogl->initialized_redraw_clip)
|
2010-11-16 08:54:15 -05:00
|
|
|
{
|
2011-08-16 11:01:22 -04:00
|
|
|
stage_cogl->bounding_redraw_clip = *stage_clip;
|
2010-11-16 08:54:15 -05:00
|
|
|
}
|
2011-05-08 19:27:10 -04:00
|
|
|
else if (stage_cogl->bounding_redraw_clip.width > 0)
|
2010-11-16 08:54:15 -05:00
|
|
|
{
|
2011-08-16 11:01:22 -04:00
|
|
|
_clutter_util_rectangle_union (&stage_cogl->bounding_redraw_clip,
|
|
|
|
stage_clip,
|
|
|
|
&stage_cogl->bounding_redraw_clip);
|
2010-11-16 08:54:15 -05:00
|
|
|
}
|
|
|
|
|
2011-05-08 19:27:10 -04:00
|
|
|
stage_cogl->initialized_redraw_clip = TRUE;
|
2010-11-16 08:54:15 -05:00
|
|
|
}
|
|
|
|
|
2011-07-12 12:16:43 -04:00
|
|
|
static gboolean
|
2011-08-16 11:01:22 -04:00
|
|
|
clutter_stage_cogl_get_redraw_clip_bounds (ClutterStageWindow *stage_window,
|
2011-07-12 12:16:43 -04:00
|
|
|
cairo_rectangle_int_t *stage_clip)
|
|
|
|
{
|
|
|
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
|
|
|
|
|
|
|
if (stage_cogl->using_clipped_redraw)
|
|
|
|
{
|
2011-08-16 11:01:22 -04:00
|
|
|
*stage_clip = stage_cogl->bounding_redraw_clip;
|
2011-07-12 12:16:43 -04:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2011-02-24 19:31:41 -05:00
|
|
|
/* XXX: This is basically identical to clutter_stage_glx_redraw */
|
2011-02-04 10:09:41 -05:00
|
|
|
static void
|
2011-05-08 19:27:10 -04:00
|
|
|
clutter_stage_cogl_redraw (ClutterStageWindow *stage_window)
|
2010-06-17 17:26:12 -04:00
|
|
|
{
|
2011-05-08 19:27:10 -04:00
|
|
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
2011-02-04 10:09:41 -05:00
|
|
|
gboolean may_use_clipped_redraw;
|
|
|
|
gboolean use_clipped_redraw;
|
2011-11-04 12:13:04 -04:00
|
|
|
gboolean can_blit_sub_buffer;
|
2013-02-06 05:05:58 -05:00
|
|
|
gboolean has_buffer_age;
|
2011-11-04 12:13:04 -04:00
|
|
|
ClutterActor *wrapper;
|
2013-02-06 05:05:58 -05:00
|
|
|
cairo_rectangle_int_t *clip_region;
|
|
|
|
gboolean force_swap;
|
2011-02-24 19:31:41 -05:00
|
|
|
|
|
|
|
CLUTTER_STATIC_TIMER (painting_timer,
|
|
|
|
"Redrawing", /* parent */
|
|
|
|
"Painting actors",
|
|
|
|
"The time spent painting actors",
|
|
|
|
0 /* no application private data */);
|
|
|
|
CLUTTER_STATIC_TIMER (swapbuffers_timer,
|
|
|
|
"Redrawing", /* parent */
|
2011-07-10 20:42:59 -04:00
|
|
|
"SwapBuffers",
|
|
|
|
"The time spent blocked by SwapBuffers",
|
2011-02-24 19:31:41 -05:00
|
|
|
0 /* no application private data */);
|
|
|
|
CLUTTER_STATIC_TIMER (blit_sub_buffer_timer,
|
|
|
|
"Redrawing", /* parent */
|
2011-07-10 20:42:59 -04:00
|
|
|
"blit_sub_buffer",
|
|
|
|
"The time spent in blit_sub_buffer",
|
2011-02-24 19:31:41 -05:00
|
|
|
0 /* no application private data */);
|
|
|
|
|
2011-05-08 19:27:10 -04:00
|
|
|
wrapper = CLUTTER_ACTOR (stage_cogl->wrapper);
|
2010-06-17 17:26:12 -04:00
|
|
|
|
2011-05-08 19:27:10 -04:00
|
|
|
if (!stage_cogl->onscreen)
|
2011-02-24 19:31:41 -05:00
|
|
|
return;
|
|
|
|
|
|
|
|
CLUTTER_TIMER_START (_clutter_uprof_context, painting_timer);
|
|
|
|
|
2011-11-04 12:13:04 -04:00
|
|
|
can_blit_sub_buffer =
|
|
|
|
cogl_clutter_winsys_has_feature (COGL_WINSYS_FEATURE_SWAP_REGION);
|
|
|
|
|
2013-02-06 05:05:58 -05:00
|
|
|
has_buffer_age = cogl_clutter_winsys_has_feature (COGL_WINSYS_FEATURE_BUFFER_AGE);
|
|
|
|
|
2011-09-21 13:05:03 -04:00
|
|
|
may_use_clipped_redraw = FALSE;
|
2011-09-30 08:58:40 -04:00
|
|
|
if (_clutter_stage_window_can_clip_redraws (stage_window) &&
|
2011-11-04 12:13:04 -04:00
|
|
|
can_blit_sub_buffer &&
|
2011-02-24 19:31:41 -05:00
|
|
|
/* NB: a zero width redraw clip == full stage redraw */
|
2011-05-08 19:27:10 -04:00
|
|
|
stage_cogl->bounding_redraw_clip.width != 0 &&
|
2010-11-16 08:54:15 -05:00
|
|
|
/* some drivers struggle to get going and produce some junk
|
|
|
|
* frames when starting up... */
|
2011-11-04 12:13:04 -04:00
|
|
|
stage_cogl->frame_count > 3)
|
2011-09-21 13:05:03 -04:00
|
|
|
{
|
2011-09-30 08:58:40 -04:00
|
|
|
may_use_clipped_redraw = TRUE;
|
2013-02-06 05:05:58 -05:00
|
|
|
clip_region = &stage_cogl->bounding_redraw_clip;
|
2011-02-24 19:31:41 -05:00
|
|
|
}
|
2010-11-16 08:54:15 -05:00
|
|
|
|
|
|
|
if (may_use_clipped_redraw &&
|
|
|
|
G_LIKELY (!(clutter_paint_debug_flags &
|
|
|
|
CLUTTER_DEBUG_DISABLE_CLIPPED_REDRAWS)))
|
|
|
|
use_clipped_redraw = TRUE;
|
|
|
|
else
|
|
|
|
use_clipped_redraw = FALSE;
|
|
|
|
|
2013-02-06 05:05:58 -05:00
|
|
|
force_swap = FALSE;
|
|
|
|
|
|
|
|
if (use_clipped_redraw)
|
|
|
|
{
|
|
|
|
if (has_buffer_age)
|
|
|
|
{
|
|
|
|
int age = cogl_onscreen_get_buffer_age (stage_cogl->onscreen);
|
|
|
|
cairo_rectangle_int_t *current_damage;
|
|
|
|
|
|
|
|
current_damage = g_new0 (cairo_rectangle_int_t, 1);
|
|
|
|
current_damage->x = clip_region->x;
|
|
|
|
current_damage->y = clip_region->y;
|
|
|
|
current_damage->width = clip_region->width;
|
|
|
|
current_damage->height = clip_region->height;
|
|
|
|
|
|
|
|
stage_cogl->damage_history = g_slist_prepend (stage_cogl->damage_history, current_damage);
|
|
|
|
|
|
|
|
if (age != 0 && !stage_cogl->dirty_backbuffer && g_slist_length (stage_cogl->damage_history) >= age)
|
|
|
|
{
|
|
|
|
int i = 0;
|
|
|
|
GSList *tmp = NULL;
|
|
|
|
for (tmp = stage_cogl->damage_history; tmp; tmp = tmp->next)
|
|
|
|
{
|
|
|
|
_clutter_util_rectangle_union (clip_region, tmp->data, clip_region);
|
|
|
|
i++;
|
|
|
|
if (i == age)
|
|
|
|
{
|
|
|
|
g_slist_free_full (tmp->next, g_free);
|
|
|
|
tmp->next = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
force_swap = TRUE;
|
|
|
|
|
|
|
|
CLUTTER_NOTE (CLIPPING, "Reusing back buffer - repairing region: x=%d, y=%d, width=%d, height=%d\n",
|
|
|
|
clip_region->x,
|
|
|
|
clip_region->y,
|
|
|
|
clip_region->width,
|
|
|
|
clip_region->height);
|
|
|
|
|
|
|
|
}
|
|
|
|
else if (age == 0 || stage_cogl->dirty_backbuffer)
|
|
|
|
{
|
|
|
|
CLUTTER_NOTE (CLIPPING, "Invalid back buffer: Resetting damage history list.\n");
|
|
|
|
g_slist_free_full (stage_cogl->damage_history, g_free);
|
|
|
|
stage_cogl->damage_history = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CLUTTER_NOTE (CLIPPING, "Unclipped redraw: Resetting damage history list.\n");
|
|
|
|
g_slist_free_full (stage_cogl->damage_history, g_free);
|
|
|
|
stage_cogl->damage_history = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (has_buffer_age && !force_swap)
|
|
|
|
use_clipped_redraw = FALSE;
|
|
|
|
|
2010-11-16 08:54:15 -05:00
|
|
|
if (use_clipped_redraw)
|
|
|
|
{
|
2011-05-08 19:27:10 -04:00
|
|
|
CLUTTER_NOTE (CLIPPING,
|
|
|
|
"Stage clip pushed: x=%d, y=%d, width=%d, height=%d\n",
|
2013-02-06 05:05:58 -05:00
|
|
|
clip_region->x,
|
|
|
|
clip_region->y,
|
|
|
|
clip_region->width,
|
|
|
|
clip_region->height);
|
2011-07-12 12:16:43 -04:00
|
|
|
|
|
|
|
stage_cogl->using_clipped_redraw = TRUE;
|
|
|
|
|
2013-02-06 05:05:58 -05:00
|
|
|
cogl_clip_push_window_rectangle (clip_region->x,
|
|
|
|
clip_region->y,
|
|
|
|
clip_region->width,
|
|
|
|
clip_region->height);
|
2011-02-09 11:16:57 -05:00
|
|
|
_clutter_stage_do_paint (CLUTTER_STAGE (wrapper),
|
2013-02-06 05:05:58 -05:00
|
|
|
clip_region);
|
2010-11-16 08:54:15 -05:00
|
|
|
cogl_clip_pop ();
|
2011-07-12 12:16:43 -04:00
|
|
|
|
|
|
|
stage_cogl->using_clipped_redraw = FALSE;
|
2010-11-16 08:54:15 -05:00
|
|
|
}
|
|
|
|
else
|
2011-05-08 19:27:10 -04:00
|
|
|
{
|
|
|
|
CLUTTER_NOTE (CLIPPING, "Unclipped stage paint\n");
|
|
|
|
|
|
|
|
/* If we are trying to debug redraw issues then we want to pass
|
|
|
|
* the bounding_redraw_clip so it can be visualized */
|
|
|
|
if (G_UNLIKELY (clutter_paint_debug_flags &
|
|
|
|
CLUTTER_DEBUG_DISABLE_CLIPPED_REDRAWS) &&
|
|
|
|
may_use_clipped_redraw)
|
|
|
|
{
|
|
|
|
_clutter_stage_do_paint (CLUTTER_STAGE (wrapper),
|
2013-02-06 05:05:58 -05:00
|
|
|
clip_region);
|
2011-05-08 19:27:10 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
_clutter_stage_do_paint (CLUTTER_STAGE (wrapper), NULL);
|
|
|
|
}
|
2010-11-16 08:54:15 -05:00
|
|
|
|
2011-02-24 19:31:41 -05:00
|
|
|
if (may_use_clipped_redraw &&
|
|
|
|
G_UNLIKELY ((clutter_paint_debug_flags & CLUTTER_DEBUG_REDRAWS)))
|
2010-11-16 08:54:15 -05:00
|
|
|
{
|
2012-04-20 12:11:47 -04:00
|
|
|
CoglFramebuffer *fb = COGL_FRAMEBUFFER (stage_cogl->onscreen);
|
|
|
|
CoglContext *ctx = cogl_framebuffer_get_context (fb);
|
2012-02-21 08:22:17 -05:00
|
|
|
static CoglPipeline *outline = NULL;
|
2011-08-16 11:01:22 -04:00
|
|
|
cairo_rectangle_int_t *clip = &stage_cogl->bounding_redraw_clip;
|
2011-02-24 19:31:41 -05:00
|
|
|
ClutterActor *actor = CLUTTER_ACTOR (wrapper);
|
2010-11-16 08:54:15 -05:00
|
|
|
float x_1 = clip->x;
|
|
|
|
float x_2 = clip->x + clip->width;
|
|
|
|
float y_1 = clip->y;
|
|
|
|
float y_2 = clip->y + clip->height;
|
2012-04-20 12:11:47 -04:00
|
|
|
CoglVertexP2 quad[4] = {
|
|
|
|
{ x_1, y_1 },
|
|
|
|
{ x_2, y_1 },
|
|
|
|
{ x_2, y_2 },
|
|
|
|
{ x_1, y_2 }
|
2010-11-16 08:54:15 -05:00
|
|
|
};
|
2012-04-20 12:11:47 -04:00
|
|
|
CoglPrimitive *prim;
|
2010-11-16 08:54:15 -05:00
|
|
|
CoglMatrix modelview;
|
|
|
|
|
|
|
|
if (outline == NULL)
|
|
|
|
{
|
2012-02-21 08:22:17 -05:00
|
|
|
outline = cogl_pipeline_new (ctx);
|
|
|
|
cogl_pipeline_set_color4ub (outline, 0xff, 0x00, 0x00, 0xff);
|
2010-11-16 08:54:15 -05:00
|
|
|
}
|
|
|
|
|
2012-04-20 12:11:47 -04:00
|
|
|
prim = cogl_primitive_new_p2 (ctx,
|
|
|
|
COGL_VERTICES_MODE_LINE_LOOP,
|
|
|
|
4, /* n_vertices */
|
|
|
|
quad);
|
|
|
|
|
|
|
|
cogl_framebuffer_push_matrix (fb);
|
2010-11-16 08:54:15 -05:00
|
|
|
cogl_matrix_init_identity (&modelview);
|
2011-02-24 19:31:41 -05:00
|
|
|
_clutter_actor_apply_modelview_transform (actor, &modelview);
|
2012-04-20 12:11:47 -04:00
|
|
|
cogl_framebuffer_set_modelview_matrix (fb, &modelview);
|
|
|
|
cogl_framebuffer_draw_primitive (COGL_FRAMEBUFFER (stage_cogl->onscreen),
|
|
|
|
outline,
|
|
|
|
prim);
|
|
|
|
cogl_framebuffer_pop_matrix (fb);
|
|
|
|
cogl_object_unref (prim);
|
2010-11-16 08:54:15 -05:00
|
|
|
}
|
|
|
|
|
2011-02-24 19:31:41 -05:00
|
|
|
CLUTTER_TIMER_STOP (_clutter_uprof_context, painting_timer);
|
2010-06-17 17:26:12 -04:00
|
|
|
|
2010-11-16 08:54:15 -05:00
|
|
|
/* push on the screen */
|
2013-02-06 05:05:58 -05:00
|
|
|
if (use_clipped_redraw && !force_swap)
|
2010-11-16 08:54:15 -05:00
|
|
|
{
|
2013-02-06 05:05:58 -05:00
|
|
|
cairo_rectangle_int_t *clip = clip_region;
|
2011-02-24 19:31:41 -05:00
|
|
|
int copy_area[4];
|
|
|
|
|
|
|
|
/* XXX: It seems there will be a race here in that the stage
|
2012-02-21 08:37:55 -05:00
|
|
|
* window may be resized before the cogl_onscreen_swap_region
|
2011-02-24 19:31:41 -05:00
|
|
|
* is handled and so we may copy the wrong region. I can't
|
|
|
|
* really see how we can handle this with the current state of X
|
|
|
|
* but at least in this case a full redraw should be queued by
|
|
|
|
* the resize anyway so it should only exhibit temporary
|
|
|
|
* artefacts.
|
|
|
|
*/
|
|
|
|
|
|
|
|
copy_area[0] = clip->x;
|
2011-07-22 07:28:29 -04:00
|
|
|
copy_area[1] = clip->y;
|
2011-02-24 19:31:41 -05:00
|
|
|
copy_area[2] = clip->width;
|
|
|
|
copy_area[3] = clip->height;
|
2010-11-16 08:54:15 -05:00
|
|
|
|
|
|
|
CLUTTER_NOTE (BACKEND,
|
2012-02-21 08:37:55 -05:00
|
|
|
"cogl_onscreen_swap_region (onscreen: %p, "
|
|
|
|
"x: %d, y: %d, "
|
|
|
|
"width: %d, height: %d)",
|
2011-05-08 19:27:10 -04:00
|
|
|
stage_cogl->onscreen,
|
2011-02-24 19:31:41 -05:00
|
|
|
copy_area[0], copy_area[1], copy_area[2], copy_area[3]);
|
|
|
|
|
2010-11-16 08:54:15 -05:00
|
|
|
|
|
|
|
CLUTTER_TIMER_START (_clutter_uprof_context, blit_sub_buffer_timer);
|
2011-02-24 19:31:41 -05:00
|
|
|
|
2012-02-21 08:37:55 -05:00
|
|
|
cogl_onscreen_swap_region (stage_cogl->onscreen, copy_area, 1);
|
2011-02-24 19:31:41 -05:00
|
|
|
|
2010-11-16 08:54:15 -05:00
|
|
|
CLUTTER_TIMER_STOP (_clutter_uprof_context, blit_sub_buffer_timer);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-02-21 08:37:55 -05:00
|
|
|
CLUTTER_NOTE (BACKEND, "cogl_onscreen_swap_buffers (onscreen: %p)",
|
2011-05-08 19:27:10 -04:00
|
|
|
stage_cogl->onscreen);
|
|
|
|
|
2012-02-21 08:37:55 -05:00
|
|
|
/* If we have swap buffer events then cogl_onscreen_swap_buffers
|
|
|
|
* will return immediately and we need to track that there is a
|
|
|
|
* swap in progress... */
|
2011-05-08 19:27:10 -04:00
|
|
|
if (clutter_feature_available (CLUTTER_FEATURE_SWAP_EVENTS))
|
|
|
|
stage_cogl->pending_swaps++;
|
2010-11-16 08:54:15 -05:00
|
|
|
|
|
|
|
CLUTTER_TIMER_START (_clutter_uprof_context, swapbuffers_timer);
|
2012-02-21 08:37:55 -05:00
|
|
|
cogl_onscreen_swap_buffers (stage_cogl->onscreen);
|
2010-11-16 08:54:15 -05:00
|
|
|
CLUTTER_TIMER_STOP (_clutter_uprof_context, swapbuffers_timer);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* reset the redraw clipping for the next paint... */
|
2011-05-08 19:27:10 -04:00
|
|
|
stage_cogl->initialized_redraw_clip = FALSE;
|
|
|
|
|
2013-02-03 05:51:19 -05:00
|
|
|
/* We have repaired the backbuffer */
|
|
|
|
stage_cogl->dirty_backbuffer = FALSE;
|
|
|
|
|
2011-05-08 19:27:10 -04:00
|
|
|
stage_cogl->frame_count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
static CoglFramebuffer *
|
|
|
|
clutter_stage_cogl_get_active_framebuffer (ClutterStageWindow *stage_window)
|
|
|
|
{
|
|
|
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
2010-11-16 08:54:15 -05:00
|
|
|
|
2011-05-08 19:27:10 -04:00
|
|
|
return COGL_FRAMEBUFFER (stage_cogl->onscreen);
|
2008-04-23 13:20:59 -04:00
|
|
|
}
|
2011-02-09 11:16:57 -05:00
|
|
|
|
2013-02-03 05:51:19 -05:00
|
|
|
static void
|
|
|
|
clutter_stage_cogl_dirty_back_buffer (ClutterStageWindow *stage_window)
|
|
|
|
{
|
|
|
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
|
|
|
|
|
|
|
stage_cogl->dirty_backbuffer = TRUE;
|
|
|
|
}
|
|
|
|
|
2013-02-06 05:05:58 -05:00
|
|
|
static void
|
|
|
|
clutter_stage_cogl_get_dirty_pixel (ClutterStageWindow *stage_window,
|
|
|
|
int *x, int *y)
|
|
|
|
{
|
|
|
|
ClutterStageCogl *stage_cogl = CLUTTER_STAGE_COGL (stage_window);
|
|
|
|
gboolean has_buffer_age = cogl_clutter_winsys_has_feature (COGL_WINSYS_FEATURE_BUFFER_AGE);
|
|
|
|
if ((stage_cogl->damage_history == NULL && has_buffer_age) || !has_buffer_age)
|
|
|
|
{
|
|
|
|
*x = 0;
|
|
|
|
*y = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cairo_rectangle_int_t *rect;
|
|
|
|
rect = (cairo_rectangle_int_t *) (stage_cogl->damage_history->data);
|
|
|
|
*x = rect->x;
|
|
|
|
*y = rect->y;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-09 11:16:57 -05:00
|
|
|
static void
|
|
|
|
clutter_stage_window_iface_init (ClutterStageWindowIface *iface)
|
|
|
|
{
|
2011-05-08 19:27:10 -04:00
|
|
|
iface->realize = clutter_stage_cogl_realize;
|
|
|
|
iface->unrealize = clutter_stage_cogl_unrealize;
|
|
|
|
iface->get_wrapper = clutter_stage_cogl_get_wrapper;
|
|
|
|
iface->get_geometry = clutter_stage_cogl_get_geometry;
|
|
|
|
iface->resize = clutter_stage_cogl_resize;
|
|
|
|
iface->show = clutter_stage_cogl_show;
|
|
|
|
iface->hide = clutter_stage_cogl_hide;
|
|
|
|
iface->get_pending_swaps = clutter_stage_cogl_get_pending_swaps;
|
|
|
|
iface->add_redraw_clip = clutter_stage_cogl_add_redraw_clip;
|
|
|
|
iface->has_redraw_clips = clutter_stage_cogl_has_redraw_clips;
|
|
|
|
iface->ignoring_redraw_clips = clutter_stage_cogl_ignoring_redraw_clips;
|
2011-07-12 12:16:43 -04:00
|
|
|
iface->get_redraw_clip_bounds = clutter_stage_cogl_get_redraw_clip_bounds;
|
2011-05-08 19:27:10 -04:00
|
|
|
iface->redraw = clutter_stage_cogl_redraw;
|
|
|
|
iface->get_active_framebuffer = clutter_stage_cogl_get_active_framebuffer;
|
2013-02-03 05:51:19 -05:00
|
|
|
iface->dirty_back_buffer = clutter_stage_cogl_dirty_back_buffer;
|
2013-02-06 05:05:58 -05:00
|
|
|
iface->get_dirty_pixel = clutter_stage_cogl_get_dirty_pixel;
|
2011-02-09 11:16:57 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-08-26 18:16:12 -04:00
|
|
|
clutter_stage_cogl_set_property (GObject *gobject,
|
|
|
|
guint prop_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
2011-02-09 11:16:57 -05:00
|
|
|
{
|
2011-08-26 18:16:12 -04:00
|
|
|
ClutterStageCogl *self = CLUTTER_STAGE_COGL (gobject);
|
|
|
|
|
|
|
|
switch (prop_id)
|
|
|
|
{
|
|
|
|
case PROP_WRAPPER:
|
2011-11-04 12:52:44 -04:00
|
|
|
self->wrapper = g_value_get_object (value);
|
2011-08-26 18:16:12 -04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case PROP_BACKEND:
|
2011-11-04 12:52:44 -04:00
|
|
|
self->backend = g_value_get_object (value);
|
2011-08-26 18:16:12 -04:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
2011-02-09 11:16:57 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-05-08 19:27:10 -04:00
|
|
|
_clutter_stage_cogl_class_init (ClutterStageCoglClass *klass)
|
2011-02-09 11:16:57 -05:00
|
|
|
{
|
|
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
2011-08-26 18:16:12 -04:00
|
|
|
gobject_class->set_property = clutter_stage_cogl_set_property;
|
|
|
|
|
2011-11-04 14:50:46 -04:00
|
|
|
g_object_class_override_property (gobject_class, PROP_WRAPPER, "wrapper");
|
|
|
|
g_object_class_override_property (gobject_class, PROP_BACKEND, "backend");
|
2011-02-09 11:16:57 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2011-05-08 19:27:10 -04:00
|
|
|
_clutter_stage_cogl_init (ClutterStageCogl *stage)
|
2011-02-09 11:16:57 -05:00
|
|
|
{
|
|
|
|
}
|