summaryrefslogtreecommitdiffstats
path: root/src/rocksdb/util/timer_test.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/rocksdb/util/timer_test.cc')
-rw-r--r--src/rocksdb/util/timer_test.cc402
1 files changed, 402 insertions, 0 deletions
diff --git a/src/rocksdb/util/timer_test.cc b/src/rocksdb/util/timer_test.cc
new file mode 100644
index 000000000..0ebfa9f3d
--- /dev/null
+++ b/src/rocksdb/util/timer_test.cc
@@ -0,0 +1,402 @@
+// Copyright (c) 2011-present, Facebook, Inc. All rights reserved.
+// This source code is licensed under both the GPLv2 (found in the
+// COPYING file in the root directory) and Apache 2.0 License
+// (found in the LICENSE.Apache file in the root directory).
+
+#include "util/timer.h"
+
+#include "db/db_test_util.h"
+#include "rocksdb/file_system.h"
+#include "test_util/mock_time_env.h"
+
+namespace ROCKSDB_NAMESPACE {
+
+class TimerTest : public testing::Test {
+ public:
+ TimerTest()
+ : mock_clock_(std::make_shared<MockSystemClock>(SystemClock::Default())) {
+ }
+
+ protected:
+ std::shared_ptr<MockSystemClock> mock_clock_;
+
+ void SetUp() override { mock_clock_->InstallTimedWaitFixCallback(); }
+
+ const int kUsPerSec = 1000000;
+};
+
+TEST_F(TimerTest, SingleScheduleOnce) {
+ const int kInitDelayUs = 1 * kUsPerSec;
+ Timer timer(mock_clock_.get());
+
+ int count = 0;
+ timer.Add([&] { count++; }, "fn_sch_test", kInitDelayUs, 0);
+
+ ASSERT_TRUE(timer.Start());
+
+ ASSERT_EQ(0, count);
+ // Wait for execution to finish
+ timer.TEST_WaitForRun(
+ [&] { mock_clock_->SleepForMicroseconds(kInitDelayUs); });
+ ASSERT_EQ(1, count);
+
+ ASSERT_TRUE(timer.Shutdown());
+}
+
+TEST_F(TimerTest, MultipleScheduleOnce) {
+ const int kInitDelay1Us = 1 * kUsPerSec;
+ const int kInitDelay2Us = 3 * kUsPerSec;
+ Timer timer(mock_clock_.get());
+
+ int count1 = 0;
+ timer.Add([&] { count1++; }, "fn_sch_test1", kInitDelay1Us, 0);
+
+ int count2 = 0;
+ timer.Add([&] { count2++; }, "fn_sch_test2", kInitDelay2Us, 0);
+
+ ASSERT_TRUE(timer.Start());
+ ASSERT_EQ(0, count1);
+ ASSERT_EQ(0, count2);
+
+ timer.TEST_WaitForRun(
+ [&] { mock_clock_->SleepForMicroseconds(kInitDelay1Us); });
+
+ ASSERT_EQ(1, count1);
+ ASSERT_EQ(0, count2);
+
+ timer.TEST_WaitForRun([&] {
+ mock_clock_->SleepForMicroseconds(kInitDelay2Us - kInitDelay1Us);
+ });
+
+ ASSERT_EQ(1, count1);
+ ASSERT_EQ(1, count2);
+
+ ASSERT_TRUE(timer.Shutdown());
+}
+
+TEST_F(TimerTest, SingleScheduleRepeatedly) {
+ const int kIterations = 5;
+ const int kInitDelayUs = 1 * kUsPerSec;
+ const int kRepeatUs = 1 * kUsPerSec;
+
+ Timer timer(mock_clock_.get());
+ int count = 0;
+ timer.Add([&] { count++; }, "fn_sch_test", kInitDelayUs, kRepeatUs);
+
+ ASSERT_TRUE(timer.Start());
+ ASSERT_EQ(0, count);
+
+ timer.TEST_WaitForRun(
+ [&] { mock_clock_->SleepForMicroseconds(kInitDelayUs); });
+
+ ASSERT_EQ(1, count);
+
+ // Wait for execution to finish
+ for (int i = 1; i < kIterations; i++) {
+ timer.TEST_WaitForRun(
+ [&] { mock_clock_->SleepForMicroseconds(kRepeatUs); });
+ }
+ ASSERT_EQ(kIterations, count);
+
+ ASSERT_TRUE(timer.Shutdown());
+}
+
+TEST_F(TimerTest, MultipleScheduleRepeatedly) {
+ const int kIterations = 5;
+ const int kInitDelay1Us = 0 * kUsPerSec;
+ const int kInitDelay2Us = 1 * kUsPerSec;
+ const int kInitDelay3Us = 0 * kUsPerSec;
+ const int kRepeatUs = 2 * kUsPerSec;
+ const int kLargeRepeatUs = 100 * kUsPerSec;
+
+ Timer timer(mock_clock_.get());
+
+ int count1 = 0;
+ timer.Add([&] { count1++; }, "fn_sch_test1", kInitDelay1Us, kRepeatUs);
+
+ int count2 = 0;
+ timer.Add([&] { count2++; }, "fn_sch_test2", kInitDelay2Us, kRepeatUs);
+
+ // Add a function with relatively large repeat interval
+ int count3 = 0;
+ timer.Add([&] { count3++; }, "fn_sch_test3", kInitDelay3Us, kLargeRepeatUs);
+
+ ASSERT_TRUE(timer.Start());
+
+ ASSERT_EQ(0, count2);
+ // Wait for execution to finish
+ for (int i = 1; i < kIterations * (kRepeatUs / kUsPerSec); i++) {
+ timer.TEST_WaitForRun(
+ [&] { mock_clock_->SleepForMicroseconds(1 * kUsPerSec); });
+ ASSERT_EQ((i + 2) / (kRepeatUs / kUsPerSec), count1);
+ ASSERT_EQ((i + 1) / (kRepeatUs / kUsPerSec), count2);
+
+ // large interval function should only run once (the first one).
+ ASSERT_EQ(1, count3);
+ }
+
+ timer.Cancel("fn_sch_test1");
+
+ // Wait for execution to finish
+ timer.TEST_WaitForRun(
+ [&] { mock_clock_->SleepForMicroseconds(1 * kUsPerSec); });
+ ASSERT_EQ(kIterations, count1);
+ ASSERT_EQ(kIterations, count2);
+ ASSERT_EQ(1, count3);
+
+ timer.Cancel("fn_sch_test2");
+
+ ASSERT_EQ(kIterations, count1);
+ ASSERT_EQ(kIterations, count2);
+
+ // execute the long interval one
+ timer.TEST_WaitForRun([&] {
+ mock_clock_->SleepForMicroseconds(
+ kLargeRepeatUs - static_cast<int>(mock_clock_->NowMicros()));
+ });
+ ASSERT_EQ(2, count3);
+
+ ASSERT_TRUE(timer.Shutdown());
+}
+
+TEST_F(TimerTest, AddAfterStartTest) {
+ const int kIterations = 5;
+ const int kInitDelayUs = 1 * kUsPerSec;
+ const int kRepeatUs = 1 * kUsPerSec;
+
+ // wait timer to run and then add a new job
+ SyncPoint::GetInstance()->LoadDependency(
+ {{"Timer::Run::Waiting", "TimerTest:AddAfterStartTest:1"}});
+ SyncPoint::GetInstance()->EnableProcessing();
+
+ Timer timer(mock_clock_.get());
+
+ ASSERT_TRUE(timer.Start());
+
+ TEST_SYNC_POINT("TimerTest:AddAfterStartTest:1");
+ int count = 0;
+ timer.Add([&] { count++; }, "fn_sch_test", kInitDelayUs, kRepeatUs);
+ ASSERT_EQ(0, count);
+ // Wait for execution to finish
+ timer.TEST_WaitForRun(
+ [&] { mock_clock_->SleepForMicroseconds(kInitDelayUs); });
+ ASSERT_EQ(1, count);
+
+ for (int i = 1; i < kIterations; i++) {
+ timer.TEST_WaitForRun(
+ [&] { mock_clock_->SleepForMicroseconds(kRepeatUs); });
+ }
+ ASSERT_EQ(kIterations, count);
+
+ ASSERT_TRUE(timer.Shutdown());
+}
+
+TEST_F(TimerTest, CancelRunningTask) {
+ static constexpr char kTestFuncName[] = "test_func";
+ const int kRepeatUs = 1 * kUsPerSec;
+ Timer timer(mock_clock_.get());
+ ASSERT_TRUE(timer.Start());
+ int* value = new int;
+ *value = 0;
+ SyncPoint::GetInstance()->DisableProcessing();
+ SyncPoint::GetInstance()->LoadDependency({
+ {"TimerTest::CancelRunningTask:test_func:0",
+ "TimerTest::CancelRunningTask:BeforeCancel"},
+ {"Timer::WaitForTaskCompleteIfNecessary:TaskExecuting",
+ "TimerTest::CancelRunningTask:test_func:1"},
+ });
+ SyncPoint::GetInstance()->EnableProcessing();
+ timer.Add(
+ [&]() {
+ *value = 1;
+ TEST_SYNC_POINT("TimerTest::CancelRunningTask:test_func:0");
+ TEST_SYNC_POINT("TimerTest::CancelRunningTask:test_func:1");
+ },
+ kTestFuncName, 0, kRepeatUs);
+ port::Thread control_thr([&]() {
+ TEST_SYNC_POINT("TimerTest::CancelRunningTask:BeforeCancel");
+ timer.Cancel(kTestFuncName);
+ // Verify that *value has been set to 1.
+ ASSERT_EQ(1, *value);
+ delete value;
+ value = nullptr;
+ });
+ mock_clock_->SleepForMicroseconds(kRepeatUs);
+ control_thr.join();
+ ASSERT_TRUE(timer.Shutdown());
+}
+
+TEST_F(TimerTest, ShutdownRunningTask) {
+ const int kRepeatUs = 1 * kUsPerSec;
+ constexpr char kTestFunc1Name[] = "test_func1";
+ constexpr char kTestFunc2Name[] = "test_func2";
+ Timer timer(mock_clock_.get());
+
+ SyncPoint::GetInstance()->DisableProcessing();
+ SyncPoint::GetInstance()->LoadDependency({
+ {"TimerTest::ShutdownRunningTest:test_func:0",
+ "TimerTest::ShutdownRunningTest:BeforeShutdown"},
+ {"Timer::WaitForTaskCompleteIfNecessary:TaskExecuting",
+ "TimerTest::ShutdownRunningTest:test_func:1"},
+ });
+ SyncPoint::GetInstance()->EnableProcessing();
+
+ ASSERT_TRUE(timer.Start());
+
+ int* value = new int;
+ *value = 0;
+ timer.Add(
+ [&]() {
+ TEST_SYNC_POINT("TimerTest::ShutdownRunningTest:test_func:0");
+ *value = 1;
+ TEST_SYNC_POINT("TimerTest::ShutdownRunningTest:test_func:1");
+ },
+ kTestFunc1Name, 0, kRepeatUs);
+
+ timer.Add([&]() { ++(*value); }, kTestFunc2Name, 0, kRepeatUs);
+
+ port::Thread control_thr([&]() {
+ TEST_SYNC_POINT("TimerTest::ShutdownRunningTest:BeforeShutdown");
+ timer.Shutdown();
+ });
+ mock_clock_->SleepForMicroseconds(kRepeatUs);
+ control_thr.join();
+ delete value;
+}
+
+TEST_F(TimerTest, AddSameFuncName) {
+ const int kInitDelayUs = 1 * kUsPerSec;
+ const int kRepeat1Us = 5 * kUsPerSec;
+ const int kRepeat2Us = 4 * kUsPerSec;
+
+ Timer timer(mock_clock_.get());
+ ASSERT_TRUE(timer.Start());
+
+ int func_counter1 = 0;
+ ASSERT_TRUE(timer.Add([&] { func_counter1++; }, "duplicated_func",
+ kInitDelayUs, kRepeat1Us));
+
+ int func2_counter = 0;
+ ASSERT_TRUE(
+ timer.Add([&] { func2_counter++; }, "func2", kInitDelayUs, kRepeat2Us));
+
+ // New function with the same name should fail to add
+ int func_counter2 = 0;
+ ASSERT_FALSE(timer.Add([&] { func_counter2++; }, "duplicated_func",
+ kInitDelayUs, kRepeat1Us));
+
+ ASSERT_EQ(0, func_counter1);
+ ASSERT_EQ(0, func2_counter);
+
+ timer.TEST_WaitForRun(
+ [&] { mock_clock_->SleepForMicroseconds(kInitDelayUs); });
+
+ ASSERT_EQ(1, func_counter1);
+ ASSERT_EQ(1, func2_counter);
+
+ timer.TEST_WaitForRun([&] { mock_clock_->SleepForMicroseconds(kRepeat1Us); });
+
+ ASSERT_EQ(2, func_counter1);
+ ASSERT_EQ(2, func2_counter);
+ ASSERT_EQ(0, func_counter2);
+
+ ASSERT_TRUE(timer.Shutdown());
+}
+
+TEST_F(TimerTest, RepeatIntervalWithFuncRunningTime) {
+ const int kInitDelayUs = 1 * kUsPerSec;
+ const int kRepeatUs = 5 * kUsPerSec;
+ const int kFuncRunningTimeUs = 1 * kUsPerSec;
+
+ Timer timer(mock_clock_.get());
+ ASSERT_TRUE(timer.Start());
+
+ int func_counter = 0;
+ timer.Add(
+ [&] {
+ mock_clock_->SleepForMicroseconds(kFuncRunningTimeUs);
+ func_counter++;
+ },
+ "func", kInitDelayUs, kRepeatUs);
+
+ ASSERT_EQ(0, func_counter);
+ timer.TEST_WaitForRun(
+ [&] { mock_clock_->SleepForMicroseconds(kInitDelayUs); });
+ ASSERT_EQ(1, func_counter);
+ ASSERT_EQ(kInitDelayUs + kFuncRunningTimeUs, mock_clock_->NowMicros());
+
+ // After repeat interval time, the function is not executed, as running
+ // the function takes some time (`kFuncRunningTimeSec`). The repeat interval
+ // is the time between ending time of the last call and starting time of the
+ // next call.
+ uint64_t next_abs_interval_time_us = kInitDelayUs + kRepeatUs;
+ timer.TEST_WaitForRun([&] {
+ mock_clock_->SetCurrentTime(next_abs_interval_time_us / kUsPerSec);
+ });
+ ASSERT_EQ(1, func_counter);
+
+ // After the function running time, it's executed again
+ timer.TEST_WaitForRun(
+ [&] { mock_clock_->SleepForMicroseconds(kFuncRunningTimeUs); });
+ ASSERT_EQ(2, func_counter);
+
+ ASSERT_TRUE(timer.Shutdown());
+}
+
+TEST_F(TimerTest, DestroyRunningTimer) {
+ const int kInitDelayUs = 1 * kUsPerSec;
+ const int kRepeatUs = 1 * kUsPerSec;
+
+ auto timer_ptr = new Timer(mock_clock_.get());
+
+ int count = 0;
+ timer_ptr->Add([&] { count++; }, "fn_sch_test", kInitDelayUs, kRepeatUs);
+ ASSERT_TRUE(timer_ptr->Start());
+
+ timer_ptr->TEST_WaitForRun(
+ [&] { mock_clock_->SleepForMicroseconds(kInitDelayUs); });
+
+ // delete a running timer should not cause any exception
+ delete timer_ptr;
+}
+
+TEST_F(TimerTest, DestroyTimerWithRunningFunc) {
+ const int kRepeatUs = 1 * kUsPerSec;
+ auto timer_ptr = new Timer(mock_clock_.get());
+
+ SyncPoint::GetInstance()->DisableProcessing();
+ SyncPoint::GetInstance()->LoadDependency({
+ {"TimerTest::DestroyTimerWithRunningFunc:test_func:0",
+ "TimerTest::DestroyTimerWithRunningFunc:BeforeDelete"},
+ {"Timer::WaitForTaskCompleteIfNecessary:TaskExecuting",
+ "TimerTest::DestroyTimerWithRunningFunc:test_func:1"},
+ });
+ SyncPoint::GetInstance()->EnableProcessing();
+
+ ASSERT_TRUE(timer_ptr->Start());
+
+ int count = 0;
+ timer_ptr->Add(
+ [&]() {
+ TEST_SYNC_POINT("TimerTest::DestroyTimerWithRunningFunc:test_func:0");
+ count++;
+ TEST_SYNC_POINT("TimerTest::DestroyTimerWithRunningFunc:test_func:1");
+ },
+ "fn_running_test", 0, kRepeatUs);
+
+ port::Thread control_thr([&] {
+ TEST_SYNC_POINT("TimerTest::DestroyTimerWithRunningFunc:BeforeDelete");
+ delete timer_ptr;
+ });
+ mock_clock_->SleepForMicroseconds(kRepeatUs);
+ control_thr.join();
+}
+
+} // namespace ROCKSDB_NAMESPACE
+
+int main(int argc, char** argv) {
+ ROCKSDB_NAMESPACE::port::InstallStackTraceHandler();
+ ::testing::InitGoogleTest(&argc, argv);
+
+ return RUN_ALL_TESTS();
+}