xref: /aosp_15_r20/external/cronet/base/trace_event/trace_event.h (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1 // Copyright 2012 The Chromium Authors
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #ifndef BASE_TRACE_EVENT_TRACE_EVENT_H_
6 #define BASE_TRACE_EVENT_TRACE_EVENT_H_
7 
8 // This header file defines implementation details of how the trace macros in
9 // trace_event_common.h collect and store trace events. Anything not
10 // implementation-specific should go in trace_event_common.h instead of here.
11 
12 #include <stddef.h>
13 #include <stdint.h>
14 
15 #include <memory>
16 #include <utility>
17 
18 #include "base/atomicops.h"
19 #include "base/base_export.h"
20 #include "base/threading/platform_thread.h"
21 #include "base/time/time.h"
22 #include "base/time/time_override.h"
23 #include "base/trace_event/builtin_categories.h"
24 #include "base/trace_event/common/trace_event_common.h"
25 #include "base/trace_event/trace_arguments.h"
26 #include "base/trace_event/trace_category.h"
27 #include "base/trace_event/trace_log.h"
28 #include "base/trace_event/traced_value_support.h"
29 #include "base/tracing_buildflags.h"
30 
31 #if !BUILDFLAG(USE_PERFETTO_CLIENT_LIBRARY)
32 
33 // By default, const char* argument values are assumed to have long-lived scope
34 // and will not be copied. Use this macro to force a const char* to be copied.
35 #define TRACE_STR_COPY(str) ::base::trace_event::TraceStringWithCopy(str)
36 
37 // By default, trace IDs are eventually converted to a single 64-bit number. Use
38 // this macro to add a scope string. For example,
39 //
40 // TRACE_EVENT_NESTABLE_ASYNC_BEGIN0(
41 //     "network", "ResourceLoad",
42 //     TRACE_ID_WITH_SCOPE("BlinkResourceID", resourceID));
43 #define TRACE_ID_WITH_SCOPE(scope, ...) \
44   trace_event_internal::TraceID::WithScope(scope, ##__VA_ARGS__)
45 
46 // Use this for ids that are unique across processes. This allows different
47 // processes to use the same id to refer to the same event.
48 #define TRACE_ID_GLOBAL(id) trace_event_internal::TraceID::GlobalId(id)
49 
50 // Use this for ids that are unique within a single process. This allows
51 // different processes to use the same id to refer to different events.
52 #define TRACE_ID_LOCAL(id) trace_event_internal::TraceID::LocalId(id)
53 
54 #define TRACE_EVENT_API_CURRENT_THREAD_ID base::PlatformThread::CurrentId()
55 
56 #define INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED_FOR_RECORDING_MODE() \
57   UNLIKELY(*INTERNAL_TRACE_EVENT_UID(category_group_enabled) &           \
58            (base::trace_event::TraceCategory::ENABLED_FOR_RECORDING |    \
59             base::trace_event::TraceCategory::ENABLED_FOR_ETW_EXPORT))
60 
61 #define INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()                  \
62   UNLIKELY(*INTERNAL_TRACE_EVENT_UID(category_group_enabled) &         \
63            (base::trace_event::TraceCategory::ENABLED_FOR_RECORDING |  \
64             base::trace_event::TraceCategory::ENABLED_FOR_ETW_EXPORT | \
65             base::trace_event::TraceCategory::ENABLED_FOR_FILTERING))
66 
67 ////////////////////////////////////////////////////////////////////////////////
68 // Implementation specific tracing API definitions.
69 
70 // Get a pointer to the enabled state of the given trace category. Only
71 // long-lived literal strings should be given as the category group. The
72 // returned pointer can be held permanently in a local static for example. If
73 // the unsigned char is non-zero, tracing is enabled. If tracing is enabled,
74 // TRACE_EVENT_API_ADD_TRACE_EVENT can be called. It's OK if tracing is disabled
75 // between the load of the tracing state and the call to
76 // TRACE_EVENT_API_ADD_TRACE_EVENT, because this flag only provides an early out
77 // for best performance when tracing is disabled.
78 // const unsigned char*
79 //     TRACE_EVENT_API_GET_CATEGORY_GROUP_ENABLED(const char* category_group)
80 #define TRACE_EVENT_API_GET_CATEGORY_GROUP_ENABLED \
81     base::trace_event::TraceLog::GetCategoryGroupEnabled
82 
83 // Get the number of times traces have been recorded. This is used to implement
84 // the TRACE_EVENT_IS_NEW_TRACE facility.
85 // unsigned int TRACE_EVENT_API_GET_NUM_TRACES_RECORDED()
86 #define TRACE_EVENT_API_GET_NUM_TRACES_RECORDED \
87   trace_event_internal::GetNumTracesRecorded
88 
89 #endif  // !BUILDFLAG(USE_PERFETTO_CLIENT_LIBRARY)
90 
91 // Legacy TRACE_EVENT_API entrypoints. Do not use from new code.
92 
93 // Add a trace event to the platform tracing system.
94 // base::trace_event::TraceEventHandle TRACE_EVENT_API_ADD_TRACE_EVENT(
95 //                    char phase,
96 //                    const unsigned char* category_group_enabled,
97 //                    const char* name,
98 //                    const char* scope,
99 //                    uint64_t id,
100 //                    base::trace_event::TraceArguments* args,
101 //                    unsigned int flags)
102 #define TRACE_EVENT_API_ADD_TRACE_EVENT trace_event_internal::AddTraceEvent
103 
104 // Add a trace event to the platform tracing system.
105 // base::trace_event::TraceEventHandle
106 // TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_BIND_ID(
107 //                    char phase,
108 //                    const unsigned char* category_group_enabled,
109 //                    const char* name,
110 //                    const char* scope,
111 //                    uint64_t id,
112 //                    uint64_t bind_id,
113 //                    base::trace_event::TraceArguments* args,
114 //                    unsigned int flags)
115 #define TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_BIND_ID \
116   trace_event_internal::AddTraceEventWithBindId
117 
118 // Add a trace event to the platform tracing system overriding the pid.
119 // The resulting event will have tid = pid == (process_id passed here).
120 // base::trace_event::TraceEventHandle
121 // TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_PROCESS_ID(
122 //                    char phase,
123 //                    const unsigned char* category_group_enabled,
124 //                    const char* name,
125 //                    const char* scope,
126 //                    uint64_t id,
127 //                    base::ProcessId process_id,
128 //                    base::trace_event::TraceArguments* args,
129 //                    unsigned int flags)
130 #define TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_PROCESS_ID \
131   trace_event_internal::AddTraceEventWithProcessId
132 
133 // Add a trace event to the platform tracing system.
134 // base::trace_event::TraceEventHandle
135 // TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_THREAD_ID_AND_TIMESTAMP(
136 //                    char phase,
137 //                    const unsigned char* category_group_enabled,
138 //                    const char* name,
139 //                    const char* scope,
140 //                    uint64_t id,
141 //                    uint64_t bind_id,
142 //                    base::PlatformThreadId thread_id,
143 //                    const TimeTicks& timestamp,
144 //                    base::trace_event::TraceArguments* args,
145 //                    unsigned int flags)
146 #define TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_THREAD_ID_AND_TIMESTAMP \
147   trace_event_internal::AddTraceEventWithThreadIdAndTimestamp
148 
149 // Set the duration field of a COMPLETE trace event.
150 // void TRACE_EVENT_API_UPDATE_TRACE_EVENT_DURATION(
151 //     const unsigned char* category_group_enabled,
152 //     const char* name,
153 //     base::trace_event::TraceEventHandle id)
154 #define TRACE_EVENT_API_UPDATE_TRACE_EVENT_DURATION \
155   trace_event_internal::UpdateTraceEventDuration
156 
157 // Adds a metadata event to the trace log. The |AppendValueAsTraceFormat| method
158 // on the convertable value will be called at flush time.
159 // TRACE_EVENT_API_ADD_METADATA_EVENT(
160 //     const unsigned char* category_group_enabled,
161 //     const char* event_name,
162 //     const char* arg_name,
163 //     std::unique_ptr<ConvertableToTraceFormat> arg_value)
164 #define TRACE_EVENT_API_ADD_METADATA_EVENT \
165     trace_event_internal::AddMetadataEvent
166 
167 // Defines atomic operations used internally by the tracing system.
168 // Acquire/release barriers are important here: crbug.com/1330114#c8.
169 #define TRACE_EVENT_API_ATOMIC_WORD base::subtle::AtomicWord
170 #define TRACE_EVENT_API_ATOMIC_LOAD(var) base::subtle::Acquire_Load(&(var))
171 #define TRACE_EVENT_API_ATOMIC_STORE(var, value) \
172   base::subtle::Release_Store(&(var), (value))
173 
174 // Defines visibility for classes in trace_event.h
175 #define TRACE_EVENT_API_CLASS_EXPORT BASE_EXPORT
176 
177 ////////////////////////////////////////////////////////////////////////////////
178 
179 // Implementation detail: trace event macros create temporary variables
180 // to keep instrumentation overhead low. These macros give each temporary
181 // variable a unique name based on the line number to prevent name collisions.
182 #if !BUILDFLAG(USE_PERFETTO_CLIENT_LIBRARY)
183 #define INTERNAL_TRACE_EVENT_UID3(a, b) trace_event_unique_##a##b
184 #define INTERNAL_TRACE_EVENT_UID2(a, b) INTERNAL_TRACE_EVENT_UID3(a, b)
185 #define INTERNAL_TRACE_EVENT_UID(name_prefix) \
186     INTERNAL_TRACE_EVENT_UID2(name_prefix, __LINE__)
187 
188 // Implementation detail: internal macro to create static category.
189 // No barriers are needed, because this code is designed to operate safely
190 // even when the unsigned char* points to garbage data (which may be the case
191 // on processors without cache coherency).
192 #define INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO_CUSTOM_VARIABLES(    \
193     category_group, atomic, category_group_enabled)                 \
194   category_group_enabled = reinterpret_cast<const unsigned char*>(  \
195       TRACE_EVENT_API_ATOMIC_LOAD(atomic));                         \
196   if (UNLIKELY(!category_group_enabled)) {                          \
197     category_group_enabled =                                        \
198         TRACE_EVENT_API_GET_CATEGORY_GROUP_ENABLED(category_group); \
199     TRACE_EVENT_API_ATOMIC_STORE(                                   \
200         atomic, reinterpret_cast<TRACE_EVENT_API_ATOMIC_WORD>(      \
201                     category_group_enabled));                       \
202   }
203 
204 #define INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO_MAYBE_AT_COMPILE_TIME(        \
205     category_group, k_category_group_enabled, category_group_enabled)        \
206   if (k_category_group_enabled) {                                            \
207     category_group_enabled = k_category_group_enabled;                       \
208   } else {                                                                   \
209     static TRACE_EVENT_API_ATOMIC_WORD INTERNAL_TRACE_EVENT_UID(atomic) = 0; \
210     INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO_CUSTOM_VARIABLES(                 \
211         category_group, INTERNAL_TRACE_EVENT_UID(atomic),                    \
212         category_group_enabled);                                             \
213   }
214 
215 #define INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group)                 \
216   static_assert(                                                               \
217       base::trace_event::BuiltinCategories::IsAllowedCategory(category_group), \
218       "Unknown tracing category is used. Please register your "                \
219       "category in base/trace_event/builtin_categories.h");                    \
220   constexpr const unsigned char* INTERNAL_TRACE_EVENT_UID(                     \
221       k_category_group_enabled) =                                              \
222       base::trace_event::TraceLog::GetBuiltinCategoryEnabled(category_group);  \
223   const unsigned char* INTERNAL_TRACE_EVENT_UID(category_group_enabled);       \
224   INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO_MAYBE_AT_COMPILE_TIME(                \
225       category_group, INTERNAL_TRACE_EVENT_UID(k_category_group_enabled),      \
226       INTERNAL_TRACE_EVENT_UID(category_group_enabled));
227 
228 // Implementation detail: internal macro to return unoverridden
229 // base::TimeTicks::Now(). This is important because in headless VirtualTime can
230 // override base:TimeTicks::Now().
231 #define INTERNAL_TRACE_TIME_TICKS_NOW() \
232   base::subtle::TimeTicksNowIgnoringOverride()
233 
234 // Implementation detail: internal macro to return unoverridden
235 // base::Time::Now(). This is important because in headless VirtualTime can
236 // override base:TimeTicks::Now().
237 #define INTERNAL_TRACE_TIME_NOW() base::subtle::TimeNowIgnoringOverride()
238 
239 // Implementation detail: internal macro to create static category and add
240 // event if the category is enabled.
241 #define INTERNAL_TRACE_EVENT_ADD(phase, category_group, name, flags, ...)  \
242   do {                                                                     \
243     INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group);                \
244     if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) {                   \
245       trace_event_internal::AddTraceEvent(                                 \
246           phase, INTERNAL_TRACE_EVENT_UID(category_group_enabled), name,   \
247           trace_event_internal::kGlobalScope, trace_event_internal::kNoId, \
248           flags, trace_event_internal::kNoId, ##__VA_ARGS__);              \
249     }                                                                      \
250   } while (0)
251 
252 // Implementation detail: internal macro to create static category and add begin
253 // event if the category is enabled. Also adds the end event when the scope
254 // ends.
255 #define INTERNAL_TRACE_EVENT_ADD_SCOPED(category_group, name, ...)           \
256   INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group);                    \
257   trace_event_internal::ScopedTracer INTERNAL_TRACE_EVENT_UID(tracer);       \
258   if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) {                       \
259     base::trace_event::TraceEventHandle INTERNAL_TRACE_EVENT_UID(h) =        \
260         trace_event_internal::AddTraceEvent(                                 \
261             TRACE_EVENT_PHASE_COMPLETE,                                      \
262             INTERNAL_TRACE_EVENT_UID(category_group_enabled), name,          \
263             trace_event_internal::kGlobalScope, trace_event_internal::kNoId, \
264             TRACE_EVENT_FLAG_NONE, trace_event_internal::kNoId,              \
265             ##__VA_ARGS__);                                                  \
266     INTERNAL_TRACE_EVENT_UID(tracer).Initialize(                             \
267         INTERNAL_TRACE_EVENT_UID(category_group_enabled), name,              \
268         INTERNAL_TRACE_EVENT_UID(h));                                        \
269   }
270 
271 #define INTERNAL_TRACE_EVENT_ADD_SCOPED_WITH_FLAGS(category_group, name,     \
272                                                    flags, ...)               \
273   INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group);                    \
274   trace_event_internal::ScopedTracer INTERNAL_TRACE_EVENT_UID(tracer);       \
275   if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) {                       \
276     base::trace_event::TraceEventHandle INTERNAL_TRACE_EVENT_UID(h) =        \
277         trace_event_internal::AddTraceEvent(                                 \
278             TRACE_EVENT_PHASE_COMPLETE,                                      \
279             INTERNAL_TRACE_EVENT_UID(category_group_enabled), name,          \
280             trace_event_internal::kGlobalScope, trace_event_internal::kNoId, \
281             flags, trace_event_internal::kNoId, ##__VA_ARGS__);              \
282     INTERNAL_TRACE_EVENT_UID(tracer).Initialize(                             \
283         INTERNAL_TRACE_EVENT_UID(category_group_enabled), name,              \
284         INTERNAL_TRACE_EVENT_UID(h));                                        \
285   }
286 
287 #define INTERNAL_TRACE_EVENT_ADD_SCOPED_WITH_FLOW(category_group, name,      \
288                                                   bind_id, flow_flags, ...)  \
289   INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group);                    \
290   trace_event_internal::ScopedTracer INTERNAL_TRACE_EVENT_UID(tracer);       \
291   if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) {                       \
292     trace_event_internal::TraceID trace_event_bind_id((bind_id));            \
293     unsigned int trace_event_flags =                                         \
294         flow_flags | trace_event_bind_id.id_flags();                         \
295     base::trace_event::TraceEventHandle INTERNAL_TRACE_EVENT_UID(h) =        \
296         trace_event_internal::AddTraceEvent(                                 \
297             TRACE_EVENT_PHASE_COMPLETE,                                      \
298             INTERNAL_TRACE_EVENT_UID(category_group_enabled), name,          \
299             trace_event_internal::kGlobalScope, trace_event_internal::kNoId, \
300             trace_event_flags, trace_event_bind_id.raw_id(), ##__VA_ARGS__); \
301     INTERNAL_TRACE_EVENT_UID(tracer).Initialize(                             \
302         INTERNAL_TRACE_EVENT_UID(category_group_enabled), name,              \
303         INTERNAL_TRACE_EVENT_UID(h));                                        \
304   }
305 
306 // Implementation detail: internal macro to create static category and add
307 // event if the category is enabled.
308 #define INTERNAL_TRACE_EVENT_ADD_WITH_ID(phase, category_group, name, id, \
309                                          flags, ...)                      \
310   do {                                                                    \
311     INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group);               \
312     if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) {                  \
313       trace_event_internal::TraceID trace_event_trace_id((id));           \
314       unsigned int trace_event_flags =                                    \
315           flags | trace_event_trace_id.id_flags();                        \
316       trace_event_internal::AddTraceEvent(                                \
317           phase, INTERNAL_TRACE_EVENT_UID(category_group_enabled), name,  \
318           trace_event_trace_id.scope(), trace_event_trace_id.raw_id(),    \
319           trace_event_flags, trace_event_internal::kNoId, ##__VA_ARGS__); \
320     }                                                                     \
321   } while (0)
322 
323 // Implementation detail: internal macro to create static category and add
324 // event if the category is enabled.
325 #define INTERNAL_TRACE_EVENT_ADD_WITH_TIMESTAMP(phase, category_group, name, \
326                                                 timestamp, flags, ...)       \
327   do {                                                                       \
328     INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group);                  \
329     if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) {                     \
330       trace_event_internal::AddTraceEventWithThreadIdAndTimestamp(           \
331           phase, INTERNAL_TRACE_EVENT_UID(category_group_enabled), name,     \
332           trace_event_internal::kGlobalScope, trace_event_internal::kNoId,   \
333           TRACE_EVENT_API_CURRENT_THREAD_ID, timestamp,                      \
334           flags | TRACE_EVENT_FLAG_EXPLICIT_TIMESTAMP,                       \
335           trace_event_internal::kNoId, ##__VA_ARGS__);                       \
336     }                                                                        \
337   } while (0)
338 
339 // Implementation detail: internal macro to create static category and add
340 // event if the category is enabled.
341 #define INTERNAL_TRACE_EVENT_ADD_WITH_ID_TID_AND_TIMESTAMP(              \
342     phase, category_group, name, id, thread_id, timestamp, flags, ...)   \
343   do {                                                                   \
344     INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group);              \
345     if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) {                 \
346       trace_event_internal::TraceID trace_event_trace_id((id));          \
347       unsigned int trace_event_flags =                                   \
348           flags | trace_event_trace_id.id_flags();                       \
349       trace_event_internal::AddTraceEventWithThreadIdAndTimestamp(       \
350           phase, INTERNAL_TRACE_EVENT_UID(category_group_enabled), name, \
351           trace_event_trace_id.scope(), trace_event_trace_id.raw_id(),   \
352           thread_id, timestamp,                                          \
353           trace_event_flags | TRACE_EVENT_FLAG_EXPLICIT_TIMESTAMP,       \
354           trace_event_internal::kNoId, ##__VA_ARGS__);                   \
355     }                                                                    \
356   } while (0)
357 
358 // Implementation detail: internal macro to create static category and add
359 // event if the category is enabled.
360 #define INTERNAL_TRACE_EVENT_ADD_WITH_ID_TID_AND_TIMESTAMPS(                 \
361     phase, category_group, name, id, thread_id, timestamp, thread_timestamp, \
362     flags)                                                                   \
363   do {                                                                       \
364     INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group);                  \
365     if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) {                     \
366       trace_event_internal::TraceID trace_event_trace_id((id));              \
367       unsigned int trace_event_flags =                                       \
368           flags | trace_event_trace_id.id_flags();                           \
369       const unsigned char* uid_category_group_enabled =                      \
370           INTERNAL_TRACE_EVENT_UID(category_group_enabled);                  \
371       trace_event_internal::AddTraceEventWithThreadIdAndTimestamps(          \
372           phase, uid_category_group_enabled, name,                           \
373           trace_event_trace_id.scope(), trace_event_trace_id.raw_id(),       \
374           thread_id, timestamp, thread_timestamp,                            \
375           trace_event_flags | TRACE_EVENT_FLAG_EXPLICIT_TIMESTAMP);          \
376     }                                                                        \
377   } while (0)
378 
379 // Implementation detail: internal macro to create static category and add
380 // metadata event if the category is enabled.
381 #define INTERNAL_TRACE_EVENT_METADATA_ADD(category_group, name, ...) \
382   do {                                                               \
383     INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group);          \
384     if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) {             \
385       TRACE_EVENT_API_ADD_METADATA_EVENT(                            \
386           INTERNAL_TRACE_EVENT_UID(category_group_enabled), name,    \
387           ##__VA_ARGS__);                                            \
388     }                                                                \
389   } while (0)
390 #endif  // BUILDFLAG(USE_PERFETTO_CLIENT_LIBRARY)
391 
392 namespace trace_event_internal {
393 
394 // Specify these values when the corresponding argument of AddTraceEvent is not
395 // used.
396 const int kZeroNumArgs = 0;
397 const std::nullptr_t kGlobalScope = nullptr;
398 const uint64_t kNoId = 0;
399 
400 // TraceID encapsulates an ID that can either be an integer or pointer.
401 class BASE_EXPORT TraceID {
402  public:
403   // Can be combined with WithScope.
404   class LocalId {
405    public:
LocalId(const void * raw_id)406     explicit LocalId(const void* raw_id)
407         : raw_id_(static_cast<uint64_t>(reinterpret_cast<uintptr_t>(raw_id))) {}
LocalId(uint64_t raw_id)408     explicit LocalId(uint64_t raw_id) : raw_id_(raw_id) {}
raw_id()409     uint64_t raw_id() const { return raw_id_; }
410 
411    private:
412     uint64_t raw_id_;
413   };
414 
415   // Can be combined with WithScope.
416   class GlobalId {
417    public:
GlobalId(uint64_t raw_id)418     explicit GlobalId(uint64_t raw_id) : raw_id_(raw_id) {}
raw_id()419     uint64_t raw_id() const { return raw_id_; }
420 
421    private:
422     uint64_t raw_id_;
423   };
424 
425   class WithScope {
426    public:
WithScope(const char * scope,uint64_t raw_id)427     WithScope(const char* scope, uint64_t raw_id)
428         : scope_(scope), raw_id_(raw_id) {}
WithScope(const char * scope,LocalId local_id)429     WithScope(const char* scope, LocalId local_id)
430         : scope_(scope), raw_id_(local_id.raw_id()) {
431       id_flags_ = TRACE_EVENT_FLAG_HAS_LOCAL_ID;
432     }
WithScope(const char * scope,GlobalId global_id)433     WithScope(const char* scope, GlobalId global_id)
434         : scope_(scope), raw_id_(global_id.raw_id()) {
435       id_flags_ = TRACE_EVENT_FLAG_HAS_GLOBAL_ID;
436     }
raw_id()437     uint64_t raw_id() const { return raw_id_; }
scope()438     const char* scope() const { return scope_; }
id_flags()439     unsigned int id_flags() const { return id_flags_; }
440 
441    private:
442     const char* scope_ = nullptr;
443     uint64_t raw_id_;
444     unsigned int id_flags_ = TRACE_EVENT_FLAG_HAS_ID;
445   };
446 
TraceID(const void * raw_id)447   explicit TraceID(const void* raw_id)
448       : raw_id_(static_cast<uint64_t>(reinterpret_cast<uintptr_t>(raw_id))) {
449     id_flags_ = TRACE_EVENT_FLAG_HAS_LOCAL_ID;
450   }
TraceID(unsigned long long raw_id)451   explicit TraceID(unsigned long long raw_id)
452       : raw_id_(static_cast<uint64_t>(raw_id)) {}
TraceID(unsigned long raw_id)453   explicit TraceID(unsigned long raw_id)
454       : raw_id_(static_cast<uint64_t>(raw_id)) {}
TraceID(unsigned int raw_id)455   explicit TraceID(unsigned int raw_id)
456       : raw_id_(static_cast<uint64_t>(raw_id)) {}
TraceID(unsigned short raw_id)457   explicit TraceID(unsigned short raw_id)
458       : raw_id_(static_cast<uint64_t>(raw_id)) {}
TraceID(unsigned char raw_id)459   explicit TraceID(unsigned char raw_id)
460       : raw_id_(static_cast<uint64_t>(raw_id)) {}
TraceID(long long raw_id)461   explicit TraceID(long long raw_id) : raw_id_(static_cast<uint64_t>(raw_id)) {}
TraceID(long raw_id)462   explicit TraceID(long raw_id) : raw_id_(static_cast<uint64_t>(raw_id)) {}
TraceID(int raw_id)463   explicit TraceID(int raw_id) : raw_id_(static_cast<uint64_t>(raw_id)) {}
TraceID(short raw_id)464   explicit TraceID(short raw_id) : raw_id_(static_cast<uint64_t>(raw_id)) {}
TraceID(signed char raw_id)465   explicit TraceID(signed char raw_id)
466       : raw_id_(static_cast<uint64_t>(raw_id)) {}
TraceID(LocalId raw_id)467   explicit TraceID(LocalId raw_id) : raw_id_(raw_id.raw_id()) {
468     id_flags_ = TRACE_EVENT_FLAG_HAS_LOCAL_ID;
469   }
TraceID(GlobalId raw_id)470   explicit TraceID(GlobalId raw_id) : raw_id_(raw_id.raw_id()) {
471     id_flags_ = TRACE_EVENT_FLAG_HAS_GLOBAL_ID;
472   }
TraceID(WithScope scoped_id)473   explicit TraceID(WithScope scoped_id)
474       : scope_(scoped_id.scope()),
475         raw_id_(scoped_id.raw_id()),
476         id_flags_(scoped_id.id_flags()) {}
477 
raw_id()478   uint64_t raw_id() const { return raw_id_; }
scope()479   const char* scope() const { return scope_; }
id_flags()480   unsigned int id_flags() const { return id_flags_; }
481 
482  private:
483   const char* scope_ = nullptr;
484   uint64_t raw_id_;
485   unsigned int id_flags_ = TRACE_EVENT_FLAG_HAS_ID;
486 };
487 
488 // These functions all internally call
489 // base::trace_event::TraceLog::GetInstance() then call the method with the same
490 // name on it. This is used to reduce the generated machine code at each
491 // TRACE_EVENTXXX macro call.
492 
493 base::trace_event::TraceEventHandle BASE_EXPORT
494 AddTraceEvent(char phase,
495               const unsigned char* category_group_enabled,
496               const char* name,
497               const char* scope,
498               uint64_t id,
499               base::trace_event::TraceArguments* args,
500               unsigned int flags);
501 
502 base::trace_event::TraceEventHandle BASE_EXPORT
503 AddTraceEventWithBindId(char phase,
504                         const unsigned char* category_group_enabled,
505                         const char* name,
506                         const char* scope,
507                         uint64_t id,
508                         uint64_t bind_id,
509                         base::trace_event::TraceArguments* args,
510                         unsigned int flags);
511 
512 base::trace_event::TraceEventHandle BASE_EXPORT
513 AddTraceEventWithProcessId(char phase,
514                            const unsigned char* category_group_enabled,
515                            const char* name,
516                            const char* scope,
517                            uint64_t id,
518                            base::ProcessId process_id,
519                            base::trace_event::TraceArguments* args,
520                            unsigned int flags);
521 
522 base::trace_event::TraceEventHandle BASE_EXPORT
523 AddTraceEventWithThreadIdAndTimestamp(
524     char phase,
525     const unsigned char* category_group_enabled,
526     const char* name,
527     const char* scope,
528     uint64_t id,
529     base::PlatformThreadId thread_id,
530     const base::TimeTicks& timestamp,
531     base::trace_event::TraceArguments* args,
532     unsigned int flags);
533 
534 base::trace_event::TraceEventHandle BASE_EXPORT
535 AddTraceEventWithThreadIdAndTimestamp(
536     char phase,
537     const unsigned char* category_group_enabled,
538     const char* name,
539     const char* scope,
540     uint64_t id,
541     uint64_t bind_id,
542     base::PlatformThreadId thread_id,
543     const base::TimeTicks& timestamp,
544     base::trace_event::TraceArguments* args,
545     unsigned int flags);
546 
547 base::trace_event::TraceEventHandle BASE_EXPORT
548 AddTraceEventWithThreadIdAndTimestamps(
549     char phase,
550     const unsigned char* category_group_enabled,
551     const char* name,
552     const char* scope,
553     uint64_t id,
554     base::PlatformThreadId thread_id,
555     const base::TimeTicks& timestamp,
556     const base::ThreadTicks& thread_timestamp,
557     unsigned int flags);
558 
559 void BASE_EXPORT AddMetadataEvent(const unsigned char* category_group_enabled,
560                                   const char* name,
561                                   base::trace_event::TraceArguments* args,
562                                   unsigned int flags);
563 
564 int BASE_EXPORT GetNumTracesRecorded();
565 
566 void BASE_EXPORT
567 UpdateTraceEventDuration(const unsigned char* category_group_enabled,
568                          const char* name,
569                          base::trace_event::TraceEventHandle handle);
570 
571 void BASE_EXPORT
572 UpdateTraceEventDurationExplicit(const unsigned char* category_group_enabled,
573                                  const char* name,
574                                  base::trace_event::TraceEventHandle handle,
575                                  base::PlatformThreadId thread_id,
576                                  bool explicit_timestamps,
577                                  const base::TimeTicks& now,
578                                  const base::ThreadTicks& thread_now);
579 
580 // These AddTraceEvent and AddTraceEventWithThreadIdAndTimestamp template
581 // functions are defined here instead of in the macro, because the arg_values
582 // could be temporary objects, such as `std::string`. In order to store
583 // pointers to the internal c_str and pass through to the tracing API,
584 // the arg_values must live throughout these procedures.
585 
586 template <class ARG1_TYPE>
587 inline base::trace_event::TraceEventHandle
AddTraceEventWithThreadIdAndTimestamp(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,uint64_t id,base::PlatformThreadId thread_id,const base::TimeTicks & timestamp,unsigned int flags,uint64_t bind_id,const char * arg1_name,ARG1_TYPE && arg1_val)588 AddTraceEventWithThreadIdAndTimestamp(
589     char phase,
590     const unsigned char* category_group_enabled,
591     const char* name,
592     const char* scope,
593     uint64_t id,
594     base::PlatformThreadId thread_id,
595     const base::TimeTicks& timestamp,
596     unsigned int flags,
597     uint64_t bind_id,
598     const char* arg1_name,
599     ARG1_TYPE&& arg1_val) {
600   base::trace_event::TraceArguments args(arg1_name,
601                                          std::forward<ARG1_TYPE>(arg1_val));
602   return TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_THREAD_ID_AND_TIMESTAMP(
603       phase, category_group_enabled, name, scope, id, bind_id, thread_id,
604       timestamp, &args, flags);
605 }
606 
607 template <class ARG1_TYPE, class ARG2_TYPE>
608 inline base::trace_event::TraceEventHandle
AddTraceEventWithThreadIdAndTimestamp(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,uint64_t id,base::PlatformThreadId thread_id,const base::TimeTicks & timestamp,unsigned int flags,uint64_t bind_id,const char * arg1_name,ARG1_TYPE && arg1_val,const char * arg2_name,ARG2_TYPE && arg2_val)609 AddTraceEventWithThreadIdAndTimestamp(
610     char phase,
611     const unsigned char* category_group_enabled,
612     const char* name,
613     const char* scope,
614     uint64_t id,
615     base::PlatformThreadId thread_id,
616     const base::TimeTicks& timestamp,
617     unsigned int flags,
618     uint64_t bind_id,
619     const char* arg1_name,
620     ARG1_TYPE&& arg1_val,
621     const char* arg2_name,
622     ARG2_TYPE&& arg2_val) {
623   base::trace_event::TraceArguments args(
624       arg1_name, std::forward<ARG1_TYPE>(arg1_val), arg2_name,
625       std::forward<ARG2_TYPE>(arg2_val));
626   return TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_THREAD_ID_AND_TIMESTAMP(
627       phase, category_group_enabled, name, scope, id, bind_id, thread_id,
628       timestamp, &args, flags);
629 }
630 
631 inline base::trace_event::TraceEventHandle
AddTraceEventWithThreadIdAndTimestamp(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,uint64_t id,base::PlatformThreadId thread_id,const base::TimeTicks & timestamp,unsigned int flags,uint64_t bind_id)632 AddTraceEventWithThreadIdAndTimestamp(
633     char phase,
634     const unsigned char* category_group_enabled,
635     const char* name,
636     const char* scope,
637     uint64_t id,
638     base::PlatformThreadId thread_id,
639     const base::TimeTicks& timestamp,
640     unsigned int flags,
641     uint64_t bind_id) {
642   return TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_THREAD_ID_AND_TIMESTAMP(
643       phase, category_group_enabled, name, scope, id, bind_id, thread_id,
644       timestamp, nullptr, flags);
645 }
646 
AddTraceEvent(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,uint64_t id,unsigned int flags,uint64_t bind_id)647 inline base::trace_event::TraceEventHandle AddTraceEvent(
648     char phase,
649     const unsigned char* category_group_enabled,
650     const char* name,
651     const char* scope,
652     uint64_t id,
653     unsigned int flags,
654     uint64_t bind_id) {
655   return AddTraceEventWithThreadIdAndTimestamp(
656       phase, category_group_enabled, name, scope, id,
657       TRACE_EVENT_API_CURRENT_THREAD_ID, TRACE_TIME_TICKS_NOW(), flags,
658       bind_id);
659 }
660 
661 template <class ARG1_TYPE>
AddTraceEvent(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,uint64_t id,unsigned int flags,uint64_t bind_id,const char * arg1_name,ARG1_TYPE && arg1_val)662 inline base::trace_event::TraceEventHandle AddTraceEvent(
663     char phase,
664     const unsigned char* category_group_enabled,
665     const char* name,
666     const char* scope,
667     uint64_t id,
668     unsigned int flags,
669     uint64_t bind_id,
670     const char* arg1_name,
671     ARG1_TYPE&& arg1_val) {
672   return AddTraceEventWithThreadIdAndTimestamp(
673       phase, category_group_enabled, name, scope, id,
674       TRACE_EVENT_API_CURRENT_THREAD_ID, TRACE_TIME_TICKS_NOW(), flags, bind_id,
675       arg1_name, std::forward<ARG1_TYPE>(arg1_val));
676 }
677 
678 template <class ARG1_TYPE, class ARG2_TYPE>
AddTraceEvent(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,uint64_t id,unsigned int flags,uint64_t bind_id,const char * arg1_name,ARG1_TYPE && arg1_val,const char * arg2_name,ARG2_TYPE && arg2_val)679 inline base::trace_event::TraceEventHandle AddTraceEvent(
680     char phase,
681     const unsigned char* category_group_enabled,
682     const char* name,
683     const char* scope,
684     uint64_t id,
685     unsigned int flags,
686     uint64_t bind_id,
687     const char* arg1_name,
688     ARG1_TYPE&& arg1_val,
689     const char* arg2_name,
690     ARG2_TYPE&& arg2_val) {
691   return AddTraceEventWithThreadIdAndTimestamp(
692       phase, category_group_enabled, name, scope, id,
693       TRACE_EVENT_API_CURRENT_THREAD_ID, TRACE_TIME_TICKS_NOW(), flags, bind_id,
694       arg1_name, std::forward<ARG1_TYPE>(arg1_val), arg2_name,
695       std::forward<ARG2_TYPE>(arg2_val));
696 }
697 
698 template <class ARG1_TYPE>
AddMetadataEvent(const unsigned char * category_group_enabled,const char * event_name,const char * arg_name,ARG1_TYPE && arg_val)699 void AddMetadataEvent(const unsigned char* category_group_enabled,
700                       const char* event_name,
701                       const char* arg_name,
702                       ARG1_TYPE&& arg_val) {
703   base::trace_event::TraceArguments args(arg_name,
704                                          std::forward<ARG1_TYPE>(arg_val));
705   trace_event_internal::AddMetadataEvent(category_group_enabled, event_name,
706                                          &args, TRACE_EVENT_FLAG_NONE);
707 }
708 
709 #if !BUILDFLAG(USE_PERFETTO_CLIENT_LIBRARY)
710 
711 // Used by TRACE_EVENTx macros. Do not use directly.
712 class TRACE_EVENT_API_CLASS_EXPORT ScopedTracer {
713  public:
714   ScopedTracer() = default;
715 
~ScopedTracer()716   ~ScopedTracer() {
717     if (category_group_enabled_ && *category_group_enabled_) {
718       TRACE_EVENT_API_UPDATE_TRACE_EVENT_DURATION(category_group_enabled_,
719                                                   name_, event_handle_);
720     }
721   }
722 
Initialize(const unsigned char * category_group_enabled,const char * name,base::trace_event::TraceEventHandle event_handle)723   void Initialize(const unsigned char* category_group_enabled,
724                   const char* name,
725                   base::trace_event::TraceEventHandle event_handle) {
726     category_group_enabled_ = category_group_enabled;
727     name_ = name;
728     event_handle_ = event_handle;
729   }
730 
731  private:
732   // NOTE: Only initialize the first member to reduce generated code size,
733   // since there is no point in initializing the other members if Initialize()
734   // is never called.
735   const unsigned char* category_group_enabled_ = nullptr;
736   const char* name_;
737   base::trace_event::TraceEventHandle event_handle_;
738 };
739 
740 // Used by TRACE_EVENT_BINARY_EFFICIENTx macro. Do not use directly.
741 class TRACE_EVENT_API_CLASS_EXPORT ScopedTraceBinaryEfficient {
742  public:
743   ScopedTraceBinaryEfficient(const char* category_group, const char* name);
744   ~ScopedTraceBinaryEfficient();
745 
746  private:
747   const unsigned char* category_group_enabled_;
748   const char* name_;
749   base::trace_event::TraceEventHandle event_handle_;
750 };
751 
752 // This macro generates less code then TRACE_EVENT0 but is also
753 // slower to execute when tracing is off. It should generally only be
754 // used with code that is seldom executed or conditionally executed
755 // when debugging.
756 // For now the category_group must be "gpu".
757 #define TRACE_EVENT_BINARY_EFFICIENT0(category_group, name) \
758     trace_event_internal::ScopedTraceBinaryEfficient \
759         INTERNAL_TRACE_EVENT_UID(scoped_trace)(category_group, name);
760 
761 #endif  // !BUILDFLAG(USE_PERFETTO_CLIENT_LIBRARY)
762 
763 }  // namespace trace_event_internal
764 
765 namespace base {
766 namespace trace_event {
767 
768 template <typename IDType, const char* category>
769 class TraceScopedTrackableObject {
770  public:
TraceScopedTrackableObject(const char * name,IDType id)771   TraceScopedTrackableObject(const char* name, IDType id)
772       : name_(name), id_(id) {
773     TRACE_EVENT_OBJECT_CREATED_WITH_ID(category, name_, id_);
774   }
775   TraceScopedTrackableObject(const TraceScopedTrackableObject&) = delete;
776   TraceScopedTrackableObject& operator=(const TraceScopedTrackableObject&) =
777       delete;
778 
snapshot(ArgType snapshot)779   template <typename ArgType> void snapshot(ArgType snapshot) {
780     TRACE_EVENT_OBJECT_SNAPSHOT_WITH_ID(category, name_, id_, snapshot);
781   }
782 
~TraceScopedTrackableObject()783   ~TraceScopedTrackableObject() {
784     TRACE_EVENT_OBJECT_DELETED_WITH_ID(category, name_, id_);
785   }
786 
787  private:
788   const char* name_;
789   IDType id_;
790 };
791 
792 }  // namespace trace_event
793 }  // namespace base
794 
795 #endif  // BASE_TRACE_EVENT_TRACE_EVENT_H_
796