xref: /aosp_15_r20/external/perfetto/src/tracing/platform_windows.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/base/build_config.h"
18*6dbdd20aSAndroid Build Coastguard Worker 
19*6dbdd20aSAndroid Build Coastguard Worker #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN)
20*6dbdd20aSAndroid Build Coastguard Worker 
21*6dbdd20aSAndroid Build Coastguard Worker #include <Windows.h>
22*6dbdd20aSAndroid Build Coastguard Worker 
23*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/thread_task_runner.h"
24*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/tracing/internal/tracing_tls.h"
25*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/tracing/platform.h"
26*6dbdd20aSAndroid Build Coastguard Worker 
27*6dbdd20aSAndroid Build Coastguard Worker // Thread Termination Callbacks.
28*6dbdd20aSAndroid Build Coastguard Worker // Windows doesn't support a per-thread destructor with its
29*6dbdd20aSAndroid Build Coastguard Worker // TLS primitives. So, we build it manually by inserting a
30*6dbdd20aSAndroid Build Coastguard Worker // function to be called on each thread's exit.
31*6dbdd20aSAndroid Build Coastguard Worker // This magic is from chromium's base/threading/thread_local_storage_win.cc
32*6dbdd20aSAndroid Build Coastguard Worker // which in turn is from http://www.codeproject.com/threads/tls.asp.
33*6dbdd20aSAndroid Build Coastguard Worker 
34*6dbdd20aSAndroid Build Coastguard Worker #ifdef _WIN64
35*6dbdd20aSAndroid Build Coastguard Worker #pragma comment(linker, "/INCLUDE:_tls_used")
36*6dbdd20aSAndroid Build Coastguard Worker #pragma comment(linker, "/INCLUDE:perfetto_thread_callback_base")
37*6dbdd20aSAndroid Build Coastguard Worker #else
38*6dbdd20aSAndroid Build Coastguard Worker #pragma comment(linker, "/INCLUDE:__tls_used")
39*6dbdd20aSAndroid Build Coastguard Worker #pragma comment(linker, "/INCLUDE:_perfetto_thread_callback_base")
40*6dbdd20aSAndroid Build Coastguard Worker #endif
41*6dbdd20aSAndroid Build Coastguard Worker 
42*6dbdd20aSAndroid Build Coastguard Worker namespace perfetto {
43*6dbdd20aSAndroid Build Coastguard Worker 
44*6dbdd20aSAndroid Build Coastguard Worker namespace {
45*6dbdd20aSAndroid Build Coastguard Worker 
46*6dbdd20aSAndroid Build Coastguard Worker class PlatformWindows : public Platform {
47*6dbdd20aSAndroid Build Coastguard Worker  public:
48*6dbdd20aSAndroid Build Coastguard Worker   static PlatformWindows* instance;
49*6dbdd20aSAndroid Build Coastguard Worker   PlatformWindows();
50*6dbdd20aSAndroid Build Coastguard Worker   ~PlatformWindows() override;
51*6dbdd20aSAndroid Build Coastguard Worker 
52*6dbdd20aSAndroid Build Coastguard Worker   ThreadLocalObject* GetOrCreateThreadLocalObject() override;
53*6dbdd20aSAndroid Build Coastguard Worker   std::unique_ptr<base::TaskRunner> CreateTaskRunner(
54*6dbdd20aSAndroid Build Coastguard Worker       const CreateTaskRunnerArgs&) override;
55*6dbdd20aSAndroid Build Coastguard Worker   std::string GetCurrentProcessName() override;
56*6dbdd20aSAndroid Build Coastguard Worker   void OnThreadExit();
57*6dbdd20aSAndroid Build Coastguard Worker 
58*6dbdd20aSAndroid Build Coastguard Worker  private:
59*6dbdd20aSAndroid Build Coastguard Worker   DWORD tls_key_{};
60*6dbdd20aSAndroid Build Coastguard Worker };
61*6dbdd20aSAndroid Build Coastguard Worker 
62*6dbdd20aSAndroid Build Coastguard Worker using ThreadLocalObject = Platform::ThreadLocalObject;
63*6dbdd20aSAndroid Build Coastguard Worker 
64*6dbdd20aSAndroid Build Coastguard Worker // static
65*6dbdd20aSAndroid Build Coastguard Worker PlatformWindows* PlatformWindows::instance = nullptr;
66*6dbdd20aSAndroid Build Coastguard Worker 
PlatformWindows()67*6dbdd20aSAndroid Build Coastguard Worker PlatformWindows::PlatformWindows() {
68*6dbdd20aSAndroid Build Coastguard Worker   instance = this;
69*6dbdd20aSAndroid Build Coastguard Worker   tls_key_ = ::TlsAlloc();
70*6dbdd20aSAndroid Build Coastguard Worker   PERFETTO_CHECK(tls_key_ != TLS_OUT_OF_INDEXES);
71*6dbdd20aSAndroid Build Coastguard Worker }
72*6dbdd20aSAndroid Build Coastguard Worker 
~PlatformWindows()73*6dbdd20aSAndroid Build Coastguard Worker PlatformWindows::~PlatformWindows() {
74*6dbdd20aSAndroid Build Coastguard Worker   ::TlsFree(tls_key_);
75*6dbdd20aSAndroid Build Coastguard Worker   instance = nullptr;
76*6dbdd20aSAndroid Build Coastguard Worker }
77*6dbdd20aSAndroid Build Coastguard Worker 
OnThreadExit()78*6dbdd20aSAndroid Build Coastguard Worker void PlatformWindows::OnThreadExit() {
79*6dbdd20aSAndroid Build Coastguard Worker   auto tls = static_cast<ThreadLocalObject*>(::TlsGetValue(tls_key_));
80*6dbdd20aSAndroid Build Coastguard Worker   if (tls) {
81*6dbdd20aSAndroid Build Coastguard Worker     // At this point we rely on the TLS object to be still set to the TracingTLS
82*6dbdd20aSAndroid Build Coastguard Worker     // we are deleting. See comments in TracingTLS::~TracingTLS().
83*6dbdd20aSAndroid Build Coastguard Worker     delete tls;
84*6dbdd20aSAndroid Build Coastguard Worker   }
85*6dbdd20aSAndroid Build Coastguard Worker }
86*6dbdd20aSAndroid Build Coastguard Worker 
GetOrCreateThreadLocalObject()87*6dbdd20aSAndroid Build Coastguard Worker ThreadLocalObject* PlatformWindows::GetOrCreateThreadLocalObject() {
88*6dbdd20aSAndroid Build Coastguard Worker   void* tls_ptr = ::TlsGetValue(tls_key_);
89*6dbdd20aSAndroid Build Coastguard Worker 
90*6dbdd20aSAndroid Build Coastguard Worker   auto* tls = static_cast<ThreadLocalObject*>(tls_ptr);
91*6dbdd20aSAndroid Build Coastguard Worker   if (!tls) {
92*6dbdd20aSAndroid Build Coastguard Worker     tls = ThreadLocalObject::CreateInstance().release();
93*6dbdd20aSAndroid Build Coastguard Worker     ::TlsSetValue(tls_key_, tls);
94*6dbdd20aSAndroid Build Coastguard Worker   }
95*6dbdd20aSAndroid Build Coastguard Worker   return tls;
96*6dbdd20aSAndroid Build Coastguard Worker }
97*6dbdd20aSAndroid Build Coastguard Worker 
CreateTaskRunner(const CreateTaskRunnerArgs & args)98*6dbdd20aSAndroid Build Coastguard Worker std::unique_ptr<base::TaskRunner> PlatformWindows::CreateTaskRunner(
99*6dbdd20aSAndroid Build Coastguard Worker     const CreateTaskRunnerArgs& args) {
100*6dbdd20aSAndroid Build Coastguard Worker   return std::unique_ptr<base::TaskRunner>(new base::ThreadTaskRunner(
101*6dbdd20aSAndroid Build Coastguard Worker       base::ThreadTaskRunner::CreateAndStart(args.name_for_debugging)));
102*6dbdd20aSAndroid Build Coastguard Worker }
103*6dbdd20aSAndroid Build Coastguard Worker 
GetCurrentProcessName()104*6dbdd20aSAndroid Build Coastguard Worker std::string PlatformWindows::GetCurrentProcessName() {
105*6dbdd20aSAndroid Build Coastguard Worker   char buf[MAX_PATH];
106*6dbdd20aSAndroid Build Coastguard Worker   auto len = ::GetModuleFileNameA(nullptr /*current*/, buf, sizeof(buf));
107*6dbdd20aSAndroid Build Coastguard Worker   std::string name(buf, static_cast<size_t>(len));
108*6dbdd20aSAndroid Build Coastguard Worker   size_t sep = name.find_last_of('\\');
109*6dbdd20aSAndroid Build Coastguard Worker   if (sep != std::string::npos)
110*6dbdd20aSAndroid Build Coastguard Worker     name = name.substr(sep + 1);
111*6dbdd20aSAndroid Build Coastguard Worker   return name;
112*6dbdd20aSAndroid Build Coastguard Worker }
113*6dbdd20aSAndroid Build Coastguard Worker 
114*6dbdd20aSAndroid Build Coastguard Worker }  // namespace
115*6dbdd20aSAndroid Build Coastguard Worker 
116*6dbdd20aSAndroid Build Coastguard Worker // static
GetDefaultPlatform()117*6dbdd20aSAndroid Build Coastguard Worker Platform* Platform::GetDefaultPlatform() {
118*6dbdd20aSAndroid Build Coastguard Worker   static PlatformWindows* thread_safe_init_instance = new PlatformWindows();
119*6dbdd20aSAndroid Build Coastguard Worker   return thread_safe_init_instance;
120*6dbdd20aSAndroid Build Coastguard Worker }
121*6dbdd20aSAndroid Build Coastguard Worker 
122*6dbdd20aSAndroid Build Coastguard Worker }  // namespace perfetto
123*6dbdd20aSAndroid Build Coastguard Worker 
124*6dbdd20aSAndroid Build Coastguard Worker // -----------------------
125*6dbdd20aSAndroid Build Coastguard Worker // Thread-local destructor
126*6dbdd20aSAndroid Build Coastguard Worker // -----------------------
127*6dbdd20aSAndroid Build Coastguard Worker 
128*6dbdd20aSAndroid Build Coastguard Worker // .CRT$XLA to .CRT$XLZ is an array of PIMAGE_TLS_CALLBACK pointers that are
129*6dbdd20aSAndroid Build Coastguard Worker // called automatically by the OS loader code (not the CRT) when the module is
130*6dbdd20aSAndroid Build Coastguard Worker // loaded and on thread creation. They are NOT called if the module has been
131*6dbdd20aSAndroid Build Coastguard Worker // loaded by a LoadLibrary() call. It must have implicitly been loaded at
132*6dbdd20aSAndroid Build Coastguard Worker // process startup.
133*6dbdd20aSAndroid Build Coastguard Worker // See VC\crt\src\tlssup.c for reference.
134*6dbdd20aSAndroid Build Coastguard Worker 
135*6dbdd20aSAndroid Build Coastguard Worker // extern "C" suppresses C++ name mangling so we know the symbol name for the
136*6dbdd20aSAndroid Build Coastguard Worker // linker /INCLUDE:symbol pragma above.
137*6dbdd20aSAndroid Build Coastguard Worker extern "C" {
138*6dbdd20aSAndroid Build Coastguard Worker // The linker must not discard perfetto_thread_callback_base. (We force a
139*6dbdd20aSAndroid Build Coastguard Worker // reference to this variable with a linker /INCLUDE:symbol pragma to ensure
140*6dbdd20aSAndroid Build Coastguard Worker // that.) If this variable is discarded, the OnThreadExit function will never be
141*6dbdd20aSAndroid Build Coastguard Worker // called.
142*6dbdd20aSAndroid Build Coastguard Worker 
143*6dbdd20aSAndroid Build Coastguard Worker void NTAPI PerfettoOnThreadExit(PVOID, DWORD, PVOID);
PerfettoOnThreadExit(PVOID,DWORD reason,PVOID)144*6dbdd20aSAndroid Build Coastguard Worker void NTAPI PerfettoOnThreadExit(PVOID, DWORD reason, PVOID) {
145*6dbdd20aSAndroid Build Coastguard Worker   if (reason == DLL_THREAD_DETACH || reason == DLL_PROCESS_DETACH) {
146*6dbdd20aSAndroid Build Coastguard Worker     if (perfetto::PlatformWindows::instance)
147*6dbdd20aSAndroid Build Coastguard Worker       perfetto::PlatformWindows::instance->OnThreadExit();
148*6dbdd20aSAndroid Build Coastguard Worker   }
149*6dbdd20aSAndroid Build Coastguard Worker }
150*6dbdd20aSAndroid Build Coastguard Worker 
151*6dbdd20aSAndroid Build Coastguard Worker #ifdef _WIN64
152*6dbdd20aSAndroid Build Coastguard Worker 
153*6dbdd20aSAndroid Build Coastguard Worker // .CRT section is merged with .rdata on x64 so it must be constant data.
154*6dbdd20aSAndroid Build Coastguard Worker #pragma const_seg(".CRT$XLP")
155*6dbdd20aSAndroid Build Coastguard Worker 
156*6dbdd20aSAndroid Build Coastguard Worker // When defining a const variable, it must have external linkage to be sure the
157*6dbdd20aSAndroid Build Coastguard Worker // linker doesn't discard it.
158*6dbdd20aSAndroid Build Coastguard Worker extern const PIMAGE_TLS_CALLBACK perfetto_thread_callback_base;
159*6dbdd20aSAndroid Build Coastguard Worker const PIMAGE_TLS_CALLBACK perfetto_thread_callback_base = PerfettoOnThreadExit;
160*6dbdd20aSAndroid Build Coastguard Worker 
161*6dbdd20aSAndroid Build Coastguard Worker // Reset the default section.
162*6dbdd20aSAndroid Build Coastguard Worker #pragma const_seg()
163*6dbdd20aSAndroid Build Coastguard Worker 
164*6dbdd20aSAndroid Build Coastguard Worker #else  // _WIN64
165*6dbdd20aSAndroid Build Coastguard Worker 
166*6dbdd20aSAndroid Build Coastguard Worker #pragma data_seg(".CRT$XLP")
167*6dbdd20aSAndroid Build Coastguard Worker PIMAGE_TLS_CALLBACK perfetto_thread_callback_base = PerfettoOnThreadExit;
168*6dbdd20aSAndroid Build Coastguard Worker // Reset the default section.
169*6dbdd20aSAndroid Build Coastguard Worker #pragma data_seg()
170*6dbdd20aSAndroid Build Coastguard Worker 
171*6dbdd20aSAndroid Build Coastguard Worker #endif  // _WIN64
172*6dbdd20aSAndroid Build Coastguard Worker 
173*6dbdd20aSAndroid Build Coastguard Worker }  // extern "C"
174*6dbdd20aSAndroid Build Coastguard Worker 
175*6dbdd20aSAndroid Build Coastguard Worker #endif  // OS_WIN
176