1 // Copyright 2012 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.
5 #include "cc/tiles/picture_layer_tiling_set.h"
11 #include "cc/playback/raster_source.h"
17 class LargestToSmallestScaleFunctor
{
19 bool operator() (PictureLayerTiling
* left
, PictureLayerTiling
* right
) {
20 return left
->contents_scale() > right
->contents_scale();
24 inline float LargerRatio(float float1
, float float2
) {
25 DCHECK_GT(float1
, 0.f
);
26 DCHECK_GT(float2
, 0.f
);
27 return float1
> float2
? float1
/ float2
: float2
/ float1
;
33 scoped_ptr
<PictureLayerTilingSet
> PictureLayerTilingSet::Create(
35 PictureLayerTilingClient
* client
,
36 size_t tiling_interest_area_padding
,
37 float skewport_target_time_in_seconds
,
38 int skewport_extrapolation_limit_in_content_pixels
) {
39 return make_scoped_ptr(new PictureLayerTilingSet(
40 tree
, client
, tiling_interest_area_padding
,
41 skewport_target_time_in_seconds
,
42 skewport_extrapolation_limit_in_content_pixels
));
45 PictureLayerTilingSet::PictureLayerTilingSet(
47 PictureLayerTilingClient
* client
,
48 size_t tiling_interest_area_padding
,
49 float skewport_target_time_in_seconds
,
50 int skewport_extrapolation_limit_in_content_pixels
)
51 : tiling_interest_area_padding_(tiling_interest_area_padding
),
52 skewport_target_time_in_seconds_(skewport_target_time_in_seconds
),
53 skewport_extrapolation_limit_in_content_pixels_(
54 skewport_extrapolation_limit_in_content_pixels
),
58 PictureLayerTilingSet::~PictureLayerTilingSet() {
61 void PictureLayerTilingSet::CopyTilingsAndPropertiesFromPendingTwin(
62 const PictureLayerTilingSet
* pending_twin_set
,
63 const scoped_refptr
<RasterSource
>& raster_source
,
64 const Region
& layer_invalidation
) {
65 if (pending_twin_set
->tilings_
.empty()) {
66 // If the twin (pending) tiling set is empty, it was not updated for the
67 // current frame. So we drop tilings from our set as well, instead of
68 // leaving behind unshared tilings that are all non-ideal.
73 bool tiling_sort_required
= false;
74 for (PictureLayerTiling
* pending_twin_tiling
: pending_twin_set
->tilings_
) {
75 float contents_scale
= pending_twin_tiling
->contents_scale();
76 PictureLayerTiling
* this_tiling
= FindTilingWithScale(contents_scale
);
78 scoped_ptr
<PictureLayerTiling
> new_tiling
= PictureLayerTiling::Create(
79 tree_
, contents_scale
, raster_source
, client_
,
80 tiling_interest_area_padding_
, skewport_target_time_in_seconds_
,
81 skewport_extrapolation_limit_in_content_pixels_
);
82 tilings_
.push_back(new_tiling
.Pass());
83 this_tiling
= tilings_
.back();
84 tiling_sort_required
= true;
86 this_tiling
->TakeTilesAndPropertiesFrom(pending_twin_tiling
,
90 if (tiling_sort_required
)
91 tilings_
.sort(LargestToSmallestScaleFunctor());
94 void PictureLayerTilingSet::UpdateTilingsToCurrentRasterSourceForActivation(
95 scoped_refptr
<RasterSource
> raster_source
,
96 const PictureLayerTilingSet
* pending_twin_set
,
97 const Region
& layer_invalidation
,
98 float minimum_contents_scale
,
99 float maximum_contents_scale
) {
100 RemoveTilingsBelowScale(minimum_contents_scale
);
101 RemoveTilingsAboveScale(maximum_contents_scale
);
103 // Copy over tilings that are shared with the |pending_twin_set| tiling set.
104 // Also, copy all of the properties from twin tilings.
105 CopyTilingsAndPropertiesFromPendingTwin(pending_twin_set
, raster_source
,
108 // If the tiling is not shared (FindTilingWithScale returns nullptr), then
109 // invalidate tiles and update them to the new raster source.
110 for (PictureLayerTiling
* tiling
: tilings_
) {
111 if (pending_twin_set
->FindTilingWithScale(tiling
->contents_scale()))
114 tiling
->SetRasterSourceAndResize(raster_source
);
115 tiling
->Invalidate(layer_invalidation
);
116 // This is needed for cases where the live tiles rect didn't change but
117 // recordings exist in the raster source that did not exist on the last
119 tiling
->CreateMissingTilesInLiveTilesRect();
121 // |this| is active set and |tiling| is not in the pending set, which means
122 // it is now NON_IDEAL_RESOLUTION. The exception is for LOW_RESOLUTION
123 // tilings, which are computed and created entirely on the active tree.
124 // Since the pending tree does not have them, we should just leave them as
125 // low resolution to not lose them.
126 if (tiling
->resolution() != LOW_RESOLUTION
)
127 tiling
->set_resolution(NON_IDEAL_RESOLUTION
);
130 VerifyTilings(pending_twin_set
);
133 void PictureLayerTilingSet::UpdateTilingsToCurrentRasterSourceForCommit(
134 scoped_refptr
<RasterSource
> raster_source
,
135 const Region
& layer_invalidation
,
136 float minimum_contents_scale
,
137 float maximum_contents_scale
) {
138 RemoveTilingsBelowScale(minimum_contents_scale
);
139 RemoveTilingsAboveScale(maximum_contents_scale
);
141 // Invalidate tiles and update them to the new raster source.
142 for (PictureLayerTiling
* tiling
: tilings_
) {
143 DCHECK_IMPLIES(tree_
== PENDING_TREE
, !tiling
->has_tiles());
144 tiling
->SetRasterSourceAndResize(raster_source
);
146 // We can commit on either active or pending trees, but only active one can
147 // have tiles at this point.
148 if (tree_
== ACTIVE_TREE
)
149 tiling
->Invalidate(layer_invalidation
);
151 // This is needed for cases where the live tiles rect didn't change but
152 // recordings exist in the raster source that did not exist on the last
154 tiling
->CreateMissingTilesInLiveTilesRect();
156 VerifyTilings(nullptr /* pending_twin_set */);
159 void PictureLayerTilingSet::UpdateRasterSourceDueToLCDChange(
160 const scoped_refptr
<RasterSource
>& raster_source
,
161 const Region
& layer_invalidation
) {
162 for (PictureLayerTiling
* tiling
: tilings_
) {
163 tiling
->SetRasterSourceAndResize(raster_source
);
164 tiling
->Invalidate(layer_invalidation
);
165 // Since the invalidation changed, we need to create any missing tiles in
166 // the live tiles rect again.
167 tiling
->CreateMissingTilesInLiveTilesRect();
171 void PictureLayerTilingSet::VerifyTilings(
172 const PictureLayerTilingSet
* pending_twin_set
) const {
174 for (PictureLayerTiling
* tiling
: tilings_
) {
175 DCHECK(tiling
->tile_size() ==
176 client_
->CalculateTileSize(tiling
->tiling_size()))
177 << "tile_size: " << tiling
->tile_size().ToString()
178 << " tiling_size: " << tiling
->tiling_size().ToString()
179 << " CalculateTileSize: "
180 << client_
->CalculateTileSize(tiling
->tiling_size()).ToString();
183 if (!tilings_
.empty()) {
184 DCHECK_LE(NumHighResTilings(), 1);
185 // When commiting from the main thread the high res tiling may get dropped,
186 // but when cloning to the active tree, there should always be one.
187 if (pending_twin_set
) {
188 DCHECK_EQ(1, NumHighResTilings())
189 << " num tilings on active: " << tilings_
.size()
190 << " num tilings on pending: " << pending_twin_set
->tilings_
.size()
191 << " num high res on pending: "
192 << pending_twin_set
->NumHighResTilings()
193 << " are on active tree: " << (tree_
== ACTIVE_TREE
);
199 void PictureLayerTilingSet::CleanUpTilings(
200 float min_acceptable_high_res_scale
,
201 float max_acceptable_high_res_scale
,
202 const std::vector
<PictureLayerTiling
*>& needed_tilings
,
203 PictureLayerTilingSet
* twin_set
) {
204 std::vector
<PictureLayerTiling
*> to_remove
;
205 for (auto* tiling
: tilings_
) {
206 // Keep all tilings within the min/max scales.
207 if (tiling
->contents_scale() >= min_acceptable_high_res_scale
&&
208 tiling
->contents_scale() <= max_acceptable_high_res_scale
) {
212 // Keep low resolution tilings.
213 if (tiling
->resolution() == LOW_RESOLUTION
)
216 // Don't remove tilings that are required.
217 if (std::find(needed_tilings
.begin(), needed_tilings
.end(), tiling
) !=
218 needed_tilings
.end()) {
222 to_remove
.push_back(tiling
);
225 for (auto* tiling
: to_remove
) {
226 DCHECK_NE(HIGH_RESOLUTION
, tiling
->resolution());
231 void PictureLayerTilingSet::RemoveNonIdealTilings() {
232 auto to_remove
= tilings_
.remove_if([](PictureLayerTiling
* t
) {
233 return t
->resolution() == NON_IDEAL_RESOLUTION
;
235 tilings_
.erase(to_remove
, tilings_
.end());
238 void PictureLayerTilingSet::MarkAllTilingsNonIdeal() {
239 for (auto* tiling
: tilings_
)
240 tiling
->set_resolution(NON_IDEAL_RESOLUTION
);
243 PictureLayerTiling
* PictureLayerTilingSet::AddTiling(
244 float contents_scale
,
245 scoped_refptr
<RasterSource
> raster_source
) {
246 for (size_t i
= 0; i
< tilings_
.size(); ++i
) {
247 DCHECK_NE(tilings_
[i
]->contents_scale(), contents_scale
);
248 DCHECK_EQ(tilings_
[i
]->raster_source(), raster_source
.get());
251 tilings_
.push_back(PictureLayerTiling::Create(
252 tree_
, contents_scale
, raster_source
, client_
,
253 tiling_interest_area_padding_
, skewport_target_time_in_seconds_
,
254 skewport_extrapolation_limit_in_content_pixels_
));
255 PictureLayerTiling
* appended
= tilings_
.back();
257 tilings_
.sort(LargestToSmallestScaleFunctor());
261 int PictureLayerTilingSet::NumHighResTilings() const {
262 return std::count_if(tilings_
.begin(), tilings_
.end(),
263 [](PictureLayerTiling
* tiling
) {
264 return tiling
->resolution() == HIGH_RESOLUTION
;
268 PictureLayerTiling
* PictureLayerTilingSet::FindTilingWithScale(
270 for (size_t i
= 0; i
< tilings_
.size(); ++i
) {
271 if (tilings_
[i
]->contents_scale() == scale
)
277 PictureLayerTiling
* PictureLayerTilingSet::FindTilingWithResolution(
278 TileResolution resolution
) const {
279 auto iter
= std::find_if(tilings_
.begin(), tilings_
.end(),
280 [resolution
](const PictureLayerTiling
* tiling
) {
281 return tiling
->resolution() == resolution
;
283 if (iter
== tilings_
.end())
288 void PictureLayerTilingSet::RemoveTilingsBelowScale(float minimum_scale
) {
290 tilings_
.remove_if([minimum_scale
](PictureLayerTiling
* tiling
) {
291 return tiling
->contents_scale() < minimum_scale
;
293 tilings_
.erase(to_remove
, tilings_
.end());
296 void PictureLayerTilingSet::RemoveTilingsAboveScale(float maximum_scale
) {
298 tilings_
.remove_if([maximum_scale
](PictureLayerTiling
* tiling
) {
299 return tiling
->contents_scale() > maximum_scale
;
301 tilings_
.erase(to_remove
, tilings_
.end());
304 void PictureLayerTilingSet::RemoveAllTilings() {
308 void PictureLayerTilingSet::Remove(PictureLayerTiling
* tiling
) {
309 ScopedPtrVector
<PictureLayerTiling
>::iterator iter
=
310 std::find(tilings_
.begin(), tilings_
.end(), tiling
);
311 if (iter
== tilings_
.end())
313 tilings_
.erase(iter
);
316 void PictureLayerTilingSet::RemoveAllTiles() {
317 for (size_t i
= 0; i
< tilings_
.size(); ++i
)
318 tilings_
[i
]->Reset();
321 float PictureLayerTilingSet::GetSnappedContentsScale(
323 float snap_to_existing_tiling_ratio
) const {
324 // If a tiling exists within the max snapping ratio, snap to its scale.
325 float snapped_contents_scale
= start_scale
;
326 float snapped_ratio
= snap_to_existing_tiling_ratio
;
327 for (const auto* tiling
: tilings_
) {
328 float tiling_contents_scale
= tiling
->contents_scale();
329 float ratio
= LargerRatio(tiling_contents_scale
, start_scale
);
330 if (ratio
< snapped_ratio
) {
331 snapped_contents_scale
= tiling_contents_scale
;
332 snapped_ratio
= ratio
;
335 return snapped_contents_scale
;
338 float PictureLayerTilingSet::GetMaximumContentsScale() const {
339 if (tilings_
.empty())
341 // The first tiling has the largest contents scale.
342 return tilings_
[0]->contents_scale();
345 bool PictureLayerTilingSet::UpdateTilePriorities(
346 const gfx::Rect
& required_rect_in_layer_space
,
347 float ideal_contents_scale
,
348 double current_frame_time_in_seconds
,
349 const Occlusion
& occlusion_in_layer_space
,
350 bool can_require_tiles_for_activation
) {
351 bool updated
= false;
352 for (auto* tiling
: tilings_
) {
353 tiling
->set_can_require_tiles_for_activation(
354 can_require_tiles_for_activation
);
355 updated
|= tiling
->ComputeTilePriorityRects(
356 required_rect_in_layer_space
, ideal_contents_scale
,
357 current_frame_time_in_seconds
, occlusion_in_layer_space
);
362 void PictureLayerTilingSet::GetAllPrioritizedTilesForTracing(
363 std::vector
<PrioritizedTile
>* prioritized_tiles
) const {
364 for (auto* tiling
: tilings_
)
365 tiling
->GetAllPrioritizedTilesForTracing(prioritized_tiles
);
368 PictureLayerTilingSet::CoverageIterator::CoverageIterator(
369 const PictureLayerTilingSet
* set
,
370 float contents_scale
,
371 const gfx::Rect
& content_rect
,
372 float ideal_contents_scale
)
374 contents_scale_(contents_scale
),
375 ideal_contents_scale_(ideal_contents_scale
),
376 current_tiling_(std::numeric_limits
<size_t>::max()) {
377 missing_region_
.Union(content_rect
);
379 size_t tilings_size
= set_
->tilings_
.size();
380 for (ideal_tiling_
= 0; ideal_tiling_
< tilings_size
; ++ideal_tiling_
) {
381 PictureLayerTiling
* tiling
= set_
->tilings_
[ideal_tiling_
];
382 if (tiling
->contents_scale() < ideal_contents_scale_
) {
383 if (ideal_tiling_
> 0)
389 if (ideal_tiling_
== tilings_size
&& ideal_tiling_
> 0)
395 PictureLayerTilingSet::CoverageIterator::~CoverageIterator() {
398 gfx::Rect
PictureLayerTilingSet::CoverageIterator::geometry_rect() const {
400 if (!region_iter_
.has_rect())
402 return region_iter_
.rect();
404 return tiling_iter_
.geometry_rect();
407 gfx::RectF
PictureLayerTilingSet::CoverageIterator::texture_rect() const {
410 return tiling_iter_
.texture_rect();
413 Tile
* PictureLayerTilingSet::CoverageIterator::operator->() const {
416 return *tiling_iter_
;
419 Tile
* PictureLayerTilingSet::CoverageIterator::operator*() const {
422 return *tiling_iter_
;
425 TileResolution
PictureLayerTilingSet::CoverageIterator::resolution() const {
426 const PictureLayerTiling
* tiling
= CurrentTiling();
428 return tiling
->resolution();
431 PictureLayerTiling
* PictureLayerTilingSet::CoverageIterator::CurrentTiling()
433 if (current_tiling_
== std::numeric_limits
<size_t>::max())
435 if (current_tiling_
>= set_
->tilings_
.size())
437 return set_
->tilings_
[current_tiling_
];
440 size_t PictureLayerTilingSet::CoverageIterator::NextTiling() const {
441 // Order returned by this method is:
442 // 1. Ideal tiling index
443 // 2. Tiling index < Ideal in decreasing order (higher res than ideal)
444 // 3. Tiling index > Ideal in increasing order (lower res than ideal)
445 // 4. Tiling index > tilings.size() (invalid index)
446 if (current_tiling_
== std::numeric_limits
<size_t>::max())
447 return ideal_tiling_
;
448 else if (current_tiling_
> ideal_tiling_
)
449 return current_tiling_
+ 1;
450 else if (current_tiling_
)
451 return current_tiling_
- 1;
453 return ideal_tiling_
+ 1;
456 PictureLayerTilingSet::CoverageIterator
&
457 PictureLayerTilingSet::CoverageIterator::operator++() {
458 bool first_time
= current_tiling_
== std::numeric_limits
<size_t>::max();
460 if (!*this && !first_time
)
466 // Loop until we find a valid place to stop.
468 while (tiling_iter_
&&
469 (!*tiling_iter_
|| !tiling_iter_
->draw_info().IsReadyToDraw())) {
470 missing_region_
.Union(tiling_iter_
.geometry_rect());
476 // If the set of current rects for this tiling is done, go to the next
477 // tiling and set up to iterate through all of the remaining holes.
478 // This will also happen the first time through the loop.
479 if (!region_iter_
.has_rect()) {
480 current_tiling_
= NextTiling();
481 current_region_
.Swap(&missing_region_
);
482 missing_region_
.Clear();
483 region_iter_
= Region::Iterator(current_region_
);
485 // All done and all filled.
486 if (!region_iter_
.has_rect()) {
487 current_tiling_
= set_
->tilings_
.size();
491 // No more valid tiles, return this checkerboard rect.
492 if (current_tiling_
>= set_
->tilings_
.size())
496 // Pop a rect off. If there are no more tilings, then these will be
497 // treated as geometry with null tiles that the caller can checkerboard.
498 gfx::Rect last_rect
= region_iter_
.rect();
501 // Done, found next checkerboard rect to return.
502 if (current_tiling_
>= set_
->tilings_
.size())
505 // Construct a new iterator for the next tiling, but we need to loop
506 // again until we get to a valid one.
507 tiling_iter_
= PictureLayerTiling::CoverageIterator(
508 set_
->tilings_
[current_tiling_
],
516 PictureLayerTilingSet::CoverageIterator::operator bool() const {
517 return current_tiling_
< set_
->tilings_
.size() || region_iter_
.has_rect();
520 void PictureLayerTilingSet::AsValueInto(
521 base::trace_event::TracedValue
* state
) const {
522 for (size_t i
= 0; i
< tilings_
.size(); ++i
) {
523 state
->BeginDictionary();
524 tilings_
[i
]->AsValueInto(state
);
525 state
->EndDictionary();
529 size_t PictureLayerTilingSet::GPUMemoryUsageInBytes() const {
531 for (size_t i
= 0; i
< tilings_
.size(); ++i
)
532 amount
+= tilings_
[i
]->GPUMemoryUsageInBytes();
536 PictureLayerTilingSet::TilingRange
PictureLayerTilingSet::GetTilingRange(
537 TilingRangeType type
) const {
538 // Doesn't seem to be the case right now but if it ever becomes a performance
539 // problem to compute these ranges each time this function is called, we can
540 // compute them only when the tiling set has changed instead.
541 size_t tilings_size
= tilings_
.size();
542 TilingRange
high_res_range(0, 0);
543 TilingRange
low_res_range(tilings_
.size(), tilings_
.size());
544 for (size_t i
= 0; i
< tilings_size
; ++i
) {
545 const PictureLayerTiling
* tiling
= tilings_
[i
];
546 if (tiling
->resolution() == HIGH_RESOLUTION
)
547 high_res_range
= TilingRange(i
, i
+ 1);
548 if (tiling
->resolution() == LOW_RESOLUTION
)
549 low_res_range
= TilingRange(i
, i
+ 1);
552 TilingRange
range(0, 0);
554 case HIGHER_THAN_HIGH_RES
:
555 range
= TilingRange(0, high_res_range
.start
);
558 range
= high_res_range
;
560 case BETWEEN_HIGH_AND_LOW_RES
:
561 // TODO(vmpstr): This code assumes that high res tiling will come before
562 // low res tiling, however there are cases where this assumption is
563 // violated. As a result, it's better to be safe in these situations,
564 // since otherwise we can end up accessing a tiling that doesn't exist.
565 // See crbug.com/429397 for high res tiling appearing after low res
566 // tiling discussion/fixes.
567 if (high_res_range
.start
<= low_res_range
.start
)
568 range
= TilingRange(high_res_range
.end
, low_res_range
.start
);
570 range
= TilingRange(low_res_range
.end
, high_res_range
.start
);
573 range
= low_res_range
;
575 case LOWER_THAN_LOW_RES
:
576 range
= TilingRange(low_res_range
.end
, tilings_size
);
580 DCHECK_LE(range
.start
, range
.end
);