|  | // Copyright 2014 The Chromium Authors | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "cc/trees/draw_property_utils.h" | 
|  |  | 
|  | #include <stddef.h> | 
|  |  | 
|  | #include <algorithm> | 
|  | #include <array> | 
|  | #include <map> | 
|  | #include <unordered_set> | 
|  | #include <utility> | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/containers/adapters.h" | 
|  | #include "base/containers/flat_map.h" | 
|  | #include "base/containers/stack.h" | 
|  | #include "base/debug/crash_logging.h" | 
|  | #include "base/debug/dump_without_crashing.h" | 
|  | #include "base/logging.h" | 
|  | #include "base/memory/raw_ptr.h" | 
|  | #include "base/notreached.h" | 
|  | #include "base/strings/string_number_conversions.h" | 
|  | #include "build/build_config.h" | 
|  | #include "cc/base/features.h" | 
|  | #include "cc/base/math_util.h" | 
|  | #include "cc/layers/draw_properties.h" | 
|  | #include "cc/layers/layer.h" | 
|  | #include "cc/layers/layer_impl.h" | 
|  | #include "cc/layers/picture_layer.h" | 
|  | #include "cc/layers/view_transition_content_layer_impl.h" | 
|  | #include "cc/paint/filter_operation.h" | 
|  | #include "cc/trees/clip_node.h" | 
|  | #include "cc/trees/effect_node.h" | 
|  | #include "cc/trees/layer_tree_impl.h" | 
|  | #include "cc/trees/property_tree.h" | 
|  | #include "cc/trees/property_tree_builder.h" | 
|  | #include "cc/trees/scroll_node.h" | 
|  | #include "cc/trees/transform_node.h" | 
|  | #include "cc/trees/viewport_property_ids.h" | 
|  | #include "components/viz/common/features.h" | 
|  | #include "components/viz/common/view_transition_element_resource_id.h" | 
|  | #include "ui/gfx/geometry/rect_conversions.h" | 
|  |  | 
|  | namespace cc { | 
|  |  | 
|  | namespace draw_property_utils { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | gfx::Rect ToEnclosingClipRect(const gfx::RectF& clip_rect) { | 
|  | constexpr float kClipError = 0.00001f; | 
|  | return gfx::ToEnclosingRectIgnoringError(clip_rect, kClipError); | 
|  | } | 
|  |  | 
|  | bool IsRootLayer(const Layer* layer) { | 
|  | return !layer->parent(); | 
|  | } | 
|  |  | 
|  | bool IsRootLayer(const LayerImpl* layer) { | 
|  | return layer->layer_tree_impl()->IsRootLayer(layer); | 
|  | } | 
|  |  | 
|  | void PostConcatSurfaceContentsScale(const EffectNode* effect_node, | 
|  | gfx::Transform* transform) { | 
|  | if (!effect_node) { | 
|  | // This can happen when PaintArtifactCompositor builds property trees as it | 
|  | // doesn't set effect ids on clip nodes. | 
|  | return; | 
|  | } | 
|  | DCHECK(effect_node->HasRenderSurface()); | 
|  | transform->PostScale(effect_node->surface_contents_scale.x(), | 
|  | effect_node->surface_contents_scale.y()); | 
|  | } | 
|  |  | 
|  | bool ConvertRectBetweenSurfaceSpaces(const PropertyTrees* property_trees, | 
|  | int source_effect_id, | 
|  | int dest_effect_id, | 
|  | gfx::RectF clip_in_source_space, | 
|  | gfx::RectF* clip_in_dest_space) { | 
|  | const EffectNode* source_effect_node = | 
|  | property_trees->effect_tree().Node(source_effect_id); | 
|  | int source_transform_id = source_effect_node->transform_id; | 
|  | const EffectNode* dest_effect_node = | 
|  | property_trees->effect_tree().Node(dest_effect_id); | 
|  | int dest_transform_id = dest_effect_node->transform_id; | 
|  | gfx::Transform source_to_dest; | 
|  | if (source_transform_id > dest_transform_id) { | 
|  | if (property_trees->GetToTarget(source_transform_id, dest_effect_id, | 
|  | &source_to_dest)) { | 
|  | ConcatInverseSurfaceContentsScale(source_effect_node, &source_to_dest); | 
|  | *clip_in_dest_space = | 
|  | MathUtil::MapClippedRect(source_to_dest, clip_in_source_space); | 
|  | } else { | 
|  | return false; | 
|  | } | 
|  | } else { | 
|  | if (property_trees->GetFromTarget(dest_transform_id, source_effect_id, | 
|  | &source_to_dest)) { | 
|  | PostConcatSurfaceContentsScale(dest_effect_node, &source_to_dest); | 
|  | *clip_in_dest_space = | 
|  | MathUtil::ProjectClippedRect(source_to_dest, clip_in_source_space); | 
|  | } else { | 
|  | return false; | 
|  | } | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | ConditionalClip ComputeTargetRectInLocalSpace( | 
|  | gfx::RectF rect, | 
|  | const PropertyTrees* property_trees, | 
|  | int target_transform_id, | 
|  | int local_transform_id, | 
|  | const int target_effect_id) { | 
|  | gfx::Transform target_to_local; | 
|  | bool success = property_trees->GetFromTarget( | 
|  | local_transform_id, target_effect_id, &target_to_local); | 
|  | // If transform is not invertible, cannot apply clip. | 
|  | if (!success) | 
|  | return ConditionalClip{false, gfx::RectF()}; | 
|  |  | 
|  | if (target_transform_id > local_transform_id) | 
|  | return ConditionalClip{true,  // is_clipped. | 
|  | MathUtil::MapClippedRect(target_to_local, rect)}; | 
|  |  | 
|  | return ConditionalClip{true,  // is_clipped. | 
|  | MathUtil::ProjectClippedRect(target_to_local, rect)}; | 
|  | } | 
|  |  | 
|  | ConditionalClip ComputeLocalRectInTargetSpace( | 
|  | gfx::RectF rect, | 
|  | const PropertyTrees* property_trees, | 
|  | int current_transform_id, | 
|  | int target_transform_id, | 
|  | int target_effect_id) { | 
|  | gfx::Transform current_to_target; | 
|  | if (!property_trees->GetToTarget(current_transform_id, target_effect_id, | 
|  | ¤t_to_target)) { | 
|  | // If transform is not invertible, cannot apply clip. | 
|  | return ConditionalClip{false, gfx::RectF()}; | 
|  | } | 
|  |  | 
|  | if (current_transform_id > target_transform_id) | 
|  | return ConditionalClip{true,  // is_clipped. | 
|  | MathUtil::MapClippedRect(current_to_target, rect)}; | 
|  |  | 
|  | return ConditionalClip{true,  // is_clipped. | 
|  | MathUtil::ProjectClippedRect(current_to_target, rect)}; | 
|  | } | 
|  |  | 
|  | ConditionalClip ComputeCurrentClip(const ClipNode* clip_node, | 
|  | const PropertyTrees* property_trees, | 
|  | int target_transform_id, | 
|  | int target_effect_id) { | 
|  | if (clip_node->transform_id != target_transform_id) { | 
|  | return ComputeLocalRectInTargetSpace(clip_node->clip, property_trees, | 
|  | clip_node->transform_id, | 
|  | target_transform_id, target_effect_id); | 
|  | } | 
|  |  | 
|  | const EffectTree& effect_tree = property_trees->effect_tree(); | 
|  | gfx::RectF current_clip = clip_node->clip; | 
|  | gfx::Vector2dF surface_contents_scale = | 
|  | effect_tree.Node(target_effect_id)->surface_contents_scale; | 
|  | // The viewport clip should not be scaled. | 
|  | if (surface_contents_scale.x() > 0 && surface_contents_scale.y() > 0 && | 
|  | clip_node->transform_id != kRootPropertyNodeId) | 
|  | current_clip.Scale(surface_contents_scale.x(), surface_contents_scale.y()); | 
|  | return ConditionalClip{true /* is_clipped */, current_clip}; | 
|  | } | 
|  |  | 
|  | bool ExpandClipForPixelMovingFilter(const PropertyTrees* property_trees, | 
|  | int target_id, | 
|  | const EffectNode* filter_node, | 
|  | gfx::RectF* clip_rect) { | 
|  | // Bring the accumulated clip to the space of the pixel-moving filter. | 
|  | gfx::RectF clip_rect_in_mapping_space; | 
|  | bool success = ConvertRectBetweenSurfaceSpaces(property_trees, target_id, | 
|  | filter_node->id, *clip_rect, | 
|  | &clip_rect_in_mapping_space); | 
|  | // If transform is not invertible, no clip will be applied. | 
|  | if (!success) | 
|  | return false; | 
|  |  | 
|  | // Do the expansion. | 
|  | SkMatrix filter_draw_matrix = | 
|  | SkMatrix::Scale(filter_node->surface_contents_scale.x(), | 
|  | filter_node->surface_contents_scale.y()); | 
|  | gfx::RectF mapped_clip_in_mapping_space(filter_node->filters.ExpandRect( | 
|  | ToEnclosingClipRect(clip_rect_in_mapping_space), filter_draw_matrix)); | 
|  |  | 
|  | // Put the expanded clip back into the original target space. | 
|  | gfx::RectF original_clip_rect = *clip_rect; | 
|  | success = ConvertRectBetweenSurfaceSpaces( | 
|  | property_trees, filter_node->id, target_id, mapped_clip_in_mapping_space, | 
|  | clip_rect); | 
|  | // If transform is not invertible, no clip will be applied. | 
|  | if (!success) | 
|  | return false; | 
|  |  | 
|  | // Ensure the clip is expanded in the target space, in case that the | 
|  | // mapped accumulated_clip doesn't contain the original. | 
|  | clip_rect->Union(original_clip_rect); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool ApplyClipNodeToAccumulatedClip(const PropertyTrees* property_trees, | 
|  | bool include_expanding_clips, | 
|  | int target_id, | 
|  | int target_transform_id, | 
|  | const ClipNode* clip_node, | 
|  | gfx::RectF* accumulated_clip) { | 
|  | if (!clip_node->AppliesLocalClip()) { | 
|  | if (!include_expanding_clips) | 
|  | return true; | 
|  | const EffectNode* filter_node = | 
|  | property_trees->effect_tree().Node(clip_node->pixel_moving_filter_id); | 
|  | DCHECK(filter_node); | 
|  | return ExpandClipForPixelMovingFilter(property_trees, target_id, | 
|  | filter_node, accumulated_clip); | 
|  | } | 
|  |  | 
|  | ConditionalClip current_clip = ComputeCurrentClip( | 
|  | clip_node, property_trees, target_transform_id, target_id); | 
|  |  | 
|  | // If transform is not invertible, no clip will be applied. | 
|  | if (!current_clip.is_clipped) | 
|  | return false; | 
|  |  | 
|  | accumulated_clip->Intersect(current_clip.clip_rect); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | ConditionalClip ComputeAccumulatedClip(PropertyTrees* property_trees, | 
|  | bool include_expanding_clips, | 
|  | int local_clip_id, | 
|  | int target_id) { | 
|  | ClipRectData* cached_data = | 
|  | property_trees->FetchClipRectFromCache(local_clip_id, target_id); | 
|  | if (cached_data->target_id != kInvalidPropertyNodeId) { | 
|  | // Cache hit | 
|  | return cached_data->clip; | 
|  | } | 
|  | cached_data->target_id = target_id; | 
|  |  | 
|  | const ClipTree& clip_tree = property_trees->clip_tree(); | 
|  | const ClipNode* clip_node = clip_tree.Node(local_clip_id); | 
|  | const EffectTree& effect_tree = property_trees->effect_tree(); | 
|  | const EffectNode* target_node = effect_tree.Node(target_id); | 
|  | int target_transform_id = target_node->transform_id; | 
|  |  | 
|  | bool cache_hit = false; | 
|  | ConditionalClip cached_clip = ConditionalClip{false, gfx::RectF()}; | 
|  | ConditionalClip unclipped = ConditionalClip{false, gfx::RectF()}; | 
|  |  | 
|  | // Collect all the clips that need to be accumulated. | 
|  | base::stack<const ClipNode*, std::vector<const ClipNode*>> parent_chain; | 
|  |  | 
|  | // If target is not direct ancestor of clip, this will find least common | 
|  | // ancestor between the target and the clip. Or, if the target has a | 
|  | // contributing layer that escapes clip, this will find the nearest ancestor | 
|  | // that doesn't. | 
|  | while (target_node->clip_id > clip_node->id || | 
|  | property_trees->effect_tree() | 
|  | .GetRenderSurface(target_node->id) | 
|  | ->has_contributing_layer_that_escapes_clip()) { | 
|  | target_node = effect_tree.Node(target_node->target_id); | 
|  | } | 
|  |  | 
|  | // Collect clip nodes up to the least common ancestor or till we get a cache | 
|  | // hit. | 
|  | while (target_node->clip_id < clip_node->id) { | 
|  | if (parent_chain.size() > 0) { | 
|  | // Search the cache. | 
|  | for (size_t i = 0; i < clip_node->cached_clip_rects.size(); ++i) { | 
|  | auto& data = clip_node->cached_clip_rects[i]; | 
|  | if (data.target_id == target_id) { | 
|  | cache_hit = true; | 
|  | cached_clip = data.clip; | 
|  | } | 
|  | } | 
|  | } | 
|  | parent_chain.push(clip_node); | 
|  | clip_node = clip_tree.parent(clip_node); | 
|  | } | 
|  |  | 
|  | if (parent_chain.size() == 0) { | 
|  | // No accumulated clip nodes. | 
|  | cached_data->clip = unclipped; | 
|  | return unclipped; | 
|  | } | 
|  |  | 
|  | clip_node = parent_chain.top(); | 
|  | parent_chain.pop(); | 
|  |  | 
|  | gfx::RectF accumulated_clip; | 
|  | if (cache_hit && cached_clip.is_clipped) { | 
|  | // Apply the first clip in parent_chain to the cached clip. | 
|  | accumulated_clip = cached_clip.clip_rect; | 
|  | bool success = ApplyClipNodeToAccumulatedClip( | 
|  | property_trees, include_expanding_clips, target_id, target_transform_id, | 
|  | clip_node, &accumulated_clip); | 
|  | if (!success) { | 
|  | // Singular transform | 
|  | cached_data->clip = unclipped; | 
|  | return unclipped; | 
|  | } | 
|  | } else { | 
|  | // No cache hit or the cached clip has no clip to apply. We need to find | 
|  | // the first clip that applies clip as there is no clip to expand. | 
|  | while (!clip_node->AppliesLocalClip() && parent_chain.size() > 0) { | 
|  | clip_node = parent_chain.top(); | 
|  | parent_chain.pop(); | 
|  | } | 
|  |  | 
|  | if (!clip_node->AppliesLocalClip()) { | 
|  | // No clip to apply. | 
|  | cached_data->clip = unclipped; | 
|  | return unclipped; | 
|  | } | 
|  | ConditionalClip current_clip = ComputeCurrentClip( | 
|  | clip_node, property_trees, target_transform_id, target_id); | 
|  | if (!current_clip.is_clipped) { | 
|  | // Singular transform | 
|  | cached_data->clip = unclipped; | 
|  | return unclipped; | 
|  | } | 
|  | accumulated_clip = current_clip.clip_rect; | 
|  | } | 
|  |  | 
|  | // Apply remaining clips | 
|  | while (parent_chain.size() > 0) { | 
|  | clip_node = parent_chain.top(); | 
|  | parent_chain.pop(); | 
|  | bool success = ApplyClipNodeToAccumulatedClip( | 
|  | property_trees, include_expanding_clips, target_id, target_transform_id, | 
|  | clip_node, &accumulated_clip); | 
|  | if (!success) { | 
|  | // Singular transform | 
|  | cached_data->clip = unclipped; | 
|  | return unclipped; | 
|  | } | 
|  | } | 
|  |  | 
|  | ConditionalClip clip = ConditionalClip{ | 
|  | true /* is_clipped */, | 
|  | accumulated_clip.IsEmpty() ? gfx::RectF() : accumulated_clip}; | 
|  | cached_data->clip = clip; | 
|  | return clip; | 
|  | } | 
|  |  | 
|  | bool HasSingularTransform(int transform_tree_index, const TransformTree& tree) { | 
|  | const TransformNode* node = tree.Node(transform_tree_index); | 
|  | return !node->is_invertible || !node->ancestors_are_invertible; | 
|  | } | 
|  |  | 
|  | int LowestCommonAncestor(int clip_id_1, | 
|  | int clip_id_2, | 
|  | const ClipTree& clip_tree) { | 
|  | const ClipNode* clip_node_1 = clip_tree.Node(clip_id_1); | 
|  | const ClipNode* clip_node_2 = clip_tree.Node(clip_id_2); | 
|  | while (clip_node_1->id != clip_node_2->id) { | 
|  | if (clip_node_1->id < clip_node_2->id) { | 
|  | clip_node_2 = clip_tree.parent(clip_node_2); | 
|  | } else { | 
|  | clip_node_1 = clip_tree.parent(clip_node_1); | 
|  | } | 
|  | } | 
|  | return clip_node_1->id; | 
|  | } | 
|  |  | 
|  | void UpdateRenderSurfaceCommonAncestorClip(LayerImpl* layer, | 
|  | const ClipTree& clip_tree) { | 
|  | RenderSurfaceImpl* render_surface = layer->render_target(); | 
|  | CHECK(render_surface); | 
|  | int clip_id = layer->clip_tree_index(); | 
|  | // Find each ancestor targets whose clip is escaped by the layer's clip, and | 
|  | // set its common_ancestor_clip_id to be the lowest common ancestor of both | 
|  | // clips. | 
|  | while (clip_id < render_surface->common_ancestor_clip_id()) { | 
|  | clip_id = LowestCommonAncestor( | 
|  | clip_id, render_surface->common_ancestor_clip_id(), clip_tree); | 
|  | render_surface->set_common_ancestor_clip_id(clip_id); | 
|  | RenderSurfaceImpl* parent_render_surface = render_surface->render_target(); | 
|  | if (parent_render_surface == render_surface) { | 
|  | break; | 
|  | } | 
|  | render_surface = parent_render_surface; | 
|  | } | 
|  | } | 
|  |  | 
|  | void ClearRenderSurfaceCommonAncestorClip(LayerImpl* layer) { | 
|  | RenderSurfaceImpl* render_surface = layer->render_target(); | 
|  | CHECK(render_surface); | 
|  | while (render_surface->has_contributing_layer_that_escapes_clip()) { | 
|  | render_surface->set_common_ancestor_clip_id(kInvalidPropertyNodeId); | 
|  | RenderSurfaceImpl* parent_render_surface = render_surface->render_target(); | 
|  | if (parent_render_surface == render_surface) { | 
|  | break; | 
|  | } | 
|  | render_surface = parent_render_surface; | 
|  | } | 
|  | } | 
|  |  | 
|  | template <typename LayerType> | 
|  | const TransformNode& TransformNodeForBackfaceVisibility( | 
|  | LayerType* layer, | 
|  | const TransformTree& tree) { | 
|  | const TransformNode* node = tree.Node(layer->transform_tree_index()); | 
|  | while (node->delegates_to_parent_for_backface) { | 
|  | const TransformNode* parent = tree.Node(node->parent_id); | 
|  | CHECK(node); | 
|  | // Backface visibility inheritance should not cross 3d sorting contexts. | 
|  | DCHECK(!node->sorting_context_id || | 
|  | parent->sorting_context_id == node->sorting_context_id); | 
|  | node = parent; | 
|  | } | 
|  | return *node; | 
|  | } | 
|  |  | 
|  | bool IsTargetSpaceTransformBackFaceVisible( | 
|  | const LayerImpl* layer, | 
|  | const PropertyTrees* property_trees) { | 
|  | const TransformNode& transform_node = TransformNodeForBackfaceVisibility( | 
|  | layer, property_trees->transform_tree()); | 
|  | gfx::Transform to_target; | 
|  | property_trees->GetToTarget( | 
|  | transform_node.id, layer->render_target_effect_tree_index(), &to_target); | 
|  |  | 
|  | return to_target.IsBackFaceVisible(); | 
|  | } | 
|  |  | 
|  | bool IsTransformToRootOf3DRenderingContextBackFaceVisible( | 
|  | const LayerImpl* layer, | 
|  | const PropertyTrees* property_trees) { | 
|  | const TransformTree& transform_tree = property_trees->transform_tree(); | 
|  |  | 
|  | const TransformNode& transform_node = | 
|  | TransformNodeForBackfaceVisibility(layer, transform_tree); | 
|  | const TransformNode* root_node = &transform_node; | 
|  |  | 
|  | int root_id = transform_node.id; | 
|  | int sorting_context_id = transform_node.sorting_context_id; | 
|  |  | 
|  | while (root_id > kRootPropertyNodeId) { | 
|  | int parent_id = root_node->parent_id; | 
|  | const TransformNode* parent_node = transform_tree.Node(parent_id); | 
|  | if (parent_node->sorting_context_id != sorting_context_id) | 
|  | break; | 
|  | root_id = parent_id; | 
|  | root_node = parent_node; | 
|  | } | 
|  |  | 
|  | // TODO(chrishtr): cache this on the transform trees if needed, similar to | 
|  | // |to_target| and |to_screen|. | 
|  | gfx::Transform to_3d_root; | 
|  | if (transform_node.id != root_id) { | 
|  | property_trees->transform_tree().CombineTransformsBetween( | 
|  | transform_node.id, root_id, &to_3d_root); | 
|  | } | 
|  | to_3d_root.PreConcat(root_node->to_parent); | 
|  | return to_3d_root.IsBackFaceVisible(); | 
|  | } | 
|  |  | 
|  | bool IsLayerBackFaceVisible(const Layer* layer, | 
|  | const PropertyTrees* property_trees) { | 
|  | // We do not skip back face invisible layers on main thread as target space | 
|  | // transform will not be available here. | 
|  | return false; | 
|  | } | 
|  |  | 
|  | bool IsLayerBackFaceVisible(const LayerImpl* layer, | 
|  | const PropertyTrees* property_trees) { | 
|  | // A layer with singular transform is not drawn. So, we can assume that its | 
|  | // backface is not visible. | 
|  | if (HasSingularTransform(layer->transform_tree_index(), | 
|  | property_trees->transform_tree())) { | 
|  | return false; | 
|  | } | 
|  | if (layer->layer_tree_impl()->settings().enable_backface_visibility_interop) { | 
|  | return IsTransformToRootOf3DRenderingContextBackFaceVisible(layer, | 
|  | property_trees); | 
|  | } else { | 
|  | return IsTargetSpaceTransformBackFaceVisible(layer, property_trees); | 
|  | } | 
|  | } | 
|  |  | 
|  | template <typename LayerType> | 
|  | bool LayerNeedsUpdate(LayerType* layer, | 
|  | bool layer_is_drawn, | 
|  | const PropertyTrees* property_trees) { | 
|  | // Layers can be skipped if any of these conditions are met. | 
|  | //   - is not drawn due to it or one of its ancestors being hidden (or having | 
|  | //     no copy requests). | 
|  | //   - does not draw content. | 
|  | //   - is transparent. | 
|  | //   - has empty bounds | 
|  | //   - the layer is not double-sided, but its back face is visible. | 
|  | // | 
|  | // Some additional conditions need to be computed at a later point after the | 
|  | // recursion is finished. | 
|  | //   - the intersection of render_surface content and layer clip_rect is empty | 
|  | //   - the visible_layer_rect is empty | 
|  | // | 
|  | // Note, if the layer should not have been drawn due to being fully | 
|  | // transparent, we would have skipped the entire subtree and never made it | 
|  | // into this function, so it is safe to omit this check here. | 
|  | if (!layer_is_drawn) | 
|  | return false; | 
|  |  | 
|  | if (!layer->draws_content()) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | if (layer->bounds().IsEmpty()) { | 
|  | // Reference filters can contribute to visual output even if the layer has | 
|  | // no other content. | 
|  | if (!property_trees->effect_tree() | 
|  | .Node(layer->effect_tree_index()) | 
|  | ->filters.HasReferenceFilter()) { | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | // The layer should not be drawn if (1) it is not double-sided and (2) the | 
|  | // back of the layer is known to be facing the screen. | 
|  | if (layer->should_check_backface_visibility() && | 
|  | IsLayerBackFaceVisible(layer, property_trees)) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | inline bool LayerShouldBeSkippedForDrawPropertiesComputation( | 
|  | Layer* layer, | 
|  | const TransformTree& transform_tree, | 
|  | const EffectTree& effect_tree) { | 
|  | const EffectNode* effect_node = effect_tree.Node(layer->effect_tree_index()); | 
|  | if (effect_node->HasRenderSurface() && effect_node->subtree_has_copy_request) | 
|  | return false; | 
|  |  | 
|  | // If the layer transform is not invertible, it should be skipped. In case the | 
|  | // transform is animating and singular, we should not skip it. | 
|  | const TransformNode* transform_node = | 
|  | transform_tree.Node(layer->transform_tree_index()); | 
|  | return !transform_node->node_and_ancestors_are_animated_or_invertible || | 
|  | !effect_node->is_drawn; | 
|  | } | 
|  |  | 
|  | gfx::Rect LayerDrawableContentRect( | 
|  | const LayerImpl* layer, | 
|  | const gfx::Rect& layer_bounds_in_target_space, | 
|  | const gfx::Rect& clip_rect) { | 
|  | if (layer->is_clipped()) | 
|  | return IntersectRects(layer_bounds_in_target_space, clip_rect); | 
|  |  | 
|  | return layer_bounds_in_target_space; | 
|  | } | 
|  |  | 
|  | void SetSurfaceIsClipped(const ClipTree& clip_tree, | 
|  | RenderSurfaceImpl* render_surface) { | 
|  | bool is_clipped = false; | 
|  | if (render_surface->EffectTreeIndex() == kContentsRootPropertyNodeId) { | 
|  | // Root render surface is always clipped. | 
|  | is_clipped = true; | 
|  | } else { | 
|  | int parent_target_clip_id = | 
|  | render_surface->render_target()->common_ancestor_clip_id(); | 
|  | for (const ClipNode* clip_node = | 
|  | clip_tree.Node(render_surface->common_ancestor_clip_id()); | 
|  | clip_node && clip_node->id != parent_target_clip_id; | 
|  | clip_node = clip_tree.parent(clip_node)) { | 
|  | if (clip_node->AppliesLocalClip()) { | 
|  | is_clipped = true; | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  | render_surface->SetIsClipped(is_clipped); | 
|  | } | 
|  |  | 
|  | void SetSurfaceDrawOpacity(const EffectTree& tree, | 
|  | RenderSurfaceImpl* render_surface) { | 
|  | // Draw opacity of a surface is the product of opacities between the surface | 
|  | // (included) and its target surface (excluded). | 
|  | const EffectNode* node = tree.Node(render_surface->EffectTreeIndex()); | 
|  | float draw_opacity = tree.EffectiveOpacity(node); | 
|  | for (node = tree.parent(node); node && !node->HasRenderSurface(); | 
|  | node = tree.parent(node)) { | 
|  | draw_opacity *= tree.EffectiveOpacity(node); | 
|  | } | 
|  | render_surface->SetDrawOpacity(draw_opacity); | 
|  | } | 
|  |  | 
|  | float LayerDrawOpacity(const LayerImpl* layer, const EffectTree& tree) { | 
|  | if (!layer->render_target()) | 
|  | return 0.f; | 
|  |  | 
|  | const EffectNode* target_node = | 
|  | tree.Node(layer->render_target()->EffectTreeIndex()); | 
|  | const EffectNode* node = tree.Node(layer->effect_tree_index()); | 
|  | if (node == target_node) | 
|  | return 1.f; | 
|  |  | 
|  | float draw_opacity = 1.f; | 
|  | while (node != target_node) { | 
|  | draw_opacity *= tree.EffectiveOpacity(node); | 
|  | node = tree.parent(node); | 
|  | } | 
|  | return draw_opacity; | 
|  | } | 
|  |  | 
|  | template <typename LayerType> | 
|  | gfx::Transform ScreenSpaceTransformInternal(LayerType* layer, | 
|  | const TransformTree& tree) { | 
|  | gfx::Transform xform = | 
|  | gfx::Transform::MakeTranslation(layer->offset_to_transform_parent().x(), | 
|  | layer->offset_to_transform_parent().y()); | 
|  | gfx::Transform ssxform = tree.ToScreen(layer->transform_tree_index()); | 
|  | xform.PostConcat(ssxform); | 
|  | return xform; | 
|  | } | 
|  |  | 
|  | void SetSurfaceClipRect(PropertyTrees* property_trees, | 
|  | RenderSurfaceImpl* render_surface) { | 
|  | if (!render_surface->is_clipped()) { | 
|  | render_surface->SetClipRect(gfx::Rect()); | 
|  | return; | 
|  | } | 
|  |  | 
|  | const EffectTree& effect_tree = property_trees->effect_tree(); | 
|  | const ClipTree& clip_tree = property_trees->clip_tree(); | 
|  | const EffectNode* effect_node = | 
|  | effect_tree.Node(render_surface->EffectTreeIndex()); | 
|  | const EffectNode* target_node = effect_tree.Node(effect_node->target_id); | 
|  | bool include_expanding_clips = false; | 
|  | if (render_surface->EffectTreeIndex() == kContentsRootPropertyNodeId) { | 
|  | render_surface->SetClipRect( | 
|  | ToEnclosingClipRect(clip_tree.Node(effect_node->clip_id)->clip)); | 
|  | } else { | 
|  | ConditionalClip accumulated_clip_rect = ComputeAccumulatedClip( | 
|  | property_trees, include_expanding_clips, | 
|  | render_surface->common_ancestor_clip_id(), target_node->id); | 
|  | accumulated_clip_rect.clip_rect.Offset( | 
|  | render_surface->render_target()->pixel_alignment_offset()); | 
|  | render_surface->SetClipRect( | 
|  | ToEnclosingClipRect(accumulated_clip_rect.clip_rect)); | 
|  | } | 
|  | } | 
|  |  | 
|  | void SetSurfaceDrawTransform(const PropertyTrees* property_trees, | 
|  | RenderSurfaceImpl* render_surface) { | 
|  | const TransformTree& transform_tree = property_trees->transform_tree(); | 
|  | const EffectTree& effect_tree = property_trees->effect_tree(); | 
|  | const TransformNode* transform_node = | 
|  | transform_tree.Node(render_surface->TransformTreeIndex()); | 
|  | const EffectNode* effect_node = | 
|  | effect_tree.Node(render_surface->EffectTreeIndex()); | 
|  | // The draw transform of root render surface is identity transform. | 
|  | if (render_surface->EffectTreeIndex() == kContentsRootPropertyNodeId) { | 
|  | render_surface->SetDrawTransform(gfx::Transform(), gfx::Vector2dF()); | 
|  | return; | 
|  | } | 
|  |  | 
|  | gfx::Transform render_surface_transform; | 
|  | const EffectNode* target_effect_node = | 
|  | effect_tree.Node(effect_node->target_id); | 
|  | property_trees->GetToTarget(transform_node->id, target_effect_node->id, | 
|  | &render_surface_transform); | 
|  |  | 
|  | ConcatInverseSurfaceContentsScale(effect_node, &render_surface_transform); | 
|  |  | 
|  | gfx::Vector2dF pixel_alignment_offset; | 
|  | // Adjust render_surface_transform by applying the render target's pixel | 
|  | // alignment before the transform, and de-applying this render surface's | 
|  | // pixel alignment to align it to screen pixels. | 
|  | render_surface_transform.PostTranslate( | 
|  | render_surface->render_target()->pixel_alignment_offset()); | 
|  | if (effect_node->render_surface_reason != | 
|  | RenderSurfaceReason::k2DScaleTransformWithCompositedDescendants && | 
|  | (base::FeatureList::IsEnabled(features::kViewTransitionFloorTransform) || | 
|  | !effect_node->view_transition_element_resource_id.IsValid())) { | 
|  | if (auto offset = draw_property_utils::PixelAlignmentOffset( | 
|  | render_surface->screen_space_transform(), | 
|  | render_surface_transform)) { | 
|  | pixel_alignment_offset = *offset; | 
|  | render_surface_transform.Translate(-pixel_alignment_offset); | 
|  | } | 
|  | } | 
|  | render_surface->SetDrawTransform(render_surface_transform, | 
|  | pixel_alignment_offset); | 
|  | } | 
|  |  | 
|  | gfx::Rect LayerVisibleRect(PropertyTrees* property_trees, LayerImpl* layer) { | 
|  | const EffectNode* effect_node = | 
|  | property_trees->effect_tree().Node(layer->effect_tree_index()); | 
|  | int lower_effect_closest_ancestor = | 
|  | effect_node->closest_ancestor_with_cached_render_surface_id; | 
|  | lower_effect_closest_ancestor = | 
|  | std::max(lower_effect_closest_ancestor, | 
|  | effect_node->closest_ancestor_with_copy_request_id); | 
|  | lower_effect_closest_ancestor = | 
|  | std::max(lower_effect_closest_ancestor, | 
|  | effect_node->closest_ancestor_being_captured_id); | 
|  | lower_effect_closest_ancestor = | 
|  | std::max(lower_effect_closest_ancestor, | 
|  | effect_node->closest_ancestor_with_shared_element_id); | 
|  | const bool non_root_with_render_surface = | 
|  | lower_effect_closest_ancestor > kContentsRootPropertyNodeId; | 
|  | gfx::Rect layer_content_rect = gfx::Rect(layer->bounds()); | 
|  |  | 
|  | gfx::RectF accumulated_clip_in_root_space; | 
|  | if (non_root_with_render_surface) { | 
|  | bool include_expanding_clips = true; | 
|  | ConditionalClip accumulated_clip = ComputeAccumulatedClip( | 
|  | property_trees, include_expanding_clips, layer->clip_tree_index(), | 
|  | lower_effect_closest_ancestor); | 
|  | if (!accumulated_clip.is_clipped) | 
|  | return layer_content_rect; | 
|  | accumulated_clip_in_root_space = accumulated_clip.clip_rect; | 
|  | } else { | 
|  | const ClipNode* clip_node = | 
|  | property_trees->clip_tree().Node(layer->clip_tree_index()); | 
|  | accumulated_clip_in_root_space = | 
|  | clip_node->cached_accumulated_rect_in_screen_space; | 
|  | } | 
|  |  | 
|  | const EffectNode* root_effect_node = | 
|  | non_root_with_render_surface | 
|  | ? property_trees->effect_tree().Node(lower_effect_closest_ancestor) | 
|  | : property_trees->effect_tree().Node(kContentsRootPropertyNodeId); | 
|  | ConditionalClip accumulated_clip_in_layer_space = | 
|  | ComputeTargetRectInLocalSpace( | 
|  | accumulated_clip_in_root_space, property_trees, | 
|  | root_effect_node->transform_id, layer->transform_tree_index(), | 
|  | root_effect_node->id); | 
|  | if (!accumulated_clip_in_layer_space.is_clipped) { | 
|  | return layer_content_rect; | 
|  | } | 
|  | gfx::RectF clip_in_layer_space = accumulated_clip_in_layer_space.clip_rect; | 
|  | clip_in_layer_space.Offset(-layer->offset_to_transform_parent()); | 
|  |  | 
|  | gfx::Rect visible_rect = ToEnclosingClipRect(clip_in_layer_space); | 
|  | visible_rect.Intersect(layer_content_rect); | 
|  | return visible_rect; | 
|  | } | 
|  |  | 
|  | ConditionalClip LayerClipRect(PropertyTrees* property_trees, LayerImpl* layer) { | 
|  | const EffectTree* effect_tree = &property_trees->effect_tree(); | 
|  | const EffectNode* effect_node = effect_tree->Node(layer->effect_tree_index()); | 
|  | const EffectNode* target_node = | 
|  | effect_node->HasRenderSurface() | 
|  | ? effect_node | 
|  | : effect_tree->Node(effect_node->target_id); | 
|  | bool include_expanding_clips = false; | 
|  | return ComputeAccumulatedClip(property_trees, include_expanding_clips, | 
|  | layer->clip_tree_index(), target_node->id); | 
|  | } | 
|  |  | 
|  | std::pair<gfx::MaskFilterInfo, bool> GetMaskFilterInfoPair( | 
|  | const PropertyTrees* property_trees, | 
|  | int effect_tree_index, | 
|  | bool for_render_surface) { | 
|  | static const std::pair<gfx::MaskFilterInfo, bool> kEmptyMaskFilterInfoPair = | 
|  | std::make_pair(gfx::MaskFilterInfo(), false); | 
|  |  | 
|  | const EffectTree* effect_tree = &property_trees->effect_tree(); | 
|  | const EffectNode* effect_node = effect_tree->Node(effect_tree_index); | 
|  | const int target_id = effect_node->target_id; | 
|  |  | 
|  | // Return empty mask info if this node has a render surface but the function | 
|  | // call was made for a non render surface. | 
|  | if (effect_node->HasRenderSurface() && !for_render_surface) | 
|  | return kEmptyMaskFilterInfoPair; | 
|  |  | 
|  | // Traverse the parent chain up to the render target to find a node which has | 
|  | // mask filter info set. | 
|  | const EffectNode* node = effect_node; | 
|  | bool found_mask_filter_info = false; | 
|  |  | 
|  | while (node) { | 
|  | found_mask_filter_info = !node->mask_filter_info.IsEmpty(); | 
|  | if (found_mask_filter_info) | 
|  | break; | 
|  |  | 
|  | // If the iteration has reached a node in the parent chain that has a render | 
|  | // surface, then break. If this iteration is for a render surface to begin | 
|  | // with, then ensure |node| is a parent of |effect_node|. | 
|  | if (node->HasRenderSurface() && | 
|  | (!for_render_surface || effect_node != node)) { | 
|  | break; | 
|  | } | 
|  |  | 
|  | // Simply break if we reached a node that is the render target. | 
|  | if (node->id == target_id) | 
|  | break; | 
|  |  | 
|  | node = effect_tree->parent(node); | 
|  | } | 
|  |  | 
|  | // While traversing up the parent chain we did not find any node with mask | 
|  | // filter info. | 
|  | if (!node || !found_mask_filter_info) | 
|  | return kEmptyMaskFilterInfoPair; | 
|  |  | 
|  | int transform_id = node->transform_id; | 
|  | std::optional<int> clip_id = node->mask_filter_info.clip_id(); | 
|  | if (clip_id) { | 
|  | const ClipTree* clip_tree = &property_trees->clip_tree(); | 
|  | const ClipNode* clip_node = clip_tree->Node(clip_id.value()); | 
|  | transform_id = clip_node->transform_id; | 
|  | } | 
|  |  | 
|  | gfx::Transform to_target; | 
|  | if (!property_trees->GetToTarget(transform_id, target_id, &to_target)) { | 
|  | return kEmptyMaskFilterInfoPair; | 
|  | } | 
|  |  | 
|  | auto result = | 
|  | std::make_pair(node->mask_filter_info, node->is_fast_rounded_corner); | 
|  | result.first.ApplyTransform(to_target); | 
|  |  | 
|  | if (result.first.IsEmpty()) { | 
|  | return kEmptyMaskFilterInfoPair; | 
|  | } | 
|  | return result; | 
|  | } | 
|  |  | 
|  | void UpdateRenderTarget(LayerTreeImpl* layer_tree_impl, | 
|  | EffectTree* effect_tree) { | 
|  | int last_backdrop_filter_disallowing_lcd_text = kInvalidNodeId; | 
|  | base::flat_map<viz::ViewTransitionElementResourceId, int> resource_to_node; | 
|  |  | 
|  | for (int i = kContentsRootPropertyNodeId; | 
|  | i < static_cast<int>(effect_tree->size()); ++i) { | 
|  | EffectNode* node = effect_tree->Node(i); | 
|  |  | 
|  | if (node->view_transition_element_resource_id.IsValid()) { | 
|  | CHECK(!resource_to_node.contains( | 
|  | node->view_transition_element_resource_id)); | 
|  | resource_to_node[node->view_transition_element_resource_id] = i; | 
|  | } | 
|  | node->view_transition_target_id = kInvalidPropertyNodeId; | 
|  |  | 
|  | if (i == kContentsRootPropertyNodeId) { | 
|  | // Render target of the node corresponding to root is itself. | 
|  | node->target_id = kContentsRootPropertyNodeId; | 
|  | } else if (effect_tree->parent(node)->HasRenderSurface()) { | 
|  | node->target_id = node->parent_id; | 
|  | } else { | 
|  | node->target_id = effect_tree->parent(node)->target_id; | 
|  | } | 
|  | if (node->has_potential_backdrop_filter_animation || | 
|  | !node->backdrop_filters.AllowsLCDText()) { | 
|  | last_backdrop_filter_disallowing_lcd_text = node->id; | 
|  | } | 
|  | node->lcd_text_disallowed_by_backdrop_filter = false; | 
|  | } | 
|  |  | 
|  | if (!resource_to_node.empty()) { | 
|  | for (auto* layer : *layer_tree_impl) { | 
|  | auto resource_id = layer->ViewTransitionResourceId(); | 
|  | if (!resource_id.IsValid()) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | auto it = resource_to_node.find(resource_id); | 
|  | if (it == resource_to_node.end()) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | auto* resource_node = effect_tree->Node(it->second); | 
|  | auto* layer_node = effect_tree->Node(layer->effect_tree_index()); | 
|  | if (layer_node->HasRenderSurface()) { | 
|  | resource_node->view_transition_target_id = layer_node->id; | 
|  | } else { | 
|  | resource_node->view_transition_target_id = layer_node->target_id; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | if (last_backdrop_filter_disallowing_lcd_text == kInvalidNodeId) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | // Update effect nodes for the backdrop filter disallowing LCD text. | 
|  | int current_target_id = | 
|  | effect_tree->Node(last_backdrop_filter_disallowing_lcd_text)->target_id; | 
|  | for (int i = last_backdrop_filter_disallowing_lcd_text - 1; | 
|  | i >= kContentsRootPropertyNodeId; --i) { | 
|  | EffectNode* node = effect_tree->Node(i); | 
|  | node->lcd_text_disallowed_by_backdrop_filter = current_target_id <= i; | 
|  | if (node->id == current_target_id) | 
|  | current_target_id = kInvalidNodeId; | 
|  | // While down to kContentsRootNodeId, move |current_target_id| forward if | 
|  | // |node| has backdrop filter. | 
|  | if (current_target_id == kInvalidNodeId && | 
|  | (node->has_potential_backdrop_filter_animation || | 
|  | !node->backdrop_filters.AllowsLCDText())) { | 
|  | current_target_id = node->target_id; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void ComputeClips(PropertyTrees* property_trees) { | 
|  | DCHECK(!property_trees->transform_tree().needs_update()); | 
|  | ClipTree* clip_tree = &property_trees->clip_tree_mutable(); | 
|  | if (!clip_tree->needs_update()) | 
|  | return; | 
|  | const int target_effect_id = kContentsRootPropertyNodeId; | 
|  | const int target_transform_id = kRootPropertyNodeId; | 
|  | const bool include_expanding_clips = true; | 
|  | for (int i = kViewportPropertyNodeId; i < static_cast<int>(clip_tree->size()); | 
|  | ++i) { | 
|  | ClipNode* clip_node = clip_tree->Node(i); | 
|  | // Clear the clip rect cache | 
|  | clip_node->cached_clip_rects.clear(); | 
|  | if (clip_node->id == kViewportPropertyNodeId) { | 
|  | clip_node->cached_accumulated_rect_in_screen_space = clip_node->clip; | 
|  | continue; | 
|  | } | 
|  | ClipNode* parent_clip_node = clip_tree->parent(clip_node); | 
|  | DCHECK(parent_clip_node); | 
|  | gfx::RectF accumulated_clip = | 
|  | parent_clip_node->cached_accumulated_rect_in_screen_space; | 
|  | bool success = ApplyClipNodeToAccumulatedClip( | 
|  | property_trees, include_expanding_clips, target_effect_id, | 
|  | target_transform_id, clip_node, &accumulated_clip); | 
|  | if (success) | 
|  | clip_node->cached_accumulated_rect_in_screen_space = accumulated_clip; | 
|  | } | 
|  | clip_tree->set_needs_update(false); | 
|  | } | 
|  |  | 
|  | void ComputeSurfaceDrawProperties(PropertyTrees* property_trees, | 
|  | RenderSurfaceImpl* render_surface) { | 
|  | SetSurfaceIsClipped(property_trees->clip_tree(), render_surface); | 
|  | SetSurfaceDrawOpacity(property_trees->effect_tree(), render_surface); | 
|  | render_surface->SetScreenSpaceTransform( | 
|  | property_trees->ToScreenSpaceTransformWithoutSurfaceContentsScale( | 
|  | render_surface->TransformTreeIndex(), | 
|  | render_surface->EffectTreeIndex())); | 
|  | SetSurfaceDrawTransform(property_trees, render_surface); | 
|  |  | 
|  | auto mask_filter_info_pair = | 
|  | GetMaskFilterInfoPair(property_trees, render_surface->EffectTreeIndex(), | 
|  | /*for_render_surface=*/true); | 
|  | mask_filter_info_pair.first.ApplyTransform(gfx::AxisTransform2d( | 
|  | 1.0f, render_surface->render_target()->pixel_alignment_offset())); | 
|  | render_surface->SetMaskFilterInfo( | 
|  | /*mask_filter_info=*/mask_filter_info_pair.first, | 
|  | /*is_fast_rounded_corner=*/mask_filter_info_pair.second); | 
|  |  | 
|  | SetSurfaceClipRect(property_trees, render_surface); | 
|  | } | 
|  |  | 
|  | void AddSurfaceToRenderSurfaceList( | 
|  | RenderSurfaceImpl* render_surface, | 
|  | RenderSurfaceList* render_surface_list, | 
|  | PropertyTrees* property_trees, | 
|  | const base::flat_set<blink::ViewTransitionToken>& | 
|  | capture_view_transition_tokens, | 
|  | std::vector<RenderSurfaceImpl*>& view_transition_capture_surfaces) { | 
|  | // |render_surface| must appear after its target, so first make sure its | 
|  | // target is in the list. | 
|  | RenderSurfaceImpl* target = render_surface->render_target(); | 
|  | bool is_root = | 
|  | render_surface->EffectTreeIndex() == kContentsRootPropertyNodeId; | 
|  |  | 
|  | // If this node is producing a snapshot for a ViewTransition then the target | 
|  | // surface (where its surface will be drawn) corresponds to the node where the | 
|  | // ViewTransitionContentLayer rendering this snapshot is drawn. | 
|  | if (render_surface->OwningEffectNode()->view_transition_target_id != | 
|  | kInvalidPropertyNodeId) { | 
|  | CHECK(!is_root); | 
|  | CHECK(render_surface->OwningEffectNode() | 
|  | ->view_transition_element_resource_id.IsValid()); | 
|  |  | 
|  | auto* vt_target = property_trees->effect_tree_mutable().GetRenderSurface( | 
|  | render_surface->OwningEffectNode()->view_transition_target_id); | 
|  | CHECK(vt_target); | 
|  |  | 
|  | if (!vt_target->is_render_surface_list_member()) { | 
|  | AddSurfaceToRenderSurfaceList( | 
|  | vt_target, render_surface_list, property_trees, | 
|  | capture_view_transition_tokens, view_transition_capture_surfaces); | 
|  | } | 
|  | } | 
|  |  | 
|  | // TODO(vmpstr): revisit this later to see if there is a better fix, as this | 
|  | // changes the render list in an incorrect way due to CC assumptions about how | 
|  | // surfaces and view-transition captures interact. | 
|  | if (!is_root && !target->is_render_surface_list_member()) { | 
|  | AddSurfaceToRenderSurfaceList(target, render_surface_list, property_trees, | 
|  | capture_view_transition_tokens, | 
|  | view_transition_capture_surfaces); | 
|  | } | 
|  | render_surface->ClearAccumulatedContentRect(); | 
|  | render_surface_list->push_back(render_surface); | 
|  | render_surface->set_is_render_surface_list_member(true); | 
|  | if (render_surface->ViewTransitionElementResourceId().MatchesToken( | 
|  | capture_view_transition_tokens)) { | 
|  | // The capture surface itself has contributions. | 
|  | render_surface->set_has_view_transition_capture_contributions(true); | 
|  | view_transition_capture_surfaces.push_back(render_surface); | 
|  | } | 
|  | if (is_root) { | 
|  | // The root surface does not contribute to any other surface, it has no | 
|  | // target. | 
|  | render_surface->set_contributes_to_drawn_surface(false); | 
|  | } else { | 
|  | bool contributes_to_drawn_surface = | 
|  | property_trees->effect_tree().ContributesToDrawnSurface( | 
|  | render_surface->EffectTreeIndex()); | 
|  | render_surface->set_contributes_to_drawn_surface( | 
|  | contributes_to_drawn_surface); | 
|  | } | 
|  |  | 
|  | ComputeSurfaceDrawProperties(property_trees, render_surface); | 
|  |  | 
|  | // Ignore occlusion from outside the surface when surface contents need to be | 
|  | // fully drawn. Layers with copy-request need to be complete.  We could be | 
|  | // smarter about layers with filters that move pixels and exclude regions | 
|  | // where both layers and the filters are occluded, but this seems like | 
|  | // overkill. | 
|  | // | 
|  | // When kAllowUndamagedNonrootRenderPassToSkip is enabled, in order for the | 
|  | // render_surface to be skipped without triggering a redraw for the changes on | 
|  | // occlusion_from_outside_target, surface contents need to befully drawn. | 
|  | // DamageTracker |has_damage_from_contributing_content_| only track occlusion | 
|  | // inside its own render_surface. Therefore we set |is_occlusion_immune| to | 
|  | // avoid the need for redraw. | 
|  | // | 
|  | // TODO(senorblanco): make this smarter for the SkImageFilter case (check for | 
|  | // pixel-moving filters) | 
|  | const bool allow_skipping_render_pass = base::FeatureList::IsEnabled( | 
|  | features::kAllowUndamagedNonrootRenderPassToSkip); | 
|  | const FilterOperations& filters = render_surface->Filters(); | 
|  | bool is_occlusion_immune = render_surface->CopyOfOutputRequired() || | 
|  | filters.HasFilterThatMovesPixels() || | 
|  | allow_skipping_render_pass; | 
|  |  | 
|  | // Setting |is_occlusion_immune| leads to an empty | 
|  | // |occlusion_from_outside_target| for a non-root render_surface. It does not | 
|  | // affect |occlusion_from_inside_target|. |occlusion_from_outside_target| is | 
|  | // always empty for the root render_surface because there is no other | 
|  | // render_surface on top to occlude the root. | 
|  | if (is_occlusion_immune) { | 
|  | render_surface->SetNearestOcclusionImmuneAncestor(render_surface); | 
|  | } else if (is_root) { | 
|  | render_surface->SetNearestOcclusionImmuneAncestor(nullptr); | 
|  | } else { | 
|  | render_surface->SetNearestOcclusionImmuneAncestor( | 
|  | render_surface->render_target()->nearest_occlusion_immune_ancestor()); | 
|  | } | 
|  | } | 
|  |  | 
|  | bool SkipForInvertibility(const LayerImpl* layer, | 
|  | PropertyTrees* property_trees) { | 
|  | const TransformNode* transform_node = | 
|  | property_trees->transform_tree().Node(layer->transform_tree_index()); | 
|  | const EffectNode* effect_node = | 
|  | property_trees->effect_tree().Node(layer->effect_tree_index()); | 
|  | bool non_root_copy_request = | 
|  | effect_node->closest_ancestor_with_copy_request_id > | 
|  | kContentsRootPropertyNodeId; | 
|  | gfx::Transform from_target; | 
|  | // If there is a copy request, we check the invertibility of the transform | 
|  | // between the node corresponding to the layer and the node corresponding to | 
|  | // the copy request. Otherwise, we are interested in the invertibility of | 
|  | // screen space transform which is already cached on the transform node. | 
|  | return non_root_copy_request | 
|  | ? !property_trees->GetFromTarget( | 
|  | layer->transform_tree_index(), | 
|  | effect_node->closest_ancestor_with_copy_request_id, | 
|  | &from_target) | 
|  | : !transform_node->ancestors_are_invertible; | 
|  | } | 
|  |  | 
|  | void ComputeLayerDrawTransforms(LayerImpl* layer, | 
|  | const PropertyTrees* property_trees) { | 
|  | const TransformNode* transform_node = | 
|  | property_trees->transform_tree().Node(layer->transform_tree_index()); | 
|  | layer->draw_properties().screen_space_transform = | 
|  | ScreenSpaceTransformInternal(layer, property_trees->transform_tree()); | 
|  | layer->draw_properties().target_space_transform = DrawTransform( | 
|  | layer, property_trees->transform_tree(), property_trees->effect_tree()); | 
|  | layer->draw_properties().screen_space_transform_is_animating = | 
|  | transform_node->to_screen_is_potentially_animated; | 
|  | auto mask_filter_info_pair = | 
|  | GetMaskFilterInfoPair(property_trees, layer->effect_tree_index(), | 
|  | /*for_render_surface=*/false); | 
|  | layer->draw_properties().mask_filter_info = mask_filter_info_pair.first; | 
|  | layer->draw_properties().is_fast_rounded_corner = | 
|  | mask_filter_info_pair.second; | 
|  | } | 
|  |  | 
|  | void ComputeLayerDrawOpacity(LayerImpl* layer, | 
|  | const PropertyTrees* property_trees) { | 
|  | layer->draw_properties().opacity = | 
|  | LayerDrawOpacity(layer, property_trees->effect_tree()); | 
|  | } | 
|  |  | 
|  | void ComputeLayerClipAndVisibleRect(LayerImpl* layer, | 
|  | PropertyTrees* property_trees) { | 
|  | ConditionalClip clip = LayerClipRect(property_trees, layer); | 
|  | // is_clipped should be set before visible rect computation as it is used | 
|  | // there. | 
|  | layer->draw_properties().is_clipped = clip.is_clipped; | 
|  | layer->draw_properties().clip_rect = ToEnclosingClipRect(clip.clip_rect); | 
|  | layer->draw_properties().visible_layer_rect = | 
|  | LayerVisibleRect(property_trees, layer); | 
|  | } | 
|  |  | 
|  | void ComputeLayerDrawableContentRect(LayerImpl* layer, | 
|  | const PropertyTrees* property_trees) { | 
|  | bool only_draws_visible_content = property_trees->effect_tree() | 
|  | .Node(layer->effect_tree_index()) | 
|  | ->only_draws_visible_content; | 
|  | gfx::Rect drawable_bounds = gfx::Rect(layer->visible_layer_rect()); | 
|  | if (!only_draws_visible_content) { | 
|  | drawable_bounds = gfx::Rect(layer->bounds()); | 
|  | } | 
|  | gfx::Rect visible_bounds_in_target_space = MathUtil::MapEnclosingClippedRect( | 
|  | layer->draw_properties().target_space_transform, drawable_bounds); | 
|  | layer->draw_properties().visible_drawable_content_rect = | 
|  | LayerDrawableContentRect(layer, visible_bounds_in_target_space, | 
|  | layer->draw_properties().clip_rect); | 
|  | } | 
|  |  | 
|  | void AdjustLayerDrawPropertiesForPixelAlignmentOffset( | 
|  | LayerImpl* layer, | 
|  | PropertyTrees* property_trees) { | 
|  | gfx::Vector2dF offset = layer->render_target()->pixel_alignment_offset(); | 
|  | if (offset.IsZero()) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | // Apply the pixel alignment offset to all draw properties that are relative | 
|  | // to the render target's space. | 
|  | layer->draw_properties().target_space_transform.PostTranslate(offset); | 
|  | layer->draw_properties().mask_filter_info.ApplyTransform( | 
|  | gfx::AxisTransform2d(1.0f, offset)); | 
|  |  | 
|  | // clip_rect and visible_drawable_content_rect are enclosing int rects. | 
|  | // We can't simply apply the pixel alignment to them because that would call | 
|  | // ToEnclosingRect() again and the result might be 1px bigger than expected. | 
|  | // That would be especially bad for clip_rect because that would expose | 
|  | // pixels out of the clip. Instead of adjustment, recompute them with the | 
|  | // adjusted target_space_transform. | 
|  | if (layer->draw_properties().is_clipped) { | 
|  | ConditionalClip clip = LayerClipRect(property_trees, layer); | 
|  | CHECK(clip.is_clipped); | 
|  | clip.clip_rect.Offset(offset); | 
|  | layer->draw_properties().clip_rect = ToEnclosingClipRect(clip.clip_rect); | 
|  | } | 
|  | // The adjusted target_space_transform will apply. | 
|  | ComputeLayerDrawableContentRect(layer, property_trees); | 
|  | } | 
|  |  | 
|  | void ComputeInitialRenderSurfaceList( | 
|  | LayerTreeImpl* layer_tree_impl, | 
|  | PropertyTrees* property_trees, | 
|  | const base::flat_set<blink::ViewTransitionToken>& | 
|  | capture_view_transition_tokens, | 
|  | RenderSurfaceList* render_surface_list, | 
|  | std::map<viz::ViewTransitionElementResourceId, | 
|  | ViewTransitionContentLayerImpl*>& view_transition_content_layers) { | 
|  | EffectTree& effect_tree = property_trees->effect_tree_mutable(); | 
|  | for (int i = kContentsRootPropertyNodeId; | 
|  | i < static_cast<int>(effect_tree.size()); ++i) { | 
|  | if (RenderSurfaceImpl* render_surface = effect_tree.GetRenderSurface(i)) { | 
|  | render_surface->set_is_render_surface_list_member(false); | 
|  | render_surface->set_has_view_transition_capture_contributions(false); | 
|  | render_surface->reset_num_contributors(); | 
|  | } | 
|  | } | 
|  |  | 
|  | std::vector<RenderSurfaceImpl*> view_transition_capture_surfaces; | 
|  |  | 
|  | RenderSurfaceImpl* root_surface = | 
|  | effect_tree.GetRenderSurface(kContentsRootPropertyNodeId); | 
|  | // The root surface always gets added to the render surface  list. | 
|  | AddSurfaceToRenderSurfaceList(root_surface, render_surface_list, | 
|  | property_trees, capture_view_transition_tokens, | 
|  | view_transition_capture_surfaces); | 
|  |  | 
|  | // For all non-skipped layers, add their target to the render surface list if | 
|  | // it's not already been added, and add their content rect to the target | 
|  | // surface's accumulated content rect. | 
|  | for (LayerImpl* layer : *layer_tree_impl) { | 
|  | DCHECK(layer); | 
|  | layer->EnsureValidPropertyTreeIndices(); | 
|  |  | 
|  | layer->set_contributes_to_drawn_render_surface(false); | 
|  | layer->set_raster_even_if_not_drawn(false); | 
|  |  | 
|  | bool is_root = layer_tree_impl->IsRootLayer(layer); | 
|  |  | 
|  | bool skip_draw_properties_computation = | 
|  | draw_property_utils::LayerShouldBeSkippedForDrawPropertiesComputation( | 
|  | layer, property_trees); | 
|  |  | 
|  | bool skip_for_invertibility = SkipForInvertibility(layer, property_trees); | 
|  |  | 
|  | bool skip_layer = !is_root && (skip_draw_properties_computation || | 
|  | skip_for_invertibility); | 
|  |  | 
|  | TransformNode* transform_node = | 
|  | property_trees->transform_tree_mutable().Node( | 
|  | layer->transform_tree_index()); | 
|  | const bool has_will_change_transform_hint = | 
|  | transform_node && transform_node->will_change_transform; | 
|  | // Raster layers that are animated but currently have a non-invertible | 
|  | // matrix, or layers that have a will-change transform hint and might | 
|  | // animate to not be backface visible soon. | 
|  | layer->set_raster_even_if_not_drawn( | 
|  | (skip_for_invertibility && !skip_draw_properties_computation) || | 
|  | has_will_change_transform_hint); | 
|  | if (skip_layer) | 
|  | continue; | 
|  |  | 
|  | bool layer_is_drawn = property_trees->effect_tree() | 
|  | .Node(layer->effect_tree_index()) | 
|  | ->is_drawn; | 
|  | bool layer_should_be_drawn = | 
|  | LayerNeedsUpdate(layer, layer_is_drawn, property_trees); | 
|  | if (!layer_should_be_drawn) | 
|  | continue; | 
|  |  | 
|  | RenderSurfaceImpl* render_target = layer->render_target(); | 
|  | if (!render_target->is_render_surface_list_member()) { | 
|  | AddSurfaceToRenderSurfaceList( | 
|  | render_target, render_surface_list, property_trees, | 
|  | capture_view_transition_tokens, view_transition_capture_surfaces); | 
|  | } | 
|  |  | 
|  | AdjustLayerDrawPropertiesForPixelAlignmentOffset(layer, property_trees); | 
|  | layer->set_contributes_to_drawn_render_surface(true); | 
|  |  | 
|  | // The layer contributes its drawable content rect to its render target. | 
|  | render_target->AccumulateContentRectFromContributingLayer( | 
|  | layer, capture_view_transition_tokens); | 
|  | render_target->increment_num_contributors(); | 
|  | if (!layer->ViewTransitionResourceId().IsValid() || | 
|  | layer->ViewTransitionResourceId().MatchesToken( | 
|  | capture_view_transition_tokens)) { | 
|  | continue; | 
|  | } | 
|  | auto* view_transition_content_layer = | 
|  | static_cast<ViewTransitionContentLayerImpl*>(layer); | 
|  | view_transition_content_layer->SetOriginatingSurfaceContentRect( | 
|  | gfx::Rect()); | 
|  | view_transition_content_layers[layer->ViewTransitionResourceId()] = | 
|  | view_transition_content_layer; | 
|  | } | 
|  |  | 
|  | // Mark any intermediate surfaces as having view transition contributions. | 
|  | for (RenderSurfaceImpl* render_surface : view_transition_capture_surfaces) { | 
|  | CHECK(render_surface->IsViewTransitionElement()); | 
|  | CHECK(render_surface->has_view_transition_capture_contributions()); | 
|  | // Find an ancestor that is also a view transition capture element. | 
|  | RenderSurfaceImpl* view_transition_target = render_surface->render_target(); | 
|  | while ( | 
|  | view_transition_target != root_surface && | 
|  | !view_transition_target->has_view_transition_capture_contributions()) { | 
|  | view_transition_target = view_transition_target->render_target(); | 
|  | } | 
|  |  | 
|  | // If we found a `view_transition_target` which has view transition capture | 
|  | // contributions, then mark the chain between render_surface and the | 
|  | // `view_transition_target` as having contributions. | 
|  | if (view_transition_target->has_view_transition_capture_contributions()) { | 
|  | RenderSurfaceImpl* intermediate_target = render_surface->render_target(); | 
|  | while (intermediate_target != view_transition_target) { | 
|  | intermediate_target->set_has_view_transition_capture_contributions( | 
|  | true); | 
|  | intermediate_target = intermediate_target->render_target(); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void ComputeSurfaceContentRects( | 
|  | const base::flat_set<blink::ViewTransitionToken>& | 
|  | capture_view_transition_tokens, | 
|  | PropertyTrees* property_trees, | 
|  | RenderSurfaceList* render_surface_list, | 
|  | int max_texture_size, | 
|  | const std::map<viz::ViewTransitionElementResourceId, | 
|  | ViewTransitionContentLayerImpl*> | 
|  | view_transition_content_layers, | 
|  | std::vector<std::pair<viz::ViewTransitionElementResourceId, gfx::RectF>>& | 
|  | view_transition_content_rects) { | 
|  | // Walk the list backwards, accumulating each surface's content rect into its | 
|  | // target's content rect. | 
|  | for (RenderSurfaceImpl* render_surface : | 
|  | base::Reversed(*render_surface_list)) { | 
|  | if (render_surface->EffectTreeIndex() == kContentsRootPropertyNodeId) { | 
|  | // The root surface's content rect is always the entire viewport. | 
|  | render_surface->SetContentRectToViewport(); | 
|  | continue; | 
|  | } | 
|  |  | 
|  | // Now all contributing drawable content rect has been accumulated to this | 
|  | // render surface, calculate the content rect. | 
|  | render_surface->CalculateContentRectFromAccumulatedContentRect( | 
|  | max_texture_size); | 
|  |  | 
|  | // Now the render surface's content rect is calculated correctly, it could | 
|  | // contribute to its render target. | 
|  | RenderSurfaceImpl* render_target = render_surface->render_target(); | 
|  | DCHECK(render_target->is_render_surface_list_member()); | 
|  | render_target->AccumulateContentRectFromContributingRenderSurface( | 
|  | render_surface, capture_view_transition_tokens); | 
|  | render_target->increment_num_contributors(); | 
|  |  | 
|  | // Collect the content rects for view transition capture surfaces, and | 
|  | // adjust the geometry for the corresponding content layer (the | 
|  | // pseudo-element's layer). | 
|  | const auto& view_transition_id = | 
|  | render_surface->OwningEffectNode()->view_transition_element_resource_id; | 
|  |  | 
|  | if (!view_transition_id.IsValid()) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | // We use the view_transition_capture_content_rect if we're in the capture | 
|  | // phase and content_rect otherwise, which is important for live captures. | 
|  | const auto& output_rect = | 
|  | render_surface->ViewTransitionElementResourceId().MatchesToken( | 
|  | capture_view_transition_tokens) | 
|  | ? render_surface->view_transition_capture_content_rect() | 
|  | : render_surface->content_rect(); | 
|  | auto unscaled_output_rect = | 
|  | render_surface->SurfaceScale().InverseOrIdentity().MapRect(output_rect); | 
|  |  | 
|  | view_transition_content_rects.emplace_back( | 
|  | view_transition_id, gfx::RectF(unscaled_output_rect)); | 
|  |  | 
|  | if (view_transition_content_layers.contains(view_transition_id)) { | 
|  | view_transition_content_layers.at(view_transition_id) | 
|  | ->SetOriginatingSurfaceContentRect(unscaled_output_rect); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void ComputeListOfNonEmptySurfaces(LayerTreeImpl* layer_tree_impl, | 
|  | PropertyTrees* property_trees, | 
|  | RenderSurfaceList* initial_surface_list, | 
|  | RenderSurfaceList* final_surface_list) { | 
|  | // Walk the initial surface list forwards. The root surface and each | 
|  | // surface with a non-empty content rect go into the final render surface | 
|  | // layer list. Surfaces with empty content rects or whose target isn't in | 
|  | // the final list do not get added to the final list. | 
|  | // Surface which require copy of output (view transition captures) are exempt | 
|  | // because their contents are required regardless of the state of the target | 
|  | // surface. | 
|  | std::unordered_set<RenderSurfaceImpl*> surfaces_to_remove; | 
|  | for (RenderSurfaceImpl* surface : *initial_surface_list) { | 
|  | bool is_root = surface->EffectTreeIndex() == kContentsRootPropertyNodeId; | 
|  | RenderSurfaceImpl* target_surface = surface->render_target(); | 
|  | if (!is_root && ((surface->content_rect().IsEmpty() && | 
|  | !surface->Filters().HasReferenceFilter()) || | 
|  | (!target_surface->is_render_surface_list_member() && | 
|  | !surface->CopyOfOutputRequired()))) { | 
|  | surface->set_is_render_surface_list_member(false); | 
|  | surfaces_to_remove.insert(surface); | 
|  | continue; | 
|  | } | 
|  |  | 
|  | // If one of the child surfaces had a CopyOfOutputRequired, it may be the | 
|  | // reason we're adding it. However, we may have skipped its target surface | 
|  | // already, so undelete the chain of all target surfaces for any surface | 
|  | // that we're adding. | 
|  | for (auto* target_to_undelete = target_surface; | 
|  | surfaces_to_remove.count(target_to_undelete); | 
|  | target_to_undelete = target_to_undelete->render_target()) { | 
|  | surface->set_is_render_surface_list_member(true); | 
|  | final_surface_list->push_back(target_to_undelete); | 
|  | surfaces_to_remove.erase(target_to_undelete); | 
|  | } | 
|  | final_surface_list->push_back(surface); | 
|  | } | 
|  |  | 
|  | for (auto* surface : surfaces_to_remove) { | 
|  | RenderSurfaceImpl* target_surface = surface->render_target(); | 
|  | target_surface->decrement_num_contributors(); | 
|  | } | 
|  |  | 
|  | if (!surfaces_to_remove.empty()) { | 
|  | for (LayerImpl* layer : *layer_tree_impl) { | 
|  | if (layer->contributes_to_drawn_render_surface()) { | 
|  | RenderSurfaceImpl* render_target = layer->render_target(); | 
|  | if (!render_target->is_render_surface_list_member()) { | 
|  | layer->set_contributes_to_drawn_render_surface(false); | 
|  | render_target->decrement_num_contributors(); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void CalculateRenderSurfaceLayerList(LayerTreeImpl* layer_tree_impl, | 
|  | PropertyTrees* property_trees, | 
|  | RenderSurfaceList* render_surface_list, | 
|  | const int max_texture_size) { | 
|  | RenderSurfaceList initial_render_surface_list; | 
|  | std::vector<std::pair<viz::ViewTransitionElementResourceId, gfx::RectF>> | 
|  | view_transition_content_rects; | 
|  | std::map<viz::ViewTransitionElementResourceId, | 
|  | ViewTransitionContentLayerImpl*> | 
|  | view_transition_content_layers; | 
|  |  | 
|  | // First compute a list that might include surfaces that later turn out to | 
|  | // have an empty content rect. After surface content rects are computed, | 
|  | // produce a final list that omits empty surfaces. | 
|  | const auto& capture_view_transition_tokens = | 
|  | layer_tree_impl->GetCaptureViewTransitionTokens(); | 
|  | ComputeInitialRenderSurfaceList( | 
|  | layer_tree_impl, property_trees, capture_view_transition_tokens, | 
|  | &initial_render_surface_list, view_transition_content_layers); | 
|  | ComputeSurfaceContentRects(capture_view_transition_tokens, property_trees, | 
|  | &initial_render_surface_list, max_texture_size, | 
|  | view_transition_content_layers, | 
|  | view_transition_content_rects); | 
|  | ComputeListOfNonEmptySurfaces(layer_tree_impl, property_trees, | 
|  | &initial_render_surface_list, | 
|  | render_surface_list); | 
|  |  | 
|  | for (const auto& [id, rect] : view_transition_content_rects) { | 
|  | layer_tree_impl->SetViewTransitionContentRect(id, rect); | 
|  | } | 
|  | } | 
|  |  | 
|  | void RecordRenderSurfaceReasonsForTracing( | 
|  | const PropertyTrees* property_trees, | 
|  | const RenderSurfaceList* render_surface_list) { | 
|  | static const auto* tracing_enabled = | 
|  | TRACE_EVENT_API_GET_CATEGORY_GROUP_ENABLED("cc"); | 
|  | if (!*tracing_enabled || | 
|  | // Don't output single root render surface. | 
|  | render_surface_list->size() <= 1) | 
|  | return; | 
|  |  | 
|  | TRACE_EVENT_INSTANT1("cc", "RenderSurfaceReasonCount", | 
|  | TRACE_EVENT_SCOPE_THREAD, "total", | 
|  | render_surface_list->size()); | 
|  |  | 
|  | // kTest is the last value which is not included for tracing. | 
|  | constexpr auto kNumReasons = static_cast<size_t>(RenderSurfaceReason::kTest); | 
|  | std::array<int, kNumReasons> reason_counts = {0}; | 
|  | for (const RenderSurfaceImpl* render_surface : *render_surface_list) { | 
|  | const auto* effect_node = | 
|  | property_trees->effect_tree().Node(render_surface->EffectTreeIndex()); | 
|  | reason_counts[static_cast<size_t>(effect_node->render_surface_reason)]++; | 
|  | } | 
|  | for (size_t i = 0; i < kNumReasons; i++) { | 
|  | if (!reason_counts[i]) | 
|  | continue; | 
|  | TRACE_EVENT_INSTANT1( | 
|  | "cc", "RenderSurfaceReasonCount", TRACE_EVENT_SCOPE_THREAD, | 
|  | RenderSurfaceReasonToString(static_cast<RenderSurfaceReason>(i)), | 
|  | reason_counts[i]); | 
|  | } | 
|  | } | 
|  |  | 
|  | void UpdateElasticOverscroll( | 
|  | PropertyTrees* property_trees, | 
|  | TransformNode* overscroll_elasticity_transform_node, | 
|  | const gfx::Vector2dF& elastic_overscroll, | 
|  | const ScrollNode* inner_viewport) { | 
|  | if (!overscroll_elasticity_transform_node) { | 
|  | DCHECK(elastic_overscroll.IsZero()); | 
|  | return; | 
|  | } | 
|  | #if BUILDFLAG(IS_ANDROID) | 
|  | if (inner_viewport && property_trees->scroll_tree() | 
|  | .container_bounds(inner_viewport->id) | 
|  | .IsEmpty()) { | 
|  | // Avoid divide by 0. Animation should not be visible for an empty viewport | 
|  | // anyway. | 
|  | return; | 
|  | } | 
|  |  | 
|  | // On android, elastic overscroll is implemented by stretching the content | 
|  | // from the overscrolled edge by applying a stretch transform | 
|  | overscroll_elasticity_transform_node->local.MakeIdentity(); | 
|  | overscroll_elasticity_transform_node->origin.SetPoint(0.f, 0.f, 0.f); | 
|  | overscroll_elasticity_transform_node->has_potential_animation = | 
|  | !elastic_overscroll.IsZero(); | 
|  |  | 
|  | if (!elastic_overscroll.IsZero() && inner_viewport) { | 
|  | // The inner viewport container size takes into account the size change as a | 
|  | // result of the top controls, see ScrollTree::container_bounds. | 
|  | gfx::Size scroller_size = | 
|  | property_trees->scroll_tree().container_bounds(inner_viewport->id); | 
|  |  | 
|  | overscroll_elasticity_transform_node->local.Scale( | 
|  | 1.f + std::abs(elastic_overscroll.x()) / scroller_size.width(), | 
|  | 1.f + std::abs(elastic_overscroll.y()) / scroller_size.height()); | 
|  |  | 
|  | // If overscrolling to the right, stretch from right. | 
|  | if (elastic_overscroll.x() > 0.f) { | 
|  | overscroll_elasticity_transform_node->origin.set_x(scroller_size.width()); | 
|  | } | 
|  |  | 
|  | // If overscrolling off the bottom, stretch from bottom. | 
|  | if (elastic_overscroll.y() > 0.f) { | 
|  | overscroll_elasticity_transform_node->origin.set_y( | 
|  | scroller_size.height()); | 
|  | } | 
|  | } | 
|  | overscroll_elasticity_transform_node->needs_local_transform_update = true; | 
|  | property_trees->transform_tree_mutable().set_needs_update(true); | 
|  |  | 
|  | #else  // BUILDFLAG(IS_ANDROID) | 
|  |  | 
|  | // On other platforms, we modify the translation offset to match the | 
|  | // overscroll amount. | 
|  | gfx::PointF overscroll_offset = | 
|  | gfx::PointAtOffsetFromOrigin(elastic_overscroll); | 
|  | if (overscroll_elasticity_transform_node->scroll_offset() == | 
|  | overscroll_offset) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | overscroll_elasticity_transform_node->SetScrollOffset( | 
|  | overscroll_offset, DamageReason::kUntracked); | 
|  |  | 
|  | overscroll_elasticity_transform_node->needs_local_transform_update = true; | 
|  | property_trees->transform_tree_mutable().set_needs_update(true); | 
|  |  | 
|  | #endif  // BUILDFLAG(IS_ANDROID) | 
|  | } | 
|  |  | 
|  | void ComputeDrawPropertiesOfVisibleLayers(const LayerImplList* layer_list, | 
|  | PropertyTrees* property_trees) { | 
|  | for (LayerImpl* layer : *layer_list) { | 
|  | ComputeLayerDrawTransforms(layer, property_trees); | 
|  | ClearRenderSurfaceCommonAncestorClip(layer); | 
|  | } | 
|  | for (LayerImpl* layer : *layer_list) { | 
|  | ComputeLayerDrawOpacity(layer, property_trees); | 
|  | UpdateRenderSurfaceCommonAncestorClip(layer, property_trees->clip_tree()); | 
|  | } | 
|  | for (LayerImpl* layer : *layer_list) { | 
|  | ComputeLayerClipAndVisibleRect(layer, property_trees); | 
|  | } | 
|  | for (LayerImpl* layer : *layer_list) { | 
|  | ComputeLayerDrawableContentRect(layer, property_trees); | 
|  | } | 
|  | } | 
|  |  | 
|  | #if DCHECK_IS_ON() | 
|  | // See property_tree_builder.cc ComputeRenderSurfaceReason. | 
|  | bool NodeMayContainBackdropBlurFilter(const EffectNode& node) { | 
|  | switch (node.render_surface_reason) { | 
|  | case RenderSurfaceReason::kMask: | 
|  | case RenderSurfaceReason::kTrilinearFiltering: | 
|  | case RenderSurfaceReason::kFilter: | 
|  | case RenderSurfaceReason::kBackdropFilter: | 
|  | return true; | 
|  | default: | 
|  | return false; | 
|  | } | 
|  | } | 
|  | #endif | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | bool LayerShouldBeSkippedForDrawPropertiesComputation( | 
|  | LayerImpl* layer, | 
|  | const PropertyTrees* property_trees) { | 
|  | const TransformTree& transform_tree = property_trees->transform_tree(); | 
|  | const EffectTree& effect_tree = property_trees->effect_tree(); | 
|  | int effect_tree_index = layer->effect_tree_index(); | 
|  | CHECK(effect_tree_index != kInvalidPropertyNodeId); | 
|  | const EffectNode* effect_node = effect_tree.Node(effect_tree_index); | 
|  | // TODO(crbug.com/390906639): Remove after bug is fixed. | 
|  | if (!effect_node) { | 
|  | SCOPED_CRASH_KEY_STRING32("cc", "Effect tree index", | 
|  | base::NumberToString(effect_tree_index)); | 
|  | base::debug::DumpWithoutCrashing(); | 
|  | } | 
|  |  | 
|  | if (effect_node->HasRenderSurface() && effect_node->subtree_has_copy_request) | 
|  | return false; | 
|  |  | 
|  | // Skip if the node's subtree is hidden and no need to cache, or capture. | 
|  | if (effect_node->subtree_hidden && !effect_node->cache_render_surface && | 
|  | !effect_node->subtree_capture_id.is_valid()) { | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // If the layer transform is not invertible, it should be skipped. In case the | 
|  | // transform is animating and singular, we should not skip it. | 
|  | const TransformNode* transform_node = | 
|  | transform_tree.Node(layer->transform_tree_index()); | 
|  |  | 
|  | if (!transform_node->node_and_ancestors_are_animated_or_invertible || | 
|  | !effect_node->is_drawn) | 
|  | return true; | 
|  | if (layer->layer_tree_impl()->settings().enable_backface_visibility_interop) { | 
|  | return layer->should_check_backface_visibility() && | 
|  | IsLayerBackFaceVisible(layer, property_trees); | 
|  | } else { | 
|  | return effect_node->hidden_by_backface_visibility; | 
|  | } | 
|  | } | 
|  |  | 
|  | bool IsLayerBackFaceVisibleForTesting(const LayerImpl* layer,  // IN-TEST | 
|  | const PropertyTrees* property_trees) { | 
|  | return IsLayerBackFaceVisible(layer, property_trees); | 
|  | } | 
|  |  | 
|  | void ConcatInverseSurfaceContentsScale(const EffectNode* effect_node, | 
|  | gfx::Transform* transform) { | 
|  | DCHECK(effect_node->HasRenderSurface()); | 
|  | if (effect_node->surface_contents_scale.x() != 0.0 && | 
|  | effect_node->surface_contents_scale.y() != 0.0) | 
|  | transform->Scale(1.0 / effect_node->surface_contents_scale.x(), | 
|  | 1.0 / effect_node->surface_contents_scale.y()); | 
|  | } | 
|  |  | 
|  | void FindLayersThatNeedUpdates(LayerTreeHost* layer_tree_host, | 
|  | LayerList* update_layer_list) { | 
|  | const PropertyTrees* property_trees = layer_tree_host->property_trees(); | 
|  | const TransformTree& transform_tree = property_trees->transform_tree(); | 
|  | const EffectTree& effect_tree = property_trees->effect_tree(); | 
|  | for (auto* layer : *layer_tree_host) { | 
|  | if (!IsRootLayer(layer) && LayerShouldBeSkippedForDrawPropertiesComputation( | 
|  | layer, transform_tree, effect_tree)) | 
|  | continue; | 
|  |  | 
|  | bool layer_is_drawn = | 
|  | effect_tree.Node(layer->effect_tree_index())->is_drawn; | 
|  |  | 
|  | if (LayerNeedsUpdate(layer, layer_is_drawn, property_trees)) { | 
|  | update_layer_list->push_back(layer); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void FindLayersThatNeedUpdates(LayerTreeImpl* layer_tree_impl, | 
|  | std::vector<LayerImpl*>* visible_layer_list) { | 
|  | const PropertyTrees* property_trees = layer_tree_impl->property_trees(); | 
|  | const EffectTree& effect_tree = property_trees->effect_tree(); | 
|  |  | 
|  | for (auto* layer_impl : *layer_tree_impl) { | 
|  | DCHECK(layer_impl); | 
|  | DCHECK(layer_impl->layer_tree_impl()); | 
|  | layer_impl->EnsureValidPropertyTreeIndices(); | 
|  |  | 
|  | if (!IsRootLayer(layer_impl) && | 
|  | LayerShouldBeSkippedForDrawPropertiesComputation(layer_impl, | 
|  | property_trees)) | 
|  | continue; | 
|  |  | 
|  | bool layer_is_drawn = | 
|  | effect_tree.Node(layer_impl->effect_tree_index())->is_drawn; | 
|  |  | 
|  | if (LayerNeedsUpdate(layer_impl, layer_is_drawn, property_trees)) | 
|  | visible_layer_list->push_back(layer_impl); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ComputeTransforms(TransformTree* transform_tree, | 
|  | const ViewportPropertyIds& viewport_property_ids) { | 
|  | transform_tree->UpdateAllTransforms(viewport_property_ids); | 
|  | } | 
|  |  | 
|  | void ComputeEffects(EffectTree* effect_tree) { | 
|  | if (!effect_tree->needs_update()) | 
|  | return; | 
|  | for (int i = kContentsRootPropertyNodeId; | 
|  | i < static_cast<int>(effect_tree->size()); ++i) | 
|  | effect_tree->UpdateEffects(i); | 
|  | effect_tree->set_needs_update(false); | 
|  | } | 
|  |  | 
|  | void UpdatePropertyTrees(LayerTreeHost* layer_tree_host) { | 
|  | DCHECK(layer_tree_host); | 
|  | auto* property_trees = layer_tree_host->property_trees(); | 
|  | DCHECK(property_trees); | 
|  | if (property_trees->transform_tree().needs_update()) { | 
|  | property_trees->clip_tree_mutable().set_needs_update(true); | 
|  | property_trees->effect_tree_mutable().set_needs_update(true); | 
|  | } | 
|  |  | 
|  | ComputeTransforms(&property_trees->transform_tree_mutable(), | 
|  | layer_tree_host->viewport_property_ids()); | 
|  | ComputeEffects(&property_trees->effect_tree_mutable()); | 
|  | // Computation of clips uses ToScreen which is updated while computing | 
|  | // transforms. So, ComputeTransforms should be before ComputeClips. | 
|  | ComputeClips(property_trees); | 
|  | } | 
|  |  | 
|  | void UpdatePropertyTreesAndRenderSurfaces(LayerTreeImpl* layer_tree_impl, | 
|  | PropertyTrees* property_trees) { | 
|  | if (property_trees->transform_tree().needs_update()) { | 
|  | property_trees->clip_tree_mutable().set_needs_update(true); | 
|  | property_trees->effect_tree_mutable().set_needs_update(true); | 
|  | } | 
|  | UpdateRenderTarget(layer_tree_impl, &property_trees->effect_tree_mutable()); | 
|  |  | 
|  | ComputeTransforms(&property_trees->transform_tree_mutable(), | 
|  | layer_tree_impl->viewport_property_ids()); | 
|  | ComputeEffects(&property_trees->effect_tree_mutable()); | 
|  | // Computation of clips uses ToScreen which is updated while computing | 
|  | // transforms. So, ComputeTransforms should be before ComputeClips. | 
|  | ComputeClips(property_trees); | 
|  | } | 
|  |  | 
|  | gfx::Transform DrawTransform(const LayerImpl* layer, | 
|  | const TransformTree& transform_tree, | 
|  | const EffectTree& effect_tree) { | 
|  | // TransformTree::ToTarget computes transform between the layer's transform | 
|  | // node and surface's transform node and scales it by the surface's content | 
|  | // scale. | 
|  | gfx::Transform xform; | 
|  | transform_tree.property_trees()->GetToTarget( | 
|  | layer->transform_tree_index(), layer->render_target_effect_tree_index(), | 
|  | &xform); | 
|  | xform.Translate(layer->offset_to_transform_parent().x(), | 
|  | layer->offset_to_transform_parent().y()); | 
|  | return xform; | 
|  | } | 
|  |  | 
|  | gfx::Transform ScreenSpaceTransform(const Layer* layer, | 
|  | const TransformTree& tree) { | 
|  | return ScreenSpaceTransformInternal(layer, tree); | 
|  | } | 
|  |  | 
|  | gfx::Transform ScreenSpaceTransform(const LayerImpl* layer, | 
|  | const TransformTree& tree) { | 
|  | return ScreenSpaceTransformInternal(layer, tree); | 
|  | } | 
|  |  | 
|  | void UpdatePageScaleFactor(PropertyTrees* property_trees, | 
|  | TransformNode* page_scale_node, | 
|  | float page_scale_factor) { | 
|  | // TODO(wjmaclean): Once Issue #845097 is resolved, we can change the nullptr | 
|  | // check below to a DCHECK. | 
|  | if (property_trees->transform_tree().page_scale_factor() == | 
|  | page_scale_factor || | 
|  | !page_scale_node) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | property_trees->transform_tree_mutable().set_page_scale_factor( | 
|  | page_scale_factor); | 
|  |  | 
|  | page_scale_node->local.MakeIdentity(); | 
|  | page_scale_node->local.Scale(page_scale_factor, page_scale_factor); | 
|  |  | 
|  | page_scale_node->needs_local_transform_update = true; | 
|  | property_trees->transform_tree_mutable().set_needs_update(true); | 
|  | } | 
|  |  | 
|  | void CalculateDrawProperties( | 
|  | LayerTreeImpl* layer_tree_impl, | 
|  | RenderSurfaceList* output_render_surface_list, | 
|  | LayerImplList* output_update_layer_list_for_testing) { | 
|  | output_render_surface_list->clear(); | 
|  |  | 
|  | LayerImplList visible_layer_list; | 
|  | // Since page scale and elastic overscroll are SyncedProperties, changes | 
|  | // on the active tree immediately affect the pending tree, so instead of | 
|  | // trying to update property trees whenever these values change, we | 
|  | // update property trees before using them. | 
|  |  | 
|  | PropertyTrees* property_trees = layer_tree_impl->property_trees(); | 
|  | UpdatePageScaleFactor(property_trees, | 
|  | layer_tree_impl->PageScaleTransformNode(), | 
|  | layer_tree_impl->current_page_scale_factor()); | 
|  | const ScrollNode* scroll_node = layer_tree_impl->InnerViewportScrollNode(); | 
|  | gfx::Vector2dF elastic_overscroll; | 
|  | if (scroll_node) { | 
|  | elastic_overscroll = | 
|  | property_trees->scroll_tree().GetElasticOverscroll(*scroll_node); | 
|  | } | 
|  | UpdateElasticOverscroll(property_trees, | 
|  | layer_tree_impl->OverscrollElasticityTransformNode(), | 
|  | elastic_overscroll, scroll_node); | 
|  | // Similarly, the device viewport and device transform are shared | 
|  | // by both trees. | 
|  | property_trees->clip_tree_mutable().SetViewportClip( | 
|  | gfx::RectF(layer_tree_impl->GetDeviceViewport())); | 
|  | property_trees->transform_tree_mutable().SetRootScaleAndTransform( | 
|  | layer_tree_impl->device_scale_factor(), layer_tree_impl->DrawTransform()); | 
|  | UpdatePropertyTreesAndRenderSurfaces(layer_tree_impl, property_trees); | 
|  |  | 
|  | { | 
|  | TRACE_EVENT0("cc", "draw_property_utils::FindLayersThatNeedUpdates"); | 
|  | FindLayersThatNeedUpdates(layer_tree_impl, &visible_layer_list); | 
|  | } | 
|  |  | 
|  | { | 
|  | TRACE_EVENT1("cc", | 
|  | "draw_property_utils::ComputeDrawPropertiesOfVisibleLayers", | 
|  | "visible_layers", visible_layer_list.size()); | 
|  | ComputeDrawPropertiesOfVisibleLayers(&visible_layer_list, property_trees); | 
|  | } | 
|  |  | 
|  | { | 
|  | TRACE_EVENT0("cc", "CalculateRenderSurfaceLayerList"); | 
|  | CalculateRenderSurfaceLayerList(layer_tree_impl, property_trees, | 
|  | output_render_surface_list, | 
|  | layer_tree_impl->max_texture_size()); | 
|  | } | 
|  |  | 
|  | #if DCHECK_IS_ON() | 
|  | if (layer_tree_impl->settings().log_on_ui_double_background_blur) | 
|  | DCHECK( | 
|  | LogDoubleBackgroundBlur(*layer_tree_impl, *output_render_surface_list)); | 
|  | #endif | 
|  |  | 
|  | RecordRenderSurfaceReasonsForTracing(property_trees, | 
|  | output_render_surface_list); | 
|  |  | 
|  | // A root layer render_surface should always exist after | 
|  | // CalculateDrawProperties. | 
|  | DCHECK(property_trees->effect_tree().GetRenderSurface( | 
|  | kContentsRootPropertyNodeId)); | 
|  |  | 
|  | if (output_update_layer_list_for_testing) | 
|  | *output_update_layer_list_for_testing = std::move(visible_layer_list); | 
|  | } | 
|  |  | 
|  | bool RasterScalesApproximatelyEqual(gfx::Vector2dF scale1, | 
|  | gfx::Vector2dF scale2) { | 
|  | // Good match if the maximum alignment error on a layer of size 10000px does | 
|  | // not exceed 0.001px. | 
|  | static constexpr float kPixelErrorThreshold = 0.001f; | 
|  | static constexpr float kScaleErrorThreshold = kPixelErrorThreshold / 10000; | 
|  | gfx::Vector2dF scale_diff = scale1 - scale2; | 
|  | return std::abs(scale_diff.x()) <= kScaleErrorThreshold && | 
|  | std::abs(scale_diff.y()) <= kScaleErrorThreshold; | 
|  | } | 
|  |  | 
|  | std::optional<gfx::Vector2dF> PixelAlignmentOffset( | 
|  | const gfx::Transform& screen_space_transform, | 
|  | const gfx::Transform& target_space_transform) { | 
|  | if (!screen_space_transform.IsScaleOrTranslation() || | 
|  | !target_space_transform.IsScaleOrTranslation()) { | 
|  | return std::nullopt; | 
|  | } | 
|  |  | 
|  | // It is only useful to align the content space to the target space if their | 
|  | // relative pixel ratio is some simple rational number. Currently we only | 
|  | // align if the relative pixel ratio is 1:1. | 
|  | if (!RasterScalesApproximatelyEqual(screen_space_transform.To2dScale(), | 
|  | target_space_transform.To2dScale())) { | 
|  | return std::nullopt; | 
|  | } | 
|  |  | 
|  | auto subpixel = [](float f) -> float { return f - std::floor(f); }; | 
|  | gfx::Vector2dF screen_translation = screen_space_transform.To2dTranslation(); | 
|  | float screen_subpixel_x = subpixel(screen_translation.x()); | 
|  | float screen_subpixel_y = subpixel(screen_translation.y()); | 
|  | gfx::Vector2dF draw_translation = target_space_transform.To2dTranslation(); | 
|  | float draw_subpixel_x = subpixel(draw_translation.x()); | 
|  | float draw_subpixel_y = subpixel(draw_translation.y()); | 
|  | // If the origin is different in the screen space and in the target space, | 
|  | // it means the render target is not aligned to physical pixels, and the | 
|  | // text content will be blurry regardless of raster translation. | 
|  | static constexpr float kPixelErrorThreshold = 0.001f; | 
|  | if (std::abs(screen_subpixel_x - draw_subpixel_x) > kPixelErrorThreshold || | 
|  | std::abs(screen_subpixel_y - draw_subpixel_y) > kPixelErrorThreshold) { | 
|  | return std::nullopt; | 
|  | } | 
|  |  | 
|  | return gfx::Vector2dF(screen_subpixel_x, screen_subpixel_y); | 
|  | } | 
|  |  | 
|  | #if DCHECK_IS_ON() | 
|  | bool LogDoubleBackgroundBlur(const LayerTreeImpl& layer_tree_impl, | 
|  | const RenderSurfaceList& render_surface_list) { | 
|  | const PropertyTrees& property_trees = *layer_tree_impl.property_trees(); | 
|  | std::vector<std::pair<const LayerImpl*, gfx::Rect>> rects; | 
|  | rects.reserve(render_surface_list.size()); | 
|  |  | 
|  | for (const RenderSurfaceImpl* render_surface : render_surface_list) { | 
|  | const auto* effect_node = | 
|  | property_trees.effect_tree().Node(render_surface->EffectTreeIndex()); | 
|  | if (NodeMayContainBackdropBlurFilter(*effect_node)) { | 
|  | const FilterOperations& filters = render_surface->BackdropFilters(); | 
|  | if (filters.HasFilterOfType(FilterOperation::BLUR)) { | 
|  | if (!render_surface->content_rect().IsEmpty()) { | 
|  | const LayerImpl* layer_impl = | 
|  | layer_tree_impl.LayerByElementId(effect_node->element_id); | 
|  | gfx::Rect screen_space_rect = MathUtil::MapEnclosingClippedRect( | 
|  | render_surface->screen_space_transform(), | 
|  | render_surface->content_rect()); | 
|  | auto it = std::ranges::find_if( | 
|  | rects, [&screen_space_rect]( | 
|  | const std::pair<const LayerImpl*, gfx::Rect>& r) { | 
|  | return r.second.Intersects(screen_space_rect); | 
|  | }); | 
|  | if (rects.end() == it) { | 
|  | rects.push_back(std::make_pair(layer_impl, screen_space_rect)); | 
|  | } else { | 
|  | LOG(ERROR) << "Double blur detected between layers: " | 
|  | << it->first->DebugName() << " and " | 
|  | << layer_impl->DebugName(); | 
|  | return false; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | return true; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | }  // namespace draw_property_utils | 
|  |  | 
|  | }  // namespace cc |