1 // Copyright (c) 2012 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 "base/timer/timer.h"
6 
7 #include <stddef.h>
8 
9 #include "base/logging.h"
10 #include "base/memory/ref_counted.h"
11 #include "base/single_thread_task_runner.h"
12 #include "base/threading/platform_thread.h"
13 #include "base/threading/thread_task_runner_handle.h"
14 
15 namespace base {
16 
17 // BaseTimerTaskInternal is a simple delegate for scheduling a callback to
18 // Timer in the thread's default task runner. It also handles the following
19 // edge cases:
20 // - deleted by the task runner.
21 // - abandoned (orphaned) by Timer.
22 class BaseTimerTaskInternal {
23  public:
BaseTimerTaskInternal(Timer * timer)24   explicit BaseTimerTaskInternal(Timer* timer)
25       : timer_(timer) {
26   }
27 
~BaseTimerTaskInternal()28   ~BaseTimerTaskInternal() {
29     // This task may be getting cleared because the task runner has been
30     // destructed.  If so, don't leave Timer with a dangling pointer
31     // to this.
32     if (timer_)
33       timer_->StopAndAbandon();
34   }
35 
Run()36   void Run() {
37     // timer_ is NULL if we were abandoned.
38     if (!timer_)
39       return;
40 
41     // *this will be deleted by the task runner, so Timer needs to
42     // forget us:
43     timer_->scheduled_task_ = NULL;
44 
45     // Although Timer should not call back into *this, let's clear
46     // the timer_ member first to be pedantic.
47     Timer* timer = timer_;
48     timer_ = NULL;
49     timer->RunScheduledTask();
50   }
51 
52   // The task remains in the MessageLoop queue, but nothing will happen when it
53   // runs.
Abandon()54   void Abandon() {
55     timer_ = NULL;
56   }
57 
58  private:
59   Timer* timer_;
60 };
61 
Timer(bool retain_user_task,bool is_repeating)62 Timer::Timer(bool retain_user_task, bool is_repeating)
63     : scheduled_task_(NULL),
64       thread_id_(0),
65       is_repeating_(is_repeating),
66       retain_user_task_(retain_user_task),
67       is_running_(false) {
68 }
69 
Timer(const tracked_objects::Location & posted_from,TimeDelta delay,const base::Closure & user_task,bool is_repeating)70 Timer::Timer(const tracked_objects::Location& posted_from,
71              TimeDelta delay,
72              const base::Closure& user_task,
73              bool is_repeating)
74     : scheduled_task_(NULL),
75       posted_from_(posted_from),
76       delay_(delay),
77       user_task_(user_task),
78       thread_id_(0),
79       is_repeating_(is_repeating),
80       retain_user_task_(true),
81       is_running_(false) {
82 }
83 
~Timer()84 Timer::~Timer() {
85   StopAndAbandon();
86 }
87 
IsRunning() const88 bool Timer::IsRunning() const {
89   return is_running_;
90 }
91 
GetCurrentDelay() const92 TimeDelta Timer::GetCurrentDelay() const {
93   return delay_;
94 }
95 
SetTaskRunner(scoped_refptr<SingleThreadTaskRunner> task_runner)96 void Timer::SetTaskRunner(scoped_refptr<SingleThreadTaskRunner> task_runner) {
97   // Do not allow changing the task runner once something has been scheduled.
98   DCHECK_EQ(thread_id_, 0);
99   task_runner_.swap(task_runner);
100 }
101 
Start(const tracked_objects::Location & posted_from,TimeDelta delay,const base::Closure & user_task)102 void Timer::Start(const tracked_objects::Location& posted_from,
103                   TimeDelta delay,
104                   const base::Closure& user_task) {
105   SetTaskInfo(posted_from, delay, user_task);
106   Reset();
107 }
108 
Stop()109 void Timer::Stop() {
110   is_running_ = false;
111   if (!retain_user_task_)
112     user_task_.Reset();
113 }
114 
Reset()115 void Timer::Reset() {
116   DCHECK(!user_task_.is_null());
117 
118   // If there's no pending task, start one up and return.
119   if (!scheduled_task_) {
120     PostNewScheduledTask(delay_);
121     return;
122   }
123 
124   // Set the new desired_run_time_.
125   if (delay_ > TimeDelta::FromMicroseconds(0))
126     desired_run_time_ = TimeTicks::Now() + delay_;
127   else
128     desired_run_time_ = TimeTicks();
129 
130   // We can use the existing scheduled task if it arrives before the new
131   // desired_run_time_.
132   if (desired_run_time_ >= scheduled_run_time_) {
133     is_running_ = true;
134     return;
135   }
136 
137   // We can't reuse the scheduled_task_, so abandon it and post a new one.
138   AbandonScheduledTask();
139   PostNewScheduledTask(delay_);
140 }
141 
SetTaskInfo(const tracked_objects::Location & posted_from,TimeDelta delay,const base::Closure & user_task)142 void Timer::SetTaskInfo(const tracked_objects::Location& posted_from,
143                         TimeDelta delay,
144                         const base::Closure& user_task) {
145   posted_from_ = posted_from;
146   delay_ = delay;
147   user_task_ = user_task;
148 }
149 
PostNewScheduledTask(TimeDelta delay)150 void Timer::PostNewScheduledTask(TimeDelta delay) {
151   DCHECK(scheduled_task_ == NULL);
152   is_running_ = true;
153   scheduled_task_ = new BaseTimerTaskInternal(this);
154   if (delay > TimeDelta::FromMicroseconds(0)) {
155     GetTaskRunner()->PostDelayedTask(posted_from_,
156         base::Bind(&BaseTimerTaskInternal::Run, base::Owned(scheduled_task_)),
157         delay);
158     scheduled_run_time_ = desired_run_time_ = TimeTicks::Now() + delay;
159   } else {
160     GetTaskRunner()->PostTask(posted_from_,
161         base::Bind(&BaseTimerTaskInternal::Run, base::Owned(scheduled_task_)));
162     scheduled_run_time_ = desired_run_time_ = TimeTicks();
163   }
164   // Remember the thread ID that posts the first task -- this will be verified
165   // later when the task is abandoned to detect misuse from multiple threads.
166   if (!thread_id_) {
167     DCHECK(GetTaskRunner()->BelongsToCurrentThread());
168     thread_id_ = static_cast<int>(PlatformThread::CurrentId());
169   }
170 }
171 
GetTaskRunner()172 scoped_refptr<SingleThreadTaskRunner> Timer::GetTaskRunner() {
173   return task_runner_.get() ? task_runner_ : ThreadTaskRunnerHandle::Get();
174 }
175 
AbandonScheduledTask()176 void Timer::AbandonScheduledTask() {
177   DCHECK(thread_id_ == 0 ||
178          thread_id_ == static_cast<int>(PlatformThread::CurrentId()));
179   if (scheduled_task_) {
180     scheduled_task_->Abandon();
181     scheduled_task_ = NULL;
182   }
183 }
184 
RunScheduledTask()185 void Timer::RunScheduledTask() {
186   // Task may have been disabled.
187   if (!is_running_)
188     return;
189 
190   // First check if we need to delay the task because of a new target time.
191   if (desired_run_time_ > scheduled_run_time_) {
192     // TimeTicks::Now() can be expensive, so only call it if we know the user
193     // has changed the desired_run_time_.
194     TimeTicks now = TimeTicks::Now();
195     // Task runner may have called us late anyway, so only post a continuation
196     // task if the desired_run_time_ is in the future.
197     if (desired_run_time_ > now) {
198       // Post a new task to span the remaining time.
199       PostNewScheduledTask(desired_run_time_ - now);
200       return;
201     }
202   }
203 
204   // Make a local copy of the task to run. The Stop method will reset the
205   // user_task_ member if retain_user_task_ is false.
206   base::Closure task = user_task_;
207 
208   if (is_repeating_)
209     PostNewScheduledTask(delay_);
210   else
211     Stop();
212 
213   task.Run();
214 
215   // No more member accesses here: *this could be deleted at this point.
216 }
217 
218 }  // namespace base
219