| Index: components/scheduler/base/time_domain_unittest.cc
|
| diff --git a/components/scheduler/base/time_domain_unittest.cc b/components/scheduler/base/time_domain_unittest.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..88328c098f1ce6b2a853c769361e22cf29e750ed
|
| --- /dev/null
|
| +++ b/components/scheduler/base/time_domain_unittest.cc
|
| @@ -0,0 +1,210 @@
|
| +// Copyright 2015 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 "components/scheduler/base/time_domain.h"
|
| +
|
| +#include "base/test/simple_test_tick_clock.h"
|
| +#include "cc/test/ordered_simple_task_runner.h"
|
| +#include "components/scheduler/base/task_queue_impl.h"
|
| +#include "components/scheduler/base/task_queue_manager.h"
|
| +#include "components/scheduler/base/task_queue_manager_delegate.h"
|
| +#include "components/scheduler/base/task_queue_manager_delegate_for_test.h"
|
| +#include "components/scheduler/base/test_time_source.h"
|
| +#include "testing/gmock/include/gmock/gmock.h"
|
| +
|
| +using testing::_;
|
| +
|
| +namespace scheduler {
|
| +
|
| +class FakeTaskQueueManagerDelegate : public TaskQueueManagerDelegate {
|
| + public:
|
| + FakeTaskQueueManagerDelegate() : belongs_to_current_thread_(true) {}
|
| +
|
| + MOCK_METHOD3(PostDelayedTask,
|
| + bool(const tracked_objects::Location& from_here,
|
| + const base::Closure& task,
|
| + base::TimeDelta dela));
|
| + MOCK_METHOD3(PostNonNestableDelayedTask,
|
| + bool(const tracked_objects::Location& from_here,
|
| + const base::Closure& task,
|
| + base::TimeDelta dela));
|
| +
|
| + base::TimeTicks NowTicks() override {
|
| + NOTREACHED();
|
| + return base::TimeTicks();
|
| + }
|
| +
|
| + bool IsNested() const override {
|
| + NOTREACHED();
|
| + return false;
|
| + }
|
| +
|
| + void OnNoMoreImmediateWork() override { NOTREACHED(); }
|
| +
|
| + double CurrentTimeSeconds() const override {
|
| + NOTREACHED();
|
| + return 0.0;
|
| + }
|
| +
|
| + void SetBelongsToCurrentThread(bool belongs_to_current_thread) {
|
| + belongs_to_current_thread_ = belongs_to_current_thread;
|
| + }
|
| +
|
| + bool RunsTasksOnCurrentThread() const override {
|
| + return belongs_to_current_thread_;
|
| + }
|
| +
|
| + private:
|
| + ~FakeTaskQueueManagerDelegate() override {}
|
| +
|
| + bool belongs_to_current_thread_;
|
| +};
|
| +
|
| +class MockTimeDomain : public TimeDomain {
|
| + public:
|
| + MockTimeDomain(TaskQueueManagerDelegate* delegate)
|
| + : TimeDomain(delegate),
|
| + now_(base::TimeTicks() + base::TimeDelta::FromSeconds(1)) {}
|
| +
|
| + using TimeDomain::NextScheduledRunTime;
|
| + using TimeDomain::NextScheduledTaskQueue;
|
| + using TimeDomain::ScheduleDelayedWork;
|
| + using TimeDomain::UnregisterQueue;
|
| + using TimeDomain::UpdateWorkQueues;
|
| +
|
| + // TimeSource implementation:
|
| + LazyNow GetLazyNow() override { return LazyNow(now_); }
|
| +
|
| + void AsValueIntoInternal(
|
| + base::trace_event::TracedValue* state) const override {}
|
| +
|
| + bool MaybeAdvanceTime() override { return false; }
|
| +
|
| + MOCK_METHOD1(ScheduleDoWork, void(base::TimeDelta delay));
|
| +
|
| + void SetNow(base::TimeTicks now) { now_ = now; }
|
| +
|
| + base::TimeTicks Now() const { return now_; }
|
| +
|
| + private:
|
| + base::TimeTicks now_;
|
| +
|
| + ~MockTimeDomain() override {}
|
| +
|
| + DISALLOW_COPY_AND_ASSIGN(MockTimeDomain);
|
| +};
|
| +
|
| +class TimeDomainTest : public testing::Test {
|
| + public:
|
| + void SetUp() final {
|
| + delegate_ = make_scoped_refptr(new FakeTaskQueueManagerDelegate);
|
| + time_domain_ = make_scoped_refptr(new MockTimeDomain(delegate_.get()));
|
| + task_queue_ = make_scoped_refptr(new internal::TaskQueueImpl(
|
| + nullptr, time_domain_, TaskQueue::Spec("test_queue"), "test.category",
|
| + "test.category"));
|
| + }
|
| +
|
| + scoped_refptr<FakeTaskQueueManagerDelegate> delegate_;
|
| + scoped_refptr<MockTimeDomain> time_domain_;
|
| + scoped_refptr<internal::TaskQueueImpl> task_queue_;
|
| +};
|
| +
|
| +TEST_F(TimeDomainTest, ScheduleDelayedWork_SameThread) {
|
| + base::TimeDelta delay = base::TimeDelta::FromMilliseconds(10);
|
| + base::TimeTicks delayed_runtime = time_domain_->Now() + delay;
|
| + EXPECT_CALL(*time_domain_.get(), ScheduleDoWork(delay));
|
| + LazyNow lazy_now = time_domain_->GetLazyNow();
|
| + time_domain_->ScheduleDelayedWork(task_queue_.get(),
|
| + time_domain_->Now() + delay, &lazy_now);
|
| +
|
| + base::TimeTicks next_scheduled_runtime;
|
| + EXPECT_TRUE(time_domain_->NextScheduledRunTime(&next_scheduled_runtime));
|
| + EXPECT_EQ(delayed_runtime, next_scheduled_runtime);
|
| +
|
| + TaskQueue* next_task_queue;
|
| + EXPECT_TRUE(time_domain_->NextScheduledTaskQueue(&next_task_queue));
|
| + EXPECT_EQ(task_queue_.get(), next_task_queue);
|
| +}
|
| +
|
| +TEST_F(TimeDomainTest, ScheduleDelayedWork_DifferentThread) {
|
| + delegate_->SetBelongsToCurrentThread(false);
|
| +
|
| + base::TimeDelta delay = base::TimeDelta::FromMilliseconds(10);
|
| + base::TimeTicks delayed_runtime = time_domain_->Now() + delay;
|
| +
|
| + EXPECT_CALL(*delegate_.get(), PostDelayedTask(_, _, base::TimeDelta()));
|
| + LazyNow lazy_now = time_domain_->GetLazyNow();
|
| + time_domain_->ScheduleDelayedWork(task_queue_.get(), delayed_runtime,
|
| + &lazy_now);
|
| +
|
| + base::TimeTicks next_scheduled_runtime;
|
| + EXPECT_FALSE(time_domain_->NextScheduledRunTime(&next_scheduled_runtime));
|
| +}
|
| +
|
| +TEST_F(TimeDomainTest, UnregisterQueue) {
|
| + scoped_refptr<internal::TaskQueueImpl> task_queue2_ =
|
| + make_scoped_refptr(new internal::TaskQueueImpl(
|
| + nullptr, time_domain_, TaskQueue::Spec("test_queue2"),
|
| + "test.category", "test.category"));
|
| +
|
| + EXPECT_CALL(*time_domain_.get(), ScheduleDoWork(_)).Times(2);
|
| + LazyNow lazy_now = time_domain_->GetLazyNow();
|
| + time_domain_->ScheduleDelayedWork(
|
| + task_queue_.get(),
|
| + time_domain_->Now() + base::TimeDelta::FromMilliseconds(10), &lazy_now);
|
| + time_domain_->ScheduleDelayedWork(
|
| + task_queue2_.get(),
|
| + time_domain_->Now() + base::TimeDelta::FromMilliseconds(100), &lazy_now);
|
| +
|
| + TaskQueue* next_task_queue;
|
| + EXPECT_TRUE(time_domain_->NextScheduledTaskQueue(&next_task_queue));
|
| + EXPECT_EQ(task_queue_.get(), next_task_queue);
|
| +
|
| + time_domain_->UnregisterQueue(task_queue_.get());
|
| + EXPECT_TRUE(time_domain_->NextScheduledTaskQueue(&next_task_queue));
|
| + EXPECT_EQ(task_queue2_.get(), next_task_queue);
|
| +
|
| + time_domain_->UnregisterQueue(task_queue2_.get());
|
| + EXPECT_FALSE(time_domain_->NextScheduledTaskQueue(&next_task_queue));
|
| +}
|
| +
|
| +TEST_F(TimeDomainTest, WakeupReadyDelayedQueues) {
|
| + scoped_refptr<MockTimeDomain> dummy_delegate(
|
| + new MockTimeDomain(delegate_.get()));
|
| + base::SimpleTestTickClock dummy_time_source;
|
| + scoped_refptr<cc::OrderedSimpleTaskRunner> dummy_task_runner(
|
| + new cc::OrderedSimpleTaskRunner(&dummy_time_source, false));
|
| + TaskQueueManager task_queue_manager(
|
| + TaskQueueManagerDelegateForTest::Create(
|
| + dummy_task_runner,
|
| + make_scoped_ptr(new TestTimeSource(&dummy_time_source))),
|
| + "test.scheduler", "test.scheduler", "scheduler.debug");
|
| + scoped_refptr<internal::TaskQueueImpl> dummy_queue =
|
| + task_queue_manager.NewTaskQueue(TaskQueue::Spec("test_queue"));
|
| +
|
| + // Post a delayed task on |dummy_queue| and advance the queue's clock so that
|
| + // next time MoveReadyDelayedTasksToIncomingQueue is called, the task will
|
| + // get moved onto the incomming queue.
|
| + base::TimeDelta dummy_delay = base::TimeDelta::FromMilliseconds(10);
|
| + dummy_queue->PostDelayedTask(FROM_HERE, base::Closure(), dummy_delay);
|
| + dummy_time_source.Advance(dummy_delay);
|
| +
|
| + // Now we can test that WakeupReadyDelayedQueues triggers calls to
|
| + // MoveReadyDelayedTasksToIncomingQueue as expected.
|
| + base::TimeDelta delay = base::TimeDelta::FromMilliseconds(50);
|
| + base::TimeTicks delayed_runtime = time_domain_->Now() + delay;
|
| + EXPECT_CALL(*time_domain_.get(), ScheduleDoWork(delay));
|
| + LazyNow lazy_now = time_domain_->GetLazyNow();
|
| + time_domain_->ScheduleDelayedWork(dummy_queue.get(), delayed_runtime,
|
| + &lazy_now);
|
| +
|
| + time_domain_->UpdateWorkQueues(false, nullptr);
|
| + EXPECT_EQ(0UL, dummy_queue->IncomingQueueSizeForTest());
|
| +
|
| + time_domain_->SetNow(delayed_runtime);
|
| + time_domain_->UpdateWorkQueues(false, nullptr);
|
| + EXPECT_EQ(1UL, dummy_queue->IncomingQueueSizeForTest());
|
| +}
|
| +
|
| +} // namespace scheduler
|
|
|