/* * Copyright (C) 2017 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/base/build_config.h" #include "perfetto/ext/base/unix_task_runner.h" #include #include "perfetto/ext/base/event_fd.h" #include "perfetto/ext/base/file_utils.h" #include "perfetto/ext/base/pipe.h" #include "perfetto/ext/base/scoped_file.h" #include "perfetto/ext/base/utils.h" #include "src/base/test/gtest_test_suite.h" #include "test/gtest_and_gmock.h" namespace perfetto { namespace base { namespace { class TaskRunnerTest : public ::testing::Test { public: UnixTaskRunner task_runner; }; TEST_F(TaskRunnerTest, PostImmediateTask) { auto& task_runner = this->task_runner; int counter = 0; task_runner.PostTask([&counter] { counter = (counter << 4) | 1; }); task_runner.PostTask([&counter] { counter = (counter << 4) | 2; }); task_runner.PostTask([&counter] { counter = (counter << 4) | 3; }); task_runner.PostTask([&counter] { counter = (counter << 4) | 4; }); task_runner.PostTask([&task_runner] { task_runner.Quit(); }); task_runner.Run(); EXPECT_EQ(0x1234, counter); } TEST_F(TaskRunnerTest, PostDelayedTask) { auto& task_runner = this->task_runner; int counter = 0; task_runner.PostDelayedTask([&counter] { counter = (counter << 4) | 1; }, 5); task_runner.PostDelayedTask([&counter] { counter = (counter << 4) | 2; }, 10); task_runner.PostDelayedTask([&counter] { counter = (counter << 4) | 3; }, 15); task_runner.PostDelayedTask([&counter] { counter = (counter << 4) | 4; }, 15); task_runner.PostDelayedTask([&task_runner] { task_runner.Quit(); }, 20); task_runner.Run(); EXPECT_EQ(0x1234, counter); } TEST_F(TaskRunnerTest, PostImmediateTaskFromTask) { auto& task_runner = this->task_runner; task_runner.PostTask([&task_runner] { task_runner.PostTask([&task_runner] { task_runner.Quit(); }); }); task_runner.Run(); } TEST_F(TaskRunnerTest, PostDelayedTaskFromTask) { auto& task_runner = this->task_runner; task_runner.PostTask([&task_runner] { task_runner.PostDelayedTask([&task_runner] { task_runner.Quit(); }, 10); }); task_runner.Run(); } TEST_F(TaskRunnerTest, PostImmediateTaskFromOtherThread) { auto& task_runner = this->task_runner; ThreadChecker thread_checker; int counter = 0; std::thread thread([&task_runner, &counter, &thread_checker] { task_runner.PostTask([&thread_checker] { EXPECT_TRUE(thread_checker.CalledOnValidThread()); }); task_runner.PostTask([&counter] { counter = (counter << 4) | 1; }); task_runner.PostTask([&counter] { counter = (counter << 4) | 2; }); task_runner.PostTask([&counter] { counter = (counter << 4) | 3; }); task_runner.PostTask([&counter] { counter = (counter << 4) | 4; }); task_runner.PostTask([&task_runner] { task_runner.Quit(); }); }); task_runner.Run(); thread.join(); EXPECT_EQ(0x1234, counter); } TEST_F(TaskRunnerTest, PostDelayedTaskFromOtherThread) { auto& task_runner = this->task_runner; std::thread thread([&task_runner] { task_runner.PostDelayedTask([&task_runner] { task_runner.Quit(); }, 10); }); task_runner.Run(); thread.join(); } TEST_F(TaskRunnerTest, AddFileDescriptorWatch) { auto& task_runner = this->task_runner; EventFd evt; task_runner.AddFileDescriptorWatch(evt.fd(), [&task_runner] { task_runner.Quit(); }); evt.Notify(); task_runner.Run(); } TEST_F(TaskRunnerTest, RemoveFileDescriptorWatch) { auto& task_runner = this->task_runner; EventFd evt; evt.Notify(); bool watch_ran = false; task_runner.AddFileDescriptorWatch(evt.fd(), [&watch_ran] { watch_ran = true; }); task_runner.RemoveFileDescriptorWatch(evt.fd()); task_runner.PostDelayedTask([&task_runner] { task_runner.Quit(); }, 10); task_runner.Run(); EXPECT_FALSE(watch_ran); } TEST_F(TaskRunnerTest, RemoveFileDescriptorWatchFromTask) { auto& task_runner = this->task_runner; EventFd evt; evt.Notify(); bool watch_ran = false; task_runner.PostTask([&task_runner, &evt] { task_runner.RemoveFileDescriptorWatch(evt.fd()); }); task_runner.AddFileDescriptorWatch(evt.fd(), [&watch_ran] { watch_ran = true; }); task_runner.PostDelayedTask([&task_runner] { task_runner.Quit(); }, 10); task_runner.Run(); EXPECT_FALSE(watch_ran); } TEST_F(TaskRunnerTest, AddFileDescriptorWatchFromAnotherWatch) { auto& task_runner = this->task_runner; EventFd evt; EventFd evt2; evt.Notify(); evt2.Notify(); task_runner.AddFileDescriptorWatch(evt.fd(), [&task_runner, &evt, &evt2] { evt.Clear(); task_runner.AddFileDescriptorWatch(evt2.fd(), [&task_runner] { task_runner.Quit(); }); }); task_runner.Run(); } TEST_F(TaskRunnerTest, RemoveFileDescriptorWatchFromAnotherWatch) { auto& task_runner = this->task_runner; EventFd evt; EventFd evt2; evt.Notify(); bool watch_ran = false; task_runner.AddFileDescriptorWatch(evt.fd(), [&task_runner, &evt, &evt2] { evt.Clear(); evt2.Notify(); task_runner.RemoveFileDescriptorWatch(evt2.fd()); }); task_runner.AddFileDescriptorWatch(evt2.fd(), [&watch_ran] { watch_ran = true; }); task_runner.PostDelayedTask([&task_runner] { task_runner.Quit(); }, 10); task_runner.Run(); EXPECT_FALSE(watch_ran); } TEST_F(TaskRunnerTest, ReplaceFileDescriptorWatchFromAnotherWatch) { auto& task_runner = this->task_runner; EventFd evt; EventFd evt2; bool watch_ran = false; evt.Notify(); task_runner.AddFileDescriptorWatch(evt.fd(), [&task_runner, &evt, &evt2] { evt.Clear(); evt2.Notify(); task_runner.RemoveFileDescriptorWatch(evt2.fd()); task_runner.AddFileDescriptorWatch(evt2.fd(), [&task_runner] { task_runner.Quit(); }); }); task_runner.AddFileDescriptorWatch(evt2.fd(), [&watch_ran] { watch_ran = true; }); task_runner.Run(); EXPECT_FALSE(watch_ran); } TEST_F(TaskRunnerTest, AddFileDescriptorWatchFromAnotherThread) { auto& task_runner = this->task_runner; EventFd evt; evt.Notify(); std::thread thread([&task_runner, &evt] { task_runner.AddFileDescriptorWatch(evt.fd(), [&task_runner] { task_runner.Quit(); }); }); task_runner.Run(); thread.join(); } TEST_F(TaskRunnerTest, FileDescriptorWatchWithMultipleEvents) { auto& task_runner = this->task_runner; EventFd evt; evt.Notify(); int event_count = 0; task_runner.AddFileDescriptorWatch( evt.fd(), [&task_runner, &evt, &event_count] { ASSERT_LT(event_count, 3); if (++event_count == 3) { task_runner.Quit(); return; } evt.Clear(); task_runner.PostTask([&evt] { evt.Notify(); }); }); task_runner.Run(); } TEST_F(TaskRunnerTest, PostManyDelayedTasks) { // Check that PostTask doesn't start failing if there are too many scheduled // wake-ups. auto& task_runner = this->task_runner; for (int i = 0; i < 0x1000; i++) task_runner.PostDelayedTask([] {}, 0); task_runner.PostDelayedTask([&task_runner] { task_runner.Quit(); }, 10); task_runner.Run(); } TEST_F(TaskRunnerTest, RunAgain) { auto& task_runner = this->task_runner; int counter = 0; task_runner.PostTask([&task_runner, &counter] { counter++; task_runner.Quit(); }); task_runner.Run(); task_runner.PostTask([&task_runner, &counter] { counter++; task_runner.Quit(); }); task_runner.Run(); EXPECT_EQ(2, counter); } void RepeatingTask(UnixTaskRunner* task_runner) { task_runner->PostTask(std::bind(&RepeatingTask, task_runner)); } TEST_F(TaskRunnerTest, FileDescriptorWatchesNotStarved) { auto& task_runner = this->task_runner; EventFd evt; evt.Notify(); task_runner.PostTask(std::bind(&RepeatingTask, &task_runner)); task_runner.AddFileDescriptorWatch(evt.fd(), [&task_runner] { task_runner.Quit(); }); task_runner.Run(); } void CountdownTask(UnixTaskRunner* task_runner, int* counter) { if (!--(*counter)) { task_runner->Quit(); return; } task_runner->PostDelayedTask(std::bind(&CountdownTask, task_runner, counter), 1); } TEST_F(TaskRunnerTest, NoDuplicateFileDescriptorWatchCallbacks) { auto& task_runner = this->task_runner; EventFd evt; evt.Notify(); bool watch_called = 0; int counter = 10; task_runner.AddFileDescriptorWatch(evt.fd(), [&evt, &watch_called] { ASSERT_FALSE(watch_called); evt.Clear(); watch_called = true; }); task_runner.PostTask(std::bind(&CountdownTask, &task_runner, &counter)); task_runner.Run(); } TEST_F(TaskRunnerTest, ReplaceFileDescriptorWatchFromOtherThread) { auto& task_runner = this->task_runner; EventFd evt; evt.Notify(); // The two watch tasks here race each other. We don't particularly care which // wins as long as one of them runs. task_runner.AddFileDescriptorWatch(evt.fd(), [&task_runner] { task_runner.Quit(); }); std::thread thread([&task_runner, &evt] { task_runner.RemoveFileDescriptorWatch(evt.fd()); task_runner.AddFileDescriptorWatch(evt.fd(), [&task_runner] { task_runner.Quit(); }); }); task_runner.Run(); thread.join(); } TEST_F(TaskRunnerTest, IsIdleForTesting) { auto& task_runner = this->task_runner; task_runner.PostTask( [&task_runner] { EXPECT_FALSE(task_runner.IsIdleForTesting()); }); task_runner.PostTask([&task_runner] { EXPECT_TRUE(task_runner.IsIdleForTesting()); task_runner.Quit(); }); task_runner.Run(); } TEST_F(TaskRunnerTest, RunsTasksOnCurrentThread) { auto& main_tr = this->task_runner; EXPECT_TRUE(main_tr.RunsTasksOnCurrentThread()); std::thread thread([&main_tr] { typename std::remove_reference::type second_tr; second_tr.PostTask([&main_tr, &second_tr] { EXPECT_FALSE(main_tr.RunsTasksOnCurrentThread()); EXPECT_TRUE(second_tr.RunsTasksOnCurrentThread()); second_tr.Quit(); }); second_tr.Run(); }); thread.join(); } TEST_F(TaskRunnerTest, FileDescriptorWatchFairness) { auto& task_runner = this->task_runner; EventFd evt[5]; std::map num_tasks; static constexpr int kNumTasksPerHandle = 100; for (auto& e : evt) { e.Notify(); task_runner.AddFileDescriptorWatch(e.fd(), [&] { if (++num_tasks[e.fd()] == kNumTasksPerHandle) { e.Clear(); task_runner.Quit(); } }); } task_runner.Run(); // The sequence evt[0], evt[1], evt[2] should be repeated N times. On the // Nth time the task runner quits. All tasks should have been running at least // N-1 times (we can't predict which one of the tasks will quit). for (auto& e : evt) { ASSERT_GE(num_tasks[e.fd()], kNumTasksPerHandle - 1); ASSERT_LE(num_tasks[e.fd()], kNumTasksPerHandle); } } #if !PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) // This tests UNIX-specific behavior on pipe closure. TEST_F(TaskRunnerTest, FileDescriptorClosedEvent) { auto& task_runner = this->task_runner; Pipe pipe = Pipe::Create(); pipe.wr.reset(); task_runner.AddFileDescriptorWatch(pipe.rd.get(), [&task_runner] { task_runner.Quit(); }); task_runner.Run(); } #endif } // namespace } // namespace base } // namespace perfetto