1 // Copyright 2014 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/resources/display_list_recording_source.h"
9 #include "cc/base/region.h"
10 #include "cc/layers/content_layer_client.h"
11 #include "cc/resources/display_item_list.h"
12 #include "cc/resources/display_list_raster_source.h"
13 #include "skia/ext/analysis_canvas.h"
17 // Layout pixel buffer around the visible layer rect to record. Any base
18 // picture that intersects the visible layer rect expanded by this distance
20 const int kPixelDistanceToRecord
= 8000;
21 // We don't perform solid color analysis on images that have more than 10 skia
23 const int kOpCountThatIsOkToAnalyze
= 10;
29 DisplayListRecordingSource::DisplayListRecordingSource()
30 : slow_down_raster_scale_factor_for_debug_(0),
31 gather_pixel_refs_(false),
32 requires_clear_(false),
33 is_solid_color_(false),
34 solid_color_(SK_ColorTRANSPARENT
),
35 background_color_(SK_ColorTRANSPARENT
),
36 pixel_record_distance_(kPixelDistanceToRecord
),
37 is_suitable_for_gpu_rasterization_(true) {
40 DisplayListRecordingSource::~DisplayListRecordingSource() {
43 bool DisplayListRecordingSource::UpdateAndExpandInvalidation(
44 ContentLayerClient
* painter
,
46 const gfx::Size
& layer_size
,
47 const gfx::Rect
& visible_layer_rect
,
49 RecordingMode recording_mode
) {
52 if (size_
!= layer_size
) {
57 gfx::Rect old_recorded_viewport
= recorded_viewport_
;
58 recorded_viewport_
= visible_layer_rect
;
59 recorded_viewport_
.Inset(-pixel_record_distance_
, -pixel_record_distance_
);
60 recorded_viewport_
.Intersect(gfx::Rect(GetSize()));
62 if (recorded_viewport_
!= old_recorded_viewport
) {
63 // Invalidate newly-exposed and no-longer-exposed areas.
64 Region
newly_exposed_region(recorded_viewport_
);
65 newly_exposed_region
.Subtract(old_recorded_viewport
);
66 invalidation
->Union(newly_exposed_region
);
68 Region
no_longer_exposed_region(old_recorded_viewport
);
69 no_longer_exposed_region
.Subtract(recorded_viewport_
);
70 invalidation
->Union(no_longer_exposed_region
);
75 if (!updated
&& !invalidation
->Intersects(recorded_viewport_
))
78 ContentLayerClient::PaintingControlSetting painting_control
=
79 ContentLayerClient::PAINTING_BEHAVIOR_NORMAL
;
81 switch (recording_mode
) {
83 // Already setup for normal recording.
85 case RECORD_WITH_SK_NULL_CANVAS
:
86 // TODO(schenney): Remove this when DisplayList recording is the only
87 // option. For now, fall through and disable construction.
88 case RECORD_WITH_PAINTING_DISABLED
:
89 painting_control
= ContentLayerClient::DISPLAY_LIST_CONSTRUCTION_DISABLED
;
91 case RECORD_WITH_CACHING_DISABLED
:
92 painting_control
= ContentLayerClient::DISPLAY_LIST_CACHING_DISABLED
;
99 if (slow_down_raster_scale_factor_for_debug_
> 1) {
100 repeat_count
= slow_down_raster_scale_factor_for_debug_
;
101 if (painting_control
!=
102 ContentLayerClient::DISPLAY_LIST_CONSTRUCTION_DISABLED
) {
103 painting_control
= ContentLayerClient::DISPLAY_LIST_CACHING_DISABLED
;
106 for (int i
= 0; i
< repeat_count
; ++i
) {
107 display_list_
= painter
->PaintContentsToDisplayList(recorded_viewport_
,
110 display_list_
->set_layer_rect(recorded_viewport_
);
111 is_suitable_for_gpu_rasterization_
=
112 display_list_
->IsSuitableForGpuRasterization();
114 DetermineIfSolidColor();
115 display_list_
->EmitTraceSnapshot();
117 display_list_
->CreateAndCacheSkPicture();
122 void DisplayListRecordingSource::DidMoveToNewCompositor() {
123 // No invalidation history to worry about here.
126 gfx::Size
DisplayListRecordingSource::GetSize() const {
130 void DisplayListRecordingSource::SetEmptyBounds() {
135 void DisplayListRecordingSource::SetSlowdownRasterScaleFactor(int factor
) {
136 slow_down_raster_scale_factor_for_debug_
= factor
;
139 void DisplayListRecordingSource::SetGatherPixelRefs(bool gather_pixel_refs
) {
140 gather_pixel_refs_
= gather_pixel_refs
;
143 void DisplayListRecordingSource::SetBackgroundColor(SkColor background_color
) {
144 background_color_
= background_color
;
147 void DisplayListRecordingSource::SetRequiresClear(bool requires_clear
) {
148 requires_clear_
= requires_clear
;
151 void DisplayListRecordingSource::SetUnsuitableForGpuRasterizationForTesting() {
152 is_suitable_for_gpu_rasterization_
= false;
155 bool DisplayListRecordingSource::IsSuitableForGpuRasterization() const {
156 return is_suitable_for_gpu_rasterization_
;
159 scoped_refptr
<RasterSource
> DisplayListRecordingSource::CreateRasterSource(
160 bool can_use_lcd_text
) const {
161 return scoped_refptr
<RasterSource
>(
162 DisplayListRasterSource::CreateFromDisplayListRecordingSource(
163 this, can_use_lcd_text
));
166 gfx::Size
DisplayListRecordingSource::GetTileGridSizeForTesting() const {
170 void DisplayListRecordingSource::DetermineIfSolidColor() {
171 DCHECK(display_list_
.get());
172 is_solid_color_
= false;
173 solid_color_
= SK_ColorTRANSPARENT
;
175 if (display_list_
->ApproximateOpCount() > kOpCountThatIsOkToAnalyze
)
178 gfx::Size layer_size
= GetSize();
179 skia::AnalysisCanvas
canvas(layer_size
.width(), layer_size
.height());
180 display_list_
->Raster(&canvas
, nullptr, 1.f
);
181 is_solid_color_
= canvas
.GetColorIfSolid(&solid_color_
);
184 void DisplayListRecordingSource::Clear() {
185 recorded_viewport_
= gfx::Rect();
186 display_list_
= NULL
;
187 is_solid_color_
= false;