blob: 0e917265415fd26e79e453e48160b41050ca6031 [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>
[email protected]cff176a2012-06-29 21:11:009
[email protected]6b16679e2012-10-27 00:44:2810#include "base/bind.h"
[email protected]ab6627372012-01-29 21:22:1311#include "base/command_line.h"
[email protected]89af4002013-09-06 07:47:0712#include "base/debug/trace_event.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"
[email protected]d359203a2013-11-29 06:16:5517#include "cc/base/latency_info_swap_promise.h"
[email protected]4d5e6762013-03-19 18:46:5718#include "cc/base/switches.h"
[email protected]3052b10f2013-03-18 07:41:2119#include "cc/input/input_handler.h"
[email protected]cc3cfaa2013-03-18 09:05:5220#include "cc/layers/layer.h"
[email protected]04c5900d2014-08-18 13:38:3621#include "cc/output/begin_frame_args.h"
[email protected]7f0d825f2013-03-18 07:24:3022#include "cc/output/context_provider.h"
simonhonga7e3ac42014-11-11 20:50:2223#include "cc/scheduler/begin_frame_source.h"
jbaumanfdc3baa2014-10-10 00:22:0924#include "cc/surfaces/surface_id_allocator.h"
[email protected]556fd292013-03-18 08:03:0425#include "cc/trees/layer_tree_host.h"
[email protected]83afcbcc2012-07-27 03:06:2726#include "third_party/skia/include/core/SkBitmap.h"
[email protected]116302fc2012-05-05 21:45:4127#include "ui/compositor/compositor_observer.h"
28#include "ui/compositor/compositor_switches.h"
[email protected]2bd1fcf02014-02-12 22:35:5329#include "ui/compositor/compositor_vsync_manager.h"
[email protected]cd9a61c72012-05-08 19:16:5930#include "ui/compositor/dip_util.h"
[email protected]116302fc2012-05-05 21:45:4131#include "ui/compositor/layer.h"
[email protected]9034a282014-06-05 03:11:4732#include "ui/compositor/layer_animator_collection.h"
[email protected]de2cf8c2013-10-25 19:46:4633#include "ui/gfx/frame_time.h"
[email protected]c9e2cbbb2012-05-12 21:17:2734#include "ui/gl/gl_context.h"
[email protected]cc2ae012012-09-21 19:35:2535#include "ui/gl/gl_switches.h"
[email protected]ab6627372012-01-29 21:22:1336
37namespace {
38
39const double kDefaultRefreshRate = 60.0;
[email protected]7ddeaab2013-04-06 00:47:0540const double kTestRefreshRate = 200.0;
[email protected]ab6627372012-01-29 21:22:1341
[email protected]6b16679e2012-10-27 00:44:2842const int kCompositorLockTimeoutMs = 67;
43
[email protected]83afcbcc2012-07-27 03:06:2744} // namespace
[email protected]c797cd42011-03-15 02:18:3645
46namespace ui {
47
[email protected]6b16679e2012-10-27 00:44:2848CompositorLock::CompositorLock(Compositor* compositor)
49 : compositor_(compositor) {
[email protected]bd30a23e2014-07-25 21:54:1550 compositor_->task_runner_->PostDelayedTask(
[email protected]6b16679e2012-10-27 00:44:2851 FROM_HERE,
52 base::Bind(&CompositorLock::CancelLock, AsWeakPtr()),
53 base::TimeDelta::FromMilliseconds(kCompositorLockTimeoutMs));
54}
55
56CompositorLock::~CompositorLock() {
57 CancelLock();
58}
59
60void CompositorLock::CancelLock() {
61 if (!compositor_)
62 return;
63 compositor_->UnlockCompositor();
64 compositor_ = NULL;
65}
66
[email protected]4183bf092014-05-17 17:33:5667Compositor::Compositor(gfx::AcceleratedWidget widget,
[email protected]bd30a23e2014-07-25 21:54:1568 ui::ContextFactory* context_factory,
69 scoped_refptr<base::SingleThreadTaskRunner> task_runner)
[email protected]4183bf092014-05-17 17:33:5670 : context_factory_(context_factory),
71 root_layer_(NULL),
72 widget_(widget),
jbaumanfdc3baa2014-10-10 00:22:0973 surface_id_allocator_(context_factory->CreateSurfaceIdAllocator()),
[email protected]4183bf092014-05-17 17:33:5674 compositor_thread_loop_(context_factory->GetCompositorMessageLoop()),
[email protected]bd30a23e2014-07-25 21:54:1575 task_runner_(task_runner),
[email protected]4183bf092014-05-17 17:33:5676 vsync_manager_(new CompositorVSyncManager()),
77 device_scale_factor_(0.0f),
isherman4f0404e12014-12-06 01:01:3378 last_started_frame_(0),
79 last_ended_frame_(0),
enne7f8fdde2014-12-10 21:32:0980 num_failed_recreate_attempts_(0),
[email protected]4183bf092014-05-17 17:33:5681 disable_schedule_composite_(false),
82 compositor_lock_(NULL),
isherman4f0404e12014-12-06 01:01:3383 defer_draw_scheduling_(false),
84 waiting_on_compositing_end_(false),
85 draw_on_compositing_end_(false),
86 swap_state_(SWAP_NONE),
weiliangc1f27b282014-10-09 17:10:3387 layer_animator_collection_(this),
jbauman1a7a5122014-10-28 00:22:5288 weak_ptr_factory_(this) {
[email protected]7aba6662013-03-12 10:17:3489 root_web_layer_ = cc::Layer::Create();
[email protected]ec05af52012-11-21 23:07:0090
avi6b10fd02014-12-23 05:51:2391 base::CommandLine* command_line = base::CommandLine::ForCurrentProcess();
[email protected]4d5e6762013-03-19 18:46:5792
[email protected]ec05af52012-11-21 23:07:0093 cc::LayerTreeSettings settings;
danakj3f76ace2014-11-18 16:56:0094 // When impl-side painting is enabled, this will ensure PictureLayers always
95 // can have LCD text, to match the previous behaviour with ContentLayers,
96 // where LCD-not-allowed notifications were ignored.
97 settings.layers_always_allowed_lcd_text = true;
jbaumanc5be44c2014-11-20 22:17:1298 settings.renderer_settings.refresh_rate =
99 context_factory_->DoesCreateTestContexts() ? kTestRefreshRate
100 : kDefaultRefreshRate;
[email protected]0c7a5612014-03-12 21:58:22101 settings.main_frame_before_activation_enabled = false;
[email protected]541f86ff2014-03-31 22:09:50102 settings.throttle_frame_production =
103 !command_line->HasSwitch(switches::kDisableGpuVsync);
[email protected]977cff12014-06-06 17:43:23104#if !defined(OS_MACOSX)
jbaumanc5be44c2014-11-20 22:17:12105 settings.renderer_settings.partial_swap_enabled =
[email protected]50e157272013-04-13 05:07:19106 !command_line->HasSwitch(cc::switches::kUIDisablePartialSwap);
[email protected]977cff12014-06-06 17:43:23107#endif
[email protected]c6d0ba72014-02-19 20:47:28108#if defined(OS_CHROMEOS)
109 settings.per_tile_painting_enabled = true;
110#endif
sunnyps340939792014-10-16 21:59:58111#if defined(OS_WIN)
isherman4f0404e12014-12-06 01:01:33112 settings.disable_hi_res_timer_tasks_on_battery = true;
jbauman616238a2014-12-03 03:17:53113 settings.renderer_settings.finish_rendering_on_resize = true;
sunnyps340939792014-10-16 21:59:58114#endif
[email protected]4d5e6762013-03-19 18:46:57115
116 // These flags should be mirrored by renderer versions in content/renderer/.
[email protected]8e0176d2013-03-21 03:14:52117 settings.initial_debug_state.show_debug_borders =
[email protected]4d5e6762013-03-19 18:46:57118 command_line->HasSwitch(cc::switches::kUIShowCompositedLayerBorders);
[email protected]8e0176d2013-03-21 03:14:52119 settings.initial_debug_state.show_fps_counter =
[email protected]4d5e6762013-03-19 18:46:57120 command_line->HasSwitch(cc::switches::kUIShowFPSCounter);
[email protected]bf9ed2c2013-12-10 22:18:39121 settings.initial_debug_state.show_layer_animation_bounds_rects =
122 command_line->HasSwitch(cc::switches::kUIShowLayerAnimationBounds);
[email protected]8e0176d2013-03-21 03:14:52123 settings.initial_debug_state.show_paint_rects =
[email protected]4d5e6762013-03-19 18:46:57124 command_line->HasSwitch(switches::kUIShowPaintRects);
[email protected]8e0176d2013-03-21 03:14:52125 settings.initial_debug_state.show_property_changed_rects =
[email protected]4d5e6762013-03-19 18:46:57126 command_line->HasSwitch(cc::switches::kUIShowPropertyChangedRects);
[email protected]8e0176d2013-03-21 03:14:52127 settings.initial_debug_state.show_surface_damage_rects =
[email protected]4d5e6762013-03-19 18:46:57128 command_line->HasSwitch(cc::switches::kUIShowSurfaceDamageRects);
[email protected]8e0176d2013-03-21 03:14:52129 settings.initial_debug_state.show_screen_space_rects =
[email protected]4d5e6762013-03-19 18:46:57130 command_line->HasSwitch(cc::switches::kUIShowScreenSpaceRects);
[email protected]8e0176d2013-03-21 03:14:52131 settings.initial_debug_state.show_replica_screen_space_rects =
[email protected]4d5e6762013-03-19 18:46:57132 command_line->HasSwitch(cc::switches::kUIShowReplicaScreenSpaceRects);
[email protected]8e0176d2013-03-21 03:14:52133 settings.initial_debug_state.show_occluding_rects =
[email protected]4d5e6762013-03-19 18:46:57134 command_line->HasSwitch(cc::switches::kUIShowOccludingRects);
[email protected]8e0176d2013-03-21 03:14:52135 settings.initial_debug_state.show_non_occluding_rects =
[email protected]4d5e6762013-03-19 18:46:57136 command_line->HasSwitch(cc::switches::kUIShowNonOccludingRects);
[email protected]302fe422012-06-11 14:49:11137
[email protected]fe3beef2014-02-06 09:20:53138 settings.initial_debug_state.SetRecordRenderingStats(
139 command_line->HasSwitch(cc::switches::kEnableGpuBenchmarking));
140
[email protected]dafdf5052014-03-13 17:02:57141 settings.impl_side_painting = IsUIImplSidePaintingEnabled();
[email protected]2cccfef2014-05-01 06:05:16142 settings.use_zero_copy = IsUIZeroCopyEnabled();
isherman4f0404e12014-12-06 01:01:33143 settings.single_thread_proxy_scheduler = false;
[email protected]dafdf5052014-03-13 17:02:57144
[email protected]2f2fd9e2013-12-05 03:36:30145 base::TimeTicks before_create = base::TimeTicks::Now();
dcheng0efb94f2014-08-25 21:49:41146 if (compositor_thread_loop_.get()) {
[email protected]943528e2013-11-07 05:01:32147 host_ = cc::LayerTreeHost::CreateThreaded(
[email protected]142b19f2014-03-14 21:50:36148 this,
[email protected]4183bf092014-05-17 17:33:56149 context_factory_->GetSharedBitmapManager(),
reveman22dd9292014-10-13 20:52:05150 context_factory_->GetGpuMemoryBufferManager(),
[email protected]142b19f2014-03-14 21:50:36151 settings,
[email protected]bd30a23e2014-07-25 21:54:15152 task_runner_,
simonhonga7e3ac42014-11-11 20:50:22153 compositor_thread_loop_,
154 nullptr);
[email protected]943528e2013-11-07 05:01:32155 } else {
[email protected]142b19f2014-03-14 21:50:36156 host_ = cc::LayerTreeHost::CreateSingleThreaded(
[email protected]27e6a212014-07-18 15:51:27157 this,
158 this,
159 context_factory_->GetSharedBitmapManager(),
reveman22dd9292014-10-13 20:52:05160 context_factory_->GetGpuMemoryBufferManager(),
[email protected]27e6a212014-07-18 15:51:27161 settings,
simonhonga7e3ac42014-11-11 20:50:22162 task_runner_,
163 nullptr);
[email protected]943528e2013-11-07 05:01:32164 }
[email protected]2f2fd9e2013-12-05 03:36:30165 UMA_HISTOGRAM_TIMES("GPU.CreateBrowserCompositor",
166 base::TimeTicks::Now() - before_create);
[email protected]804c8982013-03-13 16:32:21167 host_->SetRootLayer(root_web_layer_);
jbaumandbccae1ab2014-11-06 23:26:44168 host_->set_surface_id_namespace(surface_id_allocator_->id_namespace());
[email protected]14bd5542013-05-08 21:51:30169 host_->SetLayerTreeHostClientReady();
[email protected]82a01ac2011-09-08 16:00:18170}
171
172Compositor::~Compositor() {
[email protected]89af4002013-09-06 07:47:07173 TRACE_EVENT0("shutdown", "Compositor::destructor");
174
[email protected]6b16679e2012-10-27 00:44:28175 CancelCompositorLock();
176 DCHECK(!compositor_lock_);
177
[email protected]7ab3f272011-11-16 00:51:56178 if (root_layer_)
179 root_layer_->SetCompositor(NULL);
[email protected]2700daddd2012-07-13 19:35:37180
181 // Stop all outstanding draws before telling the ContextFactory to tear
182 // down any contexts that the |host_| may rely upon.
183 host_.reset();
184
[email protected]4183bf092014-05-17 17:33:56185 context_factory_->RemoveCompositor(this);
[email protected]82a01ac2011-09-08 16:00:18186}
187
jbauman1a7a5122014-10-28 00:22:52188void Compositor::SetOutputSurface(
189 scoped_ptr<cc::OutputSurface> output_surface) {
190 host_->SetOutputSurface(output_surface.Pass());
191}
192
[email protected]332749032011-10-22 00:32:46193void Compositor::ScheduleDraw() {
isherman4f0404e12014-12-06 01:01:33194 if (compositor_thread_loop_.get()) {
195 host_->SetNeedsCommit();
196 } else if (!defer_draw_scheduling_) {
197 defer_draw_scheduling_ = true;
198 task_runner_->PostTask(
199 FROM_HERE,
200 base::Bind(&Compositor::Draw, weak_ptr_factory_.GetWeakPtr()));
201 }
[email protected]332749032011-10-22 00:32:46202}
203
[email protected]993d6b322011-09-27 19:14:38204void Compositor::SetRootLayer(Layer* root_layer) {
[email protected]12233c362011-11-21 16:09:25205 if (root_layer_ == root_layer)
206 return;
[email protected]7ab3f272011-11-16 00:51:56207 if (root_layer_)
208 root_layer_->SetCompositor(NULL);
[email protected]993d6b322011-09-27 19:14:38209 root_layer_ = root_layer;
[email protected]7ab3f272011-11-16 00:51:56210 if (root_layer_ && !root_layer_->GetCompositor())
[email protected]993d6b322011-09-27 19:14:38211 root_layer_->SetCompositor(this);
[email protected]7aba6662013-03-12 10:17:34212 root_web_layer_->RemoveAllChildren();
[email protected]66efabe2012-08-18 03:06:06213 if (root_layer_)
[email protected]7aba6662013-03-12 10:17:34214 root_web_layer_->AddChild(root_layer_->cc_layer());
[email protected]993d6b322011-09-27 19:14:38215}
216
[email protected]ebd52522012-10-04 15:49:40217void Compositor::SetHostHasTransparentBackground(
218 bool host_has_transparent_background) {
[email protected]804c8982013-03-13 16:32:21219 host_->set_has_transparent_background(host_has_transparent_background);
[email protected]ebd52522012-10-04 15:49:40220}
221
isherman4f0404e12014-12-06 01:01:33222void Compositor::Draw() {
223 DCHECK(!compositor_thread_loop_.get());
224
225 defer_draw_scheduling_ = false;
226 if (waiting_on_compositing_end_) {
227 draw_on_compositing_end_ = true;
228 return;
229 }
230 if (!root_layer_)
231 return;
232
233 TRACE_EVENT_ASYNC_BEGIN0("ui", "Compositor::Draw", last_started_frame_ + 1);
234
235 DCHECK_NE(swap_state_, SWAP_POSTED);
236 swap_state_ = SWAP_NONE;
237
238 waiting_on_compositing_end_ = true;
239 last_started_frame_++;
240 if (!IsLocked()) {
241 // TODO(nduca): Temporary while compositor calls
242 // compositeImmediately() directly.
243 cc::BeginFrameArgs args = cc::BeginFrameArgs::Create(
244 BEGINFRAME_FROM_HERE, gfx::FrameTime::Now(), base::TimeTicks(),
245 cc::BeginFrameArgs::DefaultInterval(), cc::BeginFrameArgs::SYNCHRONOUS);
246 BeginMainFrame(args);
247 host_->Composite(args.frame_time);
248 }
249 if (swap_state_ == SWAP_NONE)
250 NotifyEnd();
251}
252
[email protected]878705be2013-04-15 22:44:02253void Compositor::ScheduleFullRedraw() {
[email protected]804c8982013-03-13 16:32:21254 host_->SetNeedsRedraw();
[email protected]7df588fbd2012-02-10 14:15:56255}
256
[email protected]878705be2013-04-15 22:44:02257void Compositor::ScheduleRedrawRect(const gfx::Rect& damage_rect) {
258 host_->SetNeedsRedrawRect(damage_rect);
259}
260
jbauman616238a2014-12-03 03:17:53261void Compositor::DisableSwapUntilResize() {
[email protected]b5e2a732014-05-13 21:27:50262 host_->FinishAllRendering();
jbauman616238a2014-12-03 03:17:53263 context_factory_->ResizeDisplay(this, gfx::Size());
[email protected]b5e2a732014-05-13 21:27:50264}
265
[email protected]66239a22013-06-05 03:38:26266void Compositor::SetLatencyInfo(const ui::LatencyInfo& latency_info) {
[email protected]d359203a2013-11-29 06:16:55267 scoped_ptr<cc::SwapPromise> swap_promise(
268 new cc::LatencyInfoSwapPromise(latency_info));
269 host_->QueueSwapPromise(swap_promise.Pass());
[email protected]66239a22013-06-05 03:38:26270}
271
[email protected]cd9a61c72012-05-08 19:16:59272void Compositor::SetScaleAndSize(float scale, const gfx::Size& size_in_pixel) {
[email protected]cff176a2012-06-29 21:11:00273 DCHECK_GT(scale, 0);
[email protected]351b8ce2012-11-27 23:37:16274 if (!size_in_pixel.IsEmpty()) {
275 size_ = size_in_pixel;
[email protected]18ce59702013-04-09 04:58:40276 host_->SetViewportSize(size_in_pixel);
[email protected]7aba6662013-03-12 10:17:34277 root_web_layer_->SetBounds(size_in_pixel);
jbauman616238a2014-12-03 03:17:53278 context_factory_->ResizeDisplay(this, size_in_pixel);
[email protected]351b8ce2012-11-27 23:37:16279 }
[email protected]2e2216e42012-05-17 15:17:00280 if (device_scale_factor_ != scale) {
[email protected]cd9a61c72012-05-08 19:16:59281 device_scale_factor_ = scale;
[email protected]caa21662014-05-14 10:02:32282 host_->SetDeviceScaleFactor(scale);
[email protected]cd9a61c72012-05-08 19:16:59283 if (root_layer_)
284 root_layer_->OnDeviceScaleFactorChanged(scale);
285 }
[email protected]ed8de92d2011-09-14 04:16:48286}
287
[email protected]87601922013-04-02 03:56:42288void Compositor::SetBackgroundColor(SkColor color) {
289 host_->set_background_color(color);
290 ScheduleDraw();
291}
292
ccameron18bbc2a2014-08-28 22:36:16293void Compositor::SetVisible(bool visible) {
294 host_->SetVisible(visible);
295}
296
[email protected]2bd1fcf02014-02-12 22:35:53297scoped_refptr<CompositorVSyncManager> Compositor::vsync_manager() const {
298 return vsync_manager_;
299}
300
[email protected]ed8de92d2011-09-14 04:16:48301void Compositor::AddObserver(CompositorObserver* observer) {
[email protected]80be33882014-08-13 09:34:19302#if defined(OS_MACOSX)
303 // Debugging instrumentation for crbug.com/401630.
304 // TODO(ccameron): remove this.
305 CHECK(observer);
[email protected]6b4504f2014-08-09 10:29:47306 if (!observer_list_.HasObserver(observer))
307 observer->observing_count_ += 1;
308#endif
309
[email protected]ed8de92d2011-09-14 04:16:48310 observer_list_.AddObserver(observer);
311}
312
313void Compositor::RemoveObserver(CompositorObserver* observer) {
[email protected]80be33882014-08-13 09:34:19314#if defined(OS_MACOSX)
315 // Debugging instrumentation for crbug.com/401630.
316 // TODO(ccameron): remove this.
[email protected]6b4504f2014-08-09 10:29:47317 if (observer_list_.HasObserver(observer))
318 observer->observing_count_ -= 1;
319#endif
320
[email protected]ed8de92d2011-09-14 04:16:48321 observer_list_.RemoveObserver(observer);
322}
323
mgiuca64ccf2362014-11-10 06:44:23324bool Compositor::HasObserver(const CompositorObserver* observer) const {
[email protected]3ce2feb2011-09-19 18:44:23325 return observer_list_.HasObserver(observer);
326}
327
[email protected]5ffe6d142014-08-01 16:04:38328void Compositor::AddAnimationObserver(CompositorAnimationObserver* observer) {
329 animation_observer_list_.AddObserver(observer);
330 host_->SetNeedsAnimate();
331}
332
333void Compositor::RemoveAnimationObserver(
334 CompositorAnimationObserver* observer) {
335 animation_observer_list_.RemoveObserver(observer);
336}
337
mgiuca64ccf2362014-11-10 06:44:23338bool Compositor::HasAnimationObserver(
339 const CompositorAnimationObserver* observer) const {
[email protected]5ffe6d142014-08-01 16:04:38340 return animation_observer_list_.HasObserver(observer);
341}
342
[email protected]04c5900d2014-08-18 13:38:36343void Compositor::BeginMainFrame(const cc::BeginFrameArgs& args) {
[email protected]5ffe6d142014-08-01 16:04:38344 FOR_EACH_OBSERVER(CompositorAnimationObserver,
345 animation_observer_list_,
[email protected]04c5900d2014-08-18 13:38:36346 OnAnimationStep(args.frame_time));
[email protected]5ffe6d142014-08-01 16:04:38347 if (animation_observer_list_.might_have_observers())
[email protected]9034a282014-06-05 03:11:47348 host_->SetNeedsAnimate();
349}
350
[email protected]408b5e22013-03-19 09:48:09351void Compositor::Layout() {
[email protected]d4ae80572012-06-06 23:02:06352 // We're sending damage that will be addressed during this composite
353 // cycle, so we don't need to schedule another composite to address it.
354 disable_schedule_composite_ = true;
[email protected]f78649ea2012-02-23 18:39:04355 if (root_layer_)
[email protected]cedc3952012-03-06 06:15:55356 root_layer_->SendDamagedRects();
[email protected]d4ae80572012-06-06 23:02:06357 disable_schedule_composite_ = false;
[email protected]ab6627372012-01-29 21:22:13358}
359
enne7f8fdde2014-12-10 21:32:09360void Compositor::RequestNewOutputSurface() {
361 bool fallback =
362 num_failed_recreate_attempts_ >= OUTPUT_SURFACE_RETRIES_BEFORE_FALLBACK;
jbauman1a7a5122014-10-28 00:22:52363 context_factory_->CreateOutputSurface(weak_ptr_factory_.GetWeakPtr(),
364 fallback);
[email protected]ab6627372012-01-29 21:22:13365}
366
enne7f8fdde2014-12-10 21:32:09367void Compositor::DidInitializeOutputSurface() {
368 num_failed_recreate_attempts_ = 0;
369}
370
371void Compositor::DidFailToInitializeOutputSurface() {
372 num_failed_recreate_attempts_++;
373
374 // Tolerate a certain number of recreation failures to work around races
375 // in the output-surface-lost machinery.
376 if (num_failed_recreate_attempts_ >= MAX_OUTPUT_SURFACE_RETRIES)
377 LOG(FATAL) << "Failed to create a fallback OutputSurface.";
378
379 base::MessageLoop::current()->PostTask(
380 FROM_HERE, base::Bind(&Compositor::RequestNewOutputSurface,
381 weak_ptr_factory_.GetWeakPtr()));
382}
383
[email protected]408b5e22013-03-19 09:48:09384void Compositor::DidCommit() {
[email protected]6b16679e2012-10-27 00:44:28385 DCHECK(!IsLocked());
[email protected]2700daddd2012-07-13 19:35:37386 FOR_EACH_OBSERVER(CompositorObserver,
387 observer_list_,
388 OnCompositingDidCommit(this));
389}
390
[email protected]408b5e22013-03-19 09:48:09391void Compositor::DidCommitAndDrawFrame() {
weiliangcb821b71f2014-12-05 01:54:13392 base::TimeTicks start_time = gfx::FrameTime::Now();
isherman4f0404e12014-12-06 01:01:33393 FOR_EACH_OBSERVER(CompositorObserver,
394 observer_list_,
weiliangcb821b71f2014-12-05 01:54:13395 OnCompositingStarted(this, start_time));
weiliangc1f27b282014-10-09 17:10:33396}
397
isherman4f0404e12014-12-06 01:01:33398void Compositor::DidCompleteSwapBuffers() {
399 if (compositor_thread_loop_.get()) {
400 NotifyEnd();
401 } else {
402 DCHECK_EQ(swap_state_, SWAP_POSTED);
403 NotifyEnd();
404 swap_state_ = SWAP_COMPLETED;
405 }
406}
407
408void Compositor::ScheduleComposite() {
409 if (!disable_schedule_composite_)
410 ScheduleDraw();
411}
412
413void Compositor::ScheduleAnimation() {
414 ScheduleComposite();
415}
416
417void Compositor::DidPostSwapBuffers() {
418 DCHECK(!compositor_thread_loop_.get());
419 DCHECK_EQ(swap_state_, SWAP_NONE);
420 swap_state_ = SWAP_POSTED;
421}
422
[email protected]4d7e46a2013-11-08 05:33:40423void Compositor::DidAbortSwapBuffers() {
isherman4f0404e12014-12-06 01:01:33424 if (!compositor_thread_loop_.get()) {
425 if (swap_state_ == SWAP_POSTED) {
426 NotifyEnd();
427 swap_state_ = SWAP_COMPLETED;
428 }
429 }
430
[email protected]4d7e46a2013-11-08 05:33:40431 FOR_EACH_OBSERVER(CompositorObserver,
432 observer_list_,
433 OnCompositingAborted(this));
434}
435
[email protected]2e77cdbb2013-04-29 13:59:14436const cc::LayerTreeDebugState& Compositor::GetLayerTreeDebugState() const {
437 return host_->debug_state();
[email protected]918f8db42013-04-27 01:53:40438}
439
[email protected]2e77cdbb2013-04-29 13:59:14440void Compositor::SetLayerTreeDebugState(
441 const cc::LayerTreeDebugState& debug_state) {
[email protected]918f8db42013-04-27 01:53:40442 host_->SetDebugState(debug_state);
443}
444
jbaumanc5be44c2014-11-20 22:17:12445const cc::RendererSettings& Compositor::GetRendererSettings() const {
446 return host_->settings().renderer_settings;
447}
448
[email protected]6b16679e2012-10-27 00:44:28449scoped_refptr<CompositorLock> Compositor::GetCompositorLock() {
450 if (!compositor_lock_) {
451 compositor_lock_ = new CompositorLock(this);
isherman4f0404e12014-12-06 01:01:33452 if (compositor_thread_loop_.get())
453 host_->SetDeferCommits(true);
[email protected]6b16679e2012-10-27 00:44:28454 FOR_EACH_OBSERVER(CompositorObserver,
455 observer_list_,
456 OnCompositingLockStateChanged(this));
457 }
458 return compositor_lock_;
459}
460
461void Compositor::UnlockCompositor() {
462 DCHECK(compositor_lock_);
463 compositor_lock_ = NULL;
isherman4f0404e12014-12-06 01:01:33464 if (compositor_thread_loop_.get())
465 host_->SetDeferCommits(false);
[email protected]6b16679e2012-10-27 00:44:28466 FOR_EACH_OBSERVER(CompositorObserver,
467 observer_list_,
468 OnCompositingLockStateChanged(this));
469}
470
471void Compositor::CancelCompositorLock() {
472 if (compositor_lock_)
473 compositor_lock_->CancelLock();
474}
475
isherman4f0404e12014-12-06 01:01:33476void Compositor::NotifyEnd() {
477 last_ended_frame_++;
478 TRACE_EVENT_ASYNC_END0("ui", "Compositor::Draw", last_ended_frame_);
479 waiting_on_compositing_end_ = false;
480 if (draw_on_compositing_end_) {
481 draw_on_compositing_end_ = false;
482
483 // Call ScheduleDraw() instead of Draw() in order to allow other
484 // CompositorObservers to be notified before starting another
485 // draw cycle.
486 ScheduleDraw();
487 }
488 FOR_EACH_OBSERVER(
489 CompositorObserver, observer_list_, OnCompositingEnded(this));
490}
491
[email protected]c797cd42011-03-15 02:18:36492} // namespace ui