blob: bac8f2435f8341a888fe62d78d6ead9eb1089c95 [file] [log] [blame]
[email protected]94f206c12012-08-25 00:09:141// Copyright 2011 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
[email protected]cc3cfaa2013-03-18 09:05:525#include "cc/layers/render_surface_impl.h"
[email protected]94f206c12012-08-25 00:09:146
avi02a4d172015-12-21 06:14:367#include <stddef.h>
8
[email protected]ac7c7f52012-11-08 06:26:509#include <algorithm>
10
[email protected]4456eee22012-10-19 18:16:3811#include "base/logging.h"
[email protected]8e61d4b2013-06-10 22:11:4812#include "base/strings/stringprintf.h"
[email protected]681ccff2013-03-18 06:13:5213#include "cc/base/math_util.h"
[email protected]6e84de22013-03-18 06:54:2714#include "cc/debug/debug_colors.h"
[email protected]cc3cfaa2013-03-18 09:05:5215#include "cc/layers/layer_impl.h"
[email protected]cc3cfaa2013-03-18 09:05:5216#include "cc/layers/render_pass_sink.h"
[email protected]89e8267a2013-03-18 07:50:5617#include "cc/quads/debug_border_draw_quad.h"
18#include "cc/quads/render_pass.h"
19#include "cc/quads/render_pass_draw_quad.h"
20#include "cc/quads/shared_quad_state.h"
[email protected]556fd292013-03-18 08:03:0421#include "cc/trees/damage_tracker.h"
ajumad9432e32015-11-30 19:43:4422#include "cc/trees/draw_property_utils.h"
23#include "cc/trees/layer_tree_impl.h"
danakja1b92e42015-02-11 21:07:4724#include "cc/trees/occlusion.h"
[email protected]55761e62012-11-21 18:55:5825#include "third_party/skia/include/core/SkImageFilter.h"
heejin.r.chungd28506ba2014-10-23 16:36:2026#include "ui/gfx/geometry/rect_conversions.h"
[email protected]c8686a02012-11-27 08:29:0027#include "ui/gfx/transform.h"
[email protected]94f206c12012-08-25 00:09:1428
[email protected]9c88e562012-09-14 22:21:3029namespace cc {
[email protected]94f206c12012-08-25 00:09:1430
[email protected]d2d915aa2013-03-08 20:18:1231RenderSurfaceImpl::RenderSurfaceImpl(LayerImpl* owning_layer)
32 : owning_layer_(owning_layer),
33 surface_property_changed_(false),
[email protected]30fe19ff2013-07-04 00:54:4534 contributes_to_drawn_surface_(false),
kulkarni.a4015690f12014-10-10 13:50:0635 nearest_occlusion_immune_ancestor_(nullptr),
[email protected]d2d915aa2013-03-08 20:18:1236 target_render_surface_layer_index_history_(0),
37 current_layer_index_history_(0) {
38 damage_tracker_ = DamageTracker::Create();
[email protected]94f206c12012-08-25 00:09:1439}
40
[email protected]d2d915aa2013-03-08 20:18:1241RenderSurfaceImpl::~RenderSurfaceImpl() {}
42
weiliangc189c1a12016-04-11 16:16:2543RenderSurfaceImpl* RenderSurfaceImpl::render_target() {
44 EffectTree& effect_tree =
45 owning_layer_->layer_tree_impl()->property_trees()->effect_tree;
46 EffectNode* node = effect_tree.Node(EffectTreeIndex());
47 EffectNode* target_node = effect_tree.Node(node->data.target_id);
48 if (target_node->id != 0)
49 return target_node->data.render_surface;
50 else
51 return this;
52}
53
54const RenderSurfaceImpl* RenderSurfaceImpl::render_target() const {
55 const EffectTree& effect_tree =
56 owning_layer_->layer_tree_impl()->property_trees()->effect_tree;
57 const EffectNode* node = effect_tree.Node(EffectTreeIndex());
58 const EffectNode* target_node = effect_tree.Node(node->data.target_id);
59 if (target_node->id != 0)
60 return target_node->data.render_surface;
61 else
62 return this;
63}
64
weiliangcc97575c2016-03-03 18:34:2765RenderSurfaceImpl::DrawProperties::DrawProperties() {
66 draw_opacity = 1.f;
67 is_clipped = false;
68}
69
70RenderSurfaceImpl::DrawProperties::~DrawProperties() {}
71
[email protected]d2d915aa2013-03-08 20:18:1272gfx::RectF RenderSurfaceImpl::DrawableContentRect() const {
73 gfx::RectF drawable_content_rect =
weiliangcc97575c2016-03-03 18:34:2774 MathUtil::MapClippedRect(draw_transform(), gfx::RectF(content_rect()));
[email protected]7aba6662013-03-12 10:17:3475 if (owning_layer_->has_replica()) {
danakj5e6ff6d2015-09-05 04:43:4476 drawable_content_rect.Union(MathUtil::MapClippedRect(
weiliangcc97575c2016-03-03 18:34:2777 replica_draw_transform(), gfx::RectF(content_rect())));
[email protected]d2d915aa2013-03-08 20:18:1278 }
senorblanco38858c52016-01-20 23:15:0079 if (!owning_layer_->filters().IsEmpty()) {
80 int left, top, right, bottom;
81 owning_layer_->filters().GetOutsets(&top, &right, &bottom, &left);
82 drawable_content_rect.Inset(-left, -top, -right, -bottom);
83 }
[email protected]d2d915aa2013-03-08 20:18:1284
jaydasika5160e672015-10-15 15:25:1485 // If the rect has a NaN coordinate, we return empty rect to avoid crashes in
86 // functions (for example, gfx::ToEnclosedRect) that are called on this rect.
87 if (std::isnan(drawable_content_rect.x()) ||
88 std::isnan(drawable_content_rect.y()) ||
89 std::isnan(drawable_content_rect.right()) ||
90 std::isnan(drawable_content_rect.bottom()))
91 return gfx::RectF();
92
[email protected]d2d915aa2013-03-08 20:18:1293 return drawable_content_rect;
[email protected]94f206c12012-08-25 00:09:1494}
95
danakja1b92e42015-02-11 21:07:4796SkColor RenderSurfaceImpl::GetDebugBorderColor() const {
97 return DebugColors::SurfaceBorderColor();
98}
99
100SkColor RenderSurfaceImpl::GetReplicaDebugBorderColor() const {
101 return DebugColors::SurfaceReplicaBorderColor();
102}
103
104float RenderSurfaceImpl::GetDebugBorderWidth() const {
105 return DebugColors::SurfaceBorderWidth(owning_layer_->layer_tree_impl());
106}
107
108float RenderSurfaceImpl::GetReplicaDebugBorderWidth() const {
109 return DebugColors::SurfaceReplicaBorderWidth(
110 owning_layer_->layer_tree_impl());
111}
112
[email protected]d2d915aa2013-03-08 20:18:12113int RenderSurfaceImpl::OwningLayerId() const {
114 return owning_layer_ ? owning_layer_->id() : 0;
[email protected]94f206c12012-08-25 00:09:14115}
116
jaydasika7ba8f922015-08-21 05:39:42117bool RenderSurfaceImpl::HasReplica() const {
118 return owning_layer_->has_replica();
119}
120
jaydasika23fb3822015-08-25 03:22:59121const LayerImpl* RenderSurfaceImpl::ReplicaLayer() const {
122 return owning_layer_->replica_layer();
123}
124
jaydasika504a0502015-07-23 19:25:44125int RenderSurfaceImpl::TransformTreeIndex() const {
126 return owning_layer_->transform_tree_index();
127}
[email protected]94f206c12012-08-25 00:09:14128
jaydasikaebf9e4ea2015-08-14 21:29:12129int RenderSurfaceImpl::ClipTreeIndex() const {
130 return owning_layer_->clip_tree_index();
131}
132
jaydasika0f4b1a92015-08-18 23:19:02133int RenderSurfaceImpl::EffectTreeIndex() const {
134 return owning_layer_->effect_tree_index();
135}
136
[email protected]0023fc72014-01-10 20:05:06137void RenderSurfaceImpl::SetClipRect(const gfx::Rect& clip_rect) {
weiliangcc97575c2016-03-03 18:34:27138 if (clip_rect == draw_properties_.clip_rect)
[email protected]d2d915aa2013-03-08 20:18:12139 return;
[email protected]94f206c12012-08-25 00:09:14140
[email protected]d2d915aa2013-03-08 20:18:12141 surface_property_changed_ = true;
weiliangcc97575c2016-03-03 18:34:27142 draw_properties_.clip_rect = clip_rect;
[email protected]94f206c12012-08-25 00:09:14143}
144
[email protected]0023fc72014-01-10 20:05:06145void RenderSurfaceImpl::SetContentRect(const gfx::Rect& content_rect) {
weiliangcc97575c2016-03-03 18:34:27146 if (content_rect == draw_properties_.content_rect)
[email protected]d2d915aa2013-03-08 20:18:12147 return;
[email protected]94f206c12012-08-25 00:09:14148
[email protected]d2d915aa2013-03-08 20:18:12149 surface_property_changed_ = true;
weiliangcc97575c2016-03-03 18:34:27150 draw_properties_.content_rect = content_rect;
[email protected]94f206c12012-08-25 00:09:14151}
152
weiliangc6da32862016-04-20 16:40:11153void RenderSurfaceImpl::SetContentRectForTesting(const gfx::Rect& rect) {
154 SetContentRect(rect);
155}
156
157gfx::Rect RenderSurfaceImpl::CalculateClippedAccumulatedContentRect() {
158 if (owning_layer_->replica_layer() || owning_layer_->HasCopyRequest() ||
159 !is_clipped())
160 return accumulated_content_rect();
161
162 if (accumulated_content_rect().IsEmpty())
163 return gfx::Rect();
164
165 // Calculate projection from the target surface rect to local
166 // space. Non-invertible draw transforms means no able to bring clipped rect
167 // in target space back to local space, early out without clip.
168 gfx::Transform target_to_surface(gfx::Transform::kSkipInitialization);
169 if (!draw_transform().GetInverse(&target_to_surface))
170 return accumulated_content_rect();
171
172 // Clip rect is in target space. Bring accumulated content rect to
173 // target space in preparation for clipping.
174 gfx::Rect accumulated_rect_in_target_space =
175 MathUtil::MapEnclosingClippedRect(draw_transform(),
176 accumulated_content_rect());
177 // If accumulated content rect is contained within clip rect, early out
178 // without clipping.
179 if (clip_rect().Contains(accumulated_rect_in_target_space))
180 return accumulated_content_rect();
181
182 gfx::Rect clipped_accumulated_rect_in_target_space = clip_rect();
183 clipped_accumulated_rect_in_target_space.Intersect(
184 accumulated_rect_in_target_space);
185
186 if (clipped_accumulated_rect_in_target_space.IsEmpty())
187 return gfx::Rect();
188
189 gfx::Rect clipped_accumulated_rect_in_local_space =
190 MathUtil::ProjectEnclosingClippedRect(
191 target_to_surface, clipped_accumulated_rect_in_target_space);
192 // Bringing clipped accumulated rect back to local space may result
193 // in inflation due to axis-alignment.
194 clipped_accumulated_rect_in_local_space.Intersect(accumulated_content_rect());
195 return clipped_accumulated_rect_in_local_space;
196}
197
198void RenderSurfaceImpl::CalculateContentRectFromAccumulatedContentRect(
199 int max_texture_size) {
200 // Root render surface use viewport, and does not calculate content rect.
201 DCHECK_NE(render_target(), this);
202
203 // Surface's content rect is the clipped accumulated content rect. By default
204 // use accumulated content rect, and then try to clip it.
205 gfx::Rect surface_content_rect = CalculateClippedAccumulatedContentRect();
206
207 // The RenderSurfaceImpl backing texture cannot exceed the maximum
208 // supported texture size.
209 surface_content_rect.set_width(
210 std::min(surface_content_rect.width(), max_texture_size));
211 surface_content_rect.set_height(
212 std::min(surface_content_rect.height(), max_texture_size));
213
214 SetContentRect(surface_content_rect);
215}
216
217void RenderSurfaceImpl::SetContentRectToViewport() {
218 // Only root render surface use viewport as content rect.
219 DCHECK_EQ(render_target(), this);
220 gfx::Rect viewport = gfx::ToEnclosingRect(owning_layer_->layer_tree_impl()
221 ->property_trees()
222 ->clip_tree.ViewportClip());
223 SetContentRect(viewport);
224}
225
weiliangc189c1a12016-04-11 16:16:25226void RenderSurfaceImpl::ClearAccumulatedContentRect() {
227 accumulated_content_rect_ = gfx::Rect();
228}
229
230void RenderSurfaceImpl::AccumulateContentRectFromContributingLayer(
231 LayerImpl* layer) {
232 DCHECK(layer->DrawsContent());
233 DCHECK_EQ(this, layer->render_target());
234
235 // Root render surface doesn't accumulate content rect, it always uses
236 // viewport for content rect.
237 if (render_target() == this)
238 return;
239
240 accumulated_content_rect_.Union(layer->drawable_content_rect());
241}
242
243void RenderSurfaceImpl::AccumulateContentRectFromContributingRenderSurface(
244 RenderSurfaceImpl* contributing_surface) {
245 DCHECK_NE(this, contributing_surface);
246 DCHECK_EQ(this, contributing_surface->render_target());
247
248 // Root render surface doesn't accumulate content rect, it always uses
249 // viewport for content rect.
250 if (render_target() == this)
251 return;
252
253 // The content rect of contributing surface is in its own space. Instead, we
254 // will use contributing surface's DrawableContentRect which is in target
255 // space (local space for this render surface) as required.
256 accumulated_content_rect_.Union(
257 gfx::ToEnclosedRect(contributing_surface->DrawableContentRect()));
jaydasika7ba8f922015-08-21 05:39:42258}
259
[email protected]d2d915aa2013-03-08 20:18:12260bool RenderSurfaceImpl::SurfacePropertyChanged() const {
261 // Surface property changes are tracked as follows:
262 //
263 // - surface_property_changed_ is flagged when the clip_rect or content_rect
264 // change. As of now, these are the only two properties that can be affected
265 // by descendant layers.
266 //
267 // - all other property changes come from the owning layer (or some ancestor
268 // layer that propagates its change to the owning layer).
269 //
270 DCHECK(owning_layer_);
[email protected]7aba6662013-03-12 10:17:34271 return surface_property_changed_ || owning_layer_->LayerPropertyChanged();
[email protected]94f206c12012-08-25 00:09:14272}
273
[email protected]d2d915aa2013-03-08 20:18:12274bool RenderSurfaceImpl::SurfacePropertyChangedOnlyFromDescendant() const {
[email protected]7aba6662013-03-12 10:17:34275 return surface_property_changed_ && !owning_layer_->LayerPropertyChanged();
[email protected]94f206c12012-08-25 00:09:14276}
277
[email protected]d2d915aa2013-03-08 20:18:12278void RenderSurfaceImpl::ClearLayerLists() {
279 layer_list_.clear();
[email protected]7d929c02012-09-20 17:26:57280}
281
[email protected]0cd7d6f72014-08-22 14:50:51282RenderPassId RenderSurfaceImpl::GetRenderPassId() {
[email protected]d2d915aa2013-03-08 20:18:12283 int layer_id = owning_layer_->id();
284 int sub_id = 0;
285 DCHECK_GT(layer_id, 0);
[email protected]0cd7d6f72014-08-22 14:50:51286 return RenderPassId(layer_id, sub_id);
[email protected]0f077a52012-09-08 01:45:24287}
288
[email protected]d2d915aa2013-03-08 20:18:12289void RenderSurfaceImpl::AppendRenderPasses(RenderPassSink* pass_sink) {
danakj60bc3bc2016-04-09 00:24:48290 std::unique_ptr<RenderPass> pass = RenderPass::Create(layer_list_.size());
weiliangcc97575c2016-03-03 18:34:27291 pass->SetNew(GetRenderPassId(), content_rect(),
292 gfx::IntersectRects(content_rect(),
[email protected]b4ead7b2014-04-07 18:12:18293 damage_tracker_->current_damage_rect()),
weiliangcc97575c2016-03-03 18:34:27294 draw_properties_.screen_space_transform);
danakja04855a2015-11-18 20:39:10295 pass_sink->AppendRenderPass(std::move(pass));
[email protected]467b3612012-08-28 07:41:16296}
297
danakja1b92e42015-02-11 21:07:47298void RenderSurfaceImpl::AppendQuads(RenderPass* render_pass,
299 const gfx::Transform& draw_transform,
300 const Occlusion& occlusion_in_content_space,
301 SkColor debug_border_color,
302 float debug_border_width,
303 LayerImpl* mask_layer,
304 AppendQuadsData* append_quads_data,
305 RenderPassId render_pass_id) {
danakj64767d902015-06-19 00:10:43306 gfx::Rect visible_layer_rect =
weiliangcc97575c2016-03-03 18:34:27307 occlusion_in_content_space.GetUnoccludedContentRect(content_rect());
danakj64767d902015-06-19 00:10:43308 if (visible_layer_rect.IsEmpty())
[email protected]ba0f8d92014-03-21 18:41:10309 return;
310
[email protected]c6707fd2014-06-23 05:50:36311 SharedQuadState* shared_quad_state =
312 render_pass->CreateAndAppendSharedQuadState();
weiliangcc97575c2016-03-03 18:34:27313 shared_quad_state->SetAll(
314 draw_transform, content_rect().size(), content_rect(),
315 draw_properties_.clip_rect, draw_properties_.is_clipped,
316 draw_properties_.draw_opacity, owning_layer_->blend_mode(),
317 owning_layer_->sorting_context_id());
[email protected]94f206c12012-08-25 00:09:14318
[email protected]7aba6662013-03-12 10:17:34319 if (owning_layer_->ShowDebugBorders()) {
[email protected]f7030c32014-07-03 18:54:34320 DebugBorderDrawQuad* debug_border_quad =
321 render_pass->CreateAndAppendDrawQuad<DebugBorderDrawQuad>();
weiliangcc97575c2016-03-03 18:34:27322 debug_border_quad->SetNew(shared_quad_state, content_rect(),
danakj64767d902015-06-19 00:10:43323 visible_layer_rect, debug_border_color,
danakja1b92e42015-02-11 21:07:47324 debug_border_width);
[email protected]d2d915aa2013-03-08 20:18:12325 }
[email protected]94f206c12012-08-25 00:09:14326
jbaumanbbd425e2015-05-19 00:33:35327 ResourceId mask_resource_id = 0;
ennef6f3fbba42014-10-16 18:16:49328 gfx::Size mask_texture_size;
329 gfx::Vector2dF mask_uv_scale;
ajumad9432e32015-11-30 19:43:44330 gfx::Transform owning_layer_draw_transform = owning_layer_->DrawTransform();
danakja1b92e42015-02-11 21:07:47331 if (mask_layer && mask_layer->DrawsContent() &&
332 !mask_layer->bounds().IsEmpty()) {
ennef6f3fbba42014-10-16 18:16:49333 mask_layer->GetContentsResourceId(&mask_resource_id, &mask_texture_size);
[email protected]d2d915aa2013-03-08 20:18:12334 gfx::Vector2dF owning_layer_draw_scale =
ajumad9432e32015-11-30 19:43:44335 MathUtil::ComputeTransform2dScaleComponents(owning_layer_draw_transform,
336 1.f);
danakjddaec912015-09-25 19:38:40337 gfx::SizeF unclipped_mask_target_size = gfx::ScaleSize(
338 gfx::SizeF(owning_layer_->bounds()), owning_layer_draw_scale.x(),
339 owning_layer_draw_scale.y());
senorblancodfcb3622016-01-27 21:48:01340 mask_uv_scale = gfx::Vector2dF(1.0f / unclipped_mask_target_size.width(),
341 1.0f / unclipped_mask_target_size.height());
[email protected]d2d915aa2013-03-08 20:18:12342 }
[email protected]94f206c12012-08-25 00:09:14343
robertnce999072016-01-05 15:06:13344 DCHECK(owning_layer_draw_transform.IsScale2d());
[email protected]7ac3d492014-08-08 21:25:32345 gfx::Vector2dF owning_layer_to_target_scale =
ajumad9432e32015-11-30 19:43:44346 owning_layer_draw_transform.Scale2d();
[email protected]7ac3d492014-08-08 21:25:32347
[email protected]f7030c32014-07-03 18:54:34348 RenderPassDrawQuad* quad =
349 render_pass->CreateAndAppendDrawQuad<RenderPassDrawQuad>();
weiliangcc97575c2016-03-03 18:34:27350 quad->SetNew(shared_quad_state, content_rect(), visible_layer_rect,
danakj64767d902015-06-19 00:10:43351 render_pass_id, mask_resource_id, mask_uv_scale,
352 mask_texture_size, owning_layer_->filters(),
[email protected]7ac3d492014-08-08 21:25:32353 owning_layer_to_target_scale,
[email protected]7aba6662013-03-12 10:17:34354 owning_layer_->background_filters());
[email protected]94f206c12012-08-25 00:09:14355}
356
[email protected]bc5e77c2012-11-05 20:00:49357} // namespace cc