blob: 061b54a0f299a804d2b260d809dd96687cc42c80 [file] [log] [blame]
// Copyright (c) 2011 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 "net/dns/serial_worker.h"
#include "base/bind.h"
#include "base/location.h"
#include "base/run_loop.h"
#include "base/single_thread_task_runner.h"
#include "base/synchronization/lock.h"
#include "base/synchronization/waitable_event.h"
#include "base/task/current_thread.h"
#include "base/threading/thread_restrictions.h"
#include "base/threading/thread_task_runner_handle.h"
#include "net/test/test_with_task_environment.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace net {
namespace {
class SerialWorkerTest : public TestWithTaskEnvironment {
public:
// The class under test
class TestSerialWorker : public SerialWorker {
public:
explicit TestSerialWorker(SerialWorkerTest* t) : test_(t) {}
void DoWork() override {
ASSERT_TRUE(test_);
test_->OnWork();
}
void OnWorkFinished() override {
ASSERT_TRUE(test_);
test_->OnWorkFinished();
}
private:
~TestSerialWorker() override = default;
SerialWorkerTest* test_;
};
// Mocks
void OnWork() {
{ // Check that OnWork is executed serially.
base::AutoLock lock(work_lock_);
EXPECT_FALSE(work_running_) << "DoRead is not called serially!";
work_running_ = true;
}
BreakNow("OnWork");
{
base::ScopedAllowBaseSyncPrimitivesForTesting
scoped_allow_base_sync_primitives;
work_allowed_.Wait();
}
// Calling from ThreadPool, but protected by work_allowed_/work_called_.
output_value_ = input_value_;
{ // This lock might be destroyed after work_called_ is signalled.
base::AutoLock lock(work_lock_);
work_running_ = false;
}
work_called_.Signal();
}
void OnWorkFinished() {
EXPECT_TRUE(task_runner_->BelongsToCurrentThread());
EXPECT_EQ(output_value_, input_value_);
BreakNow("OnWorkFinished");
}
protected:
void BreakCallback(const std::string& breakpoint) {
breakpoint_ = breakpoint;
run_loop_->Quit();
}
void BreakNow(const std::string& b) {
task_runner_->PostTask(FROM_HERE,
base::BindOnce(&SerialWorkerTest::BreakCallback,
base::Unretained(this), b));
}
void RunUntilBreak(const std::string& b) {
base::RunLoop run_loop;
ASSERT_FALSE(run_loop_);
run_loop_ = &run_loop;
run_loop_->Run();
run_loop_ = nullptr;
ASSERT_EQ(breakpoint_, b);
}
SerialWorkerTest()
: input_value_(0),
output_value_(-1),
work_allowed_(base::WaitableEvent::ResetPolicy::AUTOMATIC,
base::WaitableEvent::InitialState::NOT_SIGNALED),
work_called_(base::WaitableEvent::ResetPolicy::AUTOMATIC,
base::WaitableEvent::InitialState::NOT_SIGNALED),
work_running_(false) {}
// Helpers for tests.
// Lets OnWork run and waits for it to complete. Can only return if OnWork is
// executed on a concurrent thread.
void WaitForWork() {
RunUntilBreak("OnWork");
work_allowed_.Signal();
work_called_.Wait();
}
// test::Test methods
void SetUp() override {
task_runner_ = base::ThreadTaskRunnerHandle::Get();
worker_ = new TestSerialWorker(this);
}
void TearDown() override {
// Cancel the worker to catch if it makes a late DoWork call.
worker_->Cancel();
// Check if OnWork is stalled.
EXPECT_FALSE(work_running_) << "OnWork should be done by TearDown";
// Release it for cleanliness.
if (work_running_) {
WaitForWork();
}
}
// Input value read on WorkerPool.
int input_value_;
// Output value written on WorkerPool.
int output_value_;
// read is called on WorkerPool so we need to synchronize with it.
base::WaitableEvent work_allowed_;
base::WaitableEvent work_called_;
// Protected by read_lock_. Used to verify that read calls are serialized.
bool work_running_;
base::Lock work_lock_;
// Task runner for this thread.
scoped_refptr<base::SingleThreadTaskRunner> task_runner_;
// WatcherDelegate under test.
scoped_refptr<TestSerialWorker> worker_;
std::string breakpoint_;
base::RunLoop* run_loop_ = nullptr;
DISALLOW_COPY_AND_ASSIGN(SerialWorkerTest);
};
TEST_F(SerialWorkerTest, ExecuteAndSerializeReads) {
for (int i = 0; i < 3; ++i) {
++input_value_;
worker_->WorkNow();
WaitForWork();
RunUntilBreak("OnWorkFinished");
EXPECT_TRUE(base::CurrentThread::Get()->IsIdleForTesting());
}
// Schedule two calls. OnWork checks if it is called serially.
++input_value_;
worker_->WorkNow();
// read is blocked, so this will have to induce re-work
worker_->WorkNow();
WaitForWork();
WaitForWork();
RunUntilBreak("OnWorkFinished");
// No more tasks should remain.
EXPECT_TRUE(base::CurrentThread::Get()->IsIdleForTesting());
}
} // namespace
} // namespace net