Add TimeController to api/test/ and add a CreateTimeController API.
Creates an abstraction for an "alarm clock" which can schedule time-controller callbacks and exposes a time controller driven by an external alarm. Bug: webrtc:9719 Change-Id: I08c2aa9dba25603043bfba48f55c925716a55bae Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/158969 Reviewed-by: Mirko Bonadei <mbonadei@webrtc.org> Reviewed-by: Per Kjellander <perkj@webrtc.org> Reviewed-by: Artem Titov <titovartem@webrtc.org> Reviewed-by: Steve Anton <steveanton@webrtc.org> Commit-Queue: Bjorn Mellem <mellem@webrtc.org> Cr-Commit-Position: refs/heads/master@{#29879}
This commit is contained in:
committed by
Commit Bot
parent
3daedb6c88
commit
c4f865413a
181
test/time_controller/external_time_controller_unittest.cc
Normal file
181
test/time_controller/external_time_controller_unittest.cc
Normal file
@ -0,0 +1,181 @@
|
||||
/*
|
||||
* Copyright 2019 The WebRTC project authors. All Rights Reserved.
|
||||
*
|
||||
* Use of this source code is governed by a BSD-style license
|
||||
* that can be found in the LICENSE file in the root of the source
|
||||
* tree. An additional intellectual property rights grant can be found
|
||||
* in the file PATENTS. All contributing project authors may
|
||||
* be found in the AUTHORS file in the root of the source tree.
|
||||
*/
|
||||
|
||||
#include "test/time_controller/external_time_controller.h"
|
||||
|
||||
#include <atomic>
|
||||
#include <memory>
|
||||
#include <utility>
|
||||
|
||||
#include "rtc_base/event.h"
|
||||
#include "rtc_base/task_queue.h"
|
||||
#include "rtc_base/task_utils/repeating_task.h"
|
||||
#include "test/gmock.h"
|
||||
#include "test/gtest.h"
|
||||
|
||||
// NOTE: Since these tests rely on real time behavior, they will be flaky
|
||||
// if run on heavily loaded systems.
|
||||
namespace webrtc {
|
||||
namespace {
|
||||
using ::testing::AtLeast;
|
||||
using ::testing::Invoke;
|
||||
using ::testing::MockFunction;
|
||||
using ::testing::NiceMock;
|
||||
using ::testing::Return;
|
||||
constexpr Timestamp kStartTime = Timestamp::Seconds<1000>();
|
||||
|
||||
class FakeAlarm : public ControlledAlarmClock {
|
||||
public:
|
||||
explicit FakeAlarm(Timestamp start_time);
|
||||
|
||||
Clock* GetClock() override;
|
||||
bool ScheduleAlarmAt(Timestamp deadline) override;
|
||||
void SetCallback(std::function<void()> callback) override;
|
||||
void Sleep(TimeDelta duration) override;
|
||||
|
||||
private:
|
||||
SimulatedClock clock_;
|
||||
Timestamp deadline_;
|
||||
std::function<void()> callback_;
|
||||
};
|
||||
|
||||
FakeAlarm::FakeAlarm(Timestamp start_time)
|
||||
: clock_(start_time),
|
||||
deadline_(Timestamp::PlusInfinity()),
|
||||
callback_([] {}) {}
|
||||
|
||||
Clock* FakeAlarm::GetClock() {
|
||||
return &clock_;
|
||||
}
|
||||
|
||||
bool FakeAlarm::ScheduleAlarmAt(Timestamp deadline) {
|
||||
if (deadline < deadline_) {
|
||||
deadline_ = deadline;
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
void FakeAlarm::SetCallback(std::function<void()> callback) {
|
||||
callback_ = callback;
|
||||
}
|
||||
|
||||
void FakeAlarm::Sleep(TimeDelta duration) {
|
||||
Timestamp end_time = clock_.CurrentTime() + duration;
|
||||
|
||||
while (deadline_ <= end_time) {
|
||||
clock_.AdvanceTime(deadline_ - clock_.CurrentTime());
|
||||
deadline_ = Timestamp::PlusInfinity();
|
||||
callback_();
|
||||
}
|
||||
|
||||
clock_.AdvanceTime(end_time - clock_.CurrentTime());
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
TEST(ExternalTimeControllerTest, TaskIsStoppedOnStop) {
|
||||
const TimeDelta kShortInterval = TimeDelta::ms(5);
|
||||
const TimeDelta kLongInterval = TimeDelta::ms(20);
|
||||
const int kShortIntervalCount = 4;
|
||||
const int kMargin = 1;
|
||||
FakeAlarm alarm(kStartTime);
|
||||
ExternalTimeController time_simulation(&alarm);
|
||||
rtc::TaskQueue task_queue(
|
||||
time_simulation.GetTaskQueueFactory()->CreateTaskQueue(
|
||||
"TestQueue", TaskQueueFactory::Priority::NORMAL));
|
||||
std::atomic_int counter(0);
|
||||
auto handle = RepeatingTaskHandle::Start(task_queue.Get(), [&] {
|
||||
if (++counter >= kShortIntervalCount)
|
||||
return kLongInterval;
|
||||
return kShortInterval;
|
||||
});
|
||||
// Sleep long enough to go through the initial phase.
|
||||
time_simulation.Sleep(kShortInterval * (kShortIntervalCount + kMargin));
|
||||
EXPECT_EQ(counter.load(), kShortIntervalCount);
|
||||
|
||||
task_queue.PostTask(
|
||||
[handle = std::move(handle)]() mutable { handle.Stop(); });
|
||||
|
||||
// Sleep long enough that the task would run at least once more if not
|
||||
// stopped.
|
||||
time_simulation.Sleep(kLongInterval * 2);
|
||||
EXPECT_EQ(counter.load(), kShortIntervalCount);
|
||||
}
|
||||
|
||||
TEST(ExternalTimeControllerTest, TaskCanStopItself) {
|
||||
std::atomic_int counter(0);
|
||||
FakeAlarm alarm(kStartTime);
|
||||
ExternalTimeController time_simulation(&alarm);
|
||||
rtc::TaskQueue task_queue(
|
||||
time_simulation.GetTaskQueueFactory()->CreateTaskQueue(
|
||||
"TestQueue", TaskQueueFactory::Priority::NORMAL));
|
||||
|
||||
RepeatingTaskHandle handle;
|
||||
task_queue.PostTask([&] {
|
||||
handle = RepeatingTaskHandle::Start(task_queue.Get(), [&] {
|
||||
++counter;
|
||||
handle.Stop();
|
||||
return TimeDelta::ms(2);
|
||||
});
|
||||
});
|
||||
time_simulation.Sleep(TimeDelta::ms(10));
|
||||
EXPECT_EQ(counter.load(), 1);
|
||||
}
|
||||
|
||||
TEST(ExternalTimeControllerTest, YieldForTask) {
|
||||
FakeAlarm alarm(kStartTime);
|
||||
ExternalTimeController time_simulation(&alarm);
|
||||
|
||||
rtc::TaskQueue task_queue(
|
||||
time_simulation.GetTaskQueueFactory()->CreateTaskQueue(
|
||||
"TestQueue", TaskQueueFactory::Priority::NORMAL));
|
||||
|
||||
time_simulation.InvokeWithControlledYield([&] {
|
||||
rtc::Event event;
|
||||
task_queue.PostTask([&] { event.Set(); });
|
||||
EXPECT_TRUE(event.Wait(200));
|
||||
});
|
||||
}
|
||||
|
||||
TEST(ExternalTimeControllerTest, TasksYieldToEachOther) {
|
||||
FakeAlarm alarm(kStartTime);
|
||||
ExternalTimeController time_simulation(&alarm);
|
||||
|
||||
rtc::TaskQueue task_queue(
|
||||
time_simulation.GetTaskQueueFactory()->CreateTaskQueue(
|
||||
"TestQueue", TaskQueueFactory::Priority::NORMAL));
|
||||
rtc::TaskQueue other_queue(
|
||||
time_simulation.GetTaskQueueFactory()->CreateTaskQueue(
|
||||
"OtherQueue", TaskQueueFactory::Priority::NORMAL));
|
||||
|
||||
task_queue.PostTask([&] {
|
||||
rtc::Event event;
|
||||
other_queue.PostTask([&] { event.Set(); });
|
||||
EXPECT_TRUE(event.Wait(200));
|
||||
});
|
||||
|
||||
time_simulation.Sleep(TimeDelta::ms(300));
|
||||
}
|
||||
|
||||
TEST(ExternalTimeControllerTest, CurrentTaskQueue) {
|
||||
FakeAlarm alarm(kStartTime);
|
||||
ExternalTimeController time_simulation(&alarm);
|
||||
|
||||
rtc::TaskQueue task_queue(
|
||||
time_simulation.GetTaskQueueFactory()->CreateTaskQueue(
|
||||
"TestQueue", TaskQueueFactory::Priority::NORMAL));
|
||||
|
||||
task_queue.PostTask([&] { EXPECT_TRUE(task_queue.IsCurrent()); });
|
||||
|
||||
time_simulation.Sleep(TimeDelta::ms(10));
|
||||
}
|
||||
|
||||
} // namespace webrtc
|
||||
Reference in New Issue
Block a user