|  | // Copyright (c) 2013 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "base/deferred_sequenced_task_runner.h" | 
|  |  | 
|  | #include <utility> | 
|  |  | 
|  | #include "base/bind.h" | 
|  | #include "base/logging.h" | 
|  |  | 
|  | namespace base { | 
|  |  | 
|  | DeferredSequencedTaskRunner::DeferredTask::DeferredTask() | 
|  | : is_non_nestable(false) { | 
|  | } | 
|  |  | 
|  | DeferredSequencedTaskRunner::DeferredTask::DeferredTask(DeferredTask&& other) = | 
|  | default; | 
|  |  | 
|  | DeferredSequencedTaskRunner::DeferredTask::~DeferredTask() = default; | 
|  |  | 
|  | DeferredSequencedTaskRunner::DeferredTask& | 
|  | DeferredSequencedTaskRunner::DeferredTask::operator=(DeferredTask&& other) = | 
|  | default; | 
|  |  | 
|  | DeferredSequencedTaskRunner::DeferredSequencedTaskRunner( | 
|  | scoped_refptr<SequencedTaskRunner> target_task_runner) | 
|  | : DeferredSequencedTaskRunner() { | 
|  | DCHECK(target_task_runner); | 
|  | target_task_runner_ = std::move(target_task_runner); | 
|  | } | 
|  |  | 
|  | DeferredSequencedTaskRunner::DeferredSequencedTaskRunner() | 
|  | : created_thread_id_(PlatformThread::CurrentId()) {} | 
|  |  | 
|  | bool DeferredSequencedTaskRunner::PostDelayedTask(const Location& from_here, | 
|  | OnceClosure task, | 
|  | TimeDelta delay) { | 
|  | AutoLock lock(lock_); | 
|  | if (started_) { | 
|  | DCHECK(deferred_tasks_queue_.empty()); | 
|  | return target_task_runner_->PostDelayedTask(from_here, std::move(task), | 
|  | delay); | 
|  | } | 
|  |  | 
|  | QueueDeferredTask(from_here, std::move(task), delay, | 
|  | false /* is_non_nestable */); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool DeferredSequencedTaskRunner::RunsTasksInCurrentSequence() const { | 
|  | AutoLock lock(lock_); | 
|  | if (target_task_runner_) | 
|  | return target_task_runner_->RunsTasksInCurrentSequence(); | 
|  |  | 
|  | return created_thread_id_ == PlatformThread::CurrentId(); | 
|  | } | 
|  |  | 
|  | bool DeferredSequencedTaskRunner::PostNonNestableDelayedTask( | 
|  | const Location& from_here, | 
|  | OnceClosure task, | 
|  | TimeDelta delay) { | 
|  | AutoLock lock(lock_); | 
|  | if (started_) { | 
|  | DCHECK(deferred_tasks_queue_.empty()); | 
|  | return target_task_runner_->PostNonNestableDelayedTask( | 
|  | from_here, std::move(task), delay); | 
|  | } | 
|  | QueueDeferredTask(from_here, std::move(task), delay, | 
|  | true /* is_non_nestable */); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void DeferredSequencedTaskRunner::Start() { | 
|  | AutoLock lock(lock_); | 
|  | StartImpl(); | 
|  | } | 
|  |  | 
|  | void DeferredSequencedTaskRunner::StartWithTaskRunner( | 
|  | scoped_refptr<SequencedTaskRunner> target_task_runner) { | 
|  | AutoLock lock(lock_); | 
|  | DCHECK(!target_task_runner_); | 
|  | DCHECK(target_task_runner); | 
|  | target_task_runner_ = std::move(target_task_runner); | 
|  | StartImpl(); | 
|  | } | 
|  |  | 
|  | DeferredSequencedTaskRunner::~DeferredSequencedTaskRunner() = default; | 
|  |  | 
|  | void DeferredSequencedTaskRunner::QueueDeferredTask(const Location& from_here, | 
|  | OnceClosure task, | 
|  | TimeDelta delay, | 
|  | bool is_non_nestable) { | 
|  | lock_.AssertAcquired(); | 
|  |  | 
|  | // Use CHECK instead of DCHECK to crash earlier. See http://crbug.com/711167 | 
|  | // for details. | 
|  | CHECK(task); | 
|  |  | 
|  | DeferredTask deferred_task; | 
|  | deferred_task.posted_from = from_here; | 
|  | deferred_task.task = std::move(task); | 
|  | deferred_task.delay = delay; | 
|  | deferred_task.is_non_nestable = is_non_nestable; | 
|  | deferred_tasks_queue_.push_back(std::move(deferred_task)); | 
|  | } | 
|  |  | 
|  | void DeferredSequencedTaskRunner::StartImpl() { | 
|  | lock_.AssertAcquired();  // Callers should have grabbed the lock. | 
|  | DCHECK(!started_); | 
|  | started_ = true; | 
|  | DCHECK(target_task_runner_); | 
|  | for (std::vector<DeferredTask>::iterator i = deferred_tasks_queue_.begin(); | 
|  | i != deferred_tasks_queue_.end(); | 
|  | ++i) { | 
|  | DeferredTask& task = *i; | 
|  | if (task.is_non_nestable) { | 
|  | target_task_runner_->PostNonNestableDelayedTask( | 
|  | task.posted_from, std::move(task.task), task.delay); | 
|  | } else { | 
|  | target_task_runner_->PostDelayedTask(task.posted_from, | 
|  | std::move(task.task), task.delay); | 
|  | } | 
|  | } | 
|  | deferred_tasks_queue_.clear(); | 
|  | } | 
|  |  | 
|  | }  // namespace base |