// 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 "base/bind.h" #include "base/logging.h" namespace base { DeferredSequencedTaskRunner::DeferredTask::DeferredTask() : is_non_nestable(false) { } DeferredSequencedTaskRunner::DeferredTask::~DeferredTask() { } DeferredSequencedTaskRunner::DeferredSequencedTaskRunner( const scoped_refptr& target_task_runner) : started_(false), target_task_runner_(target_task_runner) { } DeferredSequencedTaskRunner::~DeferredSequencedTaskRunner() { } bool DeferredSequencedTaskRunner::PostDelayedTask( const tracked_objects::Location& from_here, const Closure& task, TimeDelta delay) { AutoLock lock(lock_); if (started_) { DCHECK(deferred_tasks_queue_.empty()); return target_task_runner_->PostDelayedTask(from_here, task, delay); } QueueDeferredTask(from_here, task, delay, false /* is_non_nestable */); return true; } bool DeferredSequencedTaskRunner::RunsTasksOnCurrentThread() const { return target_task_runner_->RunsTasksOnCurrentThread(); } bool DeferredSequencedTaskRunner::PostNonNestableDelayedTask( const tracked_objects::Location& from_here, const Closure& task, TimeDelta delay) { AutoLock lock(lock_); if (started_) { DCHECK(deferred_tasks_queue_.empty()); return target_task_runner_->PostNonNestableDelayedTask(from_here, task, delay); } QueueDeferredTask(from_here, task, delay, true /* is_non_nestable */); return true; } void DeferredSequencedTaskRunner::QueueDeferredTask( const tracked_objects::Location& from_here, const Closure& task, TimeDelta delay, bool is_non_nestable) { DeferredTask deferred_task; deferred_task.posted_from = from_here; deferred_task.task = task; deferred_task.delay = delay; deferred_task.is_non_nestable = is_non_nestable; deferred_tasks_queue_.push_back(deferred_task); } void DeferredSequencedTaskRunner::Start() { AutoLock lock(lock_); DCHECK(!started_); started_ = true; for (std::vector::iterator i = deferred_tasks_queue_.begin(); i != deferred_tasks_queue_.end(); ++i) { const DeferredTask& task = *i; if (task.is_non_nestable) { target_task_runner_->PostNonNestableDelayedTask(task.posted_from, task.task, task.delay); } else { target_task_runner_->PostDelayedTask(task.posted_from, task.task, task.delay); } // Replace the i-th element in the |deferred_tasks_queue_| with an empty // |DelayedTask| to ensure that |task| is destroyed before the next task // is posted. *i = DeferredTask(); } deferred_tasks_queue_.clear(); } } // namespace base