/* * 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/unix_task_runner.h" #include "gtest/gtest.h" #include "perfetto/base/build_config.h" #include "perfetto/base/scoped_file.h" #if PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID) && \ !PERFETTO_BUILDFLAG(PERFETTO_EMBEDDER_BUILD) #include "perfetto/base/android_task_runner.h" #endif #include <thread> #include "perfetto/base/file_utils.h" #include "perfetto/base/pipe.h" #include "src/base/test/gtest_test_suite.h" namespace perfetto { namespace base { namespace { template <typename T> class TaskRunnerTest : public ::testing::Test { public: T task_runner; }; #if PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID) && \ !PERFETTO_BUILDFLAG(PERFETTO_EMBEDDER_BUILD) using TaskRunnerTypes = ::testing::Types<AndroidTaskRunner, UnixTaskRunner>; #else using TaskRunnerTypes = ::testing::Types<UnixTaskRunner>; #endif TYPED_TEST_SUITE(TaskRunnerTest, TaskRunnerTypes); struct TestPipe : Pipe { TestPipe() : Pipe(Pipe::Create()) { // Make the pipe initially readable. Write(); } void Read() { char b; ssize_t rd = read(*this->rd, &b, 1); PERFETTO_DCHECK(rd == 1); } void Write() { const char b = '?'; ssize_t wr = WriteAll(*this->wr, &b, 1); PERFETTO_DCHECK(wr == 1); } }; TYPED_TEST(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); } TYPED_TEST(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); } TYPED_TEST(TaskRunnerTest, PostImmediateTaskFromTask) { auto& task_runner = this->task_runner; task_runner.PostTask([&task_runner] { task_runner.PostTask([&task_runner] { task_runner.Quit(); }); }); task_runner.Run(); } TYPED_TEST(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(); } TYPED_TEST(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); } TYPED_TEST(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(); } TYPED_TEST(TaskRunnerTest, AddFileDescriptorWatch) { auto& task_runner = this->task_runner; TestPipe pipe; task_runner.AddFileDescriptorWatch(pipe.rd.get(), [&task_runner] { task_runner.Quit(); }); task_runner.Run(); } TYPED_TEST(TaskRunnerTest, RemoveFileDescriptorWatch) { auto& task_runner = this->task_runner; TestPipe pipe; bool watch_ran = false; task_runner.AddFileDescriptorWatch(pipe.rd.get(), [&watch_ran] { watch_ran = true; }); task_runner.RemoveFileDescriptorWatch(pipe.rd.get()); task_runner.PostDelayedTask([&task_runner] { task_runner.Quit(); }, 10); task_runner.Run(); EXPECT_FALSE(watch_ran); } TYPED_TEST(TaskRunnerTest, RemoveFileDescriptorWatchFromTask) { auto& task_runner = this->task_runner; TestPipe pipe; bool watch_ran = false; task_runner.PostTask([&task_runner, &pipe] { task_runner.RemoveFileDescriptorWatch(pipe.rd.get()); }); task_runner.AddFileDescriptorWatch(pipe.rd.get(), [&watch_ran] { watch_ran = true; }); task_runner.PostDelayedTask([&task_runner] { task_runner.Quit(); }, 10); task_runner.Run(); EXPECT_FALSE(watch_ran); } TYPED_TEST(TaskRunnerTest, AddFileDescriptorWatchFromAnotherWatch) { auto& task_runner = this->task_runner; TestPipe pipe; TestPipe pipe2; task_runner.AddFileDescriptorWatch( pipe.rd.get(), [&task_runner, &pipe, &pipe2] { pipe.Read(); task_runner.AddFileDescriptorWatch( pipe2.rd.get(), [&task_runner] { task_runner.Quit(); }); }); task_runner.Run(); } TYPED_TEST(TaskRunnerTest, RemoveFileDescriptorWatchFromAnotherWatch) { auto& task_runner = this->task_runner; TestPipe pipe; TestPipe pipe2; bool watch_ran = false; task_runner.AddFileDescriptorWatch( pipe.rd.get(), [&task_runner, &pipe, &pipe2] { pipe.Read(); task_runner.RemoveFileDescriptorWatch(pipe2.rd.get()); }); task_runner.AddFileDescriptorWatch(pipe2.rd.get(), [&watch_ran] { watch_ran = true; }); task_runner.PostDelayedTask([&task_runner] { task_runner.Quit(); }, 10); task_runner.Run(); EXPECT_FALSE(watch_ran); } TYPED_TEST(TaskRunnerTest, ReplaceFileDescriptorWatchFromAnotherWatch) { auto& task_runner = this->task_runner; TestPipe pipe; TestPipe pipe2; bool watch_ran = false; task_runner.AddFileDescriptorWatch(pipe.rd.get(), [&task_runner, &pipe2] { task_runner.RemoveFileDescriptorWatch(pipe2.rd.get()); task_runner.AddFileDescriptorWatch(pipe2.rd.get(), [&task_runner] { task_runner.Quit(); }); }); task_runner.AddFileDescriptorWatch(pipe2.rd.get(), [&watch_ran] { watch_ran = true; }); task_runner.Run(); EXPECT_FALSE(watch_ran); } TYPED_TEST(TaskRunnerTest, AddFileDescriptorWatchFromAnotherThread) { auto& task_runner = this->task_runner; TestPipe pipe; std::thread thread([&task_runner, &pipe] { task_runner.AddFileDescriptorWatch(pipe.rd.get(), [&task_runner] { task_runner.Quit(); }); }); task_runner.Run(); thread.join(); } TYPED_TEST(TaskRunnerTest, FileDescriptorWatchWithMultipleEvents) { auto& task_runner = this->task_runner; TestPipe pipe; int event_count = 0; task_runner.AddFileDescriptorWatch(pipe.rd.get(), [&task_runner, &pipe, &event_count] { if (++event_count == 3) { task_runner.Quit(); return; } pipe.Read(); }); task_runner.PostTask([&pipe] { pipe.Write(); }); task_runner.PostTask([&pipe] { pipe.Write(); }); task_runner.Run(); } TYPED_TEST(TaskRunnerTest, FileDescriptorClosedEvent) { auto& task_runner = this->task_runner; TestPipe pipe; pipe.wr.reset(); task_runner.AddFileDescriptorWatch(pipe.rd.get(), [&task_runner] { task_runner.Quit(); }); task_runner.Run(); } TYPED_TEST(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(); } TYPED_TEST(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); } template <typename TaskRunner> void RepeatingTask(TaskRunner* task_runner) { task_runner->PostTask(std::bind(&RepeatingTask<TaskRunner>, task_runner)); } TYPED_TEST(TaskRunnerTest, FileDescriptorWatchesNotStarved) { auto& task_runner = this->task_runner; TestPipe pipe; task_runner.PostTask(std::bind(&RepeatingTask<TypeParam>, &task_runner)); task_runner.AddFileDescriptorWatch(pipe.rd.get(), [&task_runner] { task_runner.Quit(); }); task_runner.Run(); } template <typename TaskRunner> void CountdownTask(TaskRunner* task_runner, int* counter) { if (!--(*counter)) { task_runner->Quit(); return; } task_runner->PostTask( std::bind(&CountdownTask<TaskRunner>, task_runner, counter)); } TYPED_TEST(TaskRunnerTest, NoDuplicateFileDescriptorWatchCallbacks) { auto& task_runner = this->task_runner; TestPipe pipe; bool watch_called = 0; int counter = 10; task_runner.AddFileDescriptorWatch(pipe.rd.get(), [&pipe, &watch_called] { ASSERT_FALSE(watch_called); pipe.Read(); watch_called = true; }); task_runner.PostTask( std::bind(&CountdownTask<TypeParam>, &task_runner, &counter)); task_runner.Run(); } TYPED_TEST(TaskRunnerTest, ReplaceFileDescriptorWatchFromOtherThread) { auto& task_runner = this->task_runner; TestPipe pipe; // 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(pipe.rd.get(), [&task_runner] { task_runner.Quit(); }); std::thread thread([&task_runner, &pipe] { task_runner.RemoveFileDescriptorWatch(pipe.rd.get()); task_runner.AddFileDescriptorWatch(pipe.rd.get(), [&task_runner] { task_runner.Quit(); }); }); task_runner.Run(); thread.join(); } TYPED_TEST(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(); } TYPED_TEST(TaskRunnerTest, RunsTasksOnCurrentThread) { auto& main_tr = this->task_runner; EXPECT_TRUE(main_tr.RunsTasksOnCurrentThread()); std::thread thread([&main_tr] { typename std::remove_reference<decltype(main_tr)>::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(); } } // namespace } // namespace base } // namespace perfetto