// Copyright 2017 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_scheduler/lazy_task_runner.h" #include "base/bind.h" #include "base/bind_helpers.h" #include "base/sequence_checker_impl.h" #include "base/task_scheduler/scoped_set_task_priority_for_current_thread.h" #include "base/test/scoped_task_environment.h" #include "base/threading/thread_checker_impl.h" #include "build/build_config.h" #include "testing/gtest/include/gtest/gtest.h" #if defined(OS_WIN) #include "base/win/com_init_util.h" #endif namespace base { namespace { LazySequencedTaskRunner g_sequenced_task_runner_user_visible = LAZY_SEQUENCED_TASK_RUNNER_INITIALIZER({TaskPriority::USER_VISIBLE}); LazySequencedTaskRunner g_sequenced_task_runner_user_blocking = LAZY_SEQUENCED_TASK_RUNNER_INITIALIZER({TaskPriority::USER_BLOCKING}); LazySingleThreadTaskRunner g_single_thread_task_runner_user_visible = LAZY_SINGLE_THREAD_TASK_RUNNER_INITIALIZER( {TaskPriority::USER_VISIBLE}, SingleThreadTaskRunnerThreadMode::SHARED); LazySingleThreadTaskRunner g_single_thread_task_runner_user_blocking = LAZY_SINGLE_THREAD_TASK_RUNNER_INITIALIZER( {TaskPriority::USER_BLOCKING}, SingleThreadTaskRunnerThreadMode::SHARED); #if defined(OS_WIN) LazyCOMSTATaskRunner g_com_sta_task_runner_user_visible = LAZY_COM_STA_TASK_RUNNER_INITIALIZER( {TaskPriority::USER_VISIBLE}, SingleThreadTaskRunnerThreadMode::SHARED); LazyCOMSTATaskRunner g_com_sta_task_runner_user_blocking = LAZY_COM_STA_TASK_RUNNER_INITIALIZER( {TaskPriority::USER_BLOCKING}, SingleThreadTaskRunnerThreadMode::SHARED); #endif // defined(OS_WIN) void InitCheckers(SequenceCheckerImpl* sequence_checker, ThreadCheckerImpl* thread_checker) { sequence_checker->DetachFromSequence(); EXPECT_TRUE(sequence_checker->CalledOnValidSequence()); thread_checker->DetachFromThread(); EXPECT_TRUE(thread_checker->CalledOnValidThread()); } void ExpectSequencedEnvironment(SequenceCheckerImpl* sequence_checker, ThreadCheckerImpl* thread_checker, TaskPriority expected_priority) { EXPECT_TRUE(sequence_checker->CalledOnValidSequence()); EXPECT_FALSE(thread_checker->CalledOnValidThread()); EXPECT_EQ(expected_priority, internal::GetTaskPriorityForCurrentThread()); } void ExpectSingleThreadEnvironment(SequenceCheckerImpl* sequence_checker, ThreadCheckerImpl* thread_checker, TaskPriority expected_priority #if defined(OS_WIN) , bool expect_com_sta = false #endif ) { EXPECT_TRUE(sequence_checker->CalledOnValidSequence()); EXPECT_TRUE(thread_checker->CalledOnValidThread()); EXPECT_EQ(expected_priority, internal::GetTaskPriorityForCurrentThread()); #if defined(OS_WIN) if (expect_com_sta) win::AssertComApartmentType(win::ComApartmentType::STA); #endif } class TaskSchedulerLazyTaskRunnerEnvironmentTest : public testing::Test { protected: TaskSchedulerLazyTaskRunnerEnvironmentTest() = default; void TestTaskRunnerEnvironment(scoped_refptr<SequencedTaskRunner> task_runner, bool expect_single_thread, TaskPriority expected_priority #if defined(OS_WIN) , bool expect_com_sta = false #endif ) { SequenceCheckerImpl sequence_checker; ThreadCheckerImpl thread_checker; task_runner->PostTask(FROM_HERE, BindOnce(&InitCheckers, Unretained(&sequence_checker), Unretained(&thread_checker))); scoped_task_environment_.RunUntilIdle(); OnceClosure task = expect_single_thread ? BindOnce(&ExpectSingleThreadEnvironment, Unretained(&sequence_checker), Unretained(&thread_checker), expected_priority #if defined(OS_WIN) , expect_com_sta #endif ) : BindOnce(&ExpectSequencedEnvironment, Unretained(&sequence_checker), Unretained(&thread_checker), expected_priority); task_runner->PostTask(FROM_HERE, std::move(task)); scoped_task_environment_.RunUntilIdle(); } test::ScopedTaskEnvironment scoped_task_environment_; private: DISALLOW_COPY_AND_ASSIGN(TaskSchedulerLazyTaskRunnerEnvironmentTest); }; } // namespace TEST_F(TaskSchedulerLazyTaskRunnerEnvironmentTest, LazySequencedTaskRunnerUserVisible) { TestTaskRunnerEnvironment(g_sequenced_task_runner_user_visible.Get(), false, TaskPriority::USER_VISIBLE); } TEST_F(TaskSchedulerLazyTaskRunnerEnvironmentTest, LazySequencedTaskRunnerUserBlocking) { TestTaskRunnerEnvironment(g_sequenced_task_runner_user_blocking.Get(), false, TaskPriority::USER_BLOCKING); } TEST_F(TaskSchedulerLazyTaskRunnerEnvironmentTest, LazySingleThreadTaskRunnerUserVisible) { TestTaskRunnerEnvironment(g_single_thread_task_runner_user_visible.Get(), true, TaskPriority::USER_VISIBLE); } TEST_F(TaskSchedulerLazyTaskRunnerEnvironmentTest, LazySingleThreadTaskRunnerUserBlocking) { TestTaskRunnerEnvironment(g_single_thread_task_runner_user_blocking.Get(), true, TaskPriority::USER_BLOCKING); } #if defined(OS_WIN) TEST_F(TaskSchedulerLazyTaskRunnerEnvironmentTest, LazyCOMSTATaskRunnerUserVisible) { TestTaskRunnerEnvironment(g_com_sta_task_runner_user_visible.Get(), true, TaskPriority::USER_VISIBLE, true); } TEST_F(TaskSchedulerLazyTaskRunnerEnvironmentTest, LazyCOMSTATaskRunnerUserBlocking) { TestTaskRunnerEnvironment(g_com_sta_task_runner_user_blocking.Get(), true, TaskPriority::USER_BLOCKING, true); } #endif // defined(OS_WIN) TEST(TaskSchdulerLazyTaskRunnerTest, LazySequencedTaskRunnerReset) { for (int i = 0; i < 2; ++i) { test::ScopedTaskEnvironment scoped_task_environment; // If the TaskRunner isn't released when the test::ScopedTaskEnvironment // goes out of scope, the second invocation of the line below will access a // deleted TaskScheduler and crash. g_sequenced_task_runner_user_visible.Get()->PostTask(FROM_HERE, DoNothing()); } } TEST(TaskSchdulerLazyTaskRunnerTest, LazySingleThreadTaskRunnerReset) { for (int i = 0; i < 2; ++i) { test::ScopedTaskEnvironment scoped_task_environment; // If the TaskRunner isn't released when the test::ScopedTaskEnvironment // goes out of scope, the second invocation of the line below will access a // deleted TaskScheduler and crash. g_single_thread_task_runner_user_visible.Get()->PostTask(FROM_HERE, DoNothing()); } } #if defined(OS_WIN) TEST(TaskSchdulerLazyTaskRunnerTest, LazyCOMSTATaskRunnerReset) { for (int i = 0; i < 2; ++i) { test::ScopedTaskEnvironment scoped_task_environment; // If the TaskRunner isn't released when the test::ScopedTaskEnvironment // goes out of scope, the second invocation of the line below will access a // deleted TaskScheduler and crash. g_com_sta_task_runner_user_visible.Get()->PostTask(FROM_HERE, DoNothing()); } } #endif // defined(OS_WIN) } // namespace base