1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
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 <string>
16
17 #include "base/atomicops.h"
18 #include "base/macros.h"
19 #include "base/time/time.h"
20 #include "base/trace_event/common/trace_event_common.h"
21 #include "base/trace_event/heap_profiler.h"
22 #include "base/trace_event/trace_category.h"
23 #include "base/trace_event/trace_event_system_stats_monitor.h"
24 #include "base/trace_event/trace_log.h"
25 #include "build/build_config.h"
26
27 // By default, const char* argument values are assumed to have long-lived scope
28 // and will not be copied. Use this macro to force a const char* to be copied.
29 #define TRACE_STR_COPY(str) \
30 trace_event_internal::TraceStringWithCopy(str)
31
32 // DEPRECATED: do not use: Consider using TRACE_ID_{GLOBAL, LOCAL} macros,
33 // instead. By default, uint64_t ID argument values are not mangled with the
34 // Process ID in TRACE_EVENT_ASYNC macros. Use this macro to force Process ID
35 // mangling.
36 #define TRACE_ID_MANGLE(id) \
37 trace_event_internal::TraceID::ForceMangle(id)
38
39 // DEPRECATED: do not use: Consider using TRACE_ID_{GLOBAL, LOCAL} macros,
40 // instead. By default, pointers are mangled with the Process ID in
41 // TRACE_EVENT_ASYNC macros. Use this macro to prevent Process ID mangling.
42 #define TRACE_ID_DONT_MANGLE(id) \
43 trace_event_internal::TraceID::DontMangle(id)
44
45 // By default, trace IDs are eventually converted to a single 64-bit number. Use
46 // this macro to add a scope string. For example,
47 //
48 // TRACE_EVENT_NESTABLE_ASYNC_BEGIN0(
49 // "network", "ResourceLoad",
50 // TRACE_ID_WITH_SCOPE("BlinkResourceID", resourceID));
51 //
52 // Also, it is possible to prepend the ID with another number, like the process
53 // ID. This is useful in creatin IDs that are unique among all processes. To do
54 // that, pass two numbers after the scope string instead of one. For example,
55 //
56 // TRACE_EVENT_NESTABLE_ASYNC_BEGIN0(
57 // "network", "ResourceLoad",
58 // TRACE_ID_WITH_SCOPE("BlinkResourceID", pid, resourceID));
59 #define TRACE_ID_WITH_SCOPE(scope, ...) \
60 trace_event_internal::TraceID::WithScope(scope, ##__VA_ARGS__)
61
62 #define TRACE_ID_GLOBAL(id) trace_event_internal::TraceID::GlobalId(id)
63 #define TRACE_ID_LOCAL(id) trace_event_internal::TraceID::LocalId(id)
64
65 #define TRACE_EVENT_API_CURRENT_THREAD_ID \
66 static_cast<int>(base::PlatformThread::CurrentId())
67
68 #define INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED_FOR_RECORDING_MODE() \
69 UNLIKELY(*INTERNAL_TRACE_EVENT_UID(category_group_enabled) & \
70 (base::trace_event::TraceCategory::ENABLED_FOR_RECORDING | \
71 base::trace_event::TraceCategory::ENABLED_FOR_ETW_EXPORT))
72
73 #define INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED() \
74 UNLIKELY(*INTERNAL_TRACE_EVENT_UID(category_group_enabled) & \
75 (base::trace_event::TraceCategory::ENABLED_FOR_RECORDING | \
76 base::trace_event::TraceCategory::ENABLED_FOR_ETW_EXPORT | \
77 base::trace_event::TraceCategory::ENABLED_FOR_FILTERING))
78
79 ////////////////////////////////////////////////////////////////////////////////
80 // Implementation specific tracing API definitions.
81
82 // Get a pointer to the enabled state of the given trace category. Only
83 // long-lived literal strings should be given as the category group. The
84 // returned pointer can be held permanently in a local static for example. If
85 // the unsigned char is non-zero, tracing is enabled. If tracing is enabled,
86 // TRACE_EVENT_API_ADD_TRACE_EVENT can be called. It's OK if tracing is disabled
87 // between the load of the tracing state and the call to
88 // TRACE_EVENT_API_ADD_TRACE_EVENT, because this flag only provides an early out
89 // for best performance when tracing is disabled.
90 // const unsigned char*
91 // TRACE_EVENT_API_GET_CATEGORY_GROUP_ENABLED(const char* category_group)
92 #define TRACE_EVENT_API_GET_CATEGORY_GROUP_ENABLED \
93 base::trace_event::TraceLog::GetCategoryGroupEnabled
94
95 // Get the number of times traces have been recorded. This is used to implement
96 // the TRACE_EVENT_IS_NEW_TRACE facility.
97 // unsigned int TRACE_EVENT_API_GET_NUM_TRACES_RECORDED()
98 #define TRACE_EVENT_API_GET_NUM_TRACES_RECORDED \
99 base::trace_event::TraceLog::GetInstance()->GetNumTracesRecorded
100
101 // Add a trace event to the platform tracing system.
102 // base::trace_event::TraceEventHandle TRACE_EVENT_API_ADD_TRACE_EVENT(
103 // char phase,
104 // const unsigned char* category_group_enabled,
105 // const char* name,
106 // const char* scope,
107 // unsigned long long id,
108 // int num_args,
109 // const char** arg_names,
110 // const unsigned char* arg_types,
111 // const unsigned long long* arg_values,
112 // std::unique_ptr<ConvertableToTraceFormat>*
113 // convertable_values,
114 // unsigned int flags)
115 #define TRACE_EVENT_API_ADD_TRACE_EVENT \
116 base::trace_event::TraceLog::GetInstance()->AddTraceEvent
117
118 // Add a trace event to the platform tracing system.
119 // base::trace_event::TraceEventHandle
120 // TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_BIND_ID(
121 // char phase,
122 // const unsigned char* category_group_enabled,
123 // const char* name,
124 // const char* scope,
125 // unsigned long long id,
126 // unsigned long long bind_id,
127 // int num_args,
128 // const char** arg_names,
129 // const unsigned char* arg_types,
130 // const unsigned long long* arg_values,
131 // std::unique_ptr<ConvertableToTraceFormat>*
132 // convertable_values,
133 // unsigned int flags)
134 #define TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_BIND_ID \
135 base::trace_event::TraceLog::GetInstance()->AddTraceEventWithBindId
136
137 // Add a trace event to the platform tracing system overriding the pid.
138 // The resulting event will have tid = pid == (process_id passed here).
139 // base::trace_event::TraceEventHandle
140 // TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_PROCESS_ID(
141 // char phase,
142 // const unsigned char* category_group_enabled,
143 // const char* name,
144 // const char* scope,
145 // unsigned long long id,
146 // int process_id,
147 // int num_args,
148 // const char** arg_names,
149 // const unsigned char* arg_types,
150 // const unsigned long long* arg_values,
151 // std::unique_ptr<ConvertableToTraceFormat>*
152 // convertable_values,
153 // unsigned int flags)
154 #define TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_PROCESS_ID \
155 base::trace_event::TraceLog::GetInstance()->AddTraceEventWithProcessId
156
157 // Add a trace event to the platform tracing system.
158 // base::trace_event::TraceEventHandle
159 // TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_TIMESTAMP(
160 // char phase,
161 // const unsigned char* category_group_enabled,
162 // const char* name,
163 // const char* scope,
164 // unsigned long long id,
165 // int thread_id,
166 // const TimeTicks& timestamp,
167 // int num_args,
168 // const char** arg_names,
169 // const unsigned char* arg_types,
170 // const unsigned long long* arg_values,
171 // std::unique_ptr<ConvertableToTraceFormat>*
172 // convertable_values,
173 // unsigned int flags)
174 #define TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_THREAD_ID_AND_TIMESTAMP \
175 base::trace_event::TraceLog::GetInstance() \
176 ->AddTraceEventWithThreadIdAndTimestamp
177
178 // Set the duration field of a COMPLETE trace event.
179 // void TRACE_EVENT_API_UPDATE_TRACE_EVENT_DURATION(
180 // const unsigned char* category_group_enabled,
181 // const char* name,
182 // base::trace_event::TraceEventHandle id)
183 #define TRACE_EVENT_API_UPDATE_TRACE_EVENT_DURATION \
184 base::trace_event::TraceLog::GetInstance()->UpdateTraceEventDuration
185
186 // Adds a metadata event to the trace log. The |AppendValueAsTraceFormat| method
187 // on the convertable value will be called at flush time.
188 // TRACE_EVENT_API_ADD_METADATA_EVENT(
189 // const unsigned char* category_group_enabled,
190 // const char* event_name,
191 // const char* arg_name,
192 // std::unique_ptr<ConvertableToTraceFormat> arg_value)
193 #define TRACE_EVENT_API_ADD_METADATA_EVENT \
194 trace_event_internal::AddMetadataEvent
195
196 // Defines atomic operations used internally by the tracing system.
197 #define TRACE_EVENT_API_ATOMIC_WORD base::subtle::AtomicWord
198 #define TRACE_EVENT_API_ATOMIC_LOAD(var) base::subtle::NoBarrier_Load(&(var))
199 #define TRACE_EVENT_API_ATOMIC_STORE(var, value) \
200 base::subtle::NoBarrier_Store(&(var), (value))
201
202 // Defines visibility for classes in trace_event.h
203 #define TRACE_EVENT_API_CLASS_EXPORT BASE_EXPORT
204
205 ////////////////////////////////////////////////////////////////////////////////
206
207 // Implementation detail: trace event macros create temporary variables
208 // to keep instrumentation overhead low. These macros give each temporary
209 // variable a unique name based on the line number to prevent name collisions.
210 #define INTERNAL_TRACE_EVENT_UID3(a,b) \
211 trace_event_unique_##a##b
212 #define INTERNAL_TRACE_EVENT_UID2(a,b) \
213 INTERNAL_TRACE_EVENT_UID3(a,b)
214 #define INTERNAL_TRACE_EVENT_UID(name_prefix) \
215 INTERNAL_TRACE_EVENT_UID2(name_prefix, __LINE__)
216
217 // Implementation detail: internal macro to create static category.
218 // No barriers are needed, because this code is designed to operate safely
219 // even when the unsigned char* points to garbage data (which may be the case
220 // on processors without cache coherency).
221 #define INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO_CUSTOM_VARIABLES( \
222 category_group, atomic, category_group_enabled) \
223 category_group_enabled = \
224 reinterpret_cast<const unsigned char*>(TRACE_EVENT_API_ATOMIC_LOAD( \
225 atomic)); \
226 if (UNLIKELY(!category_group_enabled)) { \
227 category_group_enabled = \
228 TRACE_EVENT_API_GET_CATEGORY_GROUP_ENABLED(category_group); \
229 TRACE_EVENT_API_ATOMIC_STORE(atomic, \
230 reinterpret_cast<TRACE_EVENT_API_ATOMIC_WORD>( \
231 category_group_enabled)); \
232 }
233
234 #define INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group) \
235 static TRACE_EVENT_API_ATOMIC_WORD INTERNAL_TRACE_EVENT_UID(atomic) = 0; \
236 const unsigned char* INTERNAL_TRACE_EVENT_UID(category_group_enabled); \
237 INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO_CUSTOM_VARIABLES(category_group, \
238 INTERNAL_TRACE_EVENT_UID(atomic), \
239 INTERNAL_TRACE_EVENT_UID(category_group_enabled));
240
241 // Implementation detail: internal macro to create static category and add
242 // event if the category is enabled.
243 #define INTERNAL_TRACE_EVENT_ADD(phase, category_group, name, flags, ...) \
244 do { \
245 INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group); \
246 if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) { \
247 trace_event_internal::AddTraceEvent( \
248 phase, INTERNAL_TRACE_EVENT_UID(category_group_enabled), name, \
249 trace_event_internal::kGlobalScope, trace_event_internal::kNoId, \
250 flags, trace_event_internal::kNoId, ##__VA_ARGS__); \
251 } \
252 } while (0)
253
254 // Implementation detail: internal macro to create static category and add begin
255 // event if the category is enabled. Also adds the end event when the scope
256 // ends.
257 #define INTERNAL_TRACE_EVENT_ADD_SCOPED(category_group, name, ...) \
258 INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group); \
259 trace_event_internal::ScopedTracer INTERNAL_TRACE_EVENT_UID(tracer); \
260 if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) { \
261 base::trace_event::TraceEventHandle h = \
262 trace_event_internal::AddTraceEvent( \
263 TRACE_EVENT_PHASE_COMPLETE, \
264 INTERNAL_TRACE_EVENT_UID(category_group_enabled), name, \
265 trace_event_internal::kGlobalScope, trace_event_internal::kNoId, \
266 TRACE_EVENT_FLAG_NONE, trace_event_internal::kNoId, \
267 ##__VA_ARGS__); \
268 INTERNAL_TRACE_EVENT_UID(tracer).Initialize( \
269 INTERNAL_TRACE_EVENT_UID(category_group_enabled), name, h); \
270 }
271
272 #define INTERNAL_TRACE_EVENT_ADD_SCOPED_WITH_FLOW(category_group, name, \
273 bind_id, flow_flags, ...) \
274 INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group); \
275 trace_event_internal::ScopedTracer INTERNAL_TRACE_EVENT_UID(tracer); \
276 if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) { \
277 trace_event_internal::TraceID trace_event_bind_id((bind_id)); \
278 unsigned int trace_event_flags = \
279 flow_flags | trace_event_bind_id.id_flags(); \
280 base::trace_event::TraceEventHandle h = \
281 trace_event_internal::AddTraceEvent( \
282 TRACE_EVENT_PHASE_COMPLETE, \
283 INTERNAL_TRACE_EVENT_UID(category_group_enabled), name, \
284 trace_event_internal::kGlobalScope, trace_event_internal::kNoId, \
285 trace_event_flags, trace_event_bind_id.raw_id(), ##__VA_ARGS__); \
286 INTERNAL_TRACE_EVENT_UID(tracer).Initialize( \
287 INTERNAL_TRACE_EVENT_UID(category_group_enabled), name, h); \
288 }
289
290 // Implementation detail: internal macro to create static category and add
291 // event if the category is enabled.
292 #define INTERNAL_TRACE_EVENT_ADD_WITH_ID(phase, category_group, name, id, \
293 flags, ...) \
294 do { \
295 INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group); \
296 if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) { \
297 trace_event_internal::TraceID trace_event_trace_id((id)); \
298 unsigned int trace_event_flags = \
299 flags | trace_event_trace_id.id_flags(); \
300 trace_event_internal::AddTraceEvent( \
301 phase, INTERNAL_TRACE_EVENT_UID(category_group_enabled), name, \
302 trace_event_trace_id.scope(), trace_event_trace_id.raw_id(), \
303 trace_event_flags, trace_event_internal::kNoId, ##__VA_ARGS__); \
304 } \
305 } while (0)
306
307 // Implementation detail: internal macro to create static category and add
308 // event if the category is enabled.
309 #define INTERNAL_TRACE_EVENT_ADD_WITH_TIMESTAMP(phase, category_group, name, \
310 timestamp, flags, ...) \
311 do { \
312 INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group); \
313 if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) { \
314 trace_event_internal::AddTraceEventWithThreadIdAndTimestamp( \
315 phase, INTERNAL_TRACE_EVENT_UID(category_group_enabled), name, \
316 trace_event_internal::kGlobalScope, trace_event_internal::kNoId, \
317 TRACE_EVENT_API_CURRENT_THREAD_ID, timestamp, \
318 flags | TRACE_EVENT_FLAG_EXPLICIT_TIMESTAMP, \
319 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_ID_TID_AND_TIMESTAMP( \
326 phase, category_group, name, id, thread_id, 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::TraceID trace_event_trace_id((id)); \
331 unsigned int trace_event_flags = \
332 flags | trace_event_trace_id.id_flags(); \
333 trace_event_internal::AddTraceEventWithThreadIdAndTimestamp( \
334 phase, INTERNAL_TRACE_EVENT_UID(category_group_enabled), name, \
335 trace_event_trace_id.scope(), trace_event_trace_id.raw_id(), \
336 thread_id, timestamp, \
337 trace_event_flags | TRACE_EVENT_FLAG_EXPLICIT_TIMESTAMP, \
338 trace_event_internal::kNoId, ##__VA_ARGS__); \
339 } \
340 } while (0)
341
342 // The linked ID will not be mangled.
343 #define INTERNAL_TRACE_EVENT_ADD_LINK_IDS(category_group, name, id1, id2) \
344 do { \
345 INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group); \
346 if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) { \
347 trace_event_internal::TraceID source_id((id1)); \
348 unsigned int source_flags = source_id.id_flags(); \
349 trace_event_internal::TraceID target_id((id2)); \
350 trace_event_internal::AddTraceEvent( \
351 TRACE_EVENT_PHASE_LINK_IDS, \
352 INTERNAL_TRACE_EVENT_UID(category_group_enabled), name, \
353 source_id.scope(), source_id.raw_id(), source_flags, \
354 trace_event_internal::kNoId, "linked_id", \
355 target_id.AsConvertableToTraceFormat()); \
356 } \
357 } while (0)
358
359 // Implementation detail: internal macro to create static category and add
360 // metadata event if the category is enabled.
361 #define INTERNAL_TRACE_EVENT_METADATA_ADD(category_group, name, ...) \
362 do { \
363 INTERNAL_TRACE_EVENT_GET_CATEGORY_INFO(category_group); \
364 if (INTERNAL_TRACE_EVENT_CATEGORY_GROUP_ENABLED()) { \
365 TRACE_EVENT_API_ADD_METADATA_EVENT( \
366 INTERNAL_TRACE_EVENT_UID(category_group_enabled), name, \
367 ##__VA_ARGS__); \
368 } \
369 } while (0)
370
371 // Implementation detail: internal macro to enter and leave a
372 // context based on the current scope.
373 #define INTERNAL_TRACE_EVENT_SCOPED_CONTEXT(category_group, name, context) \
374 struct INTERNAL_TRACE_EVENT_UID(ScopedContext) { \
375 public: \
376 INTERNAL_TRACE_EVENT_UID(ScopedContext)(uint64_t cid) : cid_(cid) { \
377 TRACE_EVENT_ENTER_CONTEXT(category_group, name, cid_); \
378 } \
379 ~INTERNAL_TRACE_EVENT_UID(ScopedContext)() { \
380 TRACE_EVENT_LEAVE_CONTEXT(category_group, name, cid_); \
381 } \
382 \
383 private: \
384 uint64_t cid_; \
385 /* Local class friendly DISALLOW_COPY_AND_ASSIGN */ \
386 INTERNAL_TRACE_EVENT_UID(ScopedContext) \
387 (const INTERNAL_TRACE_EVENT_UID(ScopedContext)&) {}; \
388 void operator=(const INTERNAL_TRACE_EVENT_UID(ScopedContext)&) {}; \
389 }; \
390 INTERNAL_TRACE_EVENT_UID(ScopedContext) \
391 INTERNAL_TRACE_EVENT_UID(scoped_context)(context);
392
393 // Implementation detail: internal macro to trace a task execution with the
394 // location where it was posted from.
395 #define INTERNAL_TRACE_TASK_EXECUTION(run_function, task) \
396 TRACE_EVENT2("toplevel", run_function, "src_file", \
397 (task).posted_from.file_name(), "src_func", \
398 (task).posted_from.function_name()); \
399 TRACE_HEAP_PROFILER_API_SCOPED_TASK_EXECUTION INTERNAL_TRACE_EVENT_UID( \
400 task_event)((task).posted_from.file_name());
401
402 namespace trace_event_internal {
403
404 // Specify these values when the corresponding argument of AddTraceEvent is not
405 // used.
406 const int kZeroNumArgs = 0;
407 const std::nullptr_t kGlobalScope = nullptr;
408 const unsigned long long kNoId = 0;
409
410 // TraceID encapsulates an ID that can either be an integer or pointer. Pointers
411 // are by default mangled with the Process ID so that they are unlikely to
412 // collide when the same pointer is used on different processes.
413 class BASE_EXPORT TraceID {
414 public:
415 // Can be combined with WithScope.
416 class LocalId {
417 public:
LocalId(unsigned long long raw_id)418 explicit LocalId(unsigned long long raw_id) : raw_id_(raw_id) {}
raw_id()419 unsigned long long raw_id() const { return raw_id_; }
420 private:
421 unsigned long long raw_id_;
422 };
423
424 // Can be combined with WithScope.
425 class GlobalId {
426 public:
GlobalId(unsigned long long raw_id)427 explicit GlobalId(unsigned long long raw_id) : raw_id_(raw_id) {}
raw_id()428 unsigned long long raw_id() const { return raw_id_; }
429 private:
430 unsigned long long raw_id_;
431 };
432
433 class WithScope {
434 public:
WithScope(const char * scope,unsigned long long raw_id)435 WithScope(const char* scope, unsigned long long raw_id)
436 : scope_(scope), raw_id_(raw_id) {}
WithScope(const char * scope,LocalId local_id)437 WithScope(const char* scope, LocalId local_id)
438 : scope_(scope), raw_id_(local_id.raw_id()) {
439 id_flags_ = TRACE_EVENT_FLAG_HAS_LOCAL_ID;
440 }
WithScope(const char * scope,GlobalId global_id)441 WithScope(const char* scope, GlobalId global_id)
442 : scope_(scope), raw_id_(global_id.raw_id()) {
443 id_flags_ = TRACE_EVENT_FLAG_HAS_GLOBAL_ID;
444 }
WithScope(const char * scope,unsigned long long prefix,unsigned long long raw_id)445 WithScope(const char* scope,
446 unsigned long long prefix,
447 unsigned long long raw_id)
448 : scope_(scope), has_prefix_(true), prefix_(prefix), raw_id_(raw_id) {}
WithScope(const char * scope,unsigned long long prefix,GlobalId global_id)449 WithScope(const char* scope, unsigned long long prefix, GlobalId global_id)
450 : scope_(scope),
451 has_prefix_(true),
452 prefix_(prefix),
453 raw_id_(global_id.raw_id()) {
454 id_flags_ = TRACE_EVENT_FLAG_HAS_GLOBAL_ID;
455 }
raw_id()456 unsigned long long raw_id() const { return raw_id_; }
scope()457 const char* scope() const { return scope_; }
has_prefix()458 bool has_prefix() const { return has_prefix_; }
prefix()459 unsigned long long prefix() const { return prefix_; }
id_flags()460 unsigned int id_flags() const { return id_flags_; }
461
462 private:
463 const char* scope_ = nullptr;
464 bool has_prefix_ = false;
465 unsigned long long prefix_;
466 unsigned long long raw_id_;
467 unsigned int id_flags_ = TRACE_EVENT_FLAG_HAS_ID;
468 };
469
470 // DEPRECATED: consider using LocalId or GlobalId, instead.
471 class DontMangle {
472 public:
DontMangle(const void * raw_id)473 explicit DontMangle(const void* raw_id)
474 : raw_id_(static_cast<unsigned long long>(
475 reinterpret_cast<uintptr_t>(raw_id))) {}
DontMangle(unsigned long long raw_id)476 explicit DontMangle(unsigned long long raw_id) : raw_id_(raw_id) {}
DontMangle(unsigned long raw_id)477 explicit DontMangle(unsigned long raw_id) : raw_id_(raw_id) {}
DontMangle(unsigned int raw_id)478 explicit DontMangle(unsigned int raw_id) : raw_id_(raw_id) {}
DontMangle(unsigned short raw_id)479 explicit DontMangle(unsigned short raw_id) : raw_id_(raw_id) {}
DontMangle(unsigned char raw_id)480 explicit DontMangle(unsigned char raw_id) : raw_id_(raw_id) {}
DontMangle(long long raw_id)481 explicit DontMangle(long long raw_id)
482 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
DontMangle(long raw_id)483 explicit DontMangle(long raw_id)
484 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
DontMangle(int raw_id)485 explicit DontMangle(int raw_id)
486 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
DontMangle(short raw_id)487 explicit DontMangle(short raw_id)
488 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
DontMangle(signed char raw_id)489 explicit DontMangle(signed char raw_id)
490 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
raw_id()491 unsigned long long raw_id() const { return raw_id_; }
492 private:
493 unsigned long long raw_id_;
494 };
495
496 // DEPRECATED: consider using LocalId or GlobalId, instead.
497 class ForceMangle {
498 public:
ForceMangle(unsigned long long raw_id)499 explicit ForceMangle(unsigned long long raw_id) : raw_id_(raw_id) {}
ForceMangle(unsigned long raw_id)500 explicit ForceMangle(unsigned long raw_id) : raw_id_(raw_id) {}
ForceMangle(unsigned int raw_id)501 explicit ForceMangle(unsigned int raw_id) : raw_id_(raw_id) {}
ForceMangle(unsigned short raw_id)502 explicit ForceMangle(unsigned short raw_id) : raw_id_(raw_id) {}
ForceMangle(unsigned char raw_id)503 explicit ForceMangle(unsigned char raw_id) : raw_id_(raw_id) {}
ForceMangle(long long raw_id)504 explicit ForceMangle(long long raw_id)
505 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
ForceMangle(long raw_id)506 explicit ForceMangle(long raw_id)
507 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
ForceMangle(int raw_id)508 explicit ForceMangle(int raw_id)
509 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
ForceMangle(short raw_id)510 explicit ForceMangle(short raw_id)
511 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
ForceMangle(signed char raw_id)512 explicit ForceMangle(signed char raw_id)
513 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
raw_id()514 unsigned long long raw_id() const { return raw_id_; }
515 private:
516 unsigned long long raw_id_;
517 };
518
TraceID(const void * raw_id)519 TraceID(const void* raw_id) : raw_id_(static_cast<unsigned long long>(
520 reinterpret_cast<uintptr_t>(raw_id))) {
521 id_flags_ = TRACE_EVENT_FLAG_HAS_ID | TRACE_EVENT_FLAG_MANGLE_ID;
522 }
TraceID(ForceMangle raw_id)523 TraceID(ForceMangle raw_id) : raw_id_(raw_id.raw_id()) {
524 id_flags_ = TRACE_EVENT_FLAG_HAS_ID | TRACE_EVENT_FLAG_MANGLE_ID;
525 }
TraceID(DontMangle raw_id)526 TraceID(DontMangle raw_id) : raw_id_(raw_id.raw_id()) {}
TraceID(unsigned long long raw_id)527 TraceID(unsigned long long raw_id) : raw_id_(raw_id) {}
TraceID(unsigned long raw_id)528 TraceID(unsigned long raw_id) : raw_id_(raw_id) {}
TraceID(unsigned int raw_id)529 TraceID(unsigned int raw_id) : raw_id_(raw_id) {}
TraceID(unsigned short raw_id)530 TraceID(unsigned short raw_id) : raw_id_(raw_id) {}
TraceID(unsigned char raw_id)531 TraceID(unsigned char raw_id) : raw_id_(raw_id) {}
TraceID(long long raw_id)532 TraceID(long long raw_id)
533 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
TraceID(long raw_id)534 TraceID(long raw_id)
535 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
TraceID(int raw_id)536 TraceID(int raw_id)
537 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
TraceID(short raw_id)538 TraceID(short raw_id)
539 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
TraceID(signed char raw_id)540 TraceID(signed char raw_id)
541 : raw_id_(static_cast<unsigned long long>(raw_id)) {}
TraceID(LocalId raw_id)542 TraceID(LocalId raw_id) : raw_id_(raw_id.raw_id()) {
543 id_flags_ = TRACE_EVENT_FLAG_HAS_LOCAL_ID;
544 }
TraceID(GlobalId raw_id)545 TraceID(GlobalId raw_id) : raw_id_(raw_id.raw_id()) {
546 id_flags_ = TRACE_EVENT_FLAG_HAS_GLOBAL_ID;
547 }
TraceID(WithScope scoped_id)548 TraceID(WithScope scoped_id)
549 : scope_(scoped_id.scope()),
550 has_prefix_(scoped_id.has_prefix()),
551 prefix_(scoped_id.prefix()),
552 raw_id_(scoped_id.raw_id()),
553 id_flags_(scoped_id.id_flags()) {}
554
raw_id()555 unsigned long long raw_id() const { return raw_id_; }
scope()556 const char* scope() const { return scope_; }
has_prefix()557 bool has_prefix() const { return has_prefix_; }
prefix()558 unsigned long long prefix() const { return prefix_; }
id_flags()559 unsigned int id_flags() const { return id_flags_; }
560
561 std::unique_ptr<base::trace_event::ConvertableToTraceFormat>
562 AsConvertableToTraceFormat() const;
563
564 private:
565 const char* scope_ = nullptr;
566 bool has_prefix_ = false;
567 unsigned long long prefix_;
568 unsigned long long raw_id_;
569 unsigned int id_flags_ = TRACE_EVENT_FLAG_HAS_ID;
570 };
571
572 // Simple union to store various types as unsigned long long.
573 union TraceValueUnion {
574 bool as_bool;
575 unsigned long long as_uint;
576 long long as_int;
577 double as_double;
578 const void* as_pointer;
579 const char* as_string;
580 };
581
582 // Simple container for const char* that should be copied instead of retained.
583 class TraceStringWithCopy {
584 public:
TraceStringWithCopy(const char * str)585 explicit TraceStringWithCopy(const char* str) : str_(str) {}
str()586 const char* str() const { return str_; }
587 private:
588 const char* str_;
589 };
590
591 // Define SetTraceValue for each allowed type. It stores the type and
592 // value in the return arguments. This allows this API to avoid declaring any
593 // structures so that it is portable to third_party libraries.
594 #define INTERNAL_DECLARE_SET_TRACE_VALUE(actual_type, \
595 arg_expression, \
596 union_member, \
597 value_type_id) \
598 static inline void SetTraceValue( \
599 actual_type arg, \
600 unsigned char* type, \
601 unsigned long long* value) { \
602 TraceValueUnion type_value; \
603 type_value.union_member = arg_expression; \
604 *type = value_type_id; \
605 *value = type_value.as_uint; \
606 }
607 // Simpler form for int types that can be safely casted.
608 #define INTERNAL_DECLARE_SET_TRACE_VALUE_INT(actual_type, \
609 value_type_id) \
610 static inline void SetTraceValue( \
611 actual_type arg, \
612 unsigned char* type, \
613 unsigned long long* value) { \
614 *type = value_type_id; \
615 *value = static_cast<unsigned long long>(arg); \
616 }
617
INTERNAL_DECLARE_SET_TRACE_VALUE_INT(unsigned long long,TRACE_VALUE_TYPE_UINT)618 INTERNAL_DECLARE_SET_TRACE_VALUE_INT(unsigned long long, TRACE_VALUE_TYPE_UINT)
619 INTERNAL_DECLARE_SET_TRACE_VALUE_INT(unsigned long, TRACE_VALUE_TYPE_UINT)
620 INTERNAL_DECLARE_SET_TRACE_VALUE_INT(unsigned int, TRACE_VALUE_TYPE_UINT)
621 INTERNAL_DECLARE_SET_TRACE_VALUE_INT(unsigned short, TRACE_VALUE_TYPE_UINT)
622 INTERNAL_DECLARE_SET_TRACE_VALUE_INT(unsigned char, TRACE_VALUE_TYPE_UINT)
623 INTERNAL_DECLARE_SET_TRACE_VALUE_INT(long long, TRACE_VALUE_TYPE_INT)
624 INTERNAL_DECLARE_SET_TRACE_VALUE_INT(long, TRACE_VALUE_TYPE_INT)
625 INTERNAL_DECLARE_SET_TRACE_VALUE_INT(int, TRACE_VALUE_TYPE_INT)
626 INTERNAL_DECLARE_SET_TRACE_VALUE_INT(short, TRACE_VALUE_TYPE_INT)
627 INTERNAL_DECLARE_SET_TRACE_VALUE_INT(signed char, TRACE_VALUE_TYPE_INT)
628 INTERNAL_DECLARE_SET_TRACE_VALUE(bool, arg, as_bool, TRACE_VALUE_TYPE_BOOL)
629 INTERNAL_DECLARE_SET_TRACE_VALUE(double, arg, as_double,
630 TRACE_VALUE_TYPE_DOUBLE)
631 INTERNAL_DECLARE_SET_TRACE_VALUE(const void*, arg, as_pointer,
632 TRACE_VALUE_TYPE_POINTER)
633 INTERNAL_DECLARE_SET_TRACE_VALUE(const char*, arg, as_string,
634 TRACE_VALUE_TYPE_STRING)
635 INTERNAL_DECLARE_SET_TRACE_VALUE(const TraceStringWithCopy&, arg.str(),
636 as_string, TRACE_VALUE_TYPE_COPY_STRING)
637
638 #undef INTERNAL_DECLARE_SET_TRACE_VALUE
639 #undef INTERNAL_DECLARE_SET_TRACE_VALUE_INT
640
641 // std::string version of SetTraceValue so that trace arguments can be strings.
642 static inline void SetTraceValue(const std::string& arg,
643 unsigned char* type,
644 unsigned long long* value) {
645 TraceValueUnion type_value;
646 type_value.as_string = arg.c_str();
647 *type = TRACE_VALUE_TYPE_COPY_STRING;
648 *value = type_value.as_uint;
649 }
650
651 // base::Time, base::TimeTicks, etc. versions of SetTraceValue to make it easier
652 // to trace these types.
SetTraceValue(const base::Time arg,unsigned char * type,unsigned long long * value)653 static inline void SetTraceValue(const base::Time arg,
654 unsigned char* type,
655 unsigned long long* value) {
656 *type = TRACE_VALUE_TYPE_INT;
657 *value = arg.ToInternalValue();
658 }
659
SetTraceValue(const base::TimeTicks arg,unsigned char * type,unsigned long long * value)660 static inline void SetTraceValue(const base::TimeTicks arg,
661 unsigned char* type,
662 unsigned long long* value) {
663 *type = TRACE_VALUE_TYPE_INT;
664 *value = arg.ToInternalValue();
665 }
666
SetTraceValue(const base::ThreadTicks arg,unsigned char * type,unsigned long long * value)667 static inline void SetTraceValue(const base::ThreadTicks arg,
668 unsigned char* type,
669 unsigned long long* value) {
670 *type = TRACE_VALUE_TYPE_INT;
671 *value = arg.ToInternalValue();
672 }
673
674 // These AddTraceEvent and AddTraceEventWithThreadIdAndTimestamp template
675 // functions are defined here instead of in the macro, because the arg_values
676 // could be temporary objects, such as std::string. In order to store
677 // pointers to the internal c_str and pass through to the tracing API,
678 // the arg_values must live throughout these procedures.
679
680 template <class ARG1_CONVERTABLE_TYPE>
681 static inline base::trace_event::TraceEventHandle
AddTraceEventWithThreadIdAndTimestamp(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,unsigned long long id,int thread_id,const base::TimeTicks & timestamp,unsigned int flags,unsigned long long bind_id,const char * arg1_name,std::unique_ptr<ARG1_CONVERTABLE_TYPE> arg1_val)682 AddTraceEventWithThreadIdAndTimestamp(
683 char phase,
684 const unsigned char* category_group_enabled,
685 const char* name,
686 const char* scope,
687 unsigned long long id,
688 int thread_id,
689 const base::TimeTicks& timestamp,
690 unsigned int flags,
691 unsigned long long bind_id,
692 const char* arg1_name,
693 std::unique_ptr<ARG1_CONVERTABLE_TYPE> arg1_val) {
694 const int num_args = 1;
695 unsigned char arg_types[1] = { TRACE_VALUE_TYPE_CONVERTABLE };
696 std::unique_ptr<base::trace_event::ConvertableToTraceFormat>
697 convertable_values[1] = {std::move(arg1_val)};
698 return TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_THREAD_ID_AND_TIMESTAMP(
699 phase, category_group_enabled, name, scope, id, bind_id, thread_id,
700 timestamp, num_args, &arg1_name, arg_types, NULL, convertable_values,
701 flags);
702 }
703
704 template <class ARG1_TYPE, class ARG2_CONVERTABLE_TYPE>
705 static inline base::trace_event::TraceEventHandle
AddTraceEventWithThreadIdAndTimestamp(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,unsigned long long id,int thread_id,const base::TimeTicks & timestamp,unsigned int flags,unsigned long long bind_id,const char * arg1_name,const ARG1_TYPE & arg1_val,const char * arg2_name,std::unique_ptr<ARG2_CONVERTABLE_TYPE> arg2_val)706 AddTraceEventWithThreadIdAndTimestamp(
707 char phase,
708 const unsigned char* category_group_enabled,
709 const char* name,
710 const char* scope,
711 unsigned long long id,
712 int thread_id,
713 const base::TimeTicks& timestamp,
714 unsigned int flags,
715 unsigned long long bind_id,
716 const char* arg1_name,
717 const ARG1_TYPE& arg1_val,
718 const char* arg2_name,
719 std::unique_ptr<ARG2_CONVERTABLE_TYPE> arg2_val) {
720 const int num_args = 2;
721 const char* arg_names[2] = { arg1_name, arg2_name };
722
723 unsigned char arg_types[2];
724 unsigned long long arg_values[2];
725 SetTraceValue(arg1_val, &arg_types[0], &arg_values[0]);
726 arg_types[1] = TRACE_VALUE_TYPE_CONVERTABLE;
727 std::unique_ptr<base::trace_event::ConvertableToTraceFormat>
728 convertable_values[2] = {nullptr, std::move(arg2_val)};
729 return TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_THREAD_ID_AND_TIMESTAMP(
730 phase, category_group_enabled, name, scope, id, bind_id, thread_id,
731 timestamp, num_args, arg_names, arg_types, arg_values, convertable_values,
732 flags);
733 }
734
735 template <class ARG1_CONVERTABLE_TYPE, class ARG2_TYPE>
736 static inline base::trace_event::TraceEventHandle
AddTraceEventWithThreadIdAndTimestamp(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,unsigned long long id,int thread_id,const base::TimeTicks & timestamp,unsigned int flags,unsigned long long bind_id,const char * arg1_name,std::unique_ptr<ARG1_CONVERTABLE_TYPE> arg1_val,const char * arg2_name,const ARG2_TYPE & arg2_val)737 AddTraceEventWithThreadIdAndTimestamp(
738 char phase,
739 const unsigned char* category_group_enabled,
740 const char* name,
741 const char* scope,
742 unsigned long long id,
743 int thread_id,
744 const base::TimeTicks& timestamp,
745 unsigned int flags,
746 unsigned long long bind_id,
747 const char* arg1_name,
748 std::unique_ptr<ARG1_CONVERTABLE_TYPE> arg1_val,
749 const char* arg2_name,
750 const ARG2_TYPE& arg2_val) {
751 const int num_args = 2;
752 const char* arg_names[2] = { arg1_name, arg2_name };
753
754 unsigned char arg_types[2];
755 unsigned long long arg_values[2];
756 arg_types[0] = TRACE_VALUE_TYPE_CONVERTABLE;
757 arg_values[0] = 0;
758 SetTraceValue(arg2_val, &arg_types[1], &arg_values[1]);
759 std::unique_ptr<base::trace_event::ConvertableToTraceFormat>
760 convertable_values[2] = {std::move(arg1_val), nullptr};
761 return TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_THREAD_ID_AND_TIMESTAMP(
762 phase, category_group_enabled, name, scope, id, bind_id, thread_id,
763 timestamp, num_args, arg_names, arg_types, arg_values, convertable_values,
764 flags);
765 }
766
767 template <class ARG1_CONVERTABLE_TYPE, class ARG2_CONVERTABLE_TYPE>
768 static inline base::trace_event::TraceEventHandle
AddTraceEventWithThreadIdAndTimestamp(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,unsigned long long id,int thread_id,const base::TimeTicks & timestamp,unsigned int flags,unsigned long long bind_id,const char * arg1_name,std::unique_ptr<ARG1_CONVERTABLE_TYPE> arg1_val,const char * arg2_name,std::unique_ptr<ARG2_CONVERTABLE_TYPE> arg2_val)769 AddTraceEventWithThreadIdAndTimestamp(
770 char phase,
771 const unsigned char* category_group_enabled,
772 const char* name,
773 const char* scope,
774 unsigned long long id,
775 int thread_id,
776 const base::TimeTicks& timestamp,
777 unsigned int flags,
778 unsigned long long bind_id,
779 const char* arg1_name,
780 std::unique_ptr<ARG1_CONVERTABLE_TYPE> arg1_val,
781 const char* arg2_name,
782 std::unique_ptr<ARG2_CONVERTABLE_TYPE> arg2_val) {
783 const int num_args = 2;
784 const char* arg_names[2] = { arg1_name, arg2_name };
785 unsigned char arg_types[2] =
786 { TRACE_VALUE_TYPE_CONVERTABLE, TRACE_VALUE_TYPE_CONVERTABLE };
787 std::unique_ptr<base::trace_event::ConvertableToTraceFormat>
788 convertable_values[2] = {std::move(arg1_val), std::move(arg2_val)};
789 return TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_THREAD_ID_AND_TIMESTAMP(
790 phase, category_group_enabled, name, scope, id, bind_id, thread_id,
791 timestamp, num_args, arg_names, arg_types, NULL, convertable_values,
792 flags);
793 }
794
795 static inline base::trace_event::TraceEventHandle
AddTraceEventWithThreadIdAndTimestamp(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,unsigned long long id,int thread_id,const base::TimeTicks & timestamp,unsigned int flags,unsigned long long bind_id)796 AddTraceEventWithThreadIdAndTimestamp(
797 char phase,
798 const unsigned char* category_group_enabled,
799 const char* name,
800 const char* scope,
801 unsigned long long id,
802 int thread_id,
803 const base::TimeTicks& timestamp,
804 unsigned int flags,
805 unsigned long long bind_id) {
806 return TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_THREAD_ID_AND_TIMESTAMP(
807 phase, category_group_enabled, name, scope, id, bind_id, thread_id,
808 timestamp, kZeroNumArgs, NULL, NULL, NULL, NULL, flags);
809 }
810
AddTraceEvent(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,unsigned long long id,unsigned int flags,unsigned long long bind_id)811 static inline base::trace_event::TraceEventHandle AddTraceEvent(
812 char phase,
813 const unsigned char* category_group_enabled,
814 const char* name,
815 const char* scope,
816 unsigned long long id,
817 unsigned int flags,
818 unsigned long long bind_id) {
819 const int thread_id = static_cast<int>(base::PlatformThread::CurrentId());
820 const base::TimeTicks now = base::TimeTicks::Now();
821 return AddTraceEventWithThreadIdAndTimestamp(
822 phase, category_group_enabled, name, scope, id, thread_id, now, flags,
823 bind_id);
824 }
825
826 template<class ARG1_TYPE>
827 static inline base::trace_event::TraceEventHandle
AddTraceEventWithThreadIdAndTimestamp(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,unsigned long long id,int thread_id,const base::TimeTicks & timestamp,unsigned int flags,unsigned long long bind_id,const char * arg1_name,const ARG1_TYPE & arg1_val)828 AddTraceEventWithThreadIdAndTimestamp(
829 char phase,
830 const unsigned char* category_group_enabled,
831 const char* name,
832 const char* scope,
833 unsigned long long id,
834 int thread_id,
835 const base::TimeTicks& timestamp,
836 unsigned int flags,
837 unsigned long long bind_id,
838 const char* arg1_name,
839 const ARG1_TYPE& arg1_val) {
840 const int num_args = 1;
841 unsigned char arg_types[1];
842 unsigned long long arg_values[1];
843 SetTraceValue(arg1_val, &arg_types[0], &arg_values[0]);
844 return TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_THREAD_ID_AND_TIMESTAMP(
845 phase, category_group_enabled, name, scope, id, bind_id, thread_id,
846 timestamp, num_args, &arg1_name, arg_types, arg_values, NULL, flags);
847 }
848
849 template<class ARG1_TYPE>
AddTraceEvent(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,unsigned long long id,unsigned int flags,unsigned long long bind_id,const char * arg1_name,const ARG1_TYPE & arg1_val)850 static inline base::trace_event::TraceEventHandle AddTraceEvent(
851 char phase,
852 const unsigned char* category_group_enabled,
853 const char* name,
854 const char* scope,
855 unsigned long long id,
856 unsigned int flags,
857 unsigned long long bind_id,
858 const char* arg1_name,
859 const ARG1_TYPE& arg1_val) {
860 int thread_id = static_cast<int>(base::PlatformThread::CurrentId());
861 base::TimeTicks now = base::TimeTicks::Now();
862 return AddTraceEventWithThreadIdAndTimestamp(
863 phase, category_group_enabled, name, scope, id, thread_id, now, flags,
864 bind_id, arg1_name, arg1_val);
865 }
866
867 template <class ARG1_CONVERTABLE_TYPE>
AddTraceEvent(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,unsigned long long id,unsigned int flags,unsigned long long bind_id,const char * arg1_name,std::unique_ptr<ARG1_CONVERTABLE_TYPE> arg1_val)868 static inline base::trace_event::TraceEventHandle AddTraceEvent(
869 char phase,
870 const unsigned char* category_group_enabled,
871 const char* name,
872 const char* scope,
873 unsigned long long id,
874 unsigned int flags,
875 unsigned long long bind_id,
876 const char* arg1_name,
877 std::unique_ptr<ARG1_CONVERTABLE_TYPE> arg1_val) {
878 int thread_id = static_cast<int>(base::PlatformThread::CurrentId());
879 base::TimeTicks now = base::TimeTicks::Now();
880 return AddTraceEventWithThreadIdAndTimestamp(
881 phase, category_group_enabled, name, scope, id, thread_id, now, flags,
882 bind_id, arg1_name, std::move(arg1_val));
883 }
884
885 template<class ARG1_TYPE, class ARG2_TYPE>
886 static inline base::trace_event::TraceEventHandle
AddTraceEventWithThreadIdAndTimestamp(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,unsigned long long id,int thread_id,const base::TimeTicks & timestamp,unsigned int flags,unsigned long long bind_id,const char * arg1_name,const ARG1_TYPE & arg1_val,const char * arg2_name,const ARG2_TYPE & arg2_val)887 AddTraceEventWithThreadIdAndTimestamp(
888 char phase,
889 const unsigned char* category_group_enabled,
890 const char* name,
891 const char* scope,
892 unsigned long long id,
893 int thread_id,
894 const base::TimeTicks& timestamp,
895 unsigned int flags,
896 unsigned long long bind_id,
897 const char* arg1_name,
898 const ARG1_TYPE& arg1_val,
899 const char* arg2_name,
900 const ARG2_TYPE& arg2_val) {
901 const int num_args = 2;
902 const char* arg_names[2] = { arg1_name, arg2_name };
903 unsigned char arg_types[2];
904 unsigned long long arg_values[2];
905 SetTraceValue(arg1_val, &arg_types[0], &arg_values[0]);
906 SetTraceValue(arg2_val, &arg_types[1], &arg_values[1]);
907 return TRACE_EVENT_API_ADD_TRACE_EVENT_WITH_THREAD_ID_AND_TIMESTAMP(
908 phase, category_group_enabled, name, scope, id, bind_id, thread_id,
909 timestamp, num_args, arg_names, arg_types, arg_values, NULL, flags);
910 }
911
912 template <class ARG1_CONVERTABLE_TYPE, class ARG2_TYPE>
AddTraceEvent(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,unsigned long long id,unsigned int flags,unsigned long long bind_id,const char * arg1_name,std::unique_ptr<ARG1_CONVERTABLE_TYPE> arg1_val,const char * arg2_name,const ARG2_TYPE & arg2_val)913 static inline base::trace_event::TraceEventHandle AddTraceEvent(
914 char phase,
915 const unsigned char* category_group_enabled,
916 const char* name,
917 const char* scope,
918 unsigned long long id,
919 unsigned int flags,
920 unsigned long long bind_id,
921 const char* arg1_name,
922 std::unique_ptr<ARG1_CONVERTABLE_TYPE> arg1_val,
923 const char* arg2_name,
924 const ARG2_TYPE& arg2_val) {
925 int thread_id = static_cast<int>(base::PlatformThread::CurrentId());
926 base::TimeTicks now = base::TimeTicks::Now();
927 return AddTraceEventWithThreadIdAndTimestamp(
928 phase, category_group_enabled, name, scope, id, thread_id, now, flags,
929 bind_id, arg1_name, std::move(arg1_val), arg2_name, arg2_val);
930 }
931
932 template <class ARG1_TYPE, class ARG2_CONVERTABLE_TYPE>
AddTraceEvent(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,unsigned long long id,unsigned int flags,unsigned long long bind_id,const char * arg1_name,const ARG1_TYPE & arg1_val,const char * arg2_name,std::unique_ptr<ARG2_CONVERTABLE_TYPE> arg2_val)933 static inline base::trace_event::TraceEventHandle AddTraceEvent(
934 char phase,
935 const unsigned char* category_group_enabled,
936 const char* name,
937 const char* scope,
938 unsigned long long id,
939 unsigned int flags,
940 unsigned long long bind_id,
941 const char* arg1_name,
942 const ARG1_TYPE& arg1_val,
943 const char* arg2_name,
944 std::unique_ptr<ARG2_CONVERTABLE_TYPE> arg2_val) {
945 int thread_id = static_cast<int>(base::PlatformThread::CurrentId());
946 base::TimeTicks now = base::TimeTicks::Now();
947 return AddTraceEventWithThreadIdAndTimestamp(
948 phase, category_group_enabled, name, scope, id, thread_id, now, flags,
949 bind_id, arg1_name, arg1_val, arg2_name, std::move(arg2_val));
950 }
951
952 template <class ARG1_CONVERTABLE_TYPE, class ARG2_CONVERTABLE_TYPE>
AddTraceEvent(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,unsigned long long id,unsigned int flags,unsigned long long bind_id,const char * arg1_name,std::unique_ptr<ARG1_CONVERTABLE_TYPE> arg1_val,const char * arg2_name,std::unique_ptr<ARG2_CONVERTABLE_TYPE> arg2_val)953 static inline base::trace_event::TraceEventHandle AddTraceEvent(
954 char phase,
955 const unsigned char* category_group_enabled,
956 const char* name,
957 const char* scope,
958 unsigned long long id,
959 unsigned int flags,
960 unsigned long long bind_id,
961 const char* arg1_name,
962 std::unique_ptr<ARG1_CONVERTABLE_TYPE> arg1_val,
963 const char* arg2_name,
964 std::unique_ptr<ARG2_CONVERTABLE_TYPE> arg2_val) {
965 int thread_id = static_cast<int>(base::PlatformThread::CurrentId());
966 base::TimeTicks now = base::TimeTicks::Now();
967 return AddTraceEventWithThreadIdAndTimestamp(
968 phase, category_group_enabled, name, scope, id, thread_id, now, flags,
969 bind_id, arg1_name, std::move(arg1_val), arg2_name, std::move(arg2_val));
970 }
971
972 template<class ARG1_TYPE, class ARG2_TYPE>
AddTraceEvent(char phase,const unsigned char * category_group_enabled,const char * name,const char * scope,unsigned long long id,unsigned int flags,unsigned long long bind_id,const char * arg1_name,const ARG1_TYPE & arg1_val,const char * arg2_name,const ARG2_TYPE & arg2_val)973 static inline base::trace_event::TraceEventHandle AddTraceEvent(
974 char phase,
975 const unsigned char* category_group_enabled,
976 const char* name,
977 const char* scope,
978 unsigned long long id,
979 unsigned int flags,
980 unsigned long long bind_id,
981 const char* arg1_name,
982 const ARG1_TYPE& arg1_val,
983 const char* arg2_name,
984 const ARG2_TYPE& arg2_val) {
985 int thread_id = static_cast<int>(base::PlatformThread::CurrentId());
986 base::TimeTicks now = base::TimeTicks::Now();
987 return AddTraceEventWithThreadIdAndTimestamp(
988 phase, category_group_enabled, name, scope, id, thread_id, now, flags,
989 bind_id, arg1_name, arg1_val, arg2_name, arg2_val);
990 }
991
992 template <class ARG1_CONVERTABLE_TYPE>
AddMetadataEvent(const unsigned char * category_group_enabled,const char * event_name,const char * arg_name,std::unique_ptr<ARG1_CONVERTABLE_TYPE> arg_value)993 static inline void AddMetadataEvent(
994 const unsigned char* category_group_enabled,
995 const char* event_name,
996 const char* arg_name,
997 std::unique_ptr<ARG1_CONVERTABLE_TYPE> arg_value) {
998 const char* arg_names[1] = {arg_name};
999 unsigned char arg_types[1] = {TRACE_VALUE_TYPE_CONVERTABLE};
1000 std::unique_ptr<base::trace_event::ConvertableToTraceFormat>
1001 convertable_values[1] = {std::move(arg_value)};
1002 base::trace_event::TraceLog::GetInstance()->AddMetadataEvent(
1003 category_group_enabled, event_name,
1004 1, // num_args
1005 arg_names, arg_types,
1006 nullptr, // arg_values
1007 convertable_values, TRACE_EVENT_FLAG_NONE);
1008 }
1009
1010 template <class ARG1_TYPE>
AddMetadataEvent(const unsigned char * category_group_enabled,const char * event_name,const char * arg_name,const ARG1_TYPE & arg_val)1011 static void AddMetadataEvent(const unsigned char* category_group_enabled,
1012 const char* event_name,
1013 const char* arg_name,
1014 const ARG1_TYPE& arg_val) {
1015 const int num_args = 1;
1016 const char* arg_names[1] = {arg_name};
1017 unsigned char arg_types[1];
1018 unsigned long long arg_values[1];
1019 SetTraceValue(arg_val, &arg_types[0], &arg_values[0]);
1020
1021 base::trace_event::TraceLog::GetInstance()->AddMetadataEvent(
1022 category_group_enabled, event_name, num_args, arg_names, arg_types,
1023 arg_values, nullptr, TRACE_EVENT_FLAG_NONE);
1024 }
1025
1026 // Used by TRACE_EVENTx macros. Do not use directly.
1027 class TRACE_EVENT_API_CLASS_EXPORT ScopedTracer {
1028 public:
1029 // Note: members of data_ intentionally left uninitialized. See Initialize.
ScopedTracer()1030 ScopedTracer() : p_data_(NULL) {}
1031
~ScopedTracer()1032 ~ScopedTracer() {
1033 if (p_data_ && *data_.category_group_enabled) {
1034 TRACE_EVENT_API_UPDATE_TRACE_EVENT_DURATION(
1035 data_.category_group_enabled, data_.name, data_.event_handle);
1036 }
1037 }
1038
Initialize(const unsigned char * category_group_enabled,const char * name,base::trace_event::TraceEventHandle event_handle)1039 void Initialize(const unsigned char* category_group_enabled,
1040 const char* name,
1041 base::trace_event::TraceEventHandle event_handle) {
1042 data_.category_group_enabled = category_group_enabled;
1043 data_.name = name;
1044 data_.event_handle = event_handle;
1045 p_data_ = &data_;
1046 }
1047
1048 private:
1049 // This Data struct workaround is to avoid initializing all the members
1050 // in Data during construction of this object, since this object is always
1051 // constructed, even when tracing is disabled. If the members of Data were
1052 // members of this class instead, compiler warnings occur about potential
1053 // uninitialized accesses.
1054 struct Data {
1055 const unsigned char* category_group_enabled;
1056 const char* name;
1057 base::trace_event::TraceEventHandle event_handle;
1058 };
1059 Data* p_data_;
1060 Data data_;
1061 };
1062
1063 // Used by TRACE_EVENT_BINARY_EFFICIENTx macro. Do not use directly.
1064 class TRACE_EVENT_API_CLASS_EXPORT ScopedTraceBinaryEfficient {
1065 public:
1066 ScopedTraceBinaryEfficient(const char* category_group, const char* name);
1067 ~ScopedTraceBinaryEfficient();
1068
1069 private:
1070 const unsigned char* category_group_enabled_;
1071 const char* name_;
1072 base::trace_event::TraceEventHandle event_handle_;
1073 };
1074
1075 // This macro generates less code then TRACE_EVENT0 but is also
1076 // slower to execute when tracing is off. It should generally only be
1077 // used with code that is seldom executed or conditionally executed
1078 // when debugging.
1079 // For now the category_group must be "gpu".
1080 #define TRACE_EVENT_BINARY_EFFICIENT0(category_group, name) \
1081 trace_event_internal::ScopedTraceBinaryEfficient \
1082 INTERNAL_TRACE_EVENT_UID(scoped_trace)(category_group, name);
1083
1084 } // namespace trace_event_internal
1085
1086 namespace base {
1087 namespace trace_event {
1088
1089 template<typename IDType> class TraceScopedTrackableObject {
1090 public:
TraceScopedTrackableObject(const char * category_group,const char * name,IDType id)1091 TraceScopedTrackableObject(const char* category_group, const char* name,
1092 IDType id)
1093 : category_group_(category_group),
1094 name_(name),
1095 id_(id) {
1096 TRACE_EVENT_OBJECT_CREATED_WITH_ID(category_group_, name_, id_);
1097 }
1098
snapshot(ArgType snapshot)1099 template <typename ArgType> void snapshot(ArgType snapshot) {
1100 TRACE_EVENT_OBJECT_SNAPSHOT_WITH_ID(category_group_, name_, id_, snapshot);
1101 }
1102
~TraceScopedTrackableObject()1103 ~TraceScopedTrackableObject() {
1104 TRACE_EVENT_OBJECT_DELETED_WITH_ID(category_group_, name_, id_);
1105 }
1106
1107 private:
1108 const char* category_group_;
1109 const char* name_;
1110 IDType id_;
1111
1112 DISALLOW_COPY_AND_ASSIGN(TraceScopedTrackableObject);
1113 };
1114
1115 } // namespace trace_event
1116 } // namespace base
1117
1118 #endif // BASE_TRACE_EVENT_TRACE_EVENT_H_
1119