OLD | NEW |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 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 | 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 "base/timer/timer.h" | 5 #include "base/timer/timer.h" |
6 | 6 |
7 #include <stddef.h> | 7 #include <stddef.h> |
8 | 8 |
9 #include "base/logging.h" | 9 #include "base/logging.h" |
10 #include "base/memory/ref_counted.h" | 10 #include "base/memory/ref_counted.h" |
11 #include "base/single_thread_task_runner.h" | 11 #include "base/sequenced_task_runner.h" |
12 #include "base/thread_task_runner_handle.h" | 12 #include "base/threading/sequenced_task_runner_handle.h" |
13 #include "base/threading/platform_thread.h" | |
14 | 13 |
15 namespace base { | 14 namespace base { |
16 | 15 |
17 // BaseTimerTaskInternal is a simple delegate for scheduling a callback to | 16 // BaseTimerTaskInternal is a simple delegate for scheduling a callback to |
18 // Timer in the thread's default task runner. It also handles the following | 17 // Timer in the task runner. It also handles the following edge cases: |
19 // edge cases: | |
20 // - deleted by the task runner. | 18 // - deleted by the task runner. |
21 // - abandoned (orphaned) by Timer. | 19 // - abandoned (orphaned) by Timer. |
22 class BaseTimerTaskInternal { | 20 class BaseTimerTaskInternal { |
23 public: | 21 public: |
24 explicit BaseTimerTaskInternal(Timer* timer) | 22 explicit BaseTimerTaskInternal(Timer* timer) |
25 : timer_(timer) { | 23 : timer_(timer) { |
26 } | 24 } |
27 | 25 |
28 ~BaseTimerTaskInternal() { | 26 ~BaseTimerTaskInternal() { |
29 // This task may be getting cleared because the task runner has been | 27 // This task may be getting cleared because the task runner has been |
(...skipping 24 matching lines...) Expand all Loading... |
54 void Abandon() { | 52 void Abandon() { |
55 timer_ = NULL; | 53 timer_ = NULL; |
56 } | 54 } |
57 | 55 |
58 private: | 56 private: |
59 Timer* timer_; | 57 Timer* timer_; |
60 }; | 58 }; |
61 | 59 |
62 Timer::Timer(bool retain_user_task, bool is_repeating) | 60 Timer::Timer(bool retain_user_task, bool is_repeating) |
63 : scheduled_task_(NULL), | 61 : scheduled_task_(NULL), |
64 thread_id_(0), | 62 was_scheduled_(false), |
65 is_repeating_(is_repeating), | 63 is_repeating_(is_repeating), |
66 retain_user_task_(retain_user_task), | 64 retain_user_task_(retain_user_task), |
67 is_running_(false) { | 65 is_running_(false) { |
| 66 // It is safe for the timer to be created on a different thread/sequence |
| 67 // than the one from which the timer APIs are called. The first call to the |
| 68 // checker's CalledOnValidSequencedThread() method will re-bind the checker, |
| 69 // and later calls will verify that the same task runner is used. |
| 70 origin_sequence_checker_.DetachFromSequence(); |
68 } | 71 } |
69 | 72 |
70 Timer::Timer(const tracked_objects::Location& posted_from, | 73 Timer::Timer(const tracked_objects::Location& posted_from, |
71 TimeDelta delay, | 74 TimeDelta delay, |
72 const base::Closure& user_task, | 75 const base::Closure& user_task, |
73 bool is_repeating) | 76 bool is_repeating) |
74 : scheduled_task_(NULL), | 77 : scheduled_task_(NULL), |
75 posted_from_(posted_from), | 78 posted_from_(posted_from), |
76 delay_(delay), | 79 delay_(delay), |
77 user_task_(user_task), | 80 user_task_(user_task), |
78 thread_id_(0), | 81 was_scheduled_(false), |
79 is_repeating_(is_repeating), | 82 is_repeating_(is_repeating), |
80 retain_user_task_(true), | 83 retain_user_task_(true), |
81 is_running_(false) { | 84 is_running_(false) { |
| 85 // See comment in other constructor. |
| 86 origin_sequence_checker_.DetachFromSequence(); |
82 } | 87 } |
83 | 88 |
84 Timer::~Timer() { | 89 Timer::~Timer() { |
| 90 DCHECK(origin_sequence_checker_.CalledOnValidSequencedThread()); |
85 StopAndAbandon(); | 91 StopAndAbandon(); |
86 } | 92 } |
87 | 93 |
88 bool Timer::IsRunning() const { | 94 bool Timer::IsRunning() const { |
89 return is_running_; | 95 return is_running_; |
90 } | 96 } |
91 | 97 |
92 TimeDelta Timer::GetCurrentDelay() const { | 98 TimeDelta Timer::GetCurrentDelay() const { |
93 return delay_; | 99 return delay_; |
94 } | 100 } |
95 | 101 |
96 void Timer::SetTaskRunner(scoped_refptr<SingleThreadTaskRunner> task_runner) { | 102 void Timer::SetTaskRunner(scoped_refptr<SequencedTaskRunner> task_runner) { |
97 // Do not allow changing the task runner once something has been scheduled. | 103 // Do not allow changing the task runner once something has been scheduled. |
98 DCHECK_EQ(thread_id_, 0); | 104 DCHECK(!was_scheduled_); |
99 task_runner_.swap(task_runner); | 105 destination_task_runner_ = std::move(task_runner); |
100 } | 106 } |
101 | 107 |
102 void Timer::Start(const tracked_objects::Location& posted_from, | 108 void Timer::Start(const tracked_objects::Location& posted_from, |
103 TimeDelta delay, | 109 TimeDelta delay, |
104 const base::Closure& user_task) { | 110 const base::Closure& user_task) { |
105 SetTaskInfo(posted_from, delay, user_task); | 111 SetTaskInfo(posted_from, delay, user_task); |
106 Reset(); | 112 Reset(); |
107 } | 113 } |
108 | 114 |
109 void Timer::Stop() { | 115 void Timer::Stop() { |
(...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
142 void Timer::SetTaskInfo(const tracked_objects::Location& posted_from, | 148 void Timer::SetTaskInfo(const tracked_objects::Location& posted_from, |
143 TimeDelta delay, | 149 TimeDelta delay, |
144 const base::Closure& user_task) { | 150 const base::Closure& user_task) { |
145 posted_from_ = posted_from; | 151 posted_from_ = posted_from; |
146 delay_ = delay; | 152 delay_ = delay; |
147 user_task_ = user_task; | 153 user_task_ = user_task; |
148 } | 154 } |
149 | 155 |
150 void Timer::PostNewScheduledTask(TimeDelta delay) { | 156 void Timer::PostNewScheduledTask(TimeDelta delay) { |
151 DCHECK(scheduled_task_ == NULL); | 157 DCHECK(scheduled_task_ == NULL); |
| 158 DCHECK(origin_sequence_checker_.CalledOnValidSequencedThread()); |
| 159 was_scheduled_ = true; |
152 is_running_ = true; | 160 is_running_ = true; |
153 scheduled_task_ = new BaseTimerTaskInternal(this); | 161 scheduled_task_ = new BaseTimerTaskInternal(this); |
154 if (delay > TimeDelta::FromMicroseconds(0)) { | 162 if (delay > TimeDelta::FromMicroseconds(0)) { |
155 GetTaskRunner()->PostDelayedTask(posted_from_, | 163 GetTaskRunner()->PostDelayedTask(posted_from_, |
156 base::Bind(&BaseTimerTaskInternal::Run, base::Owned(scheduled_task_)), | 164 base::Bind(&BaseTimerTaskInternal::Run, base::Owned(scheduled_task_)), |
157 delay); | 165 delay); |
158 scheduled_run_time_ = desired_run_time_ = TimeTicks::Now() + delay; | 166 scheduled_run_time_ = desired_run_time_ = TimeTicks::Now() + delay; |
159 } else { | 167 } else { |
160 GetTaskRunner()->PostTask(posted_from_, | 168 GetTaskRunner()->PostTask(posted_from_, |
161 base::Bind(&BaseTimerTaskInternal::Run, base::Owned(scheduled_task_))); | 169 base::Bind(&BaseTimerTaskInternal::Run, base::Owned(scheduled_task_))); |
162 scheduled_run_time_ = desired_run_time_ = TimeTicks(); | 170 scheduled_run_time_ = desired_run_time_ = TimeTicks(); |
163 } | 171 } |
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 } | 172 } |
171 | 173 |
172 scoped_refptr<SingleThreadTaskRunner> Timer::GetTaskRunner() { | 174 scoped_refptr<SequencedTaskRunner> Timer::GetTaskRunner() { |
173 return task_runner_.get() ? task_runner_ : ThreadTaskRunnerHandle::Get(); | 175 return destination_task_runner_.get() ? destination_task_runner_ |
| 176 : SequencedTaskRunnerHandle::Get(); |
174 } | 177 } |
175 | 178 |
176 void Timer::AbandonScheduledTask() { | 179 void Timer::AbandonScheduledTask() { |
177 DCHECK(thread_id_ == 0 || | 180 DCHECK(origin_sequence_checker_.CalledOnValidSequencedThread()); |
178 thread_id_ == static_cast<int>(PlatformThread::CurrentId())); | |
179 if (scheduled_task_) { | 181 if (scheduled_task_) { |
180 scheduled_task_->Abandon(); | 182 scheduled_task_->Abandon(); |
181 scheduled_task_ = NULL; | 183 scheduled_task_ = NULL; |
182 } | 184 } |
183 } | 185 } |
184 | 186 |
185 void Timer::RunScheduledTask() { | 187 void Timer::RunScheduledTask() { |
186 // Task may have been disabled. | 188 // Task may have been disabled. |
187 if (!is_running_) | 189 if (!is_running_) |
188 return; | 190 return; |
(...skipping 20 matching lines...) Expand all Loading... |
209 PostNewScheduledTask(delay_); | 211 PostNewScheduledTask(delay_); |
210 else | 212 else |
211 Stop(); | 213 Stop(); |
212 | 214 |
213 task.Run(); | 215 task.Run(); |
214 | 216 |
215 // No more member accesses here: *this could be deleted at this point. | 217 // No more member accesses here: *this could be deleted at this point. |
216 } | 218 } |
217 | 219 |
218 } // namespace base | 220 } // namespace base |
OLD | NEW |