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