Index: base/test/scoped_task_scheduler_unittest.cc |
diff --git a/base/test/scoped_task_scheduler_unittest.cc b/base/test/scoped_task_scheduler_unittest.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..057dff74bb55bd23214060dc9cc0744c756abc37 |
--- /dev/null |
+++ b/base/test/scoped_task_scheduler_unittest.cc |
@@ -0,0 +1,252 @@ |
+// 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/test/scoped_task_scheduler.h" |
+ |
+#include "base/bind.h" |
+#include "base/logging.h" |
+#include "base/macros.h" |
+#include "base/run_loop.h" |
+#include "base/sequence_checker.h" |
+#include "base/task_scheduler/post_task.h" |
+#include "base/task_scheduler/test_utils.h" |
+#include "base/threading/sequenced_task_runner_handle.h" |
+#include "base/threading/thread_checker.h" |
+#include "base/threading/thread_task_runner_handle.h" |
+#include "build/build_config.h" |
+#include "testing/gtest/include/gtest/gtest.h" |
+ |
+namespace base { |
+namespace test { |
+ |
+TEST(ScopedTaskSchedulerTest, PostTask) { |
+ ScopedTaskScheduler scoped_task_scheduler; |
+ |
+ bool first_task_ran = false; |
+ bool second_task_ran = false; |
+ |
+ SequenceCheckerImpl sequence_checker; |
+ ThreadCheckerImpl thread_checker; |
+ |
+ // Detach |sequence_checker| and |thread_checker|. Otherwise, they are bound |
+ // to the current thread without a SequenceToken or TaskToken (i.e. |
+ // CalledOnValidSequence/Thread() will always return true on the current |
+ // thread, even when the SequenceToken or TaskToken changes). |
+ sequence_checker.DetachFromSequence(); |
+ thread_checker.DetachFromThread(); |
+ |
+ PostTask(FROM_HERE, |
+ Bind( |
+ [](SequenceCheckerImpl* sequence_checker, |
+ ThreadCheckerImpl* thread_checker, bool* first_task_ran) { |
+ EXPECT_FALSE(SequencedTaskRunnerHandle::IsSet()); |
+ EXPECT_FALSE(ThreadTaskRunnerHandle::IsSet()); |
+ EXPECT_TRUE(sequence_checker->CalledOnValidSequence()); |
+ EXPECT_TRUE(thread_checker->CalledOnValidThread()); |
+ *first_task_ran = true; |
+ }, |
+ Unretained(&sequence_checker), Unretained(&thread_checker), |
+ Unretained(&first_task_ran))); |
+ |
+ PostTask(FROM_HERE, |
+ Bind( |
+ [](SequenceCheckerImpl* sequence_checker, |
+ ThreadCheckerImpl* thread_checker, bool* second_task_ran) { |
+ EXPECT_FALSE(SequencedTaskRunnerHandle::IsSet()); |
+ EXPECT_FALSE(ThreadTaskRunnerHandle::IsSet()); |
+ EXPECT_FALSE(sequence_checker->CalledOnValidSequence()); |
+ EXPECT_FALSE(thread_checker->CalledOnValidThread()); |
+ *second_task_ran = true; |
+ }, |
+ Unretained(&sequence_checker), Unretained(&thread_checker), |
+ Unretained(&second_task_ran))); |
+ |
+ RunLoop().RunUntilIdle(); |
+ |
+ EXPECT_TRUE(first_task_ran); |
+ EXPECT_TRUE(second_task_ran); |
+} |
+ |
+TEST(ScopedTaskSchedulerTest, CreateTaskRunnerAndPostTask) { |
+ ScopedTaskScheduler scoped_task_scheduler; |
+ auto task_runner = CreateTaskRunnerWithTraits(TaskTraits()); |
+ |
+ bool first_task_ran = false; |
+ bool second_task_ran = false; |
+ |
+ SequenceCheckerImpl sequence_checker; |
+ ThreadCheckerImpl thread_checker; |
+ |
+ // Detach |sequence_checker| and |thread_checker|. Otherwise, they are bound |
+ // to the current thread without a SequenceToken or TaskToken (i.e. |
+ // CalledOnValidSequence/Thread() will always return true on the current |
+ // thread, even when the SequenceToken or TaskToken changes). |
+ sequence_checker.DetachFromSequence(); |
+ thread_checker.DetachFromThread(); |
+ |
+ task_runner->PostTask( |
+ FROM_HERE, |
+ Bind( |
+ [](SequenceCheckerImpl* sequence_checker, |
+ ThreadCheckerImpl* thread_checker, bool* first_task_ran) { |
+ EXPECT_FALSE(SequencedTaskRunnerHandle::IsSet()); |
+ EXPECT_FALSE(ThreadTaskRunnerHandle::IsSet()); |
+ EXPECT_TRUE(sequence_checker->CalledOnValidSequence()); |
+ EXPECT_TRUE(thread_checker->CalledOnValidThread()); |
+ *first_task_ran = true; |
+ }, |
+ Unretained(&sequence_checker), Unretained(&thread_checker), |
+ Unretained(&first_task_ran))); |
+ |
+ task_runner->PostTask( |
+ FROM_HERE, |
+ Bind( |
+ [](SequenceCheckerImpl* sequence_checker, |
+ ThreadCheckerImpl* thread_checker, bool* second_task_ran) { |
+ EXPECT_FALSE(SequencedTaskRunnerHandle::IsSet()); |
+ EXPECT_FALSE(ThreadTaskRunnerHandle::IsSet()); |
+ EXPECT_FALSE(sequence_checker->CalledOnValidSequence()); |
+ EXPECT_FALSE(thread_checker->CalledOnValidThread()); |
+ *second_task_ran = true; |
+ }, |
+ Unretained(&sequence_checker), Unretained(&thread_checker), |
+ Unretained(&second_task_ran))); |
+ |
+ RunLoop().RunUntilIdle(); |
+ |
+ EXPECT_TRUE(first_task_ran); |
+ EXPECT_TRUE(second_task_ran); |
+} |
+ |
+TEST(ScopedTaskSchedulerTest, CreateSequencedTaskRunnerAndPostTask) { |
+ ScopedTaskScheduler scoped_task_scheduler; |
+ auto task_runner = CreateSequencedTaskRunnerWithTraits(TaskTraits()); |
+ |
+ bool first_task_ran = false; |
+ bool second_task_ran = false; |
+ |
+ SequenceCheckerImpl sequence_checker; |
+ ThreadCheckerImpl thread_checker; |
+ |
+ // Detach |sequence_checker| and |thread_checker|. Otherwise, they are bound |
+ // to the current thread without a SequenceToken or TaskToken (i.e. |
+ // CalledOnValidSequence/Thread() will always return true on the current |
+ // thread, even when the SequenceToken or TaskToken changes). |
+ sequence_checker.DetachFromSequence(); |
+ thread_checker.DetachFromThread(); |
+ |
+ task_runner->PostTask( |
+ FROM_HERE, |
+ Bind( |
+ [](SequenceCheckerImpl* sequence_checker, |
+ ThreadCheckerImpl* thread_checker, bool* first_task_ran) { |
+ EXPECT_TRUE(SequencedTaskRunnerHandle::IsSet()); |
+ EXPECT_FALSE(ThreadTaskRunnerHandle::IsSet()); |
+ EXPECT_TRUE(sequence_checker->CalledOnValidSequence()); |
+ EXPECT_TRUE(thread_checker->CalledOnValidThread()); |
+ *first_task_ran = true; |
+ }, |
+ Unretained(&sequence_checker), Unretained(&thread_checker), |
+ Unretained(&first_task_ran))); |
+ |
+ task_runner->PostTask( |
+ FROM_HERE, |
+ Bind( |
+ [](SequenceCheckerImpl* sequence_checker, |
+ ThreadCheckerImpl* thread_checker, bool* second_task_ran) { |
+ EXPECT_TRUE(SequencedTaskRunnerHandle::IsSet()); |
+ EXPECT_FALSE(ThreadTaskRunnerHandle::IsSet()); |
+ EXPECT_TRUE(sequence_checker->CalledOnValidSequence()); |
+ EXPECT_FALSE(thread_checker->CalledOnValidThread()); |
+ *second_task_ran = true; |
+ }, |
+ Unretained(&sequence_checker), Unretained(&thread_checker), |
+ Unretained(&second_task_ran))); |
+ |
+ RunLoop().RunUntilIdle(); |
+ |
+ EXPECT_TRUE(first_task_ran); |
+ EXPECT_TRUE(second_task_ran); |
+} |
+ |
+TEST(ScopedTaskSchedulerTest, CreateSingleThreadTaskRunnerAndPostTask) { |
+ ScopedTaskScheduler scoped_task_scheduler; |
+ auto task_runner = CreateSingleThreadTaskRunnerWithTraits(TaskTraits()); |
+ |
+ bool first_task_ran = false; |
+ bool second_task_ran = false; |
+ |
+ SequenceCheckerImpl sequence_checker; |
+ ThreadCheckerImpl thread_checker; |
+ |
+ // Detach |sequence_checker| and |thread_checker|. Otherwise, they are bound |
+ // to the current thread without a SequenceToken or TaskToken (i.e. |
+ // CalledOnValidSequence/Thread() will always return true on the current |
+ // thread, even when the SequenceToken or TaskToken changes). |
+ sequence_checker.DetachFromSequence(); |
+ thread_checker.DetachFromThread(); |
+ |
+ task_runner->PostTask( |
+ FROM_HERE, |
+ Bind( |
+ [](SequenceCheckerImpl* sequence_checker, |
+ ThreadCheckerImpl* thread_checker, bool* first_task_ran) { |
+ EXPECT_TRUE(SequencedTaskRunnerHandle::IsSet()); |
+ EXPECT_TRUE(ThreadTaskRunnerHandle::IsSet()); |
+ EXPECT_TRUE(sequence_checker->CalledOnValidSequence()); |
+ EXPECT_TRUE(thread_checker->CalledOnValidThread()); |
+ *first_task_ran = true; |
+ }, |
+ Unretained(&sequence_checker), Unretained(&thread_checker), |
+ Unretained(&first_task_ran))); |
+ |
+ task_runner->PostTask( |
+ FROM_HERE, |
+ Bind( |
+ [](SequenceCheckerImpl* sequence_checker, |
+ ThreadCheckerImpl* thread_checker, bool* second_task_ran) { |
+ EXPECT_TRUE(SequencedTaskRunnerHandle::IsSet()); |
+ EXPECT_TRUE(ThreadTaskRunnerHandle::IsSet()); |
+ EXPECT_TRUE(sequence_checker->CalledOnValidSequence()); |
+ EXPECT_TRUE(thread_checker->CalledOnValidThread()); |
+ *second_task_ran = true; |
+ }, |
+ Unretained(&sequence_checker), Unretained(&thread_checker), |
+ Unretained(&second_task_ran))); |
+ |
+ RunLoop().RunUntilIdle(); |
+ |
+ EXPECT_TRUE(first_task_ran); |
+ EXPECT_TRUE(second_task_ran); |
+} |
+ |
+TEST(ScopedTaskSchedulerTest, ShutdownBehavior) { |
+ bool block_shutdown_task_ran = false; |
+ { |
+ ScopedTaskScheduler scoped_task_scheduler; |
+ PostTaskWithTraits( |
+ FROM_HERE, TaskTraits().WithShutdownBehavior( |
+ TaskShutdownBehavior::CONTINUE_ON_SHUTDOWN), |
+ Bind([]() { |
+ ADD_FAILURE() << "CONTINUE_ON_SHUTDOWN task should not run"; |
+ })); |
+ PostTaskWithTraits(FROM_HERE, TaskTraits().WithShutdownBehavior( |
+ TaskShutdownBehavior::SKIP_ON_SHUTDOWN), |
+ Bind([]() { |
+ ADD_FAILURE() |
+ << "SKIP_ON_SHUTDOWN task should not run"; |
+ })); |
+ PostTaskWithTraits(FROM_HERE, TaskTraits().WithShutdownBehavior( |
+ TaskShutdownBehavior::BLOCK_SHUTDOWN), |
+ Bind( |
+ [](bool* block_shutdown_task_ran) { |
+ *block_shutdown_task_ran = true; |
+ }, |
+ Unretained(&block_shutdown_task_ran))); |
+ } |
+ EXPECT_TRUE(block_shutdown_task_ran); |
+} |
+ |
+} // namespace test |
+} // namespace base |