blob: 3eae5f50fa0695513a2f92be95144ebc2320af37 [file] [log] [blame]
[email protected]1920930592012-01-11 14:54:481// Copyright (c) 2012 The Chromium Authors. All rights reserved.
[email protected]c797cd42011-03-15 02:18:362// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
[email protected]116302fc2012-05-05 21:45:415#include "ui/compositor/compositor.h"
[email protected]ed8de92d2011-09-14 04:16:486
[email protected]cff176a2012-06-29 21:11:007#include <algorithm>
[email protected]337bd042012-11-05 23:43:438#include <deque>
dcheng79c14492015-12-18 05:07:269#include <utility>
[email protected]cff176a2012-06-29 21:11:0010
[email protected]6b16679e2012-10-27 00:44:2811#include "base/bind.h"
[email protected]ab6627372012-01-29 21:22:1312#include "base/command_line.h"
[email protected]4e2d03e22013-07-18 04:19:5413#include "base/message_loop/message_loop.h"
[email protected]2f2fd9e2013-12-05 03:36:3014#include "base/metrics/histogram.h"
[email protected]f3652ff92013-06-11 13:54:3115#include "base/strings/string_util.h"
[email protected]49c4cf852013-09-27 19:28:2416#include "base/sys_info.h"
ssid334fb87a2015-01-27 20:12:0717#include "base/trace_event/trace_event.h"
loyso841229002015-12-21 10:03:2418#include "cc/animation/animation_host.h"
19#include "cc/animation/animation_id_provider.h"
20#include "cc/animation/animation_timeline.h"
[email protected]4d5e6762013-03-19 18:46:5721#include "cc/base/switches.h"
[email protected]3052b10f2013-03-18 07:41:2122#include "cc/input/input_handler.h"
[email protected]cc3cfaa2013-03-18 09:05:5223#include "cc/layers/layer.h"
[email protected]04c5900d2014-08-18 13:38:3624#include "cc/output/begin_frame_args.h"
[email protected]7f0d825f2013-03-18 07:24:3025#include "cc/output/context_provider.h"
jamesrf313a212015-03-16 21:27:3726#include "cc/output/latency_info_swap_promise.h"
simonhonga7e3ac42014-11-11 20:50:2227#include "cc/scheduler/begin_frame_source.h"
jbaumanfdc3baa2014-10-10 00:22:0928#include "cc/surfaces/surface_id_allocator.h"
[email protected]556fd292013-03-18 08:03:0429#include "cc/trees/layer_tree_host.h"
[email protected]83afcbcc2012-07-27 03:06:2730#include "third_party/skia/include/core/SkBitmap.h"
[email protected]116302fc2012-05-05 21:45:4131#include "ui/compositor/compositor_observer.h"
32#include "ui/compositor/compositor_switches.h"
dbeame627c522015-05-05 03:25:5233#include "ui/compositor/compositor_vsync_manager.h"
[email protected]cd9a61c72012-05-08 19:16:5934#include "ui/compositor/dip_util.h"
[email protected]116302fc2012-05-05 21:45:4135#include "ui/compositor/layer.h"
[email protected]9034a282014-06-05 03:11:4736#include "ui/compositor/layer_animator_collection.h"
[email protected]c9e2cbbb2012-05-12 21:17:2737#include "ui/gl/gl_context.h"
[email protected]cc2ae012012-09-21 19:35:2538#include "ui/gl/gl_switches.h"
[email protected]ab6627372012-01-29 21:22:1339
40namespace {
41
42const double kDefaultRefreshRate = 60.0;
[email protected]7ddeaab2013-04-06 00:47:0543const double kTestRefreshRate = 200.0;
[email protected]ab6627372012-01-29 21:22:1344
[email protected]83afcbcc2012-07-27 03:06:2745} // namespace
[email protected]c797cd42011-03-15 02:18:3646
47namespace ui {
48
[email protected]6b16679e2012-10-27 00:44:2849CompositorLock::CompositorLock(Compositor* compositor)
50 : compositor_(compositor) {
ccameron00e438cd2015-03-12 06:18:1451 if (compositor_->locks_will_time_out_) {
52 compositor_->task_runner_->PostDelayedTask(
53 FROM_HERE,
54 base::Bind(&CompositorLock::CancelLock, AsWeakPtr()),
55 base::TimeDelta::FromMilliseconds(kCompositorLockTimeoutMs));
56 }
[email protected]6b16679e2012-10-27 00:44:2857}
58
59CompositorLock::~CompositorLock() {
60 CancelLock();
61}
62
63void CompositorLock::CancelLock() {
64 if (!compositor_)
65 return;
66 compositor_->UnlockCompositor();
67 compositor_ = NULL;
68}
69
sieversb2a31d3372015-08-25 19:27:1370Compositor::Compositor(ui::ContextFactory* context_factory,
[email protected]bd30a23e2014-07-25 21:54:1571 scoped_refptr<base::SingleThreadTaskRunner> task_runner)
[email protected]4183bf092014-05-17 17:33:5672 : context_factory_(context_factory),
73 root_layer_(NULL),
sieversb2a31d3372015-08-25 19:27:1374 widget_(gfx::kNullAcceleratedWidget),
sieversca8baca2015-10-05 23:38:0275 widget_valid_(false),
76 output_surface_requested_(false),
jbaumanfdc3baa2014-10-10 00:22:0977 surface_id_allocator_(context_factory->CreateSurfaceIdAllocator()),
[email protected]bd30a23e2014-07-25 21:54:1578 task_runner_(task_runner),
dbeame627c522015-05-05 03:25:5279 vsync_manager_(new CompositorVSyncManager()),
[email protected]4183bf092014-05-17 17:33:5680 device_scale_factor_(0.0f),
isherman4f0404e12014-12-06 01:01:3381 last_started_frame_(0),
82 last_ended_frame_(0),
ccameron00e438cd2015-03-12 06:18:1483 locks_will_time_out_(true),
[email protected]4183bf092014-05-17 17:33:5684 compositor_lock_(NULL),
weiliangc1f27b282014-10-09 17:10:3385 layer_animator_collection_(this),
jbauman1a7a5122014-10-28 00:22:5286 weak_ptr_factory_(this) {
loysoa6edaaff2015-05-25 03:26:4487 root_web_layer_ = cc::Layer::Create(Layer::UILayerSettings());
[email protected]ec05af52012-11-21 23:07:0088
avi6b10fd02014-12-23 05:51:2389 base::CommandLine* command_line = base::CommandLine::ForCurrentProcess();
[email protected]4d5e6762013-03-19 18:46:5790
[email protected]ec05af52012-11-21 23:07:0091 cc::LayerTreeSettings settings;
loysoa6edaaff2015-05-25 03:26:4492
danakj0006e95b2015-08-08 00:36:5493 // This will ensure PictureLayers always can have LCD text, to match the
94 // previous behaviour with ContentLayers, where LCD-not-allowed notifications
95 // were ignored.
danakj3f76ace2014-11-18 16:56:0096 settings.layers_always_allowed_lcd_text = true;
danakj0006e95b2015-08-08 00:36:5497 // Use occlusion to allow more overlapping windows to take less memory.
98 settings.use_occlusion_for_tile_prioritization = true;
jbaumanc5be44c2014-11-20 22:17:1299 settings.renderer_settings.refresh_rate =
100 context_factory_->DoesCreateTestContexts() ? kTestRefreshRate
101 : kDefaultRefreshRate;
[email protected]0c7a5612014-03-12 21:58:22102 settings.main_frame_before_activation_enabled = false;
jincheol.jo5899bd552015-07-15 02:06:46103 if (command_line->HasSwitch(switches::kDisableGpuVsync)) {
104 std::string display_vsync_string =
105 command_line->GetSwitchValueASCII(switches::kDisableGpuVsync);
106 if (display_vsync_string == "gpu") {
107 settings.renderer_settings.disable_display_vsync = true;
108 } else if (display_vsync_string == "beginframe") {
109 settings.wait_for_beginframe_interval = false;
110 } else {
111 settings.renderer_settings.disable_display_vsync = true;
112 settings.wait_for_beginframe_interval = false;
113 }
114 }
jbaumanc5be44c2014-11-20 22:17:12115 settings.renderer_settings.partial_swap_enabled =
danakjee38c1c2015-10-15 19:22:50116 !command_line->HasSwitch(switches::kUIDisablePartialSwap);
sunnyps340939792014-10-16 21:59:58117#if defined(OS_WIN)
jbauman616238a2014-12-03 03:17:53118 settings.renderer_settings.finish_rendering_on_resize = true;
ccameron07e6ae22015-11-20 04:46:29119#elif defined(OS_MACOSX)
120 settings.renderer_settings.delay_releasing_overlay_resources = true;
sunnyps340939792014-10-16 21:59:58121#endif
[email protected]4d5e6762013-03-19 18:46:57122
123 // These flags should be mirrored by renderer versions in content/renderer/.
[email protected]8e0176d2013-03-21 03:14:52124 settings.initial_debug_state.show_debug_borders =
[email protected]4d5e6762013-03-19 18:46:57125 command_line->HasSwitch(cc::switches::kUIShowCompositedLayerBorders);
[email protected]bf9ed2c2013-12-10 22:18:39126 settings.initial_debug_state.show_layer_animation_bounds_rects =
127 command_line->HasSwitch(cc::switches::kUIShowLayerAnimationBounds);
[email protected]8e0176d2013-03-21 03:14:52128 settings.initial_debug_state.show_paint_rects =
[email protected]4d5e6762013-03-19 18:46:57129 command_line->HasSwitch(switches::kUIShowPaintRects);
[email protected]8e0176d2013-03-21 03:14:52130 settings.initial_debug_state.show_property_changed_rects =
[email protected]4d5e6762013-03-19 18:46:57131 command_line->HasSwitch(cc::switches::kUIShowPropertyChangedRects);
[email protected]8e0176d2013-03-21 03:14:52132 settings.initial_debug_state.show_surface_damage_rects =
[email protected]4d5e6762013-03-19 18:46:57133 command_line->HasSwitch(cc::switches::kUIShowSurfaceDamageRects);
[email protected]8e0176d2013-03-21 03:14:52134 settings.initial_debug_state.show_screen_space_rects =
[email protected]4d5e6762013-03-19 18:46:57135 command_line->HasSwitch(cc::switches::kUIShowScreenSpaceRects);
[email protected]8e0176d2013-03-21 03:14:52136 settings.initial_debug_state.show_replica_screen_space_rects =
[email protected]4d5e6762013-03-19 18:46:57137 command_line->HasSwitch(cc::switches::kUIShowReplicaScreenSpaceRects);
[email protected]302fe422012-06-11 14:49:11138
[email protected]fe3beef2014-02-06 09:20:53139 settings.initial_debug_state.SetRecordRenderingStats(
140 command_line->HasSwitch(cc::switches::kEnableGpuBenchmarking));
141
ajumab6adea92015-12-06 17:19:15142 if (command_line->HasSwitch(cc::switches::kDisableCompositorPropertyTrees))
143 settings.use_property_trees = false;
[email protected]2cccfef2014-05-01 06:05:16144 settings.use_zero_copy = IsUIZeroCopyEnabled();
danakjadd76822015-05-27 02:09:43145
rbyers77d1de52015-12-18 15:56:22146 settings.renderer_settings.use_rgba_4444_textures =
147 command_line->HasSwitch(switches::kUIEnableRGBA4444Textures);
revemancfd85ee2015-08-20 21:14:01148
ericrk1d17f752015-10-20 03:03:07149 // UI compositor always uses partial raster if not using zero-copy. Zero copy
150 // doesn't currently support partial raster.
151 settings.use_partial_raster = !settings.use_zero_copy;
152
revemaneb0fda22015-10-26 20:27:47153 // Use CPU_READ_WRITE_PERSISTENT memory buffers to support partial tile
154 // raster if needed.
155 gfx::BufferUsage usage =
156 settings.use_partial_raster
157 ? gfx::BufferUsage::GPU_READ_CPU_READ_WRITE_PERSISTENT
158 : gfx::BufferUsage::GPU_READ_CPU_READ_WRITE;
danakjadd76822015-05-27 02:09:43159
jie.a.chen0ffadce32015-08-07 12:59:23160 for (size_t format = 0;
161 format < static_cast<size_t>(gfx::BufferFormat::LAST) + 1; format++) {
162 DCHECK_GT(settings.use_image_texture_targets.size(), format);
163 settings.use_image_texture_targets[format] =
164 context_factory_->GetImageTextureTarget(
165 static_cast<gfx::BufferFormat>(format), usage);
166 }
danakjadd76822015-05-27 02:09:43167
vmpstrdfd22862015-09-25 17:42:41168 // Note: Only enable image decode tasks if we have more than one worker
169 // thread.
170 settings.image_decode_tasks_enabled = false;
[email protected]dafdf5052014-03-13 17:02:57171
loyso4cc0ad4e2015-12-21 23:34:03172 settings.use_compositor_animation_timelines = !command_line->HasSwitch(
173 switches::kUIDisableCompositorAnimationTimelines);
loyso85a000ae2015-06-02 01:31:27174
sohan.jyotidb8e72682015-11-04 17:46:32175#if !defined(OS_ANDROID)
176 // TODO(sohanjg): Revisit this memory usage in tile manager.
177 cc::ManagedMemoryPolicy policy(
178 512 * 1024 * 1024, gpu::MemoryAllocation::CUTOFF_ALLOW_NICE_TO_HAVE,
179 settings.memory_policy_.num_resources_limit);
180 settings.memory_policy_ = policy;
181#endif
182
[email protected]2f2fd9e2013-12-05 03:36:30183 base::TimeTicks before_create = base::TimeTicks::Now();
sadrul6780f3da2015-05-11 17:01:52184
185 cc::LayerTreeHost::InitParams params;
186 params.client = this;
187 params.shared_bitmap_manager = context_factory_->GetSharedBitmapManager();
188 params.gpu_memory_buffer_manager =
189 context_factory_->GetGpuMemoryBufferManager();
190 params.task_graph_runner = context_factory_->GetTaskGraphRunner();
191 params.settings = &settings;
192 params.main_task_runner = task_runner_;
193 host_ = cc::LayerTreeHost::CreateSingleThreaded(this, &params);
[email protected]2f2fd9e2013-12-05 03:36:30194 UMA_HISTOGRAM_TIMES("GPU.CreateBrowserCompositor",
195 base::TimeTicks::Now() - before_create);
loyso841229002015-12-21 10:03:24196
197 if (settings.use_compositor_animation_timelines) {
198 animation_timeline_ = cc::AnimationTimeline::Create(
199 cc::AnimationIdProvider::NextTimelineId());
200 host_->animation_host()->AddAnimationTimeline(animation_timeline_.get());
201 }
[email protected]804c8982013-03-13 16:32:21202 host_->SetRootLayer(root_web_layer_);
jbaumandbccae1ab2014-11-06 23:26:44203 host_->set_surface_id_namespace(surface_id_allocator_->id_namespace());
sievers71c62dd52015-10-07 01:44:39204 host_->SetVisible(true);
[email protected]82a01ac2011-09-08 16:00:18205}
206
207Compositor::~Compositor() {
[email protected]89af4002013-09-06 07:47:07208 TRACE_EVENT0("shutdown", "Compositor::destructor");
209
[email protected]6b16679e2012-10-27 00:44:28210 CancelCompositorLock();
211 DCHECK(!compositor_lock_);
212
ccameron92bcf312015-01-23 21:04:09213 FOR_EACH_OBSERVER(CompositorObserver, observer_list_,
214 OnCompositingShuttingDown(this));
215
lof84adf2ce862015-06-02 22:23:32216 FOR_EACH_OBSERVER(CompositorAnimationObserver, animation_observer_list_,
217 OnCompositingShuttingDown(this));
218
[email protected]7ab3f272011-11-16 00:51:56219 if (root_layer_)
loysoac008462015-05-27 01:05:50220 root_layer_->ResetCompositor();
[email protected]2700daddd2012-07-13 19:35:37221
loyso841229002015-12-21 10:03:24222 if (animation_timeline_)
223 host_->animation_host()->RemoveAnimationTimeline(animation_timeline_.get());
224
[email protected]2700daddd2012-07-13 19:35:37225 // Stop all outstanding draws before telling the ContextFactory to tear
226 // down any contexts that the |host_| may rely upon.
227 host_.reset();
228
[email protected]4183bf092014-05-17 17:33:56229 context_factory_->RemoveCompositor(this);
[email protected]82a01ac2011-09-08 16:00:18230}
231
jbauman1a7a5122014-10-28 00:22:52232void Compositor::SetOutputSurface(
233 scoped_ptr<cc::OutputSurface> output_surface) {
sieversca8baca2015-10-05 23:38:02234 output_surface_requested_ = false;
dcheng79c14492015-12-18 05:07:26235 host_->SetOutputSurface(std::move(output_surface));
jbauman1a7a5122014-10-28 00:22:52236}
237
[email protected]332749032011-10-22 00:32:46238void Compositor::ScheduleDraw() {
weiliangc5efa0a12015-01-29 19:56:46239 host_->SetNeedsCommit();
[email protected]332749032011-10-22 00:32:46240}
241
[email protected]993d6b322011-09-27 19:14:38242void Compositor::SetRootLayer(Layer* root_layer) {
[email protected]12233c362011-11-21 16:09:25243 if (root_layer_ == root_layer)
244 return;
[email protected]7ab3f272011-11-16 00:51:56245 if (root_layer_)
loysoac008462015-05-27 01:05:50246 root_layer_->ResetCompositor();
[email protected]993d6b322011-09-27 19:14:38247 root_layer_ = root_layer;
[email protected]7aba6662013-03-12 10:17:34248 root_web_layer_->RemoveAllChildren();
[email protected]66efabe2012-08-18 03:06:06249 if (root_layer_)
loysoac008462015-05-27 01:05:50250 root_layer_->SetCompositor(this, root_web_layer_);
[email protected]993d6b322011-09-27 19:14:38251}
252
loyso841229002015-12-21 10:03:24253cc::AnimationTimeline* Compositor::GetAnimationTimeline() const {
254 return animation_timeline_.get();
255}
256
[email protected]ebd52522012-10-04 15:49:40257void Compositor::SetHostHasTransparentBackground(
258 bool host_has_transparent_background) {
[email protected]804c8982013-03-13 16:32:21259 host_->set_has_transparent_background(host_has_transparent_background);
[email protected]ebd52522012-10-04 15:49:40260}
261
[email protected]878705be2013-04-15 22:44:02262void Compositor::ScheduleFullRedraw() {
weiliangc5efa0a12015-01-29 19:56:46263 // TODO(enne): Some callers (mac) call this function expecting that it
264 // will also commit. This should probably just redraw the screen
265 // from damage and not commit. ScheduleDraw/ScheduleRedraw need
266 // better names.
[email protected]804c8982013-03-13 16:32:21267 host_->SetNeedsRedraw();
weiliangc5efa0a12015-01-29 19:56:46268 host_->SetNeedsCommit();
[email protected]7df588fbd2012-02-10 14:15:56269}
270
[email protected]878705be2013-04-15 22:44:02271void Compositor::ScheduleRedrawRect(const gfx::Rect& damage_rect) {
weiliangc5efa0a12015-01-29 19:56:46272 // TODO(enne): Make this not commit. See ScheduleFullRedraw.
[email protected]878705be2013-04-15 22:44:02273 host_->SetNeedsRedrawRect(damage_rect);
weiliangc5efa0a12015-01-29 19:56:46274 host_->SetNeedsCommit();
[email protected]878705be2013-04-15 22:44:02275}
276
chirantan40e4cd82015-02-19 01:08:19277void Compositor::FinishAllRendering() {
278 host_->FinishAllRendering();
279}
280
jbauman616238a2014-12-03 03:17:53281void Compositor::DisableSwapUntilResize() {
[email protected]b5e2a732014-05-13 21:27:50282 host_->FinishAllRendering();
jbauman616238a2014-12-03 03:17:53283 context_factory_->ResizeDisplay(this, gfx::Size());
[email protected]b5e2a732014-05-13 21:27:50284}
285
[email protected]66239a22013-06-05 03:38:26286void Compositor::SetLatencyInfo(const ui::LatencyInfo& latency_info) {
[email protected]d359203a2013-11-29 06:16:55287 scoped_ptr<cc::SwapPromise> swap_promise(
288 new cc::LatencyInfoSwapPromise(latency_info));
dcheng79c14492015-12-18 05:07:26289 host_->QueueSwapPromise(std::move(swap_promise));
[email protected]66239a22013-06-05 03:38:26290}
291
[email protected]cd9a61c72012-05-08 19:16:59292void Compositor::SetScaleAndSize(float scale, const gfx::Size& size_in_pixel) {
[email protected]cff176a2012-06-29 21:11:00293 DCHECK_GT(scale, 0);
[email protected]351b8ce2012-11-27 23:37:16294 if (!size_in_pixel.IsEmpty()) {
295 size_ = size_in_pixel;
[email protected]18ce59702013-04-09 04:58:40296 host_->SetViewportSize(size_in_pixel);
[email protected]7aba6662013-03-12 10:17:34297 root_web_layer_->SetBounds(size_in_pixel);
jbauman616238a2014-12-03 03:17:53298 context_factory_->ResizeDisplay(this, size_in_pixel);
[email protected]351b8ce2012-11-27 23:37:16299 }
[email protected]2e2216e42012-05-17 15:17:00300 if (device_scale_factor_ != scale) {
[email protected]cd9a61c72012-05-08 19:16:59301 device_scale_factor_ = scale;
[email protected]caa21662014-05-14 10:02:32302 host_->SetDeviceScaleFactor(scale);
[email protected]cd9a61c72012-05-08 19:16:59303 if (root_layer_)
304 root_layer_->OnDeviceScaleFactorChanged(scale);
305 }
[email protected]ed8de92d2011-09-14 04:16:48306}
307
[email protected]87601922013-04-02 03:56:42308void Compositor::SetBackgroundColor(SkColor color) {
309 host_->set_background_color(color);
310 ScheduleDraw();
311}
312
ccameron18bbc2a2014-08-28 22:36:16313void Compositor::SetVisible(bool visible) {
314 host_->SetVisible(visible);
315}
316
chirantan40e4cd82015-02-19 01:08:19317bool Compositor::IsVisible() {
318 return host_->visible();
319}
320
simonhong047d61b2015-05-22 02:37:58321void Compositor::SetAuthoritativeVSyncInterval(
322 const base::TimeDelta& interval) {
323 if (base::CommandLine::ForCurrentProcess()->HasSwitch(
324 cc::switches::kEnableBeginFrameScheduling)) {
325 host_->SetAuthoritativeVSyncInterval(interval);
326 return;
327 }
328
329 vsync_manager_->SetAuthoritativeVSyncInterval(interval);
330}
331
sieversca8baca2015-10-05 23:38:02332void Compositor::SetAcceleratedWidget(gfx::AcceleratedWidget widget) {
sieversb2a31d3372015-08-25 19:27:13333 // This function should only get called once.
sieversca8baca2015-10-05 23:38:02334 DCHECK(!widget_valid_);
sieversb2a31d3372015-08-25 19:27:13335 widget_ = widget;
sieversca8baca2015-10-05 23:38:02336 widget_valid_ = true;
337 if (output_surface_requested_)
338 context_factory_->CreateOutputSurface(weak_ptr_factory_.GetWeakPtr());
339}
340
sievers13e78282015-10-06 01:54:40341gfx::AcceleratedWidget Compositor::ReleaseAcceleratedWidget() {
342 DCHECK(!IsVisible());
343 if (!host_->output_surface_lost())
344 host_->ReleaseOutputSurface();
345 context_factory_->RemoveCompositor(this);
346 widget_valid_ = false;
347 gfx::AcceleratedWidget widget = widget_;
348 widget_ = gfx::kNullAcceleratedWidget;
349 return widget;
350}
351
sieversca8baca2015-10-05 23:38:02352gfx::AcceleratedWidget Compositor::widget() const {
353 DCHECK(widget_valid_);
354 return widget_;
sieversb2a31d3372015-08-25 19:27:13355}
356
dbeame627c522015-05-05 03:25:52357scoped_refptr<CompositorVSyncManager> Compositor::vsync_manager() const {
358 return vsync_manager_;
[email protected]2bd1fcf02014-02-12 22:35:53359}
360
[email protected]ed8de92d2011-09-14 04:16:48361void Compositor::AddObserver(CompositorObserver* observer) {
362 observer_list_.AddObserver(observer);
363}
364
365void Compositor::RemoveObserver(CompositorObserver* observer) {
366 observer_list_.RemoveObserver(observer);
367}
368
mgiuca64ccf2362014-11-10 06:44:23369bool Compositor::HasObserver(const CompositorObserver* observer) const {
[email protected]3ce2feb2011-09-19 18:44:23370 return observer_list_.HasObserver(observer);
371}
372
[email protected]5ffe6d142014-08-01 16:04:38373void Compositor::AddAnimationObserver(CompositorAnimationObserver* observer) {
374 animation_observer_list_.AddObserver(observer);
375 host_->SetNeedsAnimate();
376}
377
378void Compositor::RemoveAnimationObserver(
379 CompositorAnimationObserver* observer) {
380 animation_observer_list_.RemoveObserver(observer);
381}
382
mgiuca64ccf2362014-11-10 06:44:23383bool Compositor::HasAnimationObserver(
384 const CompositorAnimationObserver* observer) const {
[email protected]5ffe6d142014-08-01 16:04:38385 return animation_observer_list_.HasObserver(observer);
386}
387
simonhong8af4c832015-03-21 07:40:51388void Compositor::AddBeginFrameObserver(CompositorBeginFrameObserver* observer) {
jdduke7620c3c82015-10-22 21:46:47389 if (!begin_frame_observer_list_.might_have_observers())
simonhong8af4c832015-03-21 07:40:51390 host_->SetChildrenNeedBeginFrames(true);
391
jdduke7620c3c82015-10-22 21:46:47392 begin_frame_observer_list_.AddObserver(observer);
393
simonhong8af4c832015-03-21 07:40:51394 if (missed_begin_frame_args_.IsValid())
395 observer->OnSendBeginFrame(missed_begin_frame_args_);
simonhong8af4c832015-03-21 07:40:51396}
397
398void Compositor::RemoveBeginFrameObserver(
399 CompositorBeginFrameObserver* observer) {
jdduke7620c3c82015-10-22 21:46:47400 begin_frame_observer_list_.RemoveObserver(observer);
simonhong8af4c832015-03-21 07:40:51401
jdduke7620c3c82015-10-22 21:46:47402 // As this call may take place while iterating over observers, unsubscription
403 // from |host_| is performed after iteration in |SendBeginFramesToChildren()|.
simonhong8af4c832015-03-21 07:40:51404}
405
[email protected]04c5900d2014-08-18 13:38:36406void Compositor::BeginMainFrame(const cc::BeginFrameArgs& args) {
[email protected]5ffe6d142014-08-01 16:04:38407 FOR_EACH_OBSERVER(CompositorAnimationObserver,
408 animation_observer_list_,
[email protected]04c5900d2014-08-18 13:38:36409 OnAnimationStep(args.frame_time));
[email protected]5ffe6d142014-08-01 16:04:38410 if (animation_observer_list_.might_have_observers())
[email protected]9034a282014-06-05 03:11:47411 host_->SetNeedsAnimate();
412}
413
rmcilroy0a19362a2015-02-18 12:34:25414void Compositor::BeginMainFrameNotExpectedSoon() {
415}
416
danakj0b5eae6c2015-04-24 22:35:11417static void SendDamagedRectsRecursive(ui::Layer* layer) {
418 layer->SendDamagedRects();
419 for (auto* child : layer->children())
420 SendDamagedRectsRecursive(child);
421}
422
wkorman7265db012015-11-03 04:08:25423void Compositor::UpdateLayerTreeHost() {
danakj0b5eae6c2015-04-24 22:35:11424 if (!root_layer())
425 return;
426 SendDamagedRectsRecursive(root_layer());
[email protected]ab6627372012-01-29 21:22:13427}
428
enne7f8fdde2014-12-10 21:32:09429void Compositor::RequestNewOutputSurface() {
sieversca8baca2015-10-05 23:38:02430 DCHECK(!output_surface_requested_);
431 output_surface_requested_ = true;
432 if (widget_valid_)
433 context_factory_->CreateOutputSurface(weak_ptr_factory_.GetWeakPtr());
[email protected]ab6627372012-01-29 21:22:13434}
435
enne7f8fdde2014-12-10 21:32:09436void Compositor::DidInitializeOutputSurface() {
enne7f8fdde2014-12-10 21:32:09437}
438
439void Compositor::DidFailToInitializeOutputSurface() {
danakj13d97082015-02-25 23:04:47440 // The OutputSurface should already be bound/initialized before being given to
441 // the Compositor.
442 NOTREACHED();
enne7f8fdde2014-12-10 21:32:09443}
444
[email protected]408b5e22013-03-19 09:48:09445void Compositor::DidCommit() {
[email protected]6b16679e2012-10-27 00:44:28446 DCHECK(!IsLocked());
[email protected]2700daddd2012-07-13 19:35:37447 FOR_EACH_OBSERVER(CompositorObserver,
448 observer_list_,
449 OnCompositingDidCommit(this));
450}
451
[email protected]408b5e22013-03-19 09:48:09452void Compositor::DidCommitAndDrawFrame() {
weiliangc1f27b282014-10-09 17:10:33453}
454
isherman4f0404e12014-12-06 01:01:33455void Compositor::DidCompleteSwapBuffers() {
weiliangc5efa0a12015-01-29 19:56:46456 FOR_EACH_OBSERVER(CompositorObserver, observer_list_,
457 OnCompositingEnded(this));
isherman4f0404e12014-12-06 01:01:33458}
459
460void Compositor::DidPostSwapBuffers() {
abhishek.ka7215854d2015-05-26 06:13:17461 base::TimeTicks start_time = base::TimeTicks::Now();
weiliangc5efa0a12015-01-29 19:56:46462 FOR_EACH_OBSERVER(CompositorObserver, observer_list_,
463 OnCompositingStarted(this, start_time));
isherman4f0404e12014-12-06 01:01:33464}
465
[email protected]4d7e46a2013-11-08 05:33:40466void Compositor::DidAbortSwapBuffers() {
[email protected]4d7e46a2013-11-08 05:33:40467 FOR_EACH_OBSERVER(CompositorObserver,
468 observer_list_,
469 OnCompositingAborted(this));
470}
471
simonhong8af4c832015-03-21 07:40:51472void Compositor::SendBeginFramesToChildren(const cc::BeginFrameArgs& args) {
jdduke7620c3c82015-10-22 21:46:47473 FOR_EACH_OBSERVER(CompositorBeginFrameObserver, begin_frame_observer_list_,
474 OnSendBeginFrame(args));
475
476 // Unsubscription is performed here, after iteration, to handle the case where
477 // the last BeginFrame observer is removed while iterating over the observers.
478 if (!begin_frame_observer_list_.might_have_observers()) {
479 host_->SetChildrenNeedBeginFrames(false);
480 // Unsubscription should reset |missed_begin_frame_args_|, avoiding stale
481 // BeginFrame dispatch when the next BeginFrame observer is added.
482 missed_begin_frame_args_ = cc::BeginFrameArgs();
483 return;
484 }
simonhong8af4c832015-03-21 07:40:51485
486 missed_begin_frame_args_ = args;
487 missed_begin_frame_args_.type = cc::BeginFrameArgs::MISSED;
488}
489
[email protected]2e77cdbb2013-04-29 13:59:14490const cc::LayerTreeDebugState& Compositor::GetLayerTreeDebugState() const {
491 return host_->debug_state();
[email protected]918f8db42013-04-27 01:53:40492}
493
[email protected]2e77cdbb2013-04-29 13:59:14494void Compositor::SetLayerTreeDebugState(
495 const cc::LayerTreeDebugState& debug_state) {
[email protected]918f8db42013-04-27 01:53:40496 host_->SetDebugState(debug_state);
497}
498
jbaumanc5be44c2014-11-20 22:17:12499const cc::RendererSettings& Compositor::GetRendererSettings() const {
500 return host_->settings().renderer_settings;
501}
502
[email protected]6b16679e2012-10-27 00:44:28503scoped_refptr<CompositorLock> Compositor::GetCompositorLock() {
504 if (!compositor_lock_) {
505 compositor_lock_ = new CompositorLock(this);
weiliangc5efa0a12015-01-29 19:56:46506 host_->SetDeferCommits(true);
[email protected]6b16679e2012-10-27 00:44:28507 FOR_EACH_OBSERVER(CompositorObserver,
508 observer_list_,
509 OnCompositingLockStateChanged(this));
510 }
511 return compositor_lock_;
512}
513
514void Compositor::UnlockCompositor() {
515 DCHECK(compositor_lock_);
516 compositor_lock_ = NULL;
weiliangc5efa0a12015-01-29 19:56:46517 host_->SetDeferCommits(false);
[email protected]6b16679e2012-10-27 00:44:28518 FOR_EACH_OBSERVER(CompositorObserver,
519 observer_list_,
520 OnCompositingLockStateChanged(this));
521}
522
523void Compositor::CancelCompositorLock() {
524 if (compositor_lock_)
525 compositor_lock_->CancelLock();
526}
527
[email protected]c797cd42011-03-15 02:18:36528} // namespace ui