| // Copyright 2016 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/task/task_scheduler/test_task_factory.h" | 
 |  | 
 | #include "base/bind.h" | 
 | #include "base/bind_helpers.h" | 
 | #include "base/callback.h" | 
 | #include "base/location.h" | 
 | #include "base/logging.h" | 
 | #include "base/synchronization/waitable_event.h" | 
 | #include "base/threading/sequenced_task_runner_handle.h" | 
 | #include "base/threading/thread_task_runner_handle.h" | 
 | #include "testing/gtest/include/gtest/gtest.h" | 
 |  | 
 | namespace base { | 
 | namespace internal { | 
 | namespace test { | 
 |  | 
 | TestTaskFactory::TestTaskFactory(scoped_refptr<TaskRunner> task_runner, | 
 |                                  ExecutionMode execution_mode) | 
 |     : cv_(&lock_), | 
 |       task_runner_(std::move(task_runner)), | 
 |       execution_mode_(execution_mode) { | 
 |   // Detach |thread_checker_| from the current thread. It will be attached to | 
 |   // the first thread that calls ThreadCheckerImpl::CalledOnValidThread(). | 
 |   thread_checker_.DetachFromThread(); | 
 | } | 
 |  | 
 | TestTaskFactory::~TestTaskFactory() { | 
 |   WaitForAllTasksToRun(); | 
 | } | 
 |  | 
 | bool TestTaskFactory::PostTask(PostNestedTask post_nested_task, | 
 |                                OnceClosure after_task_closure) { | 
 |   AutoLock auto_lock(lock_); | 
 |   return task_runner_->PostTask( | 
 |       FROM_HERE, BindOnce(&TestTaskFactory::RunTaskCallback, Unretained(this), | 
 |                           num_posted_tasks_++, post_nested_task, | 
 |                           std::move(after_task_closure))); | 
 | } | 
 |  | 
 | void TestTaskFactory::WaitForAllTasksToRun() const { | 
 |   AutoLock auto_lock(lock_); | 
 |   while (ran_tasks_.size() < num_posted_tasks_) | 
 |     cv_.Wait(); | 
 | } | 
 |  | 
 | void TestTaskFactory::RunTaskCallback(size_t task_index, | 
 |                                       PostNestedTask post_nested_task, | 
 |                                       OnceClosure after_task_closure) { | 
 |   if (post_nested_task == PostNestedTask::YES) | 
 |     PostTask(PostNestedTask::NO, Closure()); | 
 |  | 
 |   EXPECT_TRUE(task_runner_->RunsTasksInCurrentSequence()); | 
 |  | 
 |   // Verify TaskRunnerHandles are set as expected in the task's scope. | 
 |   switch (execution_mode_) { | 
 |     case ExecutionMode::PARALLEL: | 
 |       EXPECT_FALSE(ThreadTaskRunnerHandle::IsSet()); | 
 |       EXPECT_FALSE(SequencedTaskRunnerHandle::IsSet()); | 
 |       break; | 
 |     case ExecutionMode::SEQUENCED: | 
 |       EXPECT_FALSE(ThreadTaskRunnerHandle::IsSet()); | 
 |       EXPECT_TRUE(SequencedTaskRunnerHandle::IsSet()); | 
 |       EXPECT_EQ(task_runner_, SequencedTaskRunnerHandle::Get()); | 
 |       break; | 
 |     case ExecutionMode::SINGLE_THREADED: | 
 |       // SequencedTaskRunnerHandle inherits from ThreadTaskRunnerHandle so | 
 |       // both are expected to be "set" in the SINGLE_THREADED case. | 
 |       EXPECT_TRUE(ThreadTaskRunnerHandle::IsSet()); | 
 |       EXPECT_TRUE(SequencedTaskRunnerHandle::IsSet()); | 
 |       EXPECT_EQ(task_runner_, ThreadTaskRunnerHandle::Get()); | 
 |       EXPECT_EQ(task_runner_, SequencedTaskRunnerHandle::Get()); | 
 |       break; | 
 |   } | 
 |  | 
 |   { | 
 |     AutoLock auto_lock(lock_); | 
 |  | 
 |     DCHECK_LE(task_index, num_posted_tasks_); | 
 |  | 
 |     if ((execution_mode_ == ExecutionMode::SINGLE_THREADED || | 
 |          execution_mode_ == ExecutionMode::SEQUENCED) && | 
 |         task_index != ran_tasks_.size()) { | 
 |       ADD_FAILURE() << "A task didn't run in the expected order."; | 
 |     } | 
 |  | 
 |     if (execution_mode_ == ExecutionMode::SINGLE_THREADED) | 
 |       EXPECT_TRUE(thread_checker_.CalledOnValidThread()); | 
 |  | 
 |     if (ran_tasks_.find(task_index) != ran_tasks_.end()) | 
 |       ADD_FAILURE() << "A task ran more than once."; | 
 |     ran_tasks_.insert(task_index); | 
 |  | 
 |     cv_.Signal(); | 
 |   } | 
 |  | 
 |   if (!after_task_closure.is_null()) | 
 |     std::move(after_task_closure).Run(); | 
 | } | 
 |  | 
 | }  // namespace test | 
 | }  // namespace internal | 
 | }  // namespace base |