|  | // 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/sequenced_task_runner.h" | 
|  |  | 
|  | #include <utility> | 
|  |  | 
|  | #include "base/bind.h" | 
|  | #include "base/gtest_prod_util.h" | 
|  | #include "base/message_loop/message_loop.h" | 
|  | #include "base/run_loop.h" | 
|  | #include "base/threading/thread.h" | 
|  | #include "testing/gtest/include/gtest/gtest.h" | 
|  |  | 
|  | namespace base { | 
|  | namespace { | 
|  |  | 
|  | class FlagOnDelete { | 
|  | public: | 
|  | FlagOnDelete(bool* deleted, | 
|  | scoped_refptr<SequencedTaskRunner> expected_deletion_sequence) | 
|  | : deleted_(deleted), | 
|  | expected_deletion_sequence_(std::move(expected_deletion_sequence)) {} | 
|  |  | 
|  | private: | 
|  | friend class DeleteHelper<FlagOnDelete>; | 
|  | FRIEND_TEST_ALL_PREFIXES(SequencedTaskRunnerTest, | 
|  | OnTaskRunnerDeleterTargetStoppedEarly); | 
|  |  | 
|  | ~FlagOnDelete() { | 
|  | EXPECT_FALSE(*deleted_); | 
|  | *deleted_ = true; | 
|  | if (expected_deletion_sequence_) | 
|  | EXPECT_TRUE(expected_deletion_sequence_->RunsTasksInCurrentSequence()); | 
|  | } | 
|  |  | 
|  | bool* deleted_; | 
|  | const scoped_refptr<SequencedTaskRunner> expected_deletion_sequence_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(FlagOnDelete); | 
|  | }; | 
|  |  | 
|  | class SequencedTaskRunnerTest : public testing::Test { | 
|  | protected: | 
|  | SequencedTaskRunnerTest() : foreign_thread_("foreign") {} | 
|  |  | 
|  | void SetUp() override { | 
|  | main_runner_ = message_loop_.task_runner(); | 
|  |  | 
|  | foreign_thread_.Start(); | 
|  | foreign_runner_ = foreign_thread_.task_runner(); | 
|  | } | 
|  |  | 
|  | scoped_refptr<SequencedTaskRunner> main_runner_; | 
|  | scoped_refptr<SequencedTaskRunner> foreign_runner_; | 
|  |  | 
|  | Thread foreign_thread_; | 
|  |  | 
|  | private: | 
|  | MessageLoop message_loop_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(SequencedTaskRunnerTest); | 
|  | }; | 
|  |  | 
|  | using SequenceBoundUniquePtr = | 
|  | std::unique_ptr<FlagOnDelete, OnTaskRunnerDeleter>; | 
|  |  | 
|  | TEST_F(SequencedTaskRunnerTest, OnTaskRunnerDeleterOnMainThread) { | 
|  | bool deleted_on_main_thread = false; | 
|  | SequenceBoundUniquePtr ptr( | 
|  | new FlagOnDelete(&deleted_on_main_thread, main_runner_), | 
|  | OnTaskRunnerDeleter(main_runner_)); | 
|  | EXPECT_FALSE(deleted_on_main_thread); | 
|  | foreign_runner_->PostTask( | 
|  | FROM_HERE, BindOnce([](SequenceBoundUniquePtr) {}, std::move(ptr))); | 
|  |  | 
|  | { | 
|  | RunLoop run_loop; | 
|  | foreign_runner_->PostTaskAndReply(FROM_HERE, BindOnce([] {}), | 
|  | run_loop.QuitClosure()); | 
|  | run_loop.Run(); | 
|  | } | 
|  | EXPECT_TRUE(deleted_on_main_thread); | 
|  | } | 
|  |  | 
|  | TEST_F(SequencedTaskRunnerTest, OnTaskRunnerDeleterTargetStoppedEarly) { | 
|  | bool deleted_on_main_thread = false; | 
|  | FlagOnDelete* raw = new FlagOnDelete(&deleted_on_main_thread, main_runner_); | 
|  | SequenceBoundUniquePtr ptr(raw, OnTaskRunnerDeleter(foreign_runner_)); | 
|  | EXPECT_FALSE(deleted_on_main_thread); | 
|  |  | 
|  | // Stopping the target ahead of deleting |ptr| should make its | 
|  | // OnTaskRunnerDeleter no-op. | 
|  | foreign_thread_.Stop(); | 
|  | ptr = nullptr; | 
|  | EXPECT_FALSE(deleted_on_main_thread); | 
|  |  | 
|  | delete raw; | 
|  | EXPECT_TRUE(deleted_on_main_thread); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  | }  // namespace base |