xref: /aosp_15_r20/external/perfetto/src/base/periodic_task_unittest.cc (revision 6dbdd20afdafa5e3ca9b8809fa73465d530080dc)
1*6dbdd20aSAndroid Build Coastguard Worker /*
2*6dbdd20aSAndroid Build Coastguard Worker  * Copyright (C) 2021 The Android Open Source Project
3*6dbdd20aSAndroid Build Coastguard Worker  *
4*6dbdd20aSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*6dbdd20aSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*6dbdd20aSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*6dbdd20aSAndroid Build Coastguard Worker  *
8*6dbdd20aSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*6dbdd20aSAndroid Build Coastguard Worker  *
10*6dbdd20aSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*6dbdd20aSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*6dbdd20aSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*6dbdd20aSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*6dbdd20aSAndroid Build Coastguard Worker  * limitations under the License.
15*6dbdd20aSAndroid Build Coastguard Worker  */
16*6dbdd20aSAndroid Build Coastguard Worker 
17*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/periodic_task.h"
18*6dbdd20aSAndroid Build Coastguard Worker 
19*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/file_utils.h"
20*6dbdd20aSAndroid Build Coastguard Worker #include "src/base/test/test_task_runner.h"
21*6dbdd20aSAndroid Build Coastguard Worker #include "test/gtest_and_gmock.h"
22*6dbdd20aSAndroid Build Coastguard Worker 
23*6dbdd20aSAndroid Build Coastguard Worker #if PERFETTO_BUILDFLAG(PERFETTO_OS_LINUX) || \
24*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID)
25*6dbdd20aSAndroid Build Coastguard Worker #include <unistd.h>
26*6dbdd20aSAndroid Build Coastguard Worker #endif
27*6dbdd20aSAndroid Build Coastguard Worker 
28*6dbdd20aSAndroid Build Coastguard Worker #include <chrono>
29*6dbdd20aSAndroid Build Coastguard Worker #include <thread>
30*6dbdd20aSAndroid Build Coastguard Worker 
31*6dbdd20aSAndroid Build Coastguard Worker namespace perfetto {
32*6dbdd20aSAndroid Build Coastguard Worker namespace base {
33*6dbdd20aSAndroid Build Coastguard Worker 
34*6dbdd20aSAndroid Build Coastguard Worker namespace {
35*6dbdd20aSAndroid Build Coastguard Worker 
TEST(PeriodicTaskTest,PostDelayedTaskMode)36*6dbdd20aSAndroid Build Coastguard Worker TEST(PeriodicTaskTest, PostDelayedTaskMode) {
37*6dbdd20aSAndroid Build Coastguard Worker   TestTaskRunner task_runner;
38*6dbdd20aSAndroid Build Coastguard Worker   PeriodicTask pt(&task_runner);
39*6dbdd20aSAndroid Build Coastguard Worker   uint32_t num_callbacks = 0;
40*6dbdd20aSAndroid Build Coastguard Worker   auto quit_closure = task_runner.CreateCheckpoint("all_timers_done");
41*6dbdd20aSAndroid Build Coastguard Worker 
42*6dbdd20aSAndroid Build Coastguard Worker   PeriodicTask::Args args;
43*6dbdd20aSAndroid Build Coastguard Worker   args.task = [&] {
44*6dbdd20aSAndroid Build Coastguard Worker     if (++num_callbacks == 3)
45*6dbdd20aSAndroid Build Coastguard Worker       quit_closure();
46*6dbdd20aSAndroid Build Coastguard Worker   };
47*6dbdd20aSAndroid Build Coastguard Worker   args.period_ms = 1;
48*6dbdd20aSAndroid Build Coastguard Worker   args.start_first_task_immediately = true;
49*6dbdd20aSAndroid Build Coastguard Worker   pt.Start(std::move(args));
50*6dbdd20aSAndroid Build Coastguard Worker   EXPECT_EQ(num_callbacks, 1u);
51*6dbdd20aSAndroid Build Coastguard Worker   task_runner.RunUntilCheckpoint("all_timers_done");
52*6dbdd20aSAndroid Build Coastguard Worker   EXPECT_EQ(num_callbacks, 3u);
53*6dbdd20aSAndroid Build Coastguard Worker }
54*6dbdd20aSAndroid Build Coastguard Worker 
TEST(PeriodicTaskTest,OneShot)55*6dbdd20aSAndroid Build Coastguard Worker TEST(PeriodicTaskTest, OneShot) {
56*6dbdd20aSAndroid Build Coastguard Worker   TestTaskRunner task_runner;
57*6dbdd20aSAndroid Build Coastguard Worker   PeriodicTask pt(&task_runner);
58*6dbdd20aSAndroid Build Coastguard Worker   uint32_t num_callbacks = 0;
59*6dbdd20aSAndroid Build Coastguard Worker   auto quit_closure = task_runner.CreateCheckpoint("one_shot_done");
60*6dbdd20aSAndroid Build Coastguard Worker 
61*6dbdd20aSAndroid Build Coastguard Worker   PeriodicTask::Args args;
62*6dbdd20aSAndroid Build Coastguard Worker   args.use_suspend_aware_timer = true;
63*6dbdd20aSAndroid Build Coastguard Worker   args.one_shot = true;
64*6dbdd20aSAndroid Build Coastguard Worker   args.period_ms = 1;
65*6dbdd20aSAndroid Build Coastguard Worker   args.task = [&] {
66*6dbdd20aSAndroid Build Coastguard Worker     ASSERT_EQ(++num_callbacks, 1u);
67*6dbdd20aSAndroid Build Coastguard Worker     quit_closure();
68*6dbdd20aSAndroid Build Coastguard Worker   };
69*6dbdd20aSAndroid Build Coastguard Worker   pt.Start(std::move(args));
70*6dbdd20aSAndroid Build Coastguard Worker   std::this_thread::sleep_for(std::chrono::milliseconds(3));
71*6dbdd20aSAndroid Build Coastguard Worker   task_runner.RunUntilCheckpoint("one_shot_done");
72*6dbdd20aSAndroid Build Coastguard Worker   EXPECT_EQ(num_callbacks, 1u);
73*6dbdd20aSAndroid Build Coastguard Worker }
74*6dbdd20aSAndroid Build Coastguard Worker 
75*6dbdd20aSAndroid Build Coastguard Worker // Call Reset() from a callback, ensure no further calls are made.
TEST(PeriodicTaskTest,ResetFromCallback)76*6dbdd20aSAndroid Build Coastguard Worker TEST(PeriodicTaskTest, ResetFromCallback) {
77*6dbdd20aSAndroid Build Coastguard Worker   TestTaskRunner task_runner;
78*6dbdd20aSAndroid Build Coastguard Worker   PeriodicTask pt(&task_runner);
79*6dbdd20aSAndroid Build Coastguard Worker   uint32_t num_callbacks = 0;
80*6dbdd20aSAndroid Build Coastguard Worker   PeriodicTask::Args args;
81*6dbdd20aSAndroid Build Coastguard Worker   auto quit_closure = task_runner.CreateCheckpoint("quit_closure");
82*6dbdd20aSAndroid Build Coastguard Worker   args.task = [&] {
83*6dbdd20aSAndroid Build Coastguard Worker     ++num_callbacks;
84*6dbdd20aSAndroid Build Coastguard Worker     pt.Reset();
85*6dbdd20aSAndroid Build Coastguard Worker     task_runner.PostDelayedTask(quit_closure, 5);
86*6dbdd20aSAndroid Build Coastguard Worker   };
87*6dbdd20aSAndroid Build Coastguard Worker   args.period_ms = 1;
88*6dbdd20aSAndroid Build Coastguard Worker   pt.Start(std::move(args));
89*6dbdd20aSAndroid Build Coastguard Worker   EXPECT_EQ(num_callbacks, 0u);  // No immediate execution.
90*6dbdd20aSAndroid Build Coastguard Worker 
91*6dbdd20aSAndroid Build Coastguard Worker   task_runner.RunUntilCheckpoint("quit_closure");
92*6dbdd20aSAndroid Build Coastguard Worker   EXPECT_EQ(num_callbacks, 1u);
93*6dbdd20aSAndroid Build Coastguard Worker }
94*6dbdd20aSAndroid Build Coastguard Worker 
95*6dbdd20aSAndroid Build Coastguard Worker // Invalidates the timerfd, by replacing it with /dev/null, in the middle of
96*6dbdd20aSAndroid Build Coastguard Worker // the periodic ticks. That causes the next read() to fail and fall back on
97*6dbdd20aSAndroid Build Coastguard Worker // PostDelayedTask().
98*6dbdd20aSAndroid Build Coastguard Worker // On Mac and other systems where timerfd is not supported this will fall back
99*6dbdd20aSAndroid Build Coastguard Worker // on PostDelayedTask() immediately (and work).
TEST(PeriodicTaskTest,FallbackIfTimerfdFails)100*6dbdd20aSAndroid Build Coastguard Worker TEST(PeriodicTaskTest, FallbackIfTimerfdFails) {
101*6dbdd20aSAndroid Build Coastguard Worker   TestTaskRunner task_runner;
102*6dbdd20aSAndroid Build Coastguard Worker   PeriodicTask pt(&task_runner);
103*6dbdd20aSAndroid Build Coastguard Worker   uint32_t num_callbacks = 0;
104*6dbdd20aSAndroid Build Coastguard Worker   auto quit_closure = task_runner.CreateCheckpoint("all_timers_done");
105*6dbdd20aSAndroid Build Coastguard Worker 
106*6dbdd20aSAndroid Build Coastguard Worker   PeriodicTask::Args args;
107*6dbdd20aSAndroid Build Coastguard Worker   args.task = [&] {
108*6dbdd20aSAndroid Build Coastguard Worker     ++num_callbacks;
109*6dbdd20aSAndroid Build Coastguard Worker #if PERFETTO_BUILDFLAG(PERFETTO_OS_LINUX) || \
110*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID)
111*6dbdd20aSAndroid Build Coastguard Worker     if (num_callbacks == 3 && pt.timer_fd_for_testing() > 0) {
112*6dbdd20aSAndroid Build Coastguard Worker       ScopedFile dev_null = OpenFile("/dev/null", O_RDONLY);
113*6dbdd20aSAndroid Build Coastguard Worker       dup2(*dev_null, pt.timer_fd_for_testing());
114*6dbdd20aSAndroid Build Coastguard Worker     }
115*6dbdd20aSAndroid Build Coastguard Worker #else
116*6dbdd20aSAndroid Build Coastguard Worker     EXPECT_FALSE(base::ScopedPlatformHandle::ValidityChecker::IsValid(
117*6dbdd20aSAndroid Build Coastguard Worker         pt.timer_fd_for_testing()));
118*6dbdd20aSAndroid Build Coastguard Worker #endif
119*6dbdd20aSAndroid Build Coastguard Worker     if (num_callbacks == 6)
120*6dbdd20aSAndroid Build Coastguard Worker       quit_closure();
121*6dbdd20aSAndroid Build Coastguard Worker   };
122*6dbdd20aSAndroid Build Coastguard Worker   args.period_ms = 1;
123*6dbdd20aSAndroid Build Coastguard Worker   args.use_suspend_aware_timer = true;
124*6dbdd20aSAndroid Build Coastguard Worker   pt.Start(std::move(args));
125*6dbdd20aSAndroid Build Coastguard Worker   task_runner.RunUntilCheckpoint("all_timers_done");
126*6dbdd20aSAndroid Build Coastguard Worker   EXPECT_EQ(num_callbacks, 6u);
127*6dbdd20aSAndroid Build Coastguard Worker }
128*6dbdd20aSAndroid Build Coastguard Worker 
TEST(PeriodicTaskTest,DestroyedFromCallback)129*6dbdd20aSAndroid Build Coastguard Worker TEST(PeriodicTaskTest, DestroyedFromCallback) {
130*6dbdd20aSAndroid Build Coastguard Worker   TestTaskRunner task_runner;
131*6dbdd20aSAndroid Build Coastguard Worker   std::unique_ptr<PeriodicTask> pt(new PeriodicTask(&task_runner));
132*6dbdd20aSAndroid Build Coastguard Worker   uint32_t num_callbacks = 0;
133*6dbdd20aSAndroid Build Coastguard Worker   PeriodicTask::Args args;
134*6dbdd20aSAndroid Build Coastguard Worker   auto quit_closure = task_runner.CreateCheckpoint("quit_closure");
135*6dbdd20aSAndroid Build Coastguard Worker   args.task = [&] {
136*6dbdd20aSAndroid Build Coastguard Worker     ++num_callbacks;
137*6dbdd20aSAndroid Build Coastguard Worker     pt.reset();
138*6dbdd20aSAndroid Build Coastguard Worker     task_runner.PostDelayedTask(quit_closure, 5);
139*6dbdd20aSAndroid Build Coastguard Worker   };
140*6dbdd20aSAndroid Build Coastguard Worker   args.period_ms = 1;
141*6dbdd20aSAndroid Build Coastguard Worker   args.use_suspend_aware_timer = true;
142*6dbdd20aSAndroid Build Coastguard Worker   pt->Start(std::move(args));
143*6dbdd20aSAndroid Build Coastguard Worker 
144*6dbdd20aSAndroid Build Coastguard Worker   task_runner.RunUntilCheckpoint("quit_closure");
145*6dbdd20aSAndroid Build Coastguard Worker   EXPECT_EQ(num_callbacks, 1u);
146*6dbdd20aSAndroid Build Coastguard Worker   EXPECT_FALSE(pt);
147*6dbdd20aSAndroid Build Coastguard Worker }
148*6dbdd20aSAndroid Build Coastguard Worker 
TEST(PeriodicTaskTest,DestroyedFromAnotherTask)149*6dbdd20aSAndroid Build Coastguard Worker TEST(PeriodicTaskTest, DestroyedFromAnotherTask) {
150*6dbdd20aSAndroid Build Coastguard Worker   TestTaskRunner task_runner;
151*6dbdd20aSAndroid Build Coastguard Worker   std::unique_ptr<PeriodicTask> pt(new PeriodicTask(&task_runner));
152*6dbdd20aSAndroid Build Coastguard Worker   uint32_t num_callbacks = 0;
153*6dbdd20aSAndroid Build Coastguard Worker   PeriodicTask::Args args;
154*6dbdd20aSAndroid Build Coastguard Worker   auto quit_closure = task_runner.CreateCheckpoint("quit_closure");
155*6dbdd20aSAndroid Build Coastguard Worker   args.task = [&] {
156*6dbdd20aSAndroid Build Coastguard Worker     if (++num_callbacks == 2) {
157*6dbdd20aSAndroid Build Coastguard Worker       task_runner.PostTask([&] {
158*6dbdd20aSAndroid Build Coastguard Worker         pt.reset();
159*6dbdd20aSAndroid Build Coastguard Worker         task_runner.PostDelayedTask(quit_closure, 5);
160*6dbdd20aSAndroid Build Coastguard Worker       });
161*6dbdd20aSAndroid Build Coastguard Worker     }
162*6dbdd20aSAndroid Build Coastguard Worker   };
163*6dbdd20aSAndroid Build Coastguard Worker   args.period_ms = 1;
164*6dbdd20aSAndroid Build Coastguard Worker   args.use_suspend_aware_timer = true;
165*6dbdd20aSAndroid Build Coastguard Worker   pt->Start(std::move(args));
166*6dbdd20aSAndroid Build Coastguard Worker 
167*6dbdd20aSAndroid Build Coastguard Worker   task_runner.RunUntilCheckpoint("quit_closure");
168*6dbdd20aSAndroid Build Coastguard Worker   EXPECT_EQ(num_callbacks, 2u);
169*6dbdd20aSAndroid Build Coastguard Worker   EXPECT_FALSE(pt);
170*6dbdd20aSAndroid Build Coastguard Worker }
171*6dbdd20aSAndroid Build Coastguard Worker 
172*6dbdd20aSAndroid Build Coastguard Worker // Checks the generation logic.
TEST(PeriodicTaskTest,RestartWhileRunning)173*6dbdd20aSAndroid Build Coastguard Worker TEST(PeriodicTaskTest, RestartWhileRunning) {
174*6dbdd20aSAndroid Build Coastguard Worker   TestTaskRunner task_runner;
175*6dbdd20aSAndroid Build Coastguard Worker   PeriodicTask pt(&task_runner);
176*6dbdd20aSAndroid Build Coastguard Worker   uint32_t num_callbacks_a = 0;
177*6dbdd20aSAndroid Build Coastguard Worker   uint32_t num_callbacks_b = 0;
178*6dbdd20aSAndroid Build Coastguard Worker   auto quit_closure = task_runner.CreateCheckpoint("quit_closure");
179*6dbdd20aSAndroid Build Coastguard Worker 
180*6dbdd20aSAndroid Build Coastguard Worker   auto reuse = [&] {
181*6dbdd20aSAndroid Build Coastguard Worker     PeriodicTask::Args args;
182*6dbdd20aSAndroid Build Coastguard Worker     args.period_ms = 1;
183*6dbdd20aSAndroid Build Coastguard Worker     args.task = [&] {
184*6dbdd20aSAndroid Build Coastguard Worker       if (++num_callbacks_b == 3)
185*6dbdd20aSAndroid Build Coastguard Worker         quit_closure();
186*6dbdd20aSAndroid Build Coastguard Worker     };
187*6dbdd20aSAndroid Build Coastguard Worker     pt.Start(std::move(args));
188*6dbdd20aSAndroid Build Coastguard Worker   };
189*6dbdd20aSAndroid Build Coastguard Worker 
190*6dbdd20aSAndroid Build Coastguard Worker   PeriodicTask::Args args;
191*6dbdd20aSAndroid Build Coastguard Worker   args.task = [&] {
192*6dbdd20aSAndroid Build Coastguard Worker     if (++num_callbacks_a == 2)
193*6dbdd20aSAndroid Build Coastguard Worker       task_runner.PostTask(reuse);
194*6dbdd20aSAndroid Build Coastguard Worker   };
195*6dbdd20aSAndroid Build Coastguard Worker   args.period_ms = 1;
196*6dbdd20aSAndroid Build Coastguard Worker   args.use_suspend_aware_timer = true;
197*6dbdd20aSAndroid Build Coastguard Worker   pt.Start(std::move(args));
198*6dbdd20aSAndroid Build Coastguard Worker 
199*6dbdd20aSAndroid Build Coastguard Worker   task_runner.RunUntilCheckpoint("quit_closure");
200*6dbdd20aSAndroid Build Coastguard Worker   EXPECT_EQ(num_callbacks_a, 2u);
201*6dbdd20aSAndroid Build Coastguard Worker   EXPECT_EQ(num_callbacks_b, 3u);
202*6dbdd20aSAndroid Build Coastguard Worker }
203*6dbdd20aSAndroid Build Coastguard Worker 
TEST(PeriodicTaskTest,ImmediateExecution)204*6dbdd20aSAndroid Build Coastguard Worker TEST(PeriodicTaskTest, ImmediateExecution) {
205*6dbdd20aSAndroid Build Coastguard Worker   TestTaskRunner task_runner;
206*6dbdd20aSAndroid Build Coastguard Worker   PeriodicTask pt(&task_runner);
207*6dbdd20aSAndroid Build Coastguard Worker   uint32_t num_callbacks = 0;
208*6dbdd20aSAndroid Build Coastguard Worker 
209*6dbdd20aSAndroid Build Coastguard Worker   PeriodicTask::Args args;
210*6dbdd20aSAndroid Build Coastguard Worker   args.task = [&] { ++num_callbacks; };
211*6dbdd20aSAndroid Build Coastguard Worker   args.period_ms = 1;
212*6dbdd20aSAndroid Build Coastguard Worker   pt.Start(args);
213*6dbdd20aSAndroid Build Coastguard Worker   EXPECT_EQ(num_callbacks, 0u);  // No immediate execution.
214*6dbdd20aSAndroid Build Coastguard Worker 
215*6dbdd20aSAndroid Build Coastguard Worker   args.start_first_task_immediately = true;
216*6dbdd20aSAndroid Build Coastguard Worker   pt.Start(args);
217*6dbdd20aSAndroid Build Coastguard Worker   EXPECT_EQ(num_callbacks, 1u);
218*6dbdd20aSAndroid Build Coastguard Worker }
219*6dbdd20aSAndroid Build Coastguard Worker 
220*6dbdd20aSAndroid Build Coastguard Worker }  // namespace
221*6dbdd20aSAndroid Build Coastguard Worker }  // namespace base
222*6dbdd20aSAndroid Build Coastguard Worker }  // namespace perfetto
223