xref: /aosp_15_r20/external/perfetto/src/base/thread_task_runner.cc (revision 6dbdd20afdafa5e3ca9b8809fa73465d530080dc)
1*6dbdd20aSAndroid Build Coastguard Worker /*
2*6dbdd20aSAndroid Build Coastguard Worker  * Copyright (C) 2019 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/base/build_config.h"
18*6dbdd20aSAndroid Build Coastguard Worker 
19*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/thread_task_runner.h"
20*6dbdd20aSAndroid Build Coastguard Worker 
21*6dbdd20aSAndroid Build Coastguard Worker #include <condition_variable>
22*6dbdd20aSAndroid Build Coastguard Worker #include <functional>
23*6dbdd20aSAndroid Build Coastguard Worker #include <mutex>
24*6dbdd20aSAndroid Build Coastguard Worker #include <thread>
25*6dbdd20aSAndroid Build Coastguard Worker 
26*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/base/logging.h"
27*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/thread_utils.h"
28*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/unix_task_runner.h"
29*6dbdd20aSAndroid Build Coastguard Worker 
30*6dbdd20aSAndroid Build Coastguard Worker #if PERFETTO_BUILDFLAG(PERFETTO_OS_LINUX) || \
31*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID)
32*6dbdd20aSAndroid Build Coastguard Worker #include <sys/prctl.h>
33*6dbdd20aSAndroid Build Coastguard Worker #endif
34*6dbdd20aSAndroid Build Coastguard Worker 
35*6dbdd20aSAndroid Build Coastguard Worker namespace perfetto {
36*6dbdd20aSAndroid Build Coastguard Worker namespace base {
37*6dbdd20aSAndroid Build Coastguard Worker 
ThreadTaskRunner(ThreadTaskRunner && other)38*6dbdd20aSAndroid Build Coastguard Worker ThreadTaskRunner::ThreadTaskRunner(ThreadTaskRunner&& other) noexcept
39*6dbdd20aSAndroid Build Coastguard Worker     : thread_(std::move(other.thread_)), task_runner_(other.task_runner_) {
40*6dbdd20aSAndroid Build Coastguard Worker   other.task_runner_ = nullptr;
41*6dbdd20aSAndroid Build Coastguard Worker }
42*6dbdd20aSAndroid Build Coastguard Worker 
operator =(ThreadTaskRunner && other)43*6dbdd20aSAndroid Build Coastguard Worker ThreadTaskRunner& ThreadTaskRunner::operator=(ThreadTaskRunner&& other) {
44*6dbdd20aSAndroid Build Coastguard Worker   this->~ThreadTaskRunner();
45*6dbdd20aSAndroid Build Coastguard Worker   new (this) ThreadTaskRunner(std::move(other));
46*6dbdd20aSAndroid Build Coastguard Worker   return *this;
47*6dbdd20aSAndroid Build Coastguard Worker }
48*6dbdd20aSAndroid Build Coastguard Worker 
~ThreadTaskRunner()49*6dbdd20aSAndroid Build Coastguard Worker ThreadTaskRunner::~ThreadTaskRunner() {
50*6dbdd20aSAndroid Build Coastguard Worker   if (task_runner_) {
51*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_CHECK(!task_runner_->QuitCalled());
52*6dbdd20aSAndroid Build Coastguard Worker     task_runner_->Quit();
53*6dbdd20aSAndroid Build Coastguard Worker 
54*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_DCHECK(thread_.joinable());
55*6dbdd20aSAndroid Build Coastguard Worker   }
56*6dbdd20aSAndroid Build Coastguard Worker   if (thread_.joinable())
57*6dbdd20aSAndroid Build Coastguard Worker     thread_.join();
58*6dbdd20aSAndroid Build Coastguard Worker }
59*6dbdd20aSAndroid Build Coastguard Worker 
ThreadTaskRunner(const std::string & name)60*6dbdd20aSAndroid Build Coastguard Worker ThreadTaskRunner::ThreadTaskRunner(const std::string& name) : name_(name) {
61*6dbdd20aSAndroid Build Coastguard Worker   std::mutex init_lock;
62*6dbdd20aSAndroid Build Coastguard Worker   std::condition_variable init_cv;
63*6dbdd20aSAndroid Build Coastguard Worker 
64*6dbdd20aSAndroid Build Coastguard Worker   std::function<void(UnixTaskRunner*)> initializer =
65*6dbdd20aSAndroid Build Coastguard Worker       [this, &init_lock, &init_cv](UnixTaskRunner* task_runner) {
66*6dbdd20aSAndroid Build Coastguard Worker         std::lock_guard<std::mutex> lock(init_lock);
67*6dbdd20aSAndroid Build Coastguard Worker         task_runner_ = task_runner;
68*6dbdd20aSAndroid Build Coastguard Worker         // Notify while still holding the lock, as init_cv ceases to exist as
69*6dbdd20aSAndroid Build Coastguard Worker         // soon as the main thread observes a non-null task_runner_, and it can
70*6dbdd20aSAndroid Build Coastguard Worker         // wake up spuriously (i.e. before the notify if we had unlocked before
71*6dbdd20aSAndroid Build Coastguard Worker         // notifying).
72*6dbdd20aSAndroid Build Coastguard Worker         init_cv.notify_one();
73*6dbdd20aSAndroid Build Coastguard Worker       };
74*6dbdd20aSAndroid Build Coastguard Worker 
75*6dbdd20aSAndroid Build Coastguard Worker   thread_ = std::thread(&ThreadTaskRunner::RunTaskThread, this,
76*6dbdd20aSAndroid Build Coastguard Worker                         std::move(initializer));
77*6dbdd20aSAndroid Build Coastguard Worker 
78*6dbdd20aSAndroid Build Coastguard Worker   std::unique_lock<std::mutex> lock(init_lock);
79*6dbdd20aSAndroid Build Coastguard Worker   init_cv.wait(lock, [this] { return !!task_runner_; });
80*6dbdd20aSAndroid Build Coastguard Worker }
81*6dbdd20aSAndroid Build Coastguard Worker 
RunTaskThread(std::function<void (UnixTaskRunner *)> initializer)82*6dbdd20aSAndroid Build Coastguard Worker void ThreadTaskRunner::RunTaskThread(
83*6dbdd20aSAndroid Build Coastguard Worker     std::function<void(UnixTaskRunner*)> initializer) {
84*6dbdd20aSAndroid Build Coastguard Worker   if (!name_.empty()) {
85*6dbdd20aSAndroid Build Coastguard Worker     base::MaybeSetThreadName(name_);
86*6dbdd20aSAndroid Build Coastguard Worker   }
87*6dbdd20aSAndroid Build Coastguard Worker 
88*6dbdd20aSAndroid Build Coastguard Worker   UnixTaskRunner task_runner;
89*6dbdd20aSAndroid Build Coastguard Worker   task_runner.PostTask(std::bind(std::move(initializer), &task_runner));
90*6dbdd20aSAndroid Build Coastguard Worker   task_runner.Run();
91*6dbdd20aSAndroid Build Coastguard Worker }
92*6dbdd20aSAndroid Build Coastguard Worker 
PostTaskAndWaitForTesting(std::function<void ()> fn)93*6dbdd20aSAndroid Build Coastguard Worker void ThreadTaskRunner::PostTaskAndWaitForTesting(std::function<void()> fn) {
94*6dbdd20aSAndroid Build Coastguard Worker   std::mutex mutex;
95*6dbdd20aSAndroid Build Coastguard Worker   std::condition_variable cv;
96*6dbdd20aSAndroid Build Coastguard Worker 
97*6dbdd20aSAndroid Build Coastguard Worker   std::unique_lock<std::mutex> lock(mutex);
98*6dbdd20aSAndroid Build Coastguard Worker   bool done = false;
99*6dbdd20aSAndroid Build Coastguard Worker   task_runner_->PostTask([&mutex, &cv, &done, &fn] {
100*6dbdd20aSAndroid Build Coastguard Worker     fn();
101*6dbdd20aSAndroid Build Coastguard Worker 
102*6dbdd20aSAndroid Build Coastguard Worker     std::lock_guard<std::mutex> inner_lock(mutex);
103*6dbdd20aSAndroid Build Coastguard Worker     done = true;
104*6dbdd20aSAndroid Build Coastguard Worker     cv.notify_one();
105*6dbdd20aSAndroid Build Coastguard Worker   });
106*6dbdd20aSAndroid Build Coastguard Worker   cv.wait(lock, [&done] { return done; });
107*6dbdd20aSAndroid Build Coastguard Worker }
108*6dbdd20aSAndroid Build Coastguard Worker 
GetThreadCPUTimeNsForTesting()109*6dbdd20aSAndroid Build Coastguard Worker uint64_t ThreadTaskRunner::GetThreadCPUTimeNsForTesting() {
110*6dbdd20aSAndroid Build Coastguard Worker   uint64_t thread_time_ns = 0;
111*6dbdd20aSAndroid Build Coastguard Worker   PostTaskAndWaitForTesting([&thread_time_ns] {
112*6dbdd20aSAndroid Build Coastguard Worker     thread_time_ns = static_cast<uint64_t>(base::GetThreadCPUTimeNs().count());
113*6dbdd20aSAndroid Build Coastguard Worker   });
114*6dbdd20aSAndroid Build Coastguard Worker   return thread_time_ns;
115*6dbdd20aSAndroid Build Coastguard Worker }
116*6dbdd20aSAndroid Build Coastguard Worker 
PostTask(std::function<void ()> task)117*6dbdd20aSAndroid Build Coastguard Worker void ThreadTaskRunner::PostTask(std::function<void()> task) {
118*6dbdd20aSAndroid Build Coastguard Worker   task_runner_->PostTask(std::move(task));
119*6dbdd20aSAndroid Build Coastguard Worker }
120*6dbdd20aSAndroid Build Coastguard Worker 
PostDelayedTask(std::function<void ()> task,uint32_t delay_ms)121*6dbdd20aSAndroid Build Coastguard Worker void ThreadTaskRunner::PostDelayedTask(std::function<void()> task,
122*6dbdd20aSAndroid Build Coastguard Worker                                        uint32_t delay_ms) {
123*6dbdd20aSAndroid Build Coastguard Worker   task_runner_->PostDelayedTask(std::move(task), delay_ms);
124*6dbdd20aSAndroid Build Coastguard Worker }
125*6dbdd20aSAndroid Build Coastguard Worker 
AddFileDescriptorWatch(PlatformHandle handle,std::function<void ()> watch_task)126*6dbdd20aSAndroid Build Coastguard Worker void ThreadTaskRunner::AddFileDescriptorWatch(
127*6dbdd20aSAndroid Build Coastguard Worker     PlatformHandle handle,
128*6dbdd20aSAndroid Build Coastguard Worker     std::function<void()> watch_task) {
129*6dbdd20aSAndroid Build Coastguard Worker   task_runner_->AddFileDescriptorWatch(handle, std::move(watch_task));
130*6dbdd20aSAndroid Build Coastguard Worker }
131*6dbdd20aSAndroid Build Coastguard Worker 
RemoveFileDescriptorWatch(PlatformHandle handle)132*6dbdd20aSAndroid Build Coastguard Worker void ThreadTaskRunner::RemoveFileDescriptorWatch(PlatformHandle handle) {
133*6dbdd20aSAndroid Build Coastguard Worker   task_runner_->RemoveFileDescriptorWatch(handle);
134*6dbdd20aSAndroid Build Coastguard Worker }
135*6dbdd20aSAndroid Build Coastguard Worker 
RunsTasksOnCurrentThread() const136*6dbdd20aSAndroid Build Coastguard Worker bool ThreadTaskRunner::RunsTasksOnCurrentThread() const {
137*6dbdd20aSAndroid Build Coastguard Worker   return task_runner_->RunsTasksOnCurrentThread();
138*6dbdd20aSAndroid Build Coastguard Worker }
139*6dbdd20aSAndroid Build Coastguard Worker 
140*6dbdd20aSAndroid Build Coastguard Worker }  // namespace base
141*6dbdd20aSAndroid Build Coastguard Worker }  // namespace perfetto
142