/* * Copyright (C) 2021 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "perfetto/ext/base/periodic_task.h" #include "perfetto/ext/base/file_utils.h" #include "src/base/test/test_task_runner.h" #include "test/gtest_and_gmock.h" #if PERFETTO_BUILDFLAG(PERFETTO_OS_LINUX) || \ PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID) #include #endif namespace perfetto { namespace base { namespace { TEST(PeriodicTaskTest, PostDelayedTaskMode) { TestTaskRunner task_runner; PeriodicTask pt(&task_runner); uint32_t num_callbacks = 0; auto quit_closure = task_runner.CreateCheckpoint("all_timers_done"); PeriodicTask::Args args; args.task = [&] { if (++num_callbacks == 3) quit_closure(); }; args.period_ms = 1; args.start_first_task_immediately = true; pt.Start(std::move(args)); EXPECT_EQ(num_callbacks, 1u); task_runner.RunUntilCheckpoint("all_timers_done"); EXPECT_EQ(num_callbacks, 3u); } // Call Reset() from a callback, ensure no further calls are made. TEST(PeriodicTaskTest, ResetFromCallback) { TestTaskRunner task_runner; PeriodicTask pt(&task_runner); uint32_t num_callbacks = 0; PeriodicTask::Args args; auto quit_closure = task_runner.CreateCheckpoint("quit_closure"); args.task = [&] { ++num_callbacks; pt.Reset(); task_runner.PostDelayedTask(quit_closure, 5); }; args.period_ms = 1; pt.Start(std::move(args)); EXPECT_EQ(num_callbacks, 0u); // No immediate execution. task_runner.RunUntilCheckpoint("quit_closure"); EXPECT_EQ(num_callbacks, 1u); } // Invalidates the timerfd, by replacing it with /dev/null, in the middle of // the periodic ticks. That causes the next read() to fail and fall back on // PostDelayedTask(). // On Mac and other systems where timerfd is not supported this will fall back // on PostDelayedTask() immediately (and work). TEST(PeriodicTaskTest, FallbackIfTimerfdFails) { TestTaskRunner task_runner; PeriodicTask pt(&task_runner); uint32_t num_callbacks = 0; auto quit_closure = task_runner.CreateCheckpoint("all_timers_done"); PeriodicTask::Args args; args.task = [&] { ++num_callbacks; #if PERFETTO_BUILDFLAG(PERFETTO_OS_LINUX) || \ PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID) if (num_callbacks == 3 && pt.timer_fd_for_testing() > 0) { ScopedFile dev_null = OpenFile("/dev/null", O_RDONLY); dup2(*dev_null, pt.timer_fd_for_testing()); } #else EXPECT_FALSE(base::ScopedPlatformHandle::ValidityChecker::IsValid( pt.timer_fd_for_testing())); #endif if (num_callbacks == 6) quit_closure(); }; args.period_ms = 1; args.use_suspend_aware_timer = true; pt.Start(std::move(args)); task_runner.RunUntilCheckpoint("all_timers_done"); EXPECT_EQ(num_callbacks, 6u); } TEST(PeriodicTaskTest, DestroyedFromCallback) { TestTaskRunner task_runner; std::unique_ptr pt(new PeriodicTask(&task_runner)); uint32_t num_callbacks = 0; PeriodicTask::Args args; auto quit_closure = task_runner.CreateCheckpoint("quit_closure"); args.task = [&] { ++num_callbacks; pt.reset(); task_runner.PostDelayedTask(quit_closure, 5); }; args.period_ms = 1; args.use_suspend_aware_timer = true; pt->Start(std::move(args)); task_runner.RunUntilCheckpoint("quit_closure"); EXPECT_EQ(num_callbacks, 1u); EXPECT_FALSE(pt); } TEST(PeriodicTaskTest, DestroyedFromAnotherTask) { TestTaskRunner task_runner; std::unique_ptr pt(new PeriodicTask(&task_runner)); uint32_t num_callbacks = 0; PeriodicTask::Args args; auto quit_closure = task_runner.CreateCheckpoint("quit_closure"); args.task = [&] { if (++num_callbacks == 2) { task_runner.PostTask([&] { pt.reset(); task_runner.PostDelayedTask(quit_closure, 5); }); } }; args.period_ms = 1; args.use_suspend_aware_timer = true; pt->Start(std::move(args)); task_runner.RunUntilCheckpoint("quit_closure"); EXPECT_EQ(num_callbacks, 2u); EXPECT_FALSE(pt); } // Checks the generation logic. TEST(PeriodicTaskTest, RestartWhileRunning) { TestTaskRunner task_runner; PeriodicTask pt(&task_runner); uint32_t num_callbacks_a = 0; uint32_t num_callbacks_b = 0; auto quit_closure = task_runner.CreateCheckpoint("quit_closure"); auto reuse = [&] { PeriodicTask::Args args; args.period_ms = 1; args.task = [&] { if (++num_callbacks_b == 3) quit_closure(); }; pt.Start(std::move(args)); }; PeriodicTask::Args args; args.task = [&] { if (++num_callbacks_a == 2) task_runner.PostTask(reuse); }; args.period_ms = 1; args.use_suspend_aware_timer = true; pt.Start(std::move(args)); task_runner.RunUntilCheckpoint("quit_closure"); EXPECT_EQ(num_callbacks_a, 2u); EXPECT_EQ(num_callbacks_b, 3u); } TEST(PeriodicTaskTest, ImmediateExecution) { TestTaskRunner task_runner; PeriodicTask pt(&task_runner); uint32_t num_callbacks = 0; PeriodicTask::Args args; args.task = [&] { ++num_callbacks; }; args.period_ms = 1; pt.Start(args); EXPECT_EQ(num_callbacks, 0u); // No immediate execution. args.start_first_task_immediately = true; pt.Start(args); EXPECT_EQ(num_callbacks, 1u); } } // namespace } // namespace base } // namespace perfetto