Index: base/timer/timer.cc |
diff --git a/base/timer/timer.cc b/base/timer/timer.cc |
index 31eec1bb574e83b32308dfbb3a94e335d550499c..abcd70aefa1c31212945b01867e6c039c27a3b9e 100644 |
--- a/base/timer/timer.cc |
+++ b/base/timer/timer.cc |
@@ -11,16 +11,14 @@ |
#include "base/logging.h" |
#include "base/memory/ptr_util.h" |
#include "base/memory/ref_counted.h" |
-#include "base/single_thread_task_runner.h" |
#include "base/threading/platform_thread.h" |
-#include "base/threading/thread_task_runner_handle.h" |
+#include "base/threading/sequenced_task_runner_handle.h" |
#include "base/time/tick_clock.h" |
namespace base { |
-// BaseTimerTaskInternal is a simple delegate for scheduling a callback to |
-// Timer in the thread's default task runner. It also handles the following |
-// edge cases: |
+// BaseTimerTaskInternal is a simple delegate for scheduling a callback to Timer |
+// on the current sequence. It also handles the following edge cases: |
// - deleted by the task runner. |
// - abandoned (orphaned) by Timer. |
class BaseTimerTaskInternal { |
@@ -34,33 +32,31 @@ class BaseTimerTaskInternal { |
// destructed. If so, don't leave Timer with a dangling pointer |
// to this. |
if (timer_) |
- timer_->StopAndAbandon(); |
+ timer_->AbandonAndStop(); |
} |
void Run() { |
- // timer_ is NULL if we were abandoned. |
+ // |timer_| is nullptr if we were abandoned. |
if (!timer_) |
return; |
- // *this will be deleted by the task runner, so Timer needs to |
- // forget us: |
- timer_->scheduled_task_ = NULL; |
+ // |this| will be deleted by the task runner, so Timer needs to forget us: |
+ timer_->scheduled_task_ = nullptr; |
- // Although Timer should not call back into *this, let's clear |
- // the timer_ member first to be pedantic. |
+ // Although Timer should not call back into |this|, let's clear |timer_| |
+ // first to be pedantic. |
Timer* timer = timer_; |
- timer_ = NULL; |
+ timer_ = nullptr; |
timer->RunScheduledTask(); |
} |
- // The task remains in the MessageLoop queue, but nothing will happen when it |
- // runs. |
- void Abandon() { |
- timer_ = NULL; |
- } |
+ // The task remains in the queue, but nothing will happen when it runs. |
+ void Abandon() { timer_ = nullptr; } |
private: |
Timer* timer_; |
+ |
+ DISALLOW_COPY_AND_ASSIGN(BaseTimerTaskInternal); |
}; |
Timer::Timer(bool retain_user_task, bool is_repeating) |
@@ -68,11 +64,16 @@ Timer::Timer(bool retain_user_task, bool is_repeating) |
Timer::Timer(bool retain_user_task, bool is_repeating, TickClock* tick_clock) |
: scheduled_task_(nullptr), |
- thread_id_(0), |
is_repeating_(is_repeating), |
retain_user_task_(retain_user_task), |
tick_clock_(tick_clock), |
- is_running_(false) {} |
+ is_running_(false) { |
+ // It is safe for the timer to be created on a different thread/sequence than |
+ // the one from which the timer APIs are called. The first call to the |
+ // checker's CalledOnValidSequence() method will re-bind the checker, and |
+ // later calls will verify that the same task runner is used. |
+ origin_sequence_checker_.DetachFromSequence(); |
+} |
Timer::Timer(const tracked_objects::Location& posted_from, |
TimeDelta delay, |
@@ -89,44 +90,68 @@ Timer::Timer(const tracked_objects::Location& posted_from, |
posted_from_(posted_from), |
delay_(delay), |
user_task_(user_task), |
- thread_id_(0), |
is_repeating_(is_repeating), |
retain_user_task_(true), |
tick_clock_(tick_clock), |
- is_running_(false) {} |
+ is_running_(false) { |
+ // See comment in other constructor. |
+ origin_sequence_checker_.DetachFromSequence(); |
+} |
Timer::~Timer() { |
- StopAndAbandon(); |
+ // TODO(gab): Enable this once Stop() properly detaches from sequence. |
+ // DCHECK(origin_sequence_checker_.CalledOnValidSequence()); |
+ AbandonAndStop(); |
} |
bool Timer::IsRunning() const { |
+ DCHECK(origin_sequence_checker_.CalledOnValidSequence()); |
return is_running_; |
} |
TimeDelta Timer::GetCurrentDelay() const { |
+ DCHECK(origin_sequence_checker_.CalledOnValidSequence()); |
return delay_; |
} |
-void Timer::SetTaskRunner(scoped_refptr<SingleThreadTaskRunner> task_runner) { |
- // Do not allow changing the task runner once something has been scheduled. |
- DCHECK_EQ(thread_id_, 0); |
+void Timer::SetTaskRunner(scoped_refptr<SequencedTaskRunner> task_runner) { |
+ // Do not allow changing the task runner when the Timer is running. |
+ // Don't check for |origin_sequence_checker_.CalledOnValidSequence()| here to |
+ // allow the use case of constructing the Timer and immediatetly invoking |
+ // SetTaskRunner() before starting it (CalledOnValidSequence() would undo the |
+ // DetachFromSequence() from the constructor). The |!is_running| check kind of |
+ // verifies the same thing (and TSAN should catch callers that do it wrong but |
+ // somehow evade all debug checks). |
+ DCHECK(!is_running_); |
task_runner_.swap(task_runner); |
} |
void Timer::Start(const tracked_objects::Location& posted_from, |
TimeDelta delay, |
const base::Closure& user_task) { |
- SetTaskInfo(posted_from, delay, user_task); |
+ DCHECK(origin_sequence_checker_.CalledOnValidSequence()); |
+ |
+ posted_from_ = posted_from; |
+ delay_ = delay; |
+ user_task_ = user_task; |
+ |
Reset(); |
} |
void Timer::Stop() { |
+ // TODO(gab): Enable this when it's no longer called racily from |
+ // RunScheduledTask(): https://crbug.com/587199. |
+ // DCHECK(origin_sequence_checker_.CalledOnValidSequence()); |
+ |
is_running_ = false; |
if (!retain_user_task_) |
user_task_.Reset(); |
+ // No more member accesses here: |this| could be deleted after freeing |
+ // |user_task_|. |
} |
void Timer::Reset() { |
+ DCHECK(origin_sequence_checker_.CalledOnValidSequence()); |
DCHECK(!user_task_.is_null()); |
// If there's no pending task, start one up and return. |
@@ -135,41 +160,41 @@ void Timer::Reset() { |
return; |
} |
- // Set the new desired_run_time_. |
+ // Set the new |desired_run_time_|. |
if (delay_ > TimeDelta::FromMicroseconds(0)) |
desired_run_time_ = Now() + delay_; |
else |
desired_run_time_ = TimeTicks(); |
// We can use the existing scheduled task if it arrives before the new |
- // desired_run_time_. |
+ // |desired_run_time_|. |
if (desired_run_time_ >= scheduled_run_time_) { |
is_running_ = true; |
return; |
} |
- // We can't reuse the scheduled_task_, so abandon it and post a new one. |
+ // We can't reuse the |scheduled_task_|, so abandon it and post a new one. |
AbandonScheduledTask(); |
PostNewScheduledTask(delay_); |
} |
TimeTicks Timer::Now() const { |
+ // TODO(gab): Enable this when it's no longer called racily from |
+ // RunScheduledTask(): https://crbug.com/587199. |
+ // DCHECK(origin_sequence_checker_.CalledOnValidSequence()); |
return tick_clock_ ? tick_clock_->NowTicks() : TimeTicks::Now(); |
} |
-void Timer::SetTaskInfo(const tracked_objects::Location& posted_from, |
- TimeDelta delay, |
- const base::Closure& user_task) { |
- posted_from_ = posted_from; |
- delay_ = delay; |
- user_task_ = user_task; |
-} |
- |
void Timer::PostNewScheduledTask(TimeDelta delay) { |
- DCHECK(scheduled_task_ == NULL); |
+ // TODO(gab): Enable this when it's no longer called racily from |
+ // RunScheduledTask(): https://crbug.com/587199. |
+ // DCHECK(origin_sequence_checker_.CalledOnValidSequence()); |
+ DCHECK(!scheduled_task_); |
is_running_ = true; |
scheduled_task_ = new BaseTimerTaskInternal(this); |
if (delay > TimeDelta::FromMicroseconds(0)) { |
+ // TODO(gab): Posting BaseTimerTaskInternal::Run to another sequence makes |
+ // this code racy. https://crbug.com/587199 |
GetTaskRunner()->PostDelayedTask( |
posted_from_, |
base::BindOnce(&BaseTimerTaskInternal::Run, |
@@ -182,26 +207,27 @@ void Timer::PostNewScheduledTask(TimeDelta delay) { |
base::Owned(scheduled_task_))); |
scheduled_run_time_ = desired_run_time_ = TimeTicks(); |
} |
- // Remember the thread ID that posts the first task -- this will be verified |
- // later when the task is abandoned to detect misuse from multiple threads. |
- if (!thread_id_) |
- thread_id_ = static_cast<int>(PlatformThread::CurrentId()); |
} |
-scoped_refptr<SingleThreadTaskRunner> Timer::GetTaskRunner() { |
- return task_runner_.get() ? task_runner_ : ThreadTaskRunnerHandle::Get(); |
+scoped_refptr<SequencedTaskRunner> Timer::GetTaskRunner() { |
+ return task_runner_.get() ? task_runner_ : SequencedTaskRunnerHandle::Get(); |
} |
void Timer::AbandonScheduledTask() { |
- DCHECK(thread_id_ == 0 || |
- thread_id_ == static_cast<int>(PlatformThread::CurrentId())); |
+ // TODO(gab): Enable this when it's no longer called racily from |
+ // RunScheduledTask() -> Stop(): https://crbug.com/587199. |
+ // DCHECK(origin_sequence_checker_.CalledOnValidSequence()); |
if (scheduled_task_) { |
scheduled_task_->Abandon(); |
- scheduled_task_ = NULL; |
+ scheduled_task_ = nullptr; |
} |
} |
void Timer::RunScheduledTask() { |
+ // TODO(gab): Enable this when it's no longer called racily: |
+ // https://crbug.com/587199. |
+ // DCHECK(origin_sequence_checker_.CalledOnValidSequence()); |
+ |
// Task may have been disabled. |
if (!is_running_) |
return; |
@@ -209,10 +235,10 @@ void Timer::RunScheduledTask() { |
// First check if we need to delay the task because of a new target time. |
if (desired_run_time_ > scheduled_run_time_) { |
// Now() can be expensive, so only call it if we know the user has changed |
- // the desired_run_time_. |
+ // the |desired_run_time_|. |
TimeTicks now = Now(); |
// Task runner may have called us late anyway, so only post a continuation |
- // task if the desired_run_time_ is in the future. |
+ // task if the |desired_run_time_| is in the future. |
if (desired_run_time_ > now) { |
// Post a new task to span the remaining time. |
PostNewScheduledTask(desired_run_time_ - now); |
@@ -221,7 +247,7 @@ void Timer::RunScheduledTask() { |
} |
// Make a local copy of the task to run. The Stop method will reset the |
- // user_task_ member if retain_user_task_ is false. |
+ // |user_task_| member if |retain_user_task_| is false. |
base::Closure task = user_task_; |
if (is_repeating_) |
@@ -231,7 +257,7 @@ void Timer::RunScheduledTask() { |
task.Run(); |
- // No more member accesses here: *this could be deleted at this point. |
+ // No more member accesses here: |this| could be deleted at this point. |
} |
} // namespace base |