• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // 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 
5 #include "cc/scheduler/scheduler.h"
6 
7 #include <algorithm>
8 #include "base/auto_reset.h"
9 #include "base/debug/trace_event.h"
10 #include "base/logging.h"
11 #include "cc/debug/devtools_instrumentation.h"
12 #include "cc/debug/traced_value.h"
13 #include "ui/gfx/frame_time.h"
14 
15 namespace cc {
16 
Scheduler(SchedulerClient * client,const SchedulerSettings & scheduler_settings,int layer_tree_host_id)17 Scheduler::Scheduler(SchedulerClient* client,
18                      const SchedulerSettings& scheduler_settings,
19                      int layer_tree_host_id)
20     : settings_(scheduler_settings),
21       client_(client),
22       layer_tree_host_id_(layer_tree_host_id),
23       last_set_needs_begin_impl_frame_(false),
24       state_machine_(scheduler_settings),
25       inside_process_scheduled_actions_(false),
26       inside_action_(SchedulerStateMachine::ACTION_NONE),
27       weak_factory_(this) {
28   DCHECK(client_);
29   DCHECK(!state_machine_.BeginImplFrameNeeded());
30 }
31 
~Scheduler()32 Scheduler::~Scheduler() {}
33 
SetCanStart()34 void Scheduler::SetCanStart() {
35   state_machine_.SetCanStart();
36   ProcessScheduledActions();
37 }
38 
SetVisible(bool visible)39 void Scheduler::SetVisible(bool visible) {
40   state_machine_.SetVisible(visible);
41   ProcessScheduledActions();
42 }
43 
SetCanDraw(bool can_draw)44 void Scheduler::SetCanDraw(bool can_draw) {
45   state_machine_.SetCanDraw(can_draw);
46   ProcessScheduledActions();
47 }
48 
NotifyReadyToActivate()49 void Scheduler::NotifyReadyToActivate() {
50   state_machine_.NotifyReadyToActivate();
51   ProcessScheduledActions();
52 }
53 
ActivatePendingTree()54 void Scheduler::ActivatePendingTree() {
55   client_->ScheduledActionActivatePendingTree();
56 }
57 
SetNeedsCommit()58 void Scheduler::SetNeedsCommit() {
59   state_machine_.SetNeedsCommit();
60   ProcessScheduledActions();
61 }
62 
SetNeedsForcedCommitForReadback()63 void Scheduler::SetNeedsForcedCommitForReadback() {
64   state_machine_.SetNeedsCommit();
65   state_machine_.SetNeedsForcedCommitForReadback();
66   ProcessScheduledActions();
67 }
68 
SetNeedsRedraw()69 void Scheduler::SetNeedsRedraw() {
70   state_machine_.SetNeedsRedraw();
71   ProcessScheduledActions();
72 }
73 
SetNeedsManageTiles()74 void Scheduler::SetNeedsManageTiles() {
75   DCHECK(!IsInsideAction(SchedulerStateMachine::ACTION_MANAGE_TILES));
76   state_machine_.SetNeedsManageTiles();
77   ProcessScheduledActions();
78 }
79 
SetSwapUsedIncompleteTile(bool used_incomplete_tile)80 void Scheduler::SetSwapUsedIncompleteTile(bool used_incomplete_tile) {
81   state_machine_.SetSwapUsedIncompleteTile(used_incomplete_tile);
82   ProcessScheduledActions();
83 }
84 
SetSmoothnessTakesPriority(bool smoothness_takes_priority)85 void Scheduler::SetSmoothnessTakesPriority(bool smoothness_takes_priority) {
86   state_machine_.SetSmoothnessTakesPriority(smoothness_takes_priority);
87   ProcessScheduledActions();
88 }
89 
SetMainThreadNeedsLayerTextures()90 void Scheduler::SetMainThreadNeedsLayerTextures() {
91   state_machine_.SetMainThreadNeedsLayerTextures();
92   ProcessScheduledActions();
93 }
94 
FinishCommit()95 void Scheduler::FinishCommit() {
96   TRACE_EVENT0("cc", "Scheduler::FinishCommit");
97   state_machine_.FinishCommit();
98   ProcessScheduledActions();
99 }
100 
BeginMainFrameAborted(bool did_handle)101 void Scheduler::BeginMainFrameAborted(bool did_handle) {
102   TRACE_EVENT0("cc", "Scheduler::BeginMainFrameAborted");
103   state_machine_.BeginMainFrameAborted(did_handle);
104   ProcessScheduledActions();
105 }
106 
DidManageTiles()107 void Scheduler::DidManageTiles() {
108   state_machine_.DidManageTiles();
109 }
110 
DidLoseOutputSurface()111 void Scheduler::DidLoseOutputSurface() {
112   TRACE_EVENT0("cc", "Scheduler::DidLoseOutputSurface");
113   last_set_needs_begin_impl_frame_ = false;
114   begin_impl_frame_deadline_closure_.Cancel();
115   state_machine_.DidLoseOutputSurface();
116   ProcessScheduledActions();
117 }
118 
DidCreateAndInitializeOutputSurface()119 void Scheduler::DidCreateAndInitializeOutputSurface() {
120   TRACE_EVENT0("cc", "Scheduler::DidCreateAndInitializeOutputSurface");
121   DCHECK(!last_set_needs_begin_impl_frame_);
122   DCHECK(begin_impl_frame_deadline_closure_.IsCancelled());
123   state_machine_.DidCreateAndInitializeOutputSurface();
124   ProcessScheduledActions();
125 }
126 
AnticipatedDrawTime()127 base::TimeTicks Scheduler::AnticipatedDrawTime() {
128   TRACE_EVENT0("cc", "Scheduler::AnticipatedDrawTime");
129 
130   if (!last_set_needs_begin_impl_frame_ ||
131       last_begin_impl_frame_args_.interval <= base::TimeDelta())
132     return base::TimeTicks();
133 
134   base::TimeTicks now = gfx::FrameTime::Now();
135   base::TimeTicks timebase = std::max(last_begin_impl_frame_args_.frame_time,
136                                       last_begin_impl_frame_args_.deadline);
137   int64 intervals =
138       1 + ((now - timebase) / last_begin_impl_frame_args_.interval);
139   return timebase + (last_begin_impl_frame_args_.interval * intervals);
140 }
141 
LastBeginImplFrameTime()142 base::TimeTicks Scheduler::LastBeginImplFrameTime() {
143   return last_begin_impl_frame_args_.frame_time;
144 }
145 
SetupNextBeginImplFrameIfNeeded()146 void Scheduler::SetupNextBeginImplFrameIfNeeded() {
147   bool needs_begin_impl_frame =
148       state_machine_.BeginImplFrameNeeded();
149 
150   bool at_end_of_deadline =
151       state_machine_.begin_impl_frame_state() ==
152           SchedulerStateMachine::BEGIN_IMPL_FRAME_STATE_INSIDE_DEADLINE;
153 
154   bool should_call_set_needs_begin_impl_frame =
155       // Always request the BeginImplFrame immediately if it wasn't needed
156       // before.
157       (needs_begin_impl_frame && !last_set_needs_begin_impl_frame_) ||
158       // We always need to explicitly request our next BeginImplFrame.
159       at_end_of_deadline;
160 
161   if (should_call_set_needs_begin_impl_frame) {
162     client_->SetNeedsBeginImplFrame(needs_begin_impl_frame);
163     last_set_needs_begin_impl_frame_ = needs_begin_impl_frame;
164   }
165 
166   bool needs_advance_commit_state_timer = false;
167   // Setup PollForAnticipatedDrawTriggers if we need to monitor state but
168   // aren't expecting any more BeginImplFrames. This should only be needed by
169   // the synchronous compositor when BeginImplFrameNeeded is false.
170   if (state_machine_.ShouldPollForAnticipatedDrawTriggers()) {
171     DCHECK(!state_machine_.SupportsProactiveBeginImplFrame());
172     DCHECK(!needs_begin_impl_frame);
173     if (poll_for_draw_triggers_closure_.IsCancelled()) {
174       poll_for_draw_triggers_closure_.Reset(
175           base::Bind(&Scheduler::PollForAnticipatedDrawTriggers,
176                      weak_factory_.GetWeakPtr()));
177       base::MessageLoop::current()->PostDelayedTask(
178           FROM_HERE,
179           poll_for_draw_triggers_closure_.callback(),
180           last_begin_impl_frame_args_.interval);
181     }
182   } else {
183     poll_for_draw_triggers_closure_.Cancel();
184 
185     // At this point we'd prefer to advance through the commit flow by
186     // drawing a frame, however it's possible that the frame rate controller
187     // will not give us a BeginImplFrame until the commit completes.  See
188     // crbug.com/317430 for an example of a swap ack being held on commit. Thus
189     // we set a repeating timer to poll on ProcessScheduledActions until we
190     // successfully reach BeginImplFrame.
191     if (state_machine_.IsCommitStateWaiting())
192       needs_advance_commit_state_timer = true;
193   }
194   if (needs_advance_commit_state_timer !=
195       advance_commit_state_timer_.IsRunning()) {
196     if (needs_advance_commit_state_timer &&
197         last_begin_impl_frame_args_.IsValid()) {
198     // Since we'd rather get a BeginImplFrame by the normally mechanism, we set
199     // the interval to twice the interval from the previous frame.
200       advance_commit_state_timer_.Start(
201           FROM_HERE,
202           last_begin_impl_frame_args_.interval * 2,
203           base::Bind(&Scheduler::ProcessScheduledActions,
204                      base::Unretained(this)));
205     } else {
206       advance_commit_state_timer_.Stop();
207     }
208   }
209 }
210 
BeginImplFrame(const BeginFrameArgs & args)211 void Scheduler::BeginImplFrame(const BeginFrameArgs& args) {
212   TRACE_EVENT0("cc", "Scheduler::BeginImplFrame");
213   DCHECK(state_machine_.begin_impl_frame_state() ==
214          SchedulerStateMachine::BEGIN_IMPL_FRAME_STATE_IDLE);
215   DCHECK(state_machine_.HasInitializedOutputSurface());
216   last_begin_impl_frame_args_ = args;
217   last_begin_impl_frame_args_.deadline -= client_->DrawDurationEstimate();
218   state_machine_.OnBeginImplFrame(last_begin_impl_frame_args_);
219 
220   if (settings_.switch_to_low_latency_if_possible) {
221     state_machine_.SetSkipBeginMainFrameToReduceLatency(
222         state_machine_.MainThreadIsInHighLatencyMode() &&
223             CanCommitAndActivateBeforeDeadline());
224   }
225 
226   ProcessScheduledActions();
227 
228   if (!state_machine_.HasInitializedOutputSurface())
229     return;
230 
231   state_machine_.OnBeginImplFrameDeadlinePending();
232   devtools_instrumentation::didBeginFrame(layer_tree_host_id_);
233   if (settings_.using_synchronous_renderer_compositor) {
234     // The synchronous renderer compositor has to make its GL calls
235     // within this call to BeginImplFrame.
236     // TODO(brianderson): Have the OutputSurface initiate the deadline tasks
237     // so the sychronous renderer compositor can take advantage of splitting
238     // up the BeginImplFrame and deadline as well.
239     OnBeginImplFrameDeadline();
240   } else if (!settings_.deadline_scheduling_enabled) {
241     // We emulate the old non-deadline scheduler here by posting the
242     // deadline task without any delay.
243     PostBeginImplFrameDeadline(base::TimeTicks());
244   } else if (state_machine_.ShouldTriggerBeginImplFrameDeadlineEarly()) {
245     // We are ready to draw a new active tree immediately.
246     PostBeginImplFrameDeadline(base::TimeTicks());
247   } else if (state_machine_.needs_redraw()) {
248     // We have an animation or fast input path on the impl thread that wants
249     // to draw, so don't wait too long for a new active tree.
250     PostBeginImplFrameDeadline(last_begin_impl_frame_args_.deadline);
251   } else {
252     // The impl thread doesn't have anything it wants to draw and we are just
253     // waiting for a new active tree, so post the deadline for the next
254     // expected BeginImplFrame start. This allows us to draw immediately when
255     // there is a new active tree, instead of waiting for the next
256     // BeginImplFrame.
257     // TODO(brianderson): Handle long deadlines (that are past the next frame's
258     // frame time) properly instead of using this hack.
259     PostBeginImplFrameDeadline(last_begin_impl_frame_args_.frame_time +
260                                last_begin_impl_frame_args_.interval);
261   }
262 }
263 
PostBeginImplFrameDeadline(base::TimeTicks deadline)264 void Scheduler::PostBeginImplFrameDeadline(base::TimeTicks deadline) {
265   begin_impl_frame_deadline_closure_.Cancel();
266   begin_impl_frame_deadline_closure_.Reset(
267       base::Bind(&Scheduler::OnBeginImplFrameDeadline,
268                  weak_factory_.GetWeakPtr()));
269   client_->PostBeginImplFrameDeadline(
270       begin_impl_frame_deadline_closure_.callback(), deadline);
271 }
272 
OnBeginImplFrameDeadline()273 void Scheduler::OnBeginImplFrameDeadline() {
274   TRACE_EVENT0("cc", "Scheduler::OnBeginImplFrameDeadline");
275   DCHECK(state_machine_.HasInitializedOutputSurface());
276   begin_impl_frame_deadline_closure_.Cancel();
277   state_machine_.OnBeginImplFrameDeadline();
278   ProcessScheduledActions();
279 
280   if (state_machine_.HasInitializedOutputSurface()) {
281     // We only transition out of BEGIN_IMPL_FRAME_STATE_INSIDE_DEADLINE when all
282     // actions that occur back-to-back in response to entering
283     // BEGIN_IMPL_FRAME_STATE_INSIDE_DEADLINE have completed. This is important
284     // because sending the BeginMainFrame will not occur if we transition to
285     // BEGIN_IMPL_FRAME_STATE_IDLE too early.
286     state_machine_.OnBeginImplFrameIdle();
287   }
288 
289   client_->DidBeginImplFrameDeadline();
290 }
291 
PollForAnticipatedDrawTriggers()292 void Scheduler::PollForAnticipatedDrawTriggers() {
293   TRACE_EVENT0("cc", "Scheduler::PollForAnticipatedDrawTriggers");
294   poll_for_draw_triggers_closure_.Cancel();
295   state_machine_.DidEnterPollForAnticipatedDrawTriggers();
296   ProcessScheduledActions();
297   state_machine_.DidLeavePollForAnticipatedDrawTriggers();
298 }
299 
DrawAndSwapIfPossible()300 void Scheduler::DrawAndSwapIfPossible() {
301   DrawSwapReadbackResult result =
302       client_->ScheduledActionDrawAndSwapIfPossible();
303   state_machine_.DidDrawIfPossibleCompleted(result.did_draw);
304 }
305 
DrawAndSwapForced()306 void Scheduler::DrawAndSwapForced() {
307   client_->ScheduledActionDrawAndSwapForced();
308 }
309 
DrawAndReadback()310 void Scheduler::DrawAndReadback() {
311   DrawSwapReadbackResult result = client_->ScheduledActionDrawAndReadback();
312   DCHECK(!result.did_swap);
313 }
314 
ProcessScheduledActions()315 void Scheduler::ProcessScheduledActions() {
316   // We do not allow ProcessScheduledActions to be recursive.
317   // The top-level call will iteratively execute the next action for us anyway.
318   if (inside_process_scheduled_actions_)
319     return;
320 
321   base::AutoReset<bool> mark_inside(&inside_process_scheduled_actions_, true);
322 
323   SchedulerStateMachine::Action action;
324   do {
325     state_machine_.CheckInvariants();
326     action = state_machine_.NextAction();
327     TRACE_EVENT1(TRACE_DISABLED_BY_DEFAULT("cc.debug.scheduler"),
328                  "SchedulerStateMachine",
329                  "state",
330                  TracedValue::FromValue(state_machine_.AsValue().release()));
331     state_machine_.UpdateState(action);
332     base::AutoReset<SchedulerStateMachine::Action>
333         mark_inside_action(&inside_action_, action);
334     switch (action) {
335       case SchedulerStateMachine::ACTION_NONE:
336         break;
337       case SchedulerStateMachine::ACTION_SEND_BEGIN_MAIN_FRAME:
338         client_->ScheduledActionSendBeginMainFrame();
339         break;
340       case SchedulerStateMachine::ACTION_COMMIT:
341         client_->ScheduledActionCommit();
342         break;
343       case SchedulerStateMachine::ACTION_UPDATE_VISIBLE_TILES:
344         client_->ScheduledActionUpdateVisibleTiles();
345         break;
346       case SchedulerStateMachine::ACTION_ACTIVATE_PENDING_TREE:
347         ActivatePendingTree();
348         break;
349       case SchedulerStateMachine::ACTION_DRAW_AND_SWAP_IF_POSSIBLE:
350         DrawAndSwapIfPossible();
351         break;
352       case SchedulerStateMachine::ACTION_DRAW_AND_SWAP_FORCED:
353         DrawAndSwapForced();
354         break;
355       case SchedulerStateMachine::ACTION_DRAW_AND_SWAP_ABORT:
356         // No action is actually performed, but this allows the state machine to
357         // advance out of its waiting to draw state without actually drawing.
358         break;
359       case SchedulerStateMachine::ACTION_DRAW_AND_READBACK:
360         DrawAndReadback();
361         break;
362       case SchedulerStateMachine::ACTION_BEGIN_OUTPUT_SURFACE_CREATION:
363         client_->ScheduledActionBeginOutputSurfaceCreation();
364         break;
365       case SchedulerStateMachine::ACTION_ACQUIRE_LAYER_TEXTURES_FOR_MAIN_THREAD:
366         client_->ScheduledActionAcquireLayerTexturesForMainThread();
367         break;
368       case SchedulerStateMachine::ACTION_MANAGE_TILES:
369         client_->ScheduledActionManageTiles();
370         break;
371     }
372   } while (action != SchedulerStateMachine::ACTION_NONE);
373 
374   SetupNextBeginImplFrameIfNeeded();
375   client_->DidAnticipatedDrawTimeChange(AnticipatedDrawTime());
376 
377   if (state_machine_.ShouldTriggerBeginImplFrameDeadlineEarly())
378     PostBeginImplFrameDeadline(base::TimeTicks());
379 }
380 
WillDrawIfNeeded() const381 bool Scheduler::WillDrawIfNeeded() const {
382   return !state_machine_.PendingDrawsShouldBeAborted();
383 }
384 
CanCommitAndActivateBeforeDeadline() const385 bool Scheduler::CanCommitAndActivateBeforeDeadline() const {
386   // Check if the main thread computation and commit can be finished before the
387   // impl thread's deadline.
388   base::TimeTicks estimated_draw_time =
389       last_begin_impl_frame_args_.frame_time +
390       client_->BeginMainFrameToCommitDurationEstimate() +
391       client_->CommitToActivateDurationEstimate();
392 
393   return estimated_draw_time < last_begin_impl_frame_args_.deadline;
394 }
395 
396 }  // namespace cc
397