OLD | NEW |
1 // Copyright 2011 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "cc/scheduler/scheduler.h" | 5 #include "cc/scheduler/scheduler.h" |
6 | 6 |
7 #include <algorithm> | 7 #include <algorithm> |
8 | 8 |
9 #include "base/auto_reset.h" | 9 #include "base/auto_reset.h" |
10 #include "base/logging.h" | 10 #include "base/logging.h" |
11 #include "base/profiler/scoped_tracker.h" | 11 #include "base/profiler/scoped_tracker.h" |
12 #include "base/single_thread_task_runner.h" | 12 #include "base/single_thread_task_runner.h" |
13 #include "base/trace_event/trace_event.h" | 13 #include "base/trace_event/trace_event.h" |
14 #include "base/trace_event/trace_event_argument.h" | 14 #include "base/trace_event/trace_event_argument.h" |
15 #include "cc/debug/devtools_instrumentation.h" | 15 #include "cc/debug/devtools_instrumentation.h" |
16 #include "cc/debug/traced_value.h" | 16 #include "cc/debug/traced_value.h" |
| 17 #include "cc/scheduler/compositor_timing_history.h" |
17 #include "cc/scheduler/delay_based_time_source.h" | 18 #include "cc/scheduler/delay_based_time_source.h" |
18 | 19 |
19 namespace cc { | 20 namespace cc { |
20 | 21 |
21 scoped_ptr<Scheduler> Scheduler::Create( | 22 scoped_ptr<Scheduler> Scheduler::Create( |
22 SchedulerClient* client, | 23 SchedulerClient* client, |
23 const SchedulerSettings& settings, | 24 const SchedulerSettings& settings, |
24 int layer_tree_host_id, | 25 int layer_tree_host_id, |
25 base::SingleThreadTaskRunner* task_runner, | 26 base::SingleThreadTaskRunner* task_runner, |
26 BeginFrameSource* external_frame_source) { | 27 BeginFrameSource* external_frame_source, |
| 28 scoped_ptr<CompositorTimingHistory> compositor_timing_history) { |
27 scoped_ptr<SyntheticBeginFrameSource> synthetic_frame_source; | 29 scoped_ptr<SyntheticBeginFrameSource> synthetic_frame_source; |
28 if (!settings.use_external_begin_frame_source) { | 30 if (!settings.use_external_begin_frame_source) { |
29 synthetic_frame_source = SyntheticBeginFrameSource::Create( | 31 synthetic_frame_source = SyntheticBeginFrameSource::Create( |
30 task_runner, BeginFrameArgs::DefaultInterval()); | 32 task_runner, BeginFrameArgs::DefaultInterval()); |
31 } | 33 } |
32 scoped_ptr<BackToBackBeginFrameSource> unthrottled_frame_source = | 34 scoped_ptr<BackToBackBeginFrameSource> unthrottled_frame_source = |
33 BackToBackBeginFrameSource::Create(task_runner); | 35 BackToBackBeginFrameSource::Create(task_runner); |
34 return make_scoped_ptr(new Scheduler( | 36 return make_scoped_ptr(new Scheduler( |
35 client, settings, layer_tree_host_id, task_runner, external_frame_source, | 37 client, settings, layer_tree_host_id, task_runner, external_frame_source, |
36 synthetic_frame_source.Pass(), unthrottled_frame_source.Pass())); | 38 synthetic_frame_source.Pass(), unthrottled_frame_source.Pass(), |
| 39 compositor_timing_history.Pass())); |
37 } | 40 } |
38 | 41 |
39 Scheduler::Scheduler( | 42 Scheduler::Scheduler( |
40 SchedulerClient* client, | 43 SchedulerClient* client, |
41 const SchedulerSettings& settings, | 44 const SchedulerSettings& settings, |
42 int layer_tree_host_id, | 45 int layer_tree_host_id, |
43 base::SingleThreadTaskRunner* task_runner, | 46 base::SingleThreadTaskRunner* task_runner, |
44 BeginFrameSource* external_frame_source, | 47 BeginFrameSource* external_frame_source, |
45 scoped_ptr<SyntheticBeginFrameSource> synthetic_frame_source, | 48 scoped_ptr<SyntheticBeginFrameSource> synthetic_frame_source, |
46 scoped_ptr<BackToBackBeginFrameSource> unthrottled_frame_source) | 49 scoped_ptr<BackToBackBeginFrameSource> unthrottled_frame_source, |
| 50 scoped_ptr<CompositorTimingHistory> compositor_timing_history) |
47 : settings_(settings), | 51 : settings_(settings), |
48 client_(client), | 52 client_(client), |
49 layer_tree_host_id_(layer_tree_host_id), | 53 layer_tree_host_id_(layer_tree_host_id), |
50 task_runner_(task_runner), | 54 task_runner_(task_runner), |
51 external_frame_source_(external_frame_source), | 55 external_frame_source_(external_frame_source), |
52 synthetic_frame_source_(synthetic_frame_source.Pass()), | 56 synthetic_frame_source_(synthetic_frame_source.Pass()), |
53 unthrottled_frame_source_(unthrottled_frame_source.Pass()), | 57 unthrottled_frame_source_(unthrottled_frame_source.Pass()), |
54 frame_source_(BeginFrameSourceMultiplexer::Create()), | 58 frame_source_(BeginFrameSourceMultiplexer::Create()), |
55 throttle_frame_production_(false), | 59 throttle_frame_production_(false), |
| 60 compositor_timing_history_(compositor_timing_history.Pass()), |
56 begin_impl_frame_deadline_mode_( | 61 begin_impl_frame_deadline_mode_( |
57 SchedulerStateMachine::BEGIN_IMPL_FRAME_DEADLINE_MODE_NONE), | 62 SchedulerStateMachine::BEGIN_IMPL_FRAME_DEADLINE_MODE_NONE), |
58 begin_impl_frame_tracker_(BEGINFRAMETRACKER_FROM_HERE), | 63 begin_impl_frame_tracker_(BEGINFRAMETRACKER_FROM_HERE), |
59 state_machine_(settings), | 64 state_machine_(settings), |
60 inside_process_scheduled_actions_(false), | 65 inside_process_scheduled_actions_(false), |
61 inside_action_(SchedulerStateMachine::ACTION_NONE), | 66 inside_action_(SchedulerStateMachine::ACTION_NONE), |
62 weak_factory_(this) { | 67 weak_factory_(this) { |
63 TRACE_EVENT1("cc", "Scheduler::Scheduler", "settings", settings_.AsValue()); | 68 TRACE_EVENT1("cc", "Scheduler::Scheduler", "settings", settings_.AsValue()); |
64 DCHECK(client_); | 69 DCHECK(client_); |
65 DCHECK(!state_machine_.BeginFrameNeeded()); | 70 DCHECK(!state_machine_.BeginFrameNeeded()); |
(...skipping 366 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
432 void Scheduler::BeginImplFrameWithDeadline(const BeginFrameArgs& args) { | 437 void Scheduler::BeginImplFrameWithDeadline(const BeginFrameArgs& args) { |
433 bool main_thread_is_in_high_latency_mode = | 438 bool main_thread_is_in_high_latency_mode = |
434 state_machine_.MainThreadIsInHighLatencyMode(); | 439 state_machine_.MainThreadIsInHighLatencyMode(); |
435 TRACE_EVENT2("cc,benchmark", "Scheduler::BeginImplFrame", "args", | 440 TRACE_EVENT2("cc,benchmark", "Scheduler::BeginImplFrame", "args", |
436 args.AsValue(), "main_thread_is_high_latency", | 441 args.AsValue(), "main_thread_is_high_latency", |
437 main_thread_is_in_high_latency_mode); | 442 main_thread_is_in_high_latency_mode); |
438 TRACE_COUNTER1(TRACE_DISABLED_BY_DEFAULT("cc.debug.scheduler"), | 443 TRACE_COUNTER1(TRACE_DISABLED_BY_DEFAULT("cc.debug.scheduler"), |
439 "MainThreadLatency", main_thread_is_in_high_latency_mode); | 444 "MainThreadLatency", main_thread_is_in_high_latency_mode); |
440 | 445 |
441 BeginFrameArgs adjusted_args = args; | 446 BeginFrameArgs adjusted_args = args; |
442 adjusted_args.deadline -= client_->DrawDurationEstimate(); | 447 adjusted_args.deadline -= compositor_timing_history_->DrawDurationEstimate(); |
443 | 448 |
444 if (!state_machine_.impl_latency_takes_priority() && | 449 if (!state_machine_.impl_latency_takes_priority() && |
445 main_thread_is_in_high_latency_mode && | 450 main_thread_is_in_high_latency_mode && |
446 CanCommitAndActivateBeforeDeadline()) { | 451 CanCommitAndActivateBeforeDeadline()) { |
447 state_machine_.SetSkipNextBeginMainFrameToReduceLatency(); | 452 state_machine_.SetSkipNextBeginMainFrameToReduceLatency(); |
448 } | 453 } |
449 | 454 |
450 BeginImplFrame(adjusted_args); | 455 BeginImplFrame(adjusted_args); |
451 | 456 |
452 // The deadline will be scheduled in ProcessScheduledActions. | 457 // The deadline will be scheduled in ProcessScheduledActions. |
(...skipping 114 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
567 // TODO(robliao): Remove ScopedTracker below once crbug.com/461509 is fixed. | 572 // TODO(robliao): Remove ScopedTracker below once crbug.com/461509 is fixed. |
568 tracked_objects::ScopedTracker tracking_profile1( | 573 tracked_objects::ScopedTracker tracking_profile1( |
569 FROM_HERE_WITH_EXPLICIT_FUNCTION( | 574 FROM_HERE_WITH_EXPLICIT_FUNCTION( |
570 "461509 Scheduler::OnBeginImplFrameDeadline1")); | 575 "461509 Scheduler::OnBeginImplFrameDeadline1")); |
571 state_machine_.OnBeginImplFrameDeadline(); | 576 state_machine_.OnBeginImplFrameDeadline(); |
572 ProcessScheduledActions(); | 577 ProcessScheduledActions(); |
573 FinishImplFrame(); | 578 FinishImplFrame(); |
574 } | 579 } |
575 | 580 |
576 void Scheduler::DrawAndSwapIfPossible() { | 581 void Scheduler::DrawAndSwapIfPossible() { |
| 582 compositor_timing_history_->DidStartDrawing(); |
577 DrawResult result = client_->ScheduledActionDrawAndSwapIfPossible(); | 583 DrawResult result = client_->ScheduledActionDrawAndSwapIfPossible(); |
578 state_machine_.DidDrawIfPossibleCompleted(result); | 584 state_machine_.DidDrawIfPossibleCompleted(result); |
| 585 compositor_timing_history_->DidFinishDrawing(); |
| 586 } |
| 587 |
| 588 void Scheduler::DrawAndSwapForced() { |
| 589 compositor_timing_history_->DidStartDrawing(); |
| 590 client_->ScheduledActionDrawAndSwapForced(); |
| 591 compositor_timing_history_->DidFinishDrawing(); |
579 } | 592 } |
580 | 593 |
581 void Scheduler::SetDeferCommits(bool defer_commits) { | 594 void Scheduler::SetDeferCommits(bool defer_commits) { |
582 TRACE_EVENT1("cc", "Scheduler::SetDeferCommits", | 595 TRACE_EVENT1("cc", "Scheduler::SetDeferCommits", |
583 "defer_commits", | 596 "defer_commits", |
584 defer_commits); | 597 defer_commits); |
585 state_machine_.SetDeferCommits(defer_commits); | 598 state_machine_.SetDeferCommits(defer_commits); |
586 ProcessScheduledActions(); | 599 ProcessScheduledActions(); |
587 } | 600 } |
588 | 601 |
(...skipping 18 matching lines...) Expand all Loading... |
607 state_machine_.UpdateState(action); | 620 state_machine_.UpdateState(action); |
608 base::AutoReset<SchedulerStateMachine::Action> | 621 base::AutoReset<SchedulerStateMachine::Action> |
609 mark_inside_action(&inside_action_, action); | 622 mark_inside_action(&inside_action_, action); |
610 switch (action) { | 623 switch (action) { |
611 case SchedulerStateMachine::ACTION_NONE: | 624 case SchedulerStateMachine::ACTION_NONE: |
612 break; | 625 break; |
613 case SchedulerStateMachine::ACTION_ANIMATE: | 626 case SchedulerStateMachine::ACTION_ANIMATE: |
614 client_->ScheduledActionAnimate(); | 627 client_->ScheduledActionAnimate(); |
615 break; | 628 break; |
616 case SchedulerStateMachine::ACTION_SEND_BEGIN_MAIN_FRAME: | 629 case SchedulerStateMachine::ACTION_SEND_BEGIN_MAIN_FRAME: |
| 630 compositor_timing_history_->WillBeginMainFrame(); |
617 client_->ScheduledActionSendBeginMainFrame(); | 631 client_->ScheduledActionSendBeginMainFrame(); |
618 break; | 632 break; |
619 case SchedulerStateMachine::ACTION_COMMIT: { | 633 case SchedulerStateMachine::ACTION_COMMIT: { |
620 // TODO(robliao): Remove ScopedTracker below once crbug.com/461509 is | 634 // TODO(robliao): Remove ScopedTracker below once crbug.com/461509 is |
621 // fixed. | 635 // fixed. |
622 tracked_objects::ScopedTracker tracking_profile4( | 636 tracked_objects::ScopedTracker tracking_profile4( |
623 FROM_HERE_WITH_EXPLICIT_FUNCTION( | 637 FROM_HERE_WITH_EXPLICIT_FUNCTION( |
624 "461509 Scheduler::ProcessScheduledActions4")); | 638 "461509 Scheduler::ProcessScheduledActions4")); |
625 client_->ScheduledActionCommit(); | 639 client_->ScheduledActionCommit(); |
| 640 compositor_timing_history_->DidCommit(); |
626 break; | 641 break; |
627 } | 642 } |
628 case SchedulerStateMachine::ACTION_ACTIVATE_SYNC_TREE: | 643 case SchedulerStateMachine::ACTION_ACTIVATE_SYNC_TREE: |
629 client_->ScheduledActionActivateSyncTree(); | 644 client_->ScheduledActionActivateSyncTree(); |
| 645 compositor_timing_history_->DidActivateSyncTree(); |
630 break; | 646 break; |
631 case SchedulerStateMachine::ACTION_DRAW_AND_SWAP_IF_POSSIBLE: { | 647 case SchedulerStateMachine::ACTION_DRAW_AND_SWAP_IF_POSSIBLE: { |
632 // TODO(robliao): Remove ScopedTracker below once crbug.com/461509 is | 648 // TODO(robliao): Remove ScopedTracker below once crbug.com/461509 is |
633 // fixed. | 649 // fixed. |
634 tracked_objects::ScopedTracker tracking_profile6( | 650 tracked_objects::ScopedTracker tracking_profile6( |
635 FROM_HERE_WITH_EXPLICIT_FUNCTION( | 651 FROM_HERE_WITH_EXPLICIT_FUNCTION( |
636 "461509 Scheduler::ProcessScheduledActions6")); | 652 "461509 Scheduler::ProcessScheduledActions6")); |
637 DrawAndSwapIfPossible(); | 653 DrawAndSwapIfPossible(); |
638 break; | 654 break; |
639 } | 655 } |
640 case SchedulerStateMachine::ACTION_DRAW_AND_SWAP_FORCED: | 656 case SchedulerStateMachine::ACTION_DRAW_AND_SWAP_FORCED: |
641 client_->ScheduledActionDrawAndSwapForced(); | 657 DrawAndSwapForced(); |
642 break; | 658 break; |
643 case SchedulerStateMachine::ACTION_DRAW_AND_SWAP_ABORT: | 659 case SchedulerStateMachine::ACTION_DRAW_AND_SWAP_ABORT: |
644 // No action is actually performed, but this allows the state machine to | 660 // No action is actually performed, but this allows the state machine to |
645 // advance out of its waiting to draw state without actually drawing. | 661 // advance out of its waiting to draw state without actually drawing. |
646 break; | 662 break; |
647 case SchedulerStateMachine::ACTION_BEGIN_OUTPUT_SURFACE_CREATION: | 663 case SchedulerStateMachine::ACTION_BEGIN_OUTPUT_SURFACE_CREATION: |
648 client_->ScheduledActionBeginOutputSurfaceCreation(); | 664 client_->ScheduledActionBeginOutputSurfaceCreation(); |
649 break; | 665 break; |
650 case SchedulerStateMachine::ACTION_PREPARE_TILES: | 666 case SchedulerStateMachine::ACTION_PREPARE_TILES: |
651 client_->ScheduledActionPrepareTiles(); | 667 client_->ScheduledActionPrepareTiles(); |
(...skipping 44 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
696 !begin_retro_frame_task_.IsCancelled()); | 712 !begin_retro_frame_task_.IsCancelled()); |
697 state->SetBoolean("begin_impl_frame_deadline_task", | 713 state->SetBoolean("begin_impl_frame_deadline_task", |
698 !begin_impl_frame_deadline_task_.IsCancelled()); | 714 !begin_impl_frame_deadline_task_.IsCancelled()); |
699 state->SetString("inside_action", | 715 state->SetString("inside_action", |
700 SchedulerStateMachine::ActionToString(inside_action_)); | 716 SchedulerStateMachine::ActionToString(inside_action_)); |
701 state->BeginDictionary("begin_impl_frame_args"); | 717 state->BeginDictionary("begin_impl_frame_args"); |
702 begin_impl_frame_tracker_.AsValueInto(Now(), state); | 718 begin_impl_frame_tracker_.AsValueInto(Now(), state); |
703 state->EndDictionary(); | 719 state->EndDictionary(); |
704 state->EndDictionary(); | 720 state->EndDictionary(); |
705 | 721 |
706 state->BeginDictionary("client_state"); | 722 state->BeginDictionary("compositor_timing_history"); |
707 state->SetDouble("draw_duration_estimate_ms", | 723 compositor_timing_history_->AsValueInto(state); |
708 client_->DrawDurationEstimate().InMillisecondsF()); | |
709 state->SetDouble( | |
710 "begin_main_frame_to_commit_duration_estimate_ms", | |
711 client_->BeginMainFrameToCommitDurationEstimate().InMillisecondsF()); | |
712 state->SetDouble( | |
713 "commit_to_activate_duration_estimate_ms", | |
714 client_->CommitToActivateDurationEstimate().InMillisecondsF()); | |
715 state->EndDictionary(); | 724 state->EndDictionary(); |
716 } | 725 } |
717 | 726 |
718 bool Scheduler::CanCommitAndActivateBeforeDeadline() const { | 727 bool Scheduler::CanCommitAndActivateBeforeDeadline() const { |
719 BeginFrameArgs args = | 728 BeginFrameArgs args = |
720 begin_impl_frame_tracker_.DangerousMethodCurrentOrLast(); | 729 begin_impl_frame_tracker_.DangerousMethodCurrentOrLast(); |
721 | 730 |
722 // Check if the main thread computation and commit can be finished before the | 731 // Check if the main thread computation and commit can be finished before the |
723 // impl thread's deadline. | 732 // impl thread's deadline. |
724 base::TimeTicks estimated_draw_time = | 733 base::TimeTicks estimated_draw_time = |
725 args.frame_time + client_->BeginMainFrameToCommitDurationEstimate() + | 734 args.frame_time + |
726 client_->CommitToActivateDurationEstimate(); | 735 compositor_timing_history_->BeginMainFrameToCommitDurationEstimate() + |
| 736 compositor_timing_history_->CommitToActivateDurationEstimate(); |
727 | 737 |
728 TRACE_EVENT2(TRACE_DISABLED_BY_DEFAULT("cc.debug.scheduler"), | 738 TRACE_EVENT2(TRACE_DISABLED_BY_DEFAULT("cc.debug.scheduler"), |
729 "CanCommitAndActivateBeforeDeadline", | 739 "CanCommitAndActivateBeforeDeadline", |
730 "time_left_after_drawing_ms", | 740 "time_left_after_drawing_ms", |
731 (args.deadline - estimated_draw_time).InMillisecondsF(), "state", | 741 (args.deadline - estimated_draw_time).InMillisecondsF(), "state", |
732 AsValue()); | 742 AsValue()); |
733 | 743 |
734 return estimated_draw_time < args.deadline; | 744 return estimated_draw_time < args.deadline; |
735 } | 745 } |
736 | 746 |
737 bool Scheduler::IsBeginMainFrameSentOrStarted() const { | 747 bool Scheduler::IsBeginMainFrameSentOrStarted() const { |
738 return (state_machine_.commit_state() == | 748 return (state_machine_.commit_state() == |
739 SchedulerStateMachine::COMMIT_STATE_BEGIN_MAIN_FRAME_SENT || | 749 SchedulerStateMachine::COMMIT_STATE_BEGIN_MAIN_FRAME_SENT || |
740 state_machine_.commit_state() == | 750 state_machine_.commit_state() == |
741 SchedulerStateMachine::COMMIT_STATE_BEGIN_MAIN_FRAME_STARTED); | 751 SchedulerStateMachine::COMMIT_STATE_BEGIN_MAIN_FRAME_STARTED); |
742 } | 752 } |
743 | 753 |
744 } // namespace cc | 754 } // namespace cc |
OLD | NEW |