blob: bec98e499172663151737642361cd730a4fc4e77 [file] [log] [blame]
// 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);
};
GTEST_ALLOW_UNINSTANTIATED_PARAMETERIZED_TEST(SequencedTaskRunnerTest);
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