xref: /aosp_15_r20/external/perfetto/src/tracing/platform_posix.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 #if PERFETTO_BUILDFLAG(PERFETTO_OS_LINUX) ||   \
20*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID) || \
21*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_BUILDFLAG(PERFETTO_OS_FUCHSIA) || \
22*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_BUILDFLAG(PERFETTO_OS_APPLE)
23*6dbdd20aSAndroid Build Coastguard Worker 
24*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/file_utils.h"
25*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/thread_task_runner.h"
26*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/tracing/internal/tracing_tls.h"
27*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/tracing/platform.h"
28*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/tracing/trace_writer_base.h"
29*6dbdd20aSAndroid Build Coastguard Worker 
30*6dbdd20aSAndroid Build Coastguard Worker #include <pthread.h>
31*6dbdd20aSAndroid Build Coastguard Worker #include <stdlib.h>
32*6dbdd20aSAndroid Build Coastguard Worker 
33*6dbdd20aSAndroid Build Coastguard Worker namespace perfetto {
34*6dbdd20aSAndroid Build Coastguard Worker 
35*6dbdd20aSAndroid Build Coastguard Worker namespace {
36*6dbdd20aSAndroid Build Coastguard Worker 
37*6dbdd20aSAndroid Build Coastguard Worker class PlatformPosix : public Platform {
38*6dbdd20aSAndroid Build Coastguard Worker  public:
39*6dbdd20aSAndroid Build Coastguard Worker   PlatformPosix();
40*6dbdd20aSAndroid Build Coastguard Worker   ~PlatformPosix() override;
41*6dbdd20aSAndroid Build Coastguard Worker 
42*6dbdd20aSAndroid Build Coastguard Worker   ThreadLocalObject* GetOrCreateThreadLocalObject() override;
43*6dbdd20aSAndroid Build Coastguard Worker 
44*6dbdd20aSAndroid Build Coastguard Worker   std::unique_ptr<base::TaskRunner> CreateTaskRunner(
45*6dbdd20aSAndroid Build Coastguard Worker       const CreateTaskRunnerArgs&) override;
46*6dbdd20aSAndroid Build Coastguard Worker   std::string GetCurrentProcessName() override;
47*6dbdd20aSAndroid Build Coastguard Worker   void Shutdown() override;
48*6dbdd20aSAndroid Build Coastguard Worker 
49*6dbdd20aSAndroid Build Coastguard Worker  private:
50*6dbdd20aSAndroid Build Coastguard Worker   pthread_key_t tls_key_{};
51*6dbdd20aSAndroid Build Coastguard Worker };
52*6dbdd20aSAndroid Build Coastguard Worker 
53*6dbdd20aSAndroid Build Coastguard Worker PlatformPosix* g_instance = nullptr;
54*6dbdd20aSAndroid Build Coastguard Worker 
55*6dbdd20aSAndroid Build Coastguard Worker using ThreadLocalObject = Platform::ThreadLocalObject;
56*6dbdd20aSAndroid Build Coastguard Worker 
PlatformPosix()57*6dbdd20aSAndroid Build Coastguard Worker PlatformPosix::PlatformPosix() {
58*6dbdd20aSAndroid Build Coastguard Worker   PERFETTO_CHECK(!g_instance);
59*6dbdd20aSAndroid Build Coastguard Worker   g_instance = this;
60*6dbdd20aSAndroid Build Coastguard Worker   auto tls_dtor = [](void* obj) {
61*6dbdd20aSAndroid Build Coastguard Worker     // The Posix TLS implementation resets the key before calling this dtor.
62*6dbdd20aSAndroid Build Coastguard Worker     // Here we re-reset it to the object we are about to delete. This is to
63*6dbdd20aSAndroid Build Coastguard Worker     // handle re-entrant usages of tracing in the PostTask done during the dtor
64*6dbdd20aSAndroid Build Coastguard Worker     // (see comments in TracingTLS::~TracingTLS()). Chromium's platform
65*6dbdd20aSAndroid Build Coastguard Worker     // implementation (which does NOT use this platform impl) has a similar
66*6dbdd20aSAndroid Build Coastguard Worker     // workaround (https://crrev.com/c/2748300).
67*6dbdd20aSAndroid Build Coastguard Worker     pthread_setspecific(g_instance->tls_key_, obj);
68*6dbdd20aSAndroid Build Coastguard Worker     delete static_cast<ThreadLocalObject*>(obj);
69*6dbdd20aSAndroid Build Coastguard Worker     pthread_setspecific(g_instance->tls_key_, nullptr);
70*6dbdd20aSAndroid Build Coastguard Worker   };
71*6dbdd20aSAndroid Build Coastguard Worker   PERFETTO_CHECK(pthread_key_create(&tls_key_, tls_dtor) == 0);
72*6dbdd20aSAndroid Build Coastguard Worker }
73*6dbdd20aSAndroid Build Coastguard Worker 
~PlatformPosix()74*6dbdd20aSAndroid Build Coastguard Worker PlatformPosix::~PlatformPosix() {
75*6dbdd20aSAndroid Build Coastguard Worker   // pthread_key_delete doesn't call destructors, so do it manually for the
76*6dbdd20aSAndroid Build Coastguard Worker   // calling thread.
77*6dbdd20aSAndroid Build Coastguard Worker   void* tls_ptr = pthread_getspecific(tls_key_);
78*6dbdd20aSAndroid Build Coastguard Worker   delete static_cast<ThreadLocalObject*>(tls_ptr);
79*6dbdd20aSAndroid Build Coastguard Worker 
80*6dbdd20aSAndroid Build Coastguard Worker   pthread_key_delete(tls_key_);
81*6dbdd20aSAndroid Build Coastguard Worker   g_instance = nullptr;
82*6dbdd20aSAndroid Build Coastguard Worker }
83*6dbdd20aSAndroid Build Coastguard Worker 
Shutdown()84*6dbdd20aSAndroid Build Coastguard Worker void PlatformPosix::Shutdown() {
85*6dbdd20aSAndroid Build Coastguard Worker   PERFETTO_CHECK(g_instance == this);
86*6dbdd20aSAndroid Build Coastguard Worker   delete this;
87*6dbdd20aSAndroid Build Coastguard Worker   PERFETTO_CHECK(!g_instance);
88*6dbdd20aSAndroid Build Coastguard Worker   // We're not clearing out the instance in GetDefaultPlatform() since it's not
89*6dbdd20aSAndroid Build Coastguard Worker   // possible to re-initialize Perfetto after calling this function anyway.
90*6dbdd20aSAndroid Build Coastguard Worker }
91*6dbdd20aSAndroid Build Coastguard Worker 
GetOrCreateThreadLocalObject()92*6dbdd20aSAndroid Build Coastguard Worker ThreadLocalObject* PlatformPosix::GetOrCreateThreadLocalObject() {
93*6dbdd20aSAndroid Build Coastguard Worker   // In chromium this should be implemented using base::ThreadLocalStorage.
94*6dbdd20aSAndroid Build Coastguard Worker   void* tls_ptr = pthread_getspecific(tls_key_);
95*6dbdd20aSAndroid Build Coastguard Worker 
96*6dbdd20aSAndroid Build Coastguard Worker   // This is needed to handle re-entrant calls during TLS dtor.
97*6dbdd20aSAndroid Build Coastguard Worker   // See comments in platform.cc and aosp/1712371 .
98*6dbdd20aSAndroid Build Coastguard Worker   ThreadLocalObject* tls = static_cast<ThreadLocalObject*>(tls_ptr);
99*6dbdd20aSAndroid Build Coastguard Worker   if (!tls) {
100*6dbdd20aSAndroid Build Coastguard Worker     tls = ThreadLocalObject::CreateInstance().release();
101*6dbdd20aSAndroid Build Coastguard Worker     pthread_setspecific(tls_key_, tls);
102*6dbdd20aSAndroid Build Coastguard Worker   }
103*6dbdd20aSAndroid Build Coastguard Worker   return tls;
104*6dbdd20aSAndroid Build Coastguard Worker }
105*6dbdd20aSAndroid Build Coastguard Worker 
CreateTaskRunner(const CreateTaskRunnerArgs & args)106*6dbdd20aSAndroid Build Coastguard Worker std::unique_ptr<base::TaskRunner> PlatformPosix::CreateTaskRunner(
107*6dbdd20aSAndroid Build Coastguard Worker     const CreateTaskRunnerArgs& args) {
108*6dbdd20aSAndroid Build Coastguard Worker   return std::unique_ptr<base::TaskRunner>(new base::ThreadTaskRunner(
109*6dbdd20aSAndroid Build Coastguard Worker       base::ThreadTaskRunner::CreateAndStart(args.name_for_debugging)));
110*6dbdd20aSAndroid Build Coastguard Worker }
111*6dbdd20aSAndroid Build Coastguard Worker 
GetCurrentProcessName()112*6dbdd20aSAndroid Build Coastguard Worker std::string PlatformPosix::GetCurrentProcessName() {
113*6dbdd20aSAndroid Build Coastguard Worker #if PERFETTO_BUILDFLAG(PERFETTO_OS_LINUX) || \
114*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID)
115*6dbdd20aSAndroid Build Coastguard Worker   std::string cmdline;
116*6dbdd20aSAndroid Build Coastguard Worker   base::ReadFile("/proc/self/cmdline", &cmdline);
117*6dbdd20aSAndroid Build Coastguard Worker   return cmdline.substr(0, cmdline.find('\0'));
118*6dbdd20aSAndroid Build Coastguard Worker #elif PERFETTO_BUILDFLAG(PERFETTO_OS_APPLE)
119*6dbdd20aSAndroid Build Coastguard Worker   return std::string(getprogname());
120*6dbdd20aSAndroid Build Coastguard Worker #else
121*6dbdd20aSAndroid Build Coastguard Worker   return "unknown_producer";
122*6dbdd20aSAndroid Build Coastguard Worker #endif
123*6dbdd20aSAndroid Build Coastguard Worker }
124*6dbdd20aSAndroid Build Coastguard Worker 
125*6dbdd20aSAndroid Build Coastguard Worker }  // namespace
126*6dbdd20aSAndroid Build Coastguard Worker 
127*6dbdd20aSAndroid Build Coastguard Worker // static
GetDefaultPlatform()128*6dbdd20aSAndroid Build Coastguard Worker Platform* Platform::GetDefaultPlatform() {
129*6dbdd20aSAndroid Build Coastguard Worker   static PlatformPosix* instance = new PlatformPosix();
130*6dbdd20aSAndroid Build Coastguard Worker   return instance;
131*6dbdd20aSAndroid Build Coastguard Worker }
132*6dbdd20aSAndroid Build Coastguard Worker 
133*6dbdd20aSAndroid Build Coastguard Worker }  // namespace perfetto
134*6dbdd20aSAndroid Build Coastguard Worker #endif  // OS_LINUX || OS_ANDROID || OS_APPLE || OS_FUCHSIA
135