1 // Copyright 2015 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 #ifndef COMPONENTS_SCHEDULER_CHILD_IDLE_HELPER_H_
6 #define COMPONENTS_SCHEDULER_CHILD_IDLE_HELPER_H_
8 #include "base/message_loop/message_loop.h"
9 #include "components/scheduler/child/cancelable_closure_holder.h"
10 #include "components/scheduler/child/scheduler_helper.h"
11 #include "components/scheduler/child/single_thread_idle_task_runner.h"
12 #include "components/scheduler/child/task_queue_selector.h"
13 #include "components/scheduler/scheduler_export.h"
17 class SchedulerHelper
;
19 // Common scheduler functionality for Idle tasks.
20 class SCHEDULER_EXPORT IdleHelper
21 : public base::MessageLoop::TaskObserver
,
22 public SingleThreadIdleTaskRunner::Delegate
{
24 // Used to by scheduler implementations to customize idle behaviour.
25 class SCHEDULER_EXPORT Delegate
{
30 // If it's ok to enter a long idle period, return true. Otherwise return
31 // false and set next_long_idle_period_delay_out so we know when to try
33 virtual bool CanEnterLongIdlePeriod(
35 base::TimeDelta
* next_long_idle_period_delay_out
) = 0;
37 // Signals that the Long Idle Period hasn't started yet because the system
39 virtual void IsNotQuiescent() = 0;
41 // Signals that we have started an Idle Period.
42 virtual void OnIdlePeriodStarted() = 0;
44 // Signals that we have finished an Idle Period.
45 virtual void OnIdlePeriodEnded() = 0;
48 DISALLOW_COPY_AND_ASSIGN(Delegate
);
51 // Keep IdleHelper::IdlePeriodStateToString in sync with this enum.
52 enum class IdlePeriodState
{
56 IN_LONG_IDLE_PERIOD_WITH_MAX_DEADLINE
,
57 IN_LONG_IDLE_PERIOD_PAUSED
,
58 // Must be the last entry.
59 IDLE_PERIOD_STATE_COUNT
,
60 FIRST_IDLE_PERIOD_STATE
= NOT_IN_IDLE_PERIOD
,
63 // The maximum length of an idle period.
64 static const int kMaximumIdlePeriodMillis
= 50;
66 // |helper| and |delegate| are not owned by IdleHelper object and must
69 SchedulerHelper
* helper
,
71 const char* tracing_category
,
72 const char* disabled_by_default_tracing_category
,
73 const char* idle_period_tracing_name
,
74 base::TimeDelta required_quiescence_duration_before_long_idle_period
);
75 ~IdleHelper() override
;
77 // Returns the idle task runner. Tasks posted to this runner may be reordered
78 // relative to other task types and may be starved for an arbitrarily long
79 // time if no idle time is available.
80 scoped_refptr
<SingleThreadIdleTaskRunner
> IdleTaskRunner();
82 // If |required_quiescence_duration_before_long_idle_period_| is zero then
83 // immediately initiate a long idle period, otherwise check if any tasks have
84 // run recently and if so, check again after a delay of
85 // |required_quiescence_duration_before_long_idle_period_|.
86 // Calling this function will end any previous idle period immediately, and
87 // potentially again later if
88 // |required_quiescence_duration_before_long_idle_period_| is non-zero.
89 // NOTE EndIdlePeriod will disable the long idle periods.
90 void EnableLongIdlePeriod();
92 // Start an idle period with a given idle period deadline.
93 void StartIdlePeriod(IdlePeriodState new_idle_period_state
,
95 base::TimeTicks idle_period_deadline
);
97 // This will end an idle period either started with StartIdlePeriod or
98 // EnableLongIdlePeriod.
101 // Returns true if a currently running idle task could exceed its deadline
102 // without impacting user experience too much. This should only be used if
103 // there is a task which cannot be pre-empted and is likely to take longer
104 // than the largest expected idle task deadline. It should NOT be polled to
105 // check whether more work can be performed on the current idle task after
106 // its deadline has expired - post a new idle task for the continuation of the
107 // work in this case.
108 // Must be called from the thread this class was created on.
109 bool CanExceedIdleDeadlineIfRequired() const;
111 // Returns the deadline for the current idle task.
112 base::TimeTicks
CurrentIdleTaskDeadline() const;
114 // SingleThreadIdleTaskRunner::Delegate implementation:
115 void OnIdleTaskPosted() override
;
116 base::TimeTicks
WillProcessIdleTask() override
;
117 void DidProcessIdleTask() override
;
119 // base::MessageLoop::TaskObserver implementation:
120 void WillProcessTask(const base::PendingTask
& pending_task
) override
;
121 void DidProcessTask(const base::PendingTask
& pending_task
) override
;
123 IdlePeriodState
SchedulerIdlePeriodState() const;
124 static const char* IdlePeriodStateToString(IdlePeriodState state
);
127 friend class BaseIdleHelperTest
;
128 friend class IdleHelperTest
;
132 State(SchedulerHelper
* helper
,
134 const char* tracing_category
,
135 const char* disabled_by_default_tracing_category
,
136 const char* idle_period_tracing_name
);
139 void UpdateState(IdlePeriodState new_state
,
140 base::TimeTicks new_deadline
,
141 base::TimeTicks optional_now
);
142 bool IsIdlePeriodPaused() const;
144 IdlePeriodState
idle_period_state() const;
145 base::TimeTicks
idle_period_deadline() const;
147 void TraceIdleIdleTaskStart();
148 void TraceIdleIdleTaskEnd();
149 void TraceEventIdlePeriodStateChange(IdlePeriodState new_state
,
150 base::TimeTicks new_deadline
,
151 base::TimeTicks optional_now
);
154 SchedulerHelper
* helper_
; // NOT OWNED
155 Delegate
* delegate_
; // NOT OWNED
157 IdlePeriodState idle_period_state_
;
158 base::TimeTicks idle_period_deadline_
;
160 base::TraceTicks idle_period_deadline_for_tracing_
;
161 base::TraceTicks last_idle_task_trace_time_
;
162 bool nestable_events_started_
;
163 const char* tracing_category_
;
164 const char* disabled_by_default_tracing_category_
;
165 const char* idle_period_tracing_name_
;
167 DISALLOW_COPY_AND_ASSIGN(State
);
170 // The minimum duration of an idle period.
171 static const int kMinimumIdlePeriodDurationMillis
= 1;
173 // The minimum delay to wait between retrying to initiate a long idle time.
174 static const int kRetryEnableLongIdlePeriodDelayMillis
= 1;
176 // Returns the new idle period state for the next long idle period. Fills in
177 // |next_long_idle_period_delay_out| with the next time we should try to
178 // initiate the next idle period.
179 IdlePeriodState
ComputeNewLongIdlePeriodState(
180 const base::TimeTicks now
,
181 base::TimeDelta
* next_long_idle_period_delay_out
);
183 bool ShouldWaitForQuiescence();
184 void OnIdleTaskPostedOnMainThread();
185 void UpdateLongIdlePeriodStateAfterIdleTask();
187 void SetIdlePeriodState(IdlePeriodState new_state
,
188 base::TimeTicks new_deadline
,
189 base::TimeTicks optional_now
);
191 // Returns true if |state| represents being within an idle period state.
192 static bool IsInIdlePeriod(IdlePeriodState state
);
193 // Returns true if |state| represents being within a long idle period state.
194 static bool IsInLongIdlePeriod(IdlePeriodState state
);
196 SchedulerHelper
* helper_
; // NOT OWNED
197 Delegate
* delegate_
; // NOT OWNED
198 scoped_refptr
<TaskQueue
> idle_queue_
;
199 scoped_refptr
<SingleThreadIdleTaskRunner
> idle_task_runner_
;
201 CancelableClosureHolder enable_next_long_idle_period_closure_
;
202 CancelableClosureHolder on_idle_task_posted_closure_
;
206 base::TimeDelta required_quiescence_duration_before_long_idle_period_
;
208 const char* disabled_by_default_tracing_category_
;
210 base::WeakPtr
<IdleHelper
> weak_idle_helper_ptr_
;
211 base::WeakPtrFactory
<IdleHelper
> weak_factory_
;
213 DISALLOW_COPY_AND_ASSIGN(IdleHelper
);
216 } // namespace scheduler
218 #endif // COMPONENTS_SCHEDULER_CHILD_IDLE_HELPER_H_