/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */ /* * Copyright (C) 2013 Red Hat * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of the * License, or (at your option) any later version. * * This program 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 * General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, see . * * Written by: * Owen Taylor * Ray Strode * Jasper St. Pierre */ #include "config.h" #include "clutter/clutter-mutter.h" #include "compositor/clutter-utils.h" #include "compositor/meta-cullable.h" #include "compositor/region-utils.h" G_DEFINE_INTERFACE (MetaCullable, meta_cullable, CLUTTER_TYPE_ACTOR); static gboolean has_active_effects (ClutterActor *actor) { g_autoptr (GList) effects = NULL; GList *l; effects = clutter_actor_get_effects (actor); for (l = effects; l != NULL; l = l->next) { if (clutter_actor_meta_get_enabled (CLUTTER_ACTOR_META (l->data))) return TRUE; } return FALSE; } static cairo_region_t * region_apply_transform_expand_maybe_ref (cairo_region_t *region, graphene_matrix_t *transform) { if (cairo_region_is_empty (region)) return cairo_region_reference (region); return meta_region_apply_matrix_transform_expand (region, transform); } /** * SECTION:meta-cullable * @title: MetaCullable * @short_description: CPU culling operations for efficient drawing * * When we are painting a stack of 5-10 large actors, the standard * bottom-to-top method of drawing every actor results in a tremendous * amount of overdraw. If these actors are painting textures like * windows, it can easily max out the available memory bandwidth on a * low-end graphics chipset. It's even worse if window textures are * being accessed over the AGP bus. * * #MetaCullable is our solution. The basic technique applied here is to * do a pre-pass before painting where we walk each actor from top to bottom * and ask each actor to "cull itself out". We pass in a region it can copy * to clip its drawing to, and the actor can subtract its fully opaque pixels * so that actors underneath know not to draw there as well. */ /** * meta_cullable_cull_out_children: * @cullable: The #MetaCullable * @unobscured_region: The unobscured region, as passed into cull_out() * @clip_region: The clip region, as passed into cull_out() * * This is a helper method for actors that want to recurse over their * child actors, and cull them out. * * See #MetaCullable and meta_cullable_cull_out() for more details. */ void meta_cullable_cull_out_children (MetaCullable *cullable, cairo_region_t *unobscured_region, cairo_region_t *clip_region) { ClutterActor *actor = CLUTTER_ACTOR (cullable); ClutterActor *child; ClutterActorIter iter; clutter_actor_iter_init (&iter, actor); while (clutter_actor_iter_prev (&iter, &child)) { gboolean needs_culling; if (!META_IS_CULLABLE (child)) continue; needs_culling = (unobscured_region != NULL && clip_region != NULL); if (needs_culling && !CLUTTER_ACTOR_IS_VISIBLE (child)) needs_culling = FALSE; /* If an actor has effects applied, then that can change the area * it paints and the opacity, so we no longer can figure out what * portion of the actor is obscured and what portion of the screen * it obscures, so we skip the actor. * * This has a secondary beneficial effect: if a ClutterOffscreenEffect * is applied to an actor, then our clipped redraws interfere with the * caching of the FBO - even if we only need to draw a small portion * of the window right now, ClutterOffscreenEffect may use other portions * of the FBO later. So, skipping actors with effects applied also * prevents these bugs. * * Theoretically, we should check clutter_actor_get_offscreen_redirect() * as well for the same reason, but omitted for simplicity in the * hopes that no-one will do that. */ if (needs_culling && has_active_effects (child)) needs_culling = FALSE; if (needs_culling) { cairo_region_t *actor_unobscured_region, *actor_clip_region; cairo_region_t *reduced_unobscured_region, *reduced_clip_region; graphene_matrix_t actor_transform, inverted_actor_transform; clutter_actor_get_transform (child, &actor_transform); if (graphene_matrix_is_identity (&actor_transform)) { /* No transformation needed, simply pass through to child */ meta_cullable_cull_out (META_CULLABLE (child), unobscured_region, clip_region); continue; } if (!graphene_matrix_inverse (&actor_transform, &inverted_actor_transform) || !graphene_matrix_is_2d (&actor_transform)) { meta_cullable_cull_out (META_CULLABLE (child), NULL, NULL); continue; } actor_unobscured_region = region_apply_transform_expand_maybe_ref (unobscured_region, &inverted_actor_transform); actor_clip_region = region_apply_transform_expand_maybe_ref (clip_region, &inverted_actor_transform); g_assert (actor_unobscured_region && actor_clip_region); meta_cullable_cull_out (META_CULLABLE (child), actor_unobscured_region, actor_clip_region); reduced_unobscured_region = region_apply_transform_expand_maybe_ref (actor_unobscured_region, &actor_transform); reduced_clip_region = region_apply_transform_expand_maybe_ref (actor_clip_region, &actor_transform); g_assert (reduced_unobscured_region && reduced_clip_region); cairo_region_intersect (unobscured_region, reduced_unobscured_region); cairo_region_intersect (clip_region, reduced_clip_region); cairo_region_destroy (actor_unobscured_region); cairo_region_destroy (actor_clip_region); cairo_region_destroy (reduced_unobscured_region); cairo_region_destroy (reduced_clip_region); } else { meta_cullable_cull_out (META_CULLABLE (child), NULL, NULL); } } } /** * meta_cullable_reset_culling_children: * @cullable: The #MetaCullable * * This is a helper method for actors that want to recurse over their * child actors, and cull them out. * * See #MetaCullable and meta_cullable_reset_culling() for more details. */ void meta_cullable_reset_culling_children (MetaCullable *cullable) { ClutterActor *actor = CLUTTER_ACTOR (cullable); ClutterActor *child; ClutterActorIter iter; clutter_actor_iter_init (&iter, actor); while (clutter_actor_iter_next (&iter, &child)) { if (!META_IS_CULLABLE (child)) continue; meta_cullable_reset_culling (META_CULLABLE (child)); } } static void meta_cullable_default_init (MetaCullableInterface *iface) { } /** * meta_cullable_cull_out: * @cullable: The #MetaCullable * @unobscured_region: The unobscured region, in @cullable's space. * @clip_region: The clip region, in @cullable's space. * * When #MetaWindowGroup is painted, we walk over its direct cullable * children from top to bottom and ask themselves to "cull out". Cullables * can use @unobscured_region and @clip_region to clip their drawing. Actors * interested in eliminating overdraw should copy the @clip_region and only * paint those parts, as everything else has been obscured by actors above it. * * Actors that may have fully opaque parts should also subtract out a region * that is fully opaque from @unobscured_region and @clip_region. * * @unobscured_region and @clip_region are extremely similar. The difference * is that @clip_region starts off with the stage's clip, if Clutter detects * that we're doing a clipped redraw. @unobscured_region, however, starts off * with the full stage size, so actors that may want to record what parts of * their window are unobscured for e.g. scheduling repaints can do so. * * Actors that have children can also use the meta_cullable_cull_out_children() * helper method to do a simple cull across all their children. */ void meta_cullable_cull_out (MetaCullable *cullable, cairo_region_t *unobscured_region, cairo_region_t *clip_region) { META_CULLABLE_GET_IFACE (cullable)->cull_out (cullable, unobscured_region, clip_region); } /** * meta_cullable_reset_culling: * @cullable: The #MetaCullable * * Actors that copied data in their cull_out() implementation can now * reset their data, as the paint is now over. Additional paints may be * done by #ClutterClone or similar, and they should not be affected by * the culling operation. */ void meta_cullable_reset_culling (MetaCullable *cullable) { META_CULLABLE_GET_IFACE (cullable)->reset_culling (cullable); }