1use pin_project::pin_project;
6use std::ffi::CStr;
7use std::future::Future;
8use std::marker::PhantomData;
9use std::pin::Pin;
10use std::task::Poll;
11use std::{mem, ptr};
12
13pub use sys::{
14 trace_site_t, TRACE_BLOB_TYPE_DATA, TRACE_BLOB_TYPE_LAST_BRANCH, TRACE_BLOB_TYPE_PERFETTO,
15};
16
17#[derive(Copy, Clone)]
19pub enum Scope {
20 Thread,
21 Process,
22 Global,
23}
24
25impl Scope {
26 fn into_raw(self) -> sys::trace_scope_t {
27 match self {
28 Scope::Thread => sys::TRACE_SCOPE_THREAD,
29 Scope::Process => sys::TRACE_SCOPE_PROCESS,
30 Scope::Global => sys::TRACE_SCOPE_GLOBAL,
31 }
32 }
33}
34
35#[inline]
37pub fn is_enabled() -> bool {
38 unsafe { sys::trace_state() != sys::TRACE_STOPPED }
40}
41
42pub fn category_enabled(category: &'static CStr) -> bool {
44 unsafe { sys::trace_is_category_enabled(category.as_ptr()) }
47}
48
49#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
50pub enum TraceState {
51 Stopped,
52 Started,
53 Stopping,
54}
55
56pub fn trace_state() -> TraceState {
57 match unsafe { sys::trace_state() } {
58 sys::TRACE_STOPPED => TraceState::Stopped,
59 sys::TRACE_STARTED => TraceState::Started,
60 sys::TRACE_STOPPING => TraceState::Stopping,
61 s => panic!("Unknown trace state {:?}", s),
62 }
63}
64
65#[repr(transparent)]
67#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
68pub struct Id(u64);
69
70impl Id {
71 pub fn new() -> Self {
78 Self(unsafe { sys::trace_generate_nonce() })
80 }
81
82 pub fn random() -> Self {
91 let ts = zx::BootInstant::get().into_nanos() as u64;
92 let high_order = ts << 16;
93 let low_order = rand::random::<u16>() as u64;
94 Self(high_order | low_order)
95 }
96}
97
98impl From<u64> for Id {
99 fn from(u: u64) -> Self {
100 Self(u)
101 }
102}
103
104impl From<Id> for u64 {
105 fn from(id: Id) -> Self {
106 id.0
107 }
108}
109
110#[repr(transparent)]
112pub struct Arg<'a>(sys::trace_arg_t, PhantomData<&'a ()>);
113
114pub trait ArgValue {
120 fn of<'a>(key: &'a str, value: Self) -> Arg<'a>
121 where
122 Self: 'a;
123}
124
125macro_rules! arg_from {
131 ($valname:ident, $(($type:ty, $tag:expr, $value:expr))*) => {
132 $(
133 impl ArgValue for $type {
134 #[inline]
135 fn of<'a>(key: &'a str, $valname: Self) -> Arg<'a>
136 where Self: 'a
137 {
138 #[allow(unused)]
139 let $valname = $valname;
140
141 Arg(sys::trace_arg_t {
142 name_ref: trace_make_inline_string_ref(key),
143 value: sys::trace_arg_value_t {
144 type_: $tag,
145 value: $value,
146 },
147 }, PhantomData)
148 }
149 }
150 )*
151 }
152}
153
154#[rustfmt::skip]
156arg_from!(val,
157 ((), sys::TRACE_ARG_NULL, sys::trace_arg_union_t { int32_value: 0 })
158 (bool, sys::TRACE_ARG_BOOL, sys::trace_arg_union_t { bool_value: val })
159 (i32, sys::TRACE_ARG_INT32, sys::trace_arg_union_t { int32_value: val })
160 (u32, sys::TRACE_ARG_UINT32, sys::trace_arg_union_t { uint32_value: val })
161 (i64, sys::TRACE_ARG_INT64, sys::trace_arg_union_t { int64_value: val })
162 (u64, sys::TRACE_ARG_UINT64, sys::trace_arg_union_t { uint64_value: val })
163 (isize, sys::TRACE_ARG_INT64, sys::trace_arg_union_t { int64_value: val as i64 })
164 (usize, sys::TRACE_ARG_UINT64, sys::trace_arg_union_t { uint64_value: val as u64 })
165 (f64, sys::TRACE_ARG_DOUBLE, sys::trace_arg_union_t { double_value: val })
166 (zx::Koid, sys::TRACE_ARG_KOID, sys::trace_arg_union_t { koid_value: val.raw_koid() })
167);
168
169impl<T> ArgValue for *const T {
170 #[inline]
171 fn of<'a>(key: &'a str, val: Self) -> Arg<'a>
172 where
173 Self: 'a,
174 {
175 Arg(
176 sys::trace_arg_t {
177 name_ref: trace_make_inline_string_ref(key),
178 value: sys::trace_arg_value_t {
179 type_: sys::TRACE_ARG_POINTER,
180 value: sys::trace_arg_union_t { pointer_value: val as usize },
181 },
182 },
183 PhantomData,
184 )
185 }
186}
187
188impl<T> ArgValue for *mut T {
189 #[inline]
190 fn of<'a>(key: &'a str, val: Self) -> Arg<'a>
191 where
192 Self: 'a,
193 {
194 Arg(
195 sys::trace_arg_t {
196 name_ref: trace_make_inline_string_ref(key),
197 value: sys::trace_arg_value_t {
198 type_: sys::TRACE_ARG_POINTER,
199 value: sys::trace_arg_union_t { pointer_value: val as usize },
200 },
201 },
202 PhantomData,
203 )
204 }
205}
206
207impl<'a> ArgValue for &'a str {
208 #[inline]
209 fn of<'b>(key: &'b str, val: Self) -> Arg<'b>
210 where
211 Self: 'b,
212 {
213 Arg(
214 sys::trace_arg_t {
215 name_ref: trace_make_inline_string_ref(key),
216 value: sys::trace_arg_value_t {
217 type_: sys::TRACE_ARG_STRING,
218 value: sys::trace_arg_union_t {
219 string_value_ref: trace_make_inline_string_ref(val),
220 },
221 },
222 },
223 PhantomData,
224 )
225 }
226}
227
228#[macro_export]
250macro_rules! instant {
251 ($category:expr, $name:expr, $scope:expr $(, $key:expr => $val:expr)*) => {
252 {
253 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
254 if let Some(context) = $crate::TraceCategoryContext::acquire_cached($category, &CACHE) {
255 $crate::instant(&context, $name, $scope, &[$($crate::ArgValue::of($key, $val)),*]);
256 }
257 }
258 }
259}
260
261#[inline]
264pub fn instant(
265 context: &TraceCategoryContext,
266 name: &'static CStr,
267 scope: Scope,
268 args: &[Arg<'_>],
269) {
270 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
271
272 let name_ref = context.register_string_literal(name);
273 context.write_instant(name_ref, scope, args);
274}
275
276#[macro_export]
290macro_rules! alert {
291 ($category:expr, $name:expr) => {
292 $crate::alert($category, $name)
293 };
294}
295
296pub fn alert(category: &'static CStr, name: &'static CStr) {
298 unsafe {
302 let mut category_ref = mem::MaybeUninit::<sys::trace_string_ref_t>::uninit();
303 let context =
304 sys::trace_acquire_context_for_category(category.as_ptr(), category_ref.as_mut_ptr());
305 if context != ptr::null() {
306 sys::trace_context_send_alert(context, name.as_ptr());
307 sys::trace_release_context(context);
308 }
309 }
310}
311
312#[macro_export]
329macro_rules! counter {
330 ($category:expr, $name:expr, $counter_id:expr $(, $key:expr => $val:expr)*) => {
331 {
332 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
333 if let Some(context) = $crate::TraceCategoryContext::acquire_cached($category, &CACHE) {
334 $crate::counter(&context, $name, $counter_id,
335 &[$($crate::ArgValue::of($key, $val)),*])
336 }
337 }
338 }
339}
340
341pub fn counter(
351 context: &TraceCategoryContext,
352 name: &'static CStr,
353 counter_id: u64,
354 args: &[Arg<'_>],
355) {
356 assert!(args.len() >= 1, "trace counter args must include at least one numeric argument");
357 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
358
359 let name_ref = context.register_string_literal(name);
360 context.write_counter(name_ref, counter_id, args);
361}
362
363#[must_use = "DurationScope must be `end`ed to be recorded"]
366pub struct DurationScope<'a> {
367 category: &'static CStr,
368 name: &'static CStr,
369 args: &'a [Arg<'a>],
370 start_time: zx::BootTicks,
371}
372
373impl<'a> DurationScope<'a> {
374 pub fn begin(category: &'static CStr, name: &'static CStr, args: &'a [Arg<'_>]) -> Self {
377 let start_time = zx::BootTicks::get();
378 Self { category, name, args, start_time }
379 }
380}
381
382impl<'a> Drop for DurationScope<'a> {
383 fn drop(&mut self) {
384 if let Some(context) = TraceCategoryContext::acquire(self.category) {
385 let name_ref = context.register_string_literal(self.name);
386 context.write_duration(name_ref, self.start_time, self.args);
387 }
388 }
389}
390
391pub fn complete_duration(
393 category: &'static CStr,
394 name: &'static CStr,
395 start_time: zx::BootTicks,
396 args: &[Arg<'_>],
397) {
398 if let Some(context) = TraceCategoryContext::acquire(category) {
399 let name_ref = context.register_string_literal(name);
400 context.write_duration(name_ref, start_time, args);
401 }
402}
403
404#[macro_export]
439macro_rules! duration {
440 ($category:expr, $name:expr $(, $key:expr => $val:expr)* $(,)?) => {
441 let mut args;
442 let _scope = {
443 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
444 if let Some(_context) =
451 $crate::TraceCategoryContext::acquire_cached($category, &CACHE) {
452 args = [$($crate::ArgValue::of($key, $val)),*];
453 Some($crate::duration($category, $name, &args))
454 } else {
455 None
456 }
457 };
458 }
459}
460
461pub fn duration<'a>(
473 category: &'static CStr,
474 name: &'static CStr,
475 args: &'a [Arg<'_>],
476) -> DurationScope<'a> {
477 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
478 DurationScope::begin(category, name, args)
479}
480
481#[macro_export]
495macro_rules! duration_begin {
496 ($category:expr, $name:expr $(, $key:expr => $val:expr)* $(,)?) => {
497 {
498 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
499 if let Some(context) = $crate::TraceCategoryContext::acquire_cached($category, &CACHE) {
500 $crate::duration_begin(&context, $name,
501 &[$($crate::ArgValue::of($key, $val)),*])
502 }
503 }
504 };
505}
506
507#[macro_export]
521macro_rules! duration_end {
522 ($category:expr, $name:expr $(, $key:expr => $val:expr)* $(,)?) => {
523 {
524 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
525 if let Some(context) = $crate::TraceCategoryContext::acquire_cached($category, &CACHE) {
526 $crate::duration_end(&context, $name, &[$($crate::ArgValue::of($key, $val)),*])
527 }
528 }
529 };
530}
531
532pub fn duration_begin(context: &TraceCategoryContext, name: &'static CStr, args: &[Arg<'_>]) {
543 let ticks = zx::BootTicks::get();
544 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
545
546 let name_ref = context.register_string_literal(name);
547 context.write_duration_begin(ticks, name_ref, args);
548}
549
550pub fn duration_end(context: &TraceCategoryContext, name: &'static CStr, args: &[Arg<'_>]) {
560 let ticks = zx::BootTicks::get();
561 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
562
563 let name_ref = context.register_string_literal(name);
564 context.write_duration_end(ticks, name_ref, args);
565}
566
567#[must_use = "emits an end event when dropped, so if dropped immediately creates an essentially \
570 zero length duration that should just be an instant instead"]
571pub struct AsyncScope {
572 id: Id,
575 category: &'static CStr,
576 name: &'static CStr,
577}
578impl AsyncScope {
579 pub fn begin(id: Id, category: &'static CStr, name: &'static CStr, args: &[Arg<'_>]) -> Self {
582 async_begin(id, category, name, args);
583 Self { id, category, name }
584 }
585
586 pub fn end(self, args: &[Arg<'_>]) {
589 let Self { id, category, name } = self;
590 async_end(id, category, name, args);
591 std::mem::forget(self);
592 }
593}
594
595impl Drop for AsyncScope {
596 fn drop(&mut self) {
597 let Self { id, category, name } = *self;
601 async_end(id, category, name, &[]);
602 }
603}
604
605pub fn async_enter(
614 id: Id,
615 category: &'static CStr,
616 name: &'static CStr,
617 args: &[Arg<'_>],
618) -> AsyncScope {
619 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
620 AsyncScope::begin(id, category, name, args)
621}
622
623#[macro_export]
654macro_rules! async_enter {
655 ($id:expr, $category:expr, $name:expr $(, $key:expr => $val:expr)*) => {
656 {
657 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
658 if let Some(context) = $crate::TraceCategoryContext::acquire_cached($category, &CACHE) {
659 Some($crate::AsyncScope::begin($id, $category, $name, &[$($crate::ArgValue::of($key, $val)),*]))
660 } else {
661 None
662 }
663 }
664 }
665}
666
667#[macro_export]
691macro_rules! async_instant {
692 ($id:expr, $category:expr, $name:expr $(, $key:expr => $val:expr)*) => {
693 {
694 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
695 if let Some(context) = $crate::TraceCategoryContext::acquire_cached($category, &CACHE) {
696 $crate::async_instant($id, &context, $name, &[$($crate::ArgValue::of($key, $val)),*]);
697 }
698 }
699 }
700}
701
702pub fn async_begin(id: Id, category: &'static CStr, name: &'static CStr, args: &[Arg<'_>]) {
714 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
715
716 if let Some(context) = TraceCategoryContext::acquire(category) {
717 let name_ref = context.register_string_literal(name);
718 context.write_async_begin(id, name_ref, args);
719 }
720}
721
722pub fn async_end(id: Id, category: &'static CStr, name: &'static CStr, args: &[Arg<'_>]) {
734 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
735
736 if let Some(context) = TraceCategoryContext::acquire(category) {
737 let name_ref = context.register_string_literal(name);
738 context.write_async_end(id, name_ref, args);
739 }
740}
741
742pub fn async_instant(
754 id: Id,
755 context: &TraceCategoryContext,
756 name: &'static CStr,
757 args: &[Arg<'_>],
758) {
759 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
760
761 let name_ref = context.register_string_literal(name);
762 context.write_async_instant(id, name_ref, args);
763}
764
765#[macro_export]
766macro_rules! blob {
767 ($category:expr, $name:expr, $bytes:expr $(, $key:expr => $val:expr)*) => {
768 {
769 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
770 if let Some(context) = $crate::TraceCategoryContext::acquire_cached($category, &CACHE) {
771 $crate::blob_fn(&context, $name, $bytes, &[$($crate::ArgValue::of($key, $val)),*])
772 }
773 }
774 }
775}
776pub fn blob_fn(
777 context: &TraceCategoryContext,
778 name: &'static CStr,
779 bytes: &[u8],
780 args: &[Arg<'_>],
781) {
782 let name_ref = context.register_string_literal(name);
783 context.write_blob(name_ref, bytes, args);
784}
785
786#[macro_export]
801macro_rules! flow_begin {
802 ($category:expr, $name:expr, $flow_id:expr $(, $key:expr => $val:expr)*) => {
803 {
804 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
805 if let Some(context) = $crate::TraceCategoryContext::acquire_cached($category, &CACHE) {
806 $crate::flow_begin(&context, $name, $flow_id,
807 &[$($crate::ArgValue::of($key, $val)),*])
808 }
809 }
810 }
811}
812
813#[macro_export]
828macro_rules! flow_step {
829 ($category:expr, $name:expr, $flow_id:expr $(, $key:expr => $val:expr)*) => {
830 {
831 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
832 if let Some(context) = $crate::TraceCategoryContext::acquire_cached($category, &CACHE) {
833 $crate::flow_step(&context, $name, $flow_id,
834 &[$($crate::ArgValue::of($key, $val)),*])
835 }
836 }
837 }
838}
839
840#[macro_export]
855macro_rules! flow_end {
856 ($category:expr, $name:expr, $flow_id:expr $(, $key:expr => $val:expr)*) => {
857 {
858 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
859 if let Some(context) = $crate::TraceCategoryContext::acquire_cached($category, &CACHE) {
860 $crate::flow_end(&context, $name, $flow_id,
861 &[$($crate::ArgValue::of($key, $val)),*])
862 }
863 }
864 }
865}
866
867pub fn flow_begin(
886 context: &TraceCategoryContext,
887 name: &'static CStr,
888 flow_id: Id,
889 args: &[Arg<'_>],
890) {
891 let ticks = zx::BootTicks::get();
892 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
893
894 let name_ref = context.register_string_literal(name);
895 context.write_flow_begin(ticks, name_ref, flow_id, args);
896}
897
898pub fn flow_end(
915 context: &TraceCategoryContext,
916 name: &'static CStr,
917 flow_id: Id,
918 args: &[Arg<'_>],
919) {
920 let ticks = zx::BootTicks::get();
921 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
922
923 let name_ref = context.register_string_literal(name);
924 context.write_flow_end(ticks, name_ref, flow_id, args);
925}
926
927pub fn flow_step(
944 context: &TraceCategoryContext,
945 name: &'static CStr,
946 flow_id: Id,
947 args: &[Arg<'_>],
948) {
949 let ticks = zx::BootTicks::get();
950 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
951
952 let name_ref = context.register_string_literal(name);
953 context.write_flow_step(ticks, name_ref, flow_id, args);
954}
955
956#[macro_export]
970macro_rules! instant_flow_begin {
971 (
972 $category:expr,
973 $event_name:expr,
974 $flow_name:expr,
975 $flow_id:expr
976 $(, $key:expr => $val:expr)*
977 ) => {
978 {
979 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
980 if let Some(context) = $crate::TraceCategoryContext::acquire_cached($category, &CACHE) {
981 $crate::instant_flow_begin(
982 &context,
983 $event_name,
984 $flow_name,
985 $flow_id,
986 &[$($crate::ArgValue::of($key, $val)),*],
987 )
988 }
989 }
990 }
991}
992
993#[macro_export]
1007macro_rules! instant_flow_end {
1008 (
1009 $category:expr,
1010 $event_name:expr,
1011 $flow_name:expr,
1012 $flow_id:expr
1013 $(, $key:expr => $val:expr)*
1014 ) => {
1015 {
1016 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
1017 if let Some(context) = $crate::TraceCategoryContext::acquire_cached($category, &CACHE) {
1018 $crate::instant_flow_end(
1019 &context,
1020 $event_name,
1021 $flow_name,
1022 $flow_id,
1023 &[$($crate::ArgValue::of($key, $val)),*],
1024 )
1025 }
1026 }
1027 }
1028}
1029
1030#[macro_export]
1044macro_rules! instant_flow_step {
1045 (
1046 $category:expr,
1047 $event_name:expr,
1048 $flow_name:expr,
1049 $flow_id:expr
1050 $(, $key:expr => $val:expr)*
1051 ) => {
1052 {
1053 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
1054 if let Some(context) = $crate::TraceCategoryContext::acquire_cached($category, &CACHE) {
1055 $crate::instant_flow_step(
1056 &context,
1057 $event_name,
1058 $flow_name,
1059 $flow_id,
1060 &[$($crate::ArgValue::of($key, $val)),*],
1061 )
1062 }
1063 }
1064 }
1065}
1066
1067pub fn instant_flow_begin(
1079 context: &TraceCategoryContext,
1080 event_name: &'static CStr,
1081 flow_name: &'static CStr,
1082 flow_id: Id,
1083 args: &[Arg<'_>],
1084) {
1085 let ticks = zx::BootTicks::get();
1086 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
1087
1088 let event_name_ref = context.register_string_literal(event_name);
1089 let flow_name_ref = context.register_string_literal(flow_name);
1090
1091 context.write_duration_begin(ticks, event_name_ref, args);
1092 context.write_flow_begin(ticks, flow_name_ref, flow_id, args);
1093 context.write_duration_end(ticks, event_name_ref, args);
1094}
1095
1096pub fn instant_flow_end(
1108 context: &TraceCategoryContext,
1109 event_name: &'static CStr,
1110 flow_name: &'static CStr,
1111 flow_id: Id,
1112 args: &[Arg<'_>],
1113) {
1114 let ticks = zx::BootTicks::get();
1115 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
1116
1117 let event_name_ref = context.register_string_literal(event_name);
1118 let flow_name_ref = context.register_string_literal(flow_name);
1119
1120 context.write_duration_begin(ticks, event_name_ref, args);
1121 context.write_flow_end(ticks, flow_name_ref, flow_id, args);
1122 context.write_duration_end(ticks, event_name_ref, args);
1123}
1124
1125pub fn instant_flow_step(
1137 context: &TraceCategoryContext,
1138 event_name: &'static CStr,
1139 flow_name: &'static CStr,
1140 flow_id: Id,
1141 args: &[Arg<'_>],
1142) {
1143 let ticks = zx::BootTicks::get();
1144 assert!(args.len() <= 15, "no more than 15 trace arguments are supported");
1145
1146 let event_name_ref = context.register_string_literal(event_name);
1147 let flow_name_ref = context.register_string_literal(flow_name);
1148
1149 context.write_duration_begin(ticks, event_name_ref, args);
1150 context.write_flow_step(ticks, flow_name_ref, flow_id, args);
1151 context.write_duration_end(ticks, event_name_ref, args);
1152}
1153
1154const fn trace_make_empty_string_ref() -> sys::trace_string_ref_t {
1156 sys::trace_string_ref_t {
1157 encoded_value: sys::TRACE_ENCODED_STRING_REF_EMPTY,
1158 inline_string: ptr::null(),
1159 }
1160}
1161
1162#[inline]
1163fn trim_to_last_char_boundary(string: &str, max_len: usize) -> &[u8] {
1164 let mut len = string.len();
1165 if string.len() > max_len {
1166 len = max_len;
1170 while len > 0 {
1171 if string.is_char_boundary(len - 1) && string.is_char_boundary(len) {
1172 break;
1173 }
1174 len -= 1;
1175 }
1176 }
1177 &string.as_bytes()[0..len]
1178}
1179
1180#[inline]
1183fn trace_make_inline_string_ref(string: &str) -> sys::trace_string_ref_t {
1184 let len = string.len() as u32;
1185 if len == 0 {
1186 return trace_make_empty_string_ref();
1187 }
1188
1189 let string =
1190 trim_to_last_char_boundary(string, sys::TRACE_ENCODED_STRING_REF_MAX_LENGTH as usize);
1191
1192 sys::trace_string_ref_t {
1193 encoded_value: sys::TRACE_ENCODED_STRING_REF_INLINE_FLAG | len,
1194 inline_string: string.as_ptr() as *const libc::c_char,
1195 }
1196}
1197
1198pub struct TraceCategoryContext {
1200 raw: *const sys::trace_context_t,
1201 category_ref: sys::trace_string_ref_t,
1202}
1203
1204impl TraceCategoryContext {
1205 #[inline]
1206 pub fn acquire_cached(
1207 category: &'static CStr,
1208 site: &sys::trace_site_t,
1209 ) -> Option<TraceCategoryContext> {
1210 unsafe {
1211 let mut category_ref = mem::MaybeUninit::<sys::trace_string_ref_t>::uninit();
1216 let raw = sys::trace_acquire_context_for_category_cached(
1217 category.as_ptr(),
1218 site.as_ptr(),
1219 category_ref.as_mut_ptr(),
1220 );
1221 if raw != ptr::null() {
1222 Some(TraceCategoryContext { raw, category_ref: category_ref.assume_init() })
1223 } else {
1224 None
1225 }
1226 }
1227 }
1228
1229 pub fn acquire(category: &'static CStr) -> Option<TraceCategoryContext> {
1230 unsafe {
1231 let mut category_ref = mem::MaybeUninit::<sys::trace_string_ref_t>::uninit();
1232 let raw = sys::trace_acquire_context_for_category(
1233 category.as_ptr(),
1234 category_ref.as_mut_ptr(),
1235 );
1236 if raw != ptr::null() {
1237 Some(TraceCategoryContext { raw, category_ref: category_ref.assume_init() })
1238 } else {
1239 None
1240 }
1241 }
1242 }
1243
1244 #[inline]
1245 pub fn register_string_literal(&self, name: &'static CStr) -> sys::trace_string_ref_t {
1246 unsafe {
1247 let mut name_ref = mem::MaybeUninit::<sys::trace_string_ref_t>::uninit();
1248 sys::trace_context_register_string_literal(
1249 self.raw,
1250 name.as_ptr(),
1251 name_ref.as_mut_ptr(),
1252 );
1253 name_ref.assume_init()
1254 }
1255 }
1256
1257 #[inline]
1258 fn register_current_thread(&self) -> sys::trace_thread_ref_t {
1259 unsafe {
1260 let mut thread_ref = mem::MaybeUninit::<sys::trace_thread_ref_t>::uninit();
1261 sys::trace_context_register_current_thread(self.raw, thread_ref.as_mut_ptr());
1262 thread_ref.assume_init()
1263 }
1264 }
1265
1266 #[inline]
1267 pub fn write_instant(&self, name_ref: sys::trace_string_ref_t, scope: Scope, args: &[Arg<'_>]) {
1268 let ticks = zx::BootTicks::get();
1269 let thread_ref = self.register_current_thread();
1270 unsafe {
1271 sys::trace_context_write_instant_event_record(
1272 self.raw,
1273 ticks.into_raw(),
1274 &thread_ref,
1275 &self.category_ref,
1276 &name_ref,
1277 scope.into_raw(),
1278 args.as_ptr() as *const sys::trace_arg_t,
1279 args.len(),
1280 );
1281 }
1282 }
1283
1284 pub fn write_instant_with_inline_name(&self, name: &str, scope: Scope, args: &[Arg<'_>]) {
1285 let name_ref = trace_make_inline_string_ref(name);
1286 self.write_instant(name_ref, scope, args)
1287 }
1288
1289 fn write_counter(&self, name_ref: sys::trace_string_ref_t, counter_id: u64, args: &[Arg<'_>]) {
1290 let ticks = zx::BootTicks::get();
1291 let thread_ref = self.register_current_thread();
1292 unsafe {
1293 sys::trace_context_write_counter_event_record(
1294 self.raw,
1295 ticks.into_raw(),
1296 &thread_ref,
1297 &self.category_ref,
1298 &name_ref,
1299 counter_id,
1300 args.as_ptr() as *const sys::trace_arg_t,
1301 args.len(),
1302 );
1303 }
1304 }
1305
1306 pub fn write_counter_with_inline_name(&self, name: &str, counter_id: u64, args: &[Arg<'_>]) {
1307 let name_ref = trace_make_inline_string_ref(name);
1308 self.write_counter(name_ref, counter_id, args);
1309 }
1310
1311 fn write_duration(
1312 &self,
1313 name_ref: sys::trace_string_ref_t,
1314 start_time: zx::BootTicks,
1315 args: &[Arg<'_>],
1316 ) {
1317 let ticks = zx::BootTicks::get();
1318 let thread_ref = self.register_current_thread();
1319 unsafe {
1320 sys::trace_context_write_duration_event_record(
1321 self.raw,
1322 start_time.into_raw(),
1323 ticks.into_raw(),
1324 &thread_ref,
1325 &self.category_ref,
1326 &name_ref,
1327 args.as_ptr() as *const sys::trace_arg_t,
1328 args.len(),
1329 );
1330 }
1331 }
1332
1333 pub fn write_duration_with_inline_name(
1334 &self,
1335 name: &str,
1336 start_time: zx::BootTicks,
1337 args: &[Arg<'_>],
1338 ) {
1339 let name_ref = trace_make_inline_string_ref(name);
1340 self.write_duration(name_ref, start_time, args);
1341 }
1342
1343 fn write_duration_begin(
1344 &self,
1345 ticks: zx::BootTicks,
1346 name_ref: sys::trace_string_ref_t,
1347 args: &[Arg<'_>],
1348 ) {
1349 let thread_ref = self.register_current_thread();
1350 unsafe {
1351 sys::trace_context_write_duration_begin_event_record(
1352 self.raw,
1353 ticks.into_raw(),
1354 &thread_ref,
1355 &self.category_ref,
1356 &name_ref,
1357 args.as_ptr() as *const sys::trace_arg_t,
1358 args.len(),
1359 );
1360 }
1361 }
1362
1363 pub fn write_duration_begin_with_inline_name(&self, name: &str, args: &[Arg<'_>]) {
1364 let name_ref = trace_make_inline_string_ref(name);
1365 self.write_duration_begin(zx::BootTicks::get(), name_ref, args);
1366 }
1367
1368 fn write_duration_end(
1369 &self,
1370 ticks: zx::BootTicks,
1371 name_ref: sys::trace_string_ref_t,
1372 args: &[Arg<'_>],
1373 ) {
1374 let thread_ref = self.register_current_thread();
1375 unsafe {
1376 sys::trace_context_write_duration_end_event_record(
1377 self.raw,
1378 ticks.into_raw(),
1379 &thread_ref,
1380 &self.category_ref,
1381 &name_ref,
1382 args.as_ptr() as *const sys::trace_arg_t,
1383 args.len(),
1384 );
1385 }
1386 }
1387
1388 pub fn write_duration_end_with_inline_name(&self, name: &str, args: &[Arg<'_>]) {
1389 let name_ref = trace_make_inline_string_ref(name);
1390 self.write_duration_end(zx::BootTicks::get(), name_ref, args);
1391 }
1392
1393 fn write_async_begin(&self, id: Id, name_ref: sys::trace_string_ref_t, args: &[Arg<'_>]) {
1394 let ticks = zx::BootTicks::get();
1395 let thread_ref = self.register_current_thread();
1396 unsafe {
1397 sys::trace_context_write_async_begin_event_record(
1398 self.raw,
1399 ticks.into_raw(),
1400 &thread_ref,
1401 &self.category_ref,
1402 &name_ref,
1403 id.into(),
1404 args.as_ptr() as *const sys::trace_arg_t,
1405 args.len(),
1406 );
1407 }
1408 }
1409
1410 pub fn write_async_begin_with_inline_name(&self, id: Id, name: &str, args: &[Arg<'_>]) {
1411 let name_ref = trace_make_inline_string_ref(name);
1412 self.write_async_begin(id, name_ref, args);
1413 }
1414
1415 fn write_async_end(&self, id: Id, name_ref: sys::trace_string_ref_t, args: &[Arg<'_>]) {
1416 let ticks = zx::BootTicks::get();
1417 let thread_ref = self.register_current_thread();
1418 unsafe {
1419 sys::trace_context_write_async_end_event_record(
1420 self.raw,
1421 ticks.into_raw(),
1422 &thread_ref,
1423 &self.category_ref,
1424 &name_ref,
1425 id.into(),
1426 args.as_ptr() as *const sys::trace_arg_t,
1427 args.len(),
1428 );
1429 }
1430 }
1431
1432 pub fn write_async_end_with_inline_name(&self, id: Id, name: &str, args: &[Arg<'_>]) {
1433 let name_ref = trace_make_inline_string_ref(name);
1434 self.write_async_end(id, name_ref, args);
1435 }
1436
1437 fn write_async_instant(&self, id: Id, name_ref: sys::trace_string_ref_t, args: &[Arg<'_>]) {
1438 let ticks = zx::BootTicks::get();
1439 let thread_ref = self.register_current_thread();
1440 unsafe {
1441 sys::trace_context_write_async_instant_event_record(
1442 self.raw,
1443 ticks.into_raw(),
1444 &thread_ref,
1445 &self.category_ref,
1446 &name_ref,
1447 id.into(),
1448 args.as_ptr() as *const sys::trace_arg_t,
1449 args.len(),
1450 );
1451 }
1452 }
1453
1454 fn write_blob(&self, name_ref: sys::trace_string_ref_t, bytes: &[u8], args: &[Arg<'_>]) {
1455 let ticks = zx::BootTicks::get();
1456 let thread_ref = self.register_current_thread();
1457 unsafe {
1458 sys::trace_context_write_blob_event_record(
1459 self.raw,
1460 ticks.into_raw(),
1461 &thread_ref,
1462 &self.category_ref,
1463 &name_ref,
1464 bytes.as_ptr() as *const core::ffi::c_void,
1465 bytes.len(),
1466 args.as_ptr() as *const sys::trace_arg_t,
1467 args.len(),
1468 );
1469 }
1470 }
1471
1472 fn write_flow_begin(
1473 &self,
1474 ticks: zx::BootTicks,
1475 name_ref: sys::trace_string_ref_t,
1476 flow_id: Id,
1477 args: &[Arg<'_>],
1478 ) {
1479 let thread_ref = self.register_current_thread();
1480 unsafe {
1481 sys::trace_context_write_flow_begin_event_record(
1482 self.raw,
1483 ticks.into_raw(),
1484 &thread_ref,
1485 &self.category_ref,
1486 &name_ref,
1487 flow_id.into(),
1488 args.as_ptr() as *const sys::trace_arg_t,
1489 args.len(),
1490 );
1491 }
1492 }
1493
1494 fn write_flow_end(
1495 &self,
1496 ticks: zx::BootTicks,
1497 name_ref: sys::trace_string_ref_t,
1498 flow_id: Id,
1499 args: &[Arg<'_>],
1500 ) {
1501 let thread_ref = self.register_current_thread();
1502 unsafe {
1503 sys::trace_context_write_flow_end_event_record(
1504 self.raw,
1505 ticks.into_raw(),
1506 &thread_ref,
1507 &self.category_ref,
1508 &name_ref,
1509 flow_id.into(),
1510 args.as_ptr() as *const sys::trace_arg_t,
1511 args.len(),
1512 );
1513 }
1514 }
1515
1516 fn write_flow_step(
1517 &self,
1518 ticks: zx::BootTicks,
1519 name_ref: sys::trace_string_ref_t,
1520 flow_id: Id,
1521 args: &[Arg<'_>],
1522 ) {
1523 let thread_ref = self.register_current_thread();
1524 unsafe {
1525 sys::trace_context_write_flow_step_event_record(
1526 self.raw,
1527 ticks.into_raw(),
1528 &thread_ref,
1529 &self.category_ref,
1530 &name_ref,
1531 flow_id.into(),
1532 args.as_ptr() as *const sys::trace_arg_t,
1533 args.len(),
1534 );
1535 }
1536 }
1537}
1538
1539impl std::ops::Drop for TraceCategoryContext {
1540 fn drop(&mut self) {
1541 unsafe {
1542 sys::trace_release_context(self.raw);
1543 }
1544 }
1545}
1546
1547pub struct Context {
1549 context: *const sys::trace_context_t,
1550}
1551
1552impl Context {
1553 #[inline]
1554 pub fn acquire() -> Option<Self> {
1555 let context = unsafe { sys::trace_acquire_context() };
1556 if context.is_null() {
1557 None
1558 } else {
1559 Some(Self { context })
1560 }
1561 }
1562
1563 #[inline]
1564 pub fn register_string_literal(&self, s: &'static CStr) -> sys::trace_string_ref_t {
1565 unsafe {
1566 let mut s_ref = mem::MaybeUninit::<sys::trace_string_ref_t>::uninit();
1567 sys::trace_context_register_string_literal(
1568 self.context,
1569 s.as_ptr(),
1570 s_ref.as_mut_ptr(),
1571 );
1572 s_ref.assume_init()
1573 }
1574 }
1575
1576 pub fn write_blob_record(
1577 &self,
1578 type_: sys::trace_blob_type_t,
1579 name_ref: &sys::trace_string_ref_t,
1580 data: &[u8],
1581 ) {
1582 unsafe {
1583 sys::trace_context_write_blob_record(
1584 self.context,
1585 type_,
1586 name_ref as *const sys::trace_string_ref_t,
1587 data.as_ptr() as *const libc::c_void,
1588 data.len(),
1589 );
1590 }
1591 }
1592}
1593
1594impl std::ops::Drop for Context {
1595 fn drop(&mut self) {
1596 unsafe { sys::trace_release_context(self.context) }
1597 }
1598}
1599
1600pub struct ProlongedContext {
1601 context: *const sys::trace_prolonged_context_t,
1602}
1603
1604impl ProlongedContext {
1605 pub fn acquire() -> Option<Self> {
1606 let context = unsafe { sys::trace_acquire_prolonged_context() };
1607 if context.is_null() {
1608 None
1609 } else {
1610 Some(Self { context })
1611 }
1612 }
1613}
1614
1615impl Drop for ProlongedContext {
1616 fn drop(&mut self) {
1617 unsafe { sys::trace_release_prolonged_context(self.context) }
1618 }
1619}
1620
1621unsafe impl Send for ProlongedContext {}
1622
1623mod sys {
1624 #![allow(non_camel_case_types, unused)]
1625 use zx::sys::{zx_handle_t, zx_koid_t, zx_obj_type_t, zx_status_t, zx_ticks_t};
1626
1627 pub type trace_ticks_t = zx_ticks_t;
1628 pub type trace_counter_id_t = u64;
1629 pub type trace_async_id_t = u64;
1630 pub type trace_flow_id_t = u64;
1631 pub type trace_thread_state_t = u32;
1632 pub type trace_cpu_number_t = u32;
1633 pub type trace_string_index_t = u32;
1634 pub type trace_thread_index_t = u32;
1635 pub type trace_context_t = libc::c_void;
1636 pub type trace_prolonged_context_t = libc::c_void;
1637
1638 pub type trace_encoded_string_ref_t = u32;
1639 pub const TRACE_ENCODED_STRING_REF_EMPTY: trace_encoded_string_ref_t = 0;
1640 pub const TRACE_ENCODED_STRING_REF_INLINE_FLAG: trace_encoded_string_ref_t = 0x8000;
1641 pub const TRACE_ENCODED_STRING_REF_LENGTH_MASK: trace_encoded_string_ref_t = 0x7fff;
1642 pub const TRACE_ENCODED_STRING_REF_MAX_LENGTH: trace_encoded_string_ref_t = 32000;
1643 pub const TRACE_ENCODED_STRING_REF_MIN_INDEX: trace_encoded_string_ref_t = 0x1;
1644 pub const TRACE_ENCODED_STRING_REF_MAX_INDEX: trace_encoded_string_ref_t = 0x7fff;
1645
1646 pub type trace_encoded_thread_ref_t = u32;
1647 pub const TRACE_ENCODED_THREAD_REF_INLINE: trace_encoded_thread_ref_t = 0;
1648 pub const TRACE_ENCODED_THREAD_MIN_INDEX: trace_encoded_thread_ref_t = 0x01;
1649 pub const TRACE_ENCODED_THREAD_MAX_INDEX: trace_encoded_thread_ref_t = 0xff;
1650
1651 pub type trace_state_t = libc::c_int;
1652 pub const TRACE_STOPPED: trace_state_t = 0;
1653 pub const TRACE_STARTED: trace_state_t = 1;
1654 pub const TRACE_STOPPING: trace_state_t = 2;
1655
1656 pub type trace_scope_t = libc::c_int;
1657 pub const TRACE_SCOPE_THREAD: trace_scope_t = 0;
1658 pub const TRACE_SCOPE_PROCESS: trace_scope_t = 1;
1659 pub const TRACE_SCOPE_GLOBAL: trace_scope_t = 2;
1660
1661 pub type trace_blob_type_t = libc::c_int;
1662 pub const TRACE_BLOB_TYPE_DATA: trace_blob_type_t = 1;
1663 pub const TRACE_BLOB_TYPE_LAST_BRANCH: trace_blob_type_t = 2;
1664 pub const TRACE_BLOB_TYPE_PERFETTO: trace_blob_type_t = 3;
1665
1666 #[repr(C)]
1667 #[derive(Copy, Clone)]
1668 pub struct trace_string_ref_t {
1669 pub encoded_value: trace_encoded_string_ref_t,
1670 pub inline_string: *const libc::c_char,
1671 }
1672
1673 pub type trace_site_t = std::sync::atomic::AtomicU64;
1677
1678 unsafe impl Send for trace_string_ref_t {}
1688 unsafe impl Sync for trace_string_ref_t {}
1689
1690 #[repr(C)]
1691 pub struct trace_thread_ref_t {
1692 pub encoded_value: trace_encoded_thread_ref_t,
1693 pub inline_process_koid: zx_koid_t,
1694 pub inline_thread_koid: zx_koid_t,
1695 }
1696
1697 #[repr(C)]
1698 pub struct trace_arg_t {
1699 pub name_ref: trace_string_ref_t,
1700 pub value: trace_arg_value_t,
1701 }
1702
1703 #[repr(C)]
1704 pub union trace_arg_union_t {
1705 pub int32_value: i32,
1706 pub uint32_value: u32,
1707 pub int64_value: i64,
1708 pub uint64_value: u64,
1709 pub double_value: libc::c_double,
1710 pub string_value_ref: trace_string_ref_t,
1711 pub pointer_value: libc::uintptr_t,
1712 pub koid_value: zx_koid_t,
1713 pub bool_value: bool,
1714 pub reserved_for_future_expansion: [libc::uintptr_t; 2],
1715 }
1716
1717 pub type trace_arg_type_t = libc::c_int;
1718 pub const TRACE_ARG_NULL: trace_arg_type_t = 0;
1719 pub const TRACE_ARG_INT32: trace_arg_type_t = 1;
1720 pub const TRACE_ARG_UINT32: trace_arg_type_t = 2;
1721 pub const TRACE_ARG_INT64: trace_arg_type_t = 3;
1722 pub const TRACE_ARG_UINT64: trace_arg_type_t = 4;
1723 pub const TRACE_ARG_DOUBLE: trace_arg_type_t = 5;
1724 pub const TRACE_ARG_STRING: trace_arg_type_t = 6;
1725 pub const TRACE_ARG_POINTER: trace_arg_type_t = 7;
1726 pub const TRACE_ARG_KOID: trace_arg_type_t = 8;
1727 pub const TRACE_ARG_BOOL: trace_arg_type_t = 9;
1728
1729 #[repr(C)]
1730 pub struct trace_arg_value_t {
1731 pub type_: trace_arg_type_t,
1732 pub value: trace_arg_union_t,
1733 }
1734
1735 #[repr(C)]
1736 pub struct trace_handler_ops_t {
1737 pub is_category_enabled:
1738 unsafe fn(handler: *const trace_handler_t, category: *const libc::c_char) -> bool,
1739 pub trace_started: unsafe fn(handler: *const trace_handler_t),
1740 pub trace_stopped: unsafe fn(
1741 handler: *const trace_handler_t,
1742 async_ptr: *const (), disposition: zx_status_t,
1744 buffer_bytes_written: libc::size_t,
1745 ),
1746 pub buffer_overflow: unsafe fn(handler: *const trace_handler_t),
1747 }
1748
1749 #[repr(C)]
1750 pub struct trace_handler_t {
1751 pub ops: *const trace_handler_ops_t,
1752 }
1753
1754 extern "C" {
1756 pub fn trace_context_is_category_enabled(
1759 context: *const trace_context_t,
1760 category_literal: *const libc::c_char,
1761 ) -> bool;
1762
1763 pub fn trace_context_register_string_copy(
1764 context: *const trace_context_t,
1765 string: *const libc::c_char,
1766 length: libc::size_t,
1767 out_ref: *mut trace_string_ref_t,
1768 );
1769
1770 pub fn trace_context_register_string_literal(
1771 context: *const trace_context_t,
1772 string_literal: *const libc::c_char,
1773 out_ref: *mut trace_string_ref_t,
1774 );
1775
1776 pub fn trace_context_register_category_literal(
1777 context: *const trace_context_t,
1778 category_literal: *const libc::c_char,
1779 out_ref: *mut trace_string_ref_t,
1780 ) -> bool;
1781
1782 pub fn trace_context_register_current_thread(
1783 context: *const trace_context_t,
1784 out_ref: *mut trace_thread_ref_t,
1785 );
1786
1787 pub fn trace_context_register_thread(
1788 context: *const trace_context_t,
1789 process_koid: zx_koid_t,
1790 thread_koid: zx_koid_t,
1791 out_ref: *mut trace_thread_ref_t,
1792 );
1793
1794 pub fn trace_context_write_kernel_object_record(
1795 context: *const trace_context_t,
1796 koid: zx_koid_t,
1797 type_: zx_obj_type_t,
1798 args: *const trace_arg_t,
1799 num_args: libc::size_t,
1800 );
1801
1802 pub fn trace_context_write_kernel_object_record_for_handle(
1803 context: *const trace_context_t,
1804 handle: zx_handle_t,
1805 args: *const trace_arg_t,
1806 num_args: libc::size_t,
1807 );
1808
1809 pub fn trace_context_write_process_info_record(
1810 context: *const trace_context_t,
1811 process_koid: zx_koid_t,
1812 process_name_ref: *const trace_string_ref_t,
1813 );
1814
1815 pub fn trace_context_write_thread_info_record(
1816 context: *const trace_context_t,
1817 process_koid: zx_koid_t,
1818 thread_koid: zx_koid_t,
1819 thread_name_ref: *const trace_string_ref_t,
1820 );
1821
1822 pub fn trace_context_write_context_switch_record(
1823 context: *const trace_context_t,
1824 event_time: trace_ticks_t,
1825 cpu_number: trace_cpu_number_t,
1826 outgoing_thread_state: trace_thread_state_t,
1827 outgoing_thread_ref: *const trace_thread_ref_t,
1828 incoming_thread_ref: *const trace_thread_ref_t,
1829 );
1830
1831 pub fn trace_context_write_log_record(
1832 context: *const trace_context_t,
1833 event_time: trace_ticks_t,
1834 thread_ref: *const trace_thread_ref_t,
1835 log_message: *const libc::c_char,
1836 log_message_length: libc::size_t,
1837 );
1838
1839 pub fn trace_context_write_instant_event_record(
1840 context: *const trace_context_t,
1841 event_time: trace_ticks_t,
1842 thread_ref: *const trace_thread_ref_t,
1843 category_ref: *const trace_string_ref_t,
1844 name_ref: *const trace_string_ref_t,
1845 scope: trace_scope_t,
1846 args: *const trace_arg_t,
1847 num_args: libc::size_t,
1848 );
1849
1850 pub fn trace_context_send_alert(context: *const trace_context_t, name: *const libc::c_char);
1851
1852 pub fn trace_context_write_counter_event_record(
1853 context: *const trace_context_t,
1854 event_time: trace_ticks_t,
1855 thread_ref: *const trace_thread_ref_t,
1856 category_ref: *const trace_string_ref_t,
1857 name_ref: *const trace_string_ref_t,
1858 counter_id: trace_counter_id_t,
1859 args: *const trace_arg_t,
1860 num_args: libc::size_t,
1861 );
1862
1863 pub fn trace_context_write_duration_event_record(
1864 context: *const trace_context_t,
1865 start_time: trace_ticks_t,
1866 end_time: trace_ticks_t,
1867 thread_ref: *const trace_thread_ref_t,
1868 category_ref: *const trace_string_ref_t,
1869 name_ref: *const trace_string_ref_t,
1870 args: *const trace_arg_t,
1871 num_args: libc::size_t,
1872 );
1873
1874 pub fn trace_context_write_blob_event_record(
1875 context: *const trace_context_t,
1876 event_time: trace_ticks_t,
1877 thread_ref: *const trace_thread_ref_t,
1878 category_ref: *const trace_string_ref_t,
1879 name_ref: *const trace_string_ref_t,
1880 blob: *const libc::c_void,
1881 blob_size: libc::size_t,
1882 args: *const trace_arg_t,
1883 num_args: libc::size_t,
1884 );
1885
1886 pub fn trace_context_write_duration_begin_event_record(
1887 context: *const trace_context_t,
1888 event_time: trace_ticks_t,
1889 thread_ref: *const trace_thread_ref_t,
1890 category_ref: *const trace_string_ref_t,
1891 name_ref: *const trace_string_ref_t,
1892 args: *const trace_arg_t,
1893 num_args: libc::size_t,
1894 );
1895
1896 pub fn trace_context_write_duration_end_event_record(
1897 context: *const trace_context_t,
1898 event_time: trace_ticks_t,
1899 thread_ref: *const trace_thread_ref_t,
1900 category_ref: *const trace_string_ref_t,
1901 name_ref: *const trace_string_ref_t,
1902 args: *const trace_arg_t,
1903 num_args: libc::size_t,
1904 );
1905
1906 pub fn trace_context_write_async_begin_event_record(
1907 context: *const trace_context_t,
1908 event_time: trace_ticks_t,
1909 thread_ref: *const trace_thread_ref_t,
1910 category_ref: *const trace_string_ref_t,
1911 name_ref: *const trace_string_ref_t,
1912 async_id: trace_async_id_t,
1913 args: *const trace_arg_t,
1914 num_args: libc::size_t,
1915 );
1916
1917 pub fn trace_context_write_async_instant_event_record(
1918 context: *const trace_context_t,
1919 event_time: trace_ticks_t,
1920 thread_ref: *const trace_thread_ref_t,
1921 category_ref: *const trace_string_ref_t,
1922 name_ref: *const trace_string_ref_t,
1923 async_id: trace_async_id_t,
1924 args: *const trace_arg_t,
1925 num_args: libc::size_t,
1926 );
1927
1928 pub fn trace_context_write_async_end_event_record(
1929 context: *const trace_context_t,
1930 event_time: trace_ticks_t,
1931 thread_ref: *const trace_thread_ref_t,
1932 category_ref: *const trace_string_ref_t,
1933 name_ref: *const trace_string_ref_t,
1934 async_id: trace_async_id_t,
1935 args: *const trace_arg_t,
1936 num_args: libc::size_t,
1937 );
1938
1939 pub fn trace_context_write_flow_begin_event_record(
1940 context: *const trace_context_t,
1941 event_time: trace_ticks_t,
1942 thread_ref: *const trace_thread_ref_t,
1943 category_ref: *const trace_string_ref_t,
1944 name_ref: *const trace_string_ref_t,
1945 flow_id: trace_flow_id_t,
1946 args: *const trace_arg_t,
1947 num_args: libc::size_t,
1948 );
1949
1950 pub fn trace_context_write_flow_step_event_record(
1951 context: *const trace_context_t,
1952 event_time: trace_ticks_t,
1953 thread_ref: *const trace_thread_ref_t,
1954 category_ref: *const trace_string_ref_t,
1955 name_ref: *const trace_string_ref_t,
1956 flow_id: trace_flow_id_t,
1957 args: *const trace_arg_t,
1958 num_args: libc::size_t,
1959 );
1960
1961 pub fn trace_context_write_flow_end_event_record(
1962 context: *const trace_context_t,
1963 event_time: trace_ticks_t,
1964 thread_ref: *const trace_thread_ref_t,
1965 category_ref: *const trace_string_ref_t,
1966 name_ref: *const trace_string_ref_t,
1967 flow_id: trace_flow_id_t,
1968 args: *const trace_arg_t,
1969 num_args: libc::size_t,
1970 );
1971
1972 pub fn trace_context_write_initialization_record(
1973 context: *const trace_context_t,
1974 ticks_per_second: u64,
1975 );
1976
1977 pub fn trace_context_write_string_record(
1978 context: *const trace_context_t,
1979 index: trace_string_index_t,
1980 string: *const libc::c_char,
1981 length: libc::size_t,
1982 );
1983
1984 pub fn trace_context_write_thread_record(
1985 context: *const trace_context_t,
1986 index: trace_thread_index_t,
1987 procss_koid: zx_koid_t,
1988 thread_koid: zx_koid_t,
1989 );
1990
1991 pub fn trace_context_write_blob_record(
1992 context: *const trace_context_t,
1993 type_: trace_blob_type_t,
1994 name_ref: *const trace_string_ref_t,
1995 data: *const libc::c_void,
1996 size: libc::size_t,
1997 );
1998
1999 pub fn trace_context_alloc_record(
2000 context: *const trace_context_t,
2001 num_bytes: libc::size_t,
2002 ) -> *const libc::c_void;
2003
2004 pub fn trace_stop_engine(disposition: zx_status_t) -> zx_status_t;
2014
2015 pub fn trace_generate_nonce() -> u64;
2018
2019 pub fn trace_state() -> trace_state_t;
2020
2021 pub fn trace_is_category_enabled(category_literal: *const libc::c_char) -> bool;
2022
2023 pub fn trace_acquire_context() -> *const trace_context_t;
2024
2025 pub fn trace_acquire_context_for_category(
2026 category_literal: *const libc::c_char,
2027 out_ref: *mut trace_string_ref_t,
2028 ) -> *const trace_context_t;
2029
2030 pub fn trace_acquire_context_for_category_cached(
2031 category_literal: *const libc::c_char,
2032 trace_site: *const u64,
2033 out_ref: *mut trace_string_ref_t,
2034 ) -> *const trace_context_t;
2035
2036 pub fn trace_release_context(context: *const trace_context_t);
2037
2038 pub fn trace_acquire_prolonged_context() -> *const trace_prolonged_context_t;
2039
2040 pub fn trace_release_prolonged_context(context: *const trace_prolonged_context_t);
2041
2042 pub fn trace_register_observer(event: zx_handle_t) -> zx_status_t;
2043
2044 pub fn trace_unregister_observer(event: zx_handle_t) -> zx_status_t;
2045
2046 pub fn trace_notify_observer_updated(event: zx_handle_t);
2047 }
2048}
2049
2050pub struct TraceFutureArgs<'a> {
2053 pub category: &'static CStr,
2054 pub name: &'static CStr,
2055
2056 pub context: Option<TraceCategoryContext>,
2060
2061 pub args: Vec<Arg<'a>>,
2064
2065 pub flow_id: Option<Id>,
2068
2069 pub _use_trace_future_args: (),
2071}
2072
2073#[doc(hidden)]
2074#[macro_export]
2075macro_rules! __impl_trace_future_args {
2076 ($category:expr, $name:expr, $flow_id:expr $(, $key:expr => $val:expr)*) => {{
2077 {
2078 static CACHE: $crate::trace_site_t = $crate::trace_site_t::new(0);
2079 let context = $crate::TraceCategoryContext::acquire_cached($category, &CACHE);
2080 let args = if context.is_some() {
2081 vec![$($crate::ArgValue::of($key, $val)),*]
2082 } else {
2083 vec![]
2084 };
2085 $crate::TraceFutureArgs {
2086 category: $category,
2087 name: $name,
2088 context: context,
2089 args: args,
2090 flow_id: $flow_id,
2091 _use_trace_future_args: (),
2092 }
2093 }
2094 }};
2095}
2096
2097#[macro_export]
2109macro_rules! trace_future_args {
2110 ($category:expr, $name:expr $(, $key:expr => $val:expr)*) => {
2111 $crate::__impl_trace_future_args!($category, $name, None $(,$key => $val)*)
2112 };
2113 ($category:expr, $name:expr, $flow_id:expr $(, $key:expr => $val:expr)*) => {
2114 $crate::__impl_trace_future_args!($category, $name, Some($flow_id) $(,$key => $val)*)
2115 };
2116}
2117
2118pub trait TraceFutureExt: Future + Sized {
2120 fn trace<'a>(self, args: TraceFutureArgs<'a>) -> TraceFuture<'a, Self> {
2134 TraceFuture::new(args, self)
2135 }
2136}
2137
2138impl<T: Future + Sized> TraceFutureExt for T {}
2139
2140#[pin_project(PinnedDrop)]
2143pub struct TraceFuture<'a, Fut: Future> {
2144 category: &'static CStr,
2145 name: &'static CStr,
2146 args: Vec<Arg<'a>>,
2147 flow_id: Option<Id>,
2148 #[pin]
2149 future: Fut,
2150}
2151
2152impl<'a, Fut: Future> TraceFuture<'a, Fut> {
2153 pub fn new(args: TraceFutureArgs<'a>, future: Fut) -> Self {
2154 debug_assert!(
2155 args.context.is_some() || args.args.is_empty(),
2156 "There should not be any trace arguments when the category is disabled"
2157 );
2158 let mut this = Self {
2159 category: args.category,
2160 name: args.name,
2161 args: args.args,
2162 flow_id: args.flow_id,
2163 future: future,
2164 };
2165 if let Some(context) = args.context {
2166 this.trace_create(context);
2167 }
2168 this
2169 }
2170
2171 #[cold]
2172 fn trace_create(&mut self, context: TraceCategoryContext) {
2173 let name_ref = context.register_string_literal(self.name);
2174 let flow_id = self.flow_id.get_or_insert_with(Id::new);
2175 let duration_start = zx::BootTicks::get();
2176 context.write_flow_begin(zx::BootTicks::get(), name_ref, *flow_id, &[]);
2177 self.args.push(ArgValue::of("state", "created"));
2178 context.write_duration(name_ref, duration_start, &self.args);
2179 self.args.pop();
2180 }
2181
2182 #[cold]
2183 fn trace_poll(
2184 self: Pin<&mut Self>,
2185 context: TraceCategoryContext,
2186 cx: &mut std::task::Context<'_>,
2187 ) -> Poll<Fut::Output> {
2188 let this = self.project();
2189 let name_ref = context.register_string_literal(this.name);
2190 let flow_id = this.flow_id.get_or_insert_with(Id::new);
2191 let duration_start = zx::BootTicks::get();
2192 context.write_flow_step(zx::BootTicks::get(), name_ref, *flow_id, &[]);
2193 let result = this.future.poll(cx);
2194 let result_str: &'static str = if result.is_pending() { "pending" } else { "ready" };
2195 this.args.push(ArgValue::of("state", result_str));
2196 context.write_duration(name_ref, duration_start, &this.args);
2197 this.args.pop();
2198 result
2199 }
2200
2201 #[cold]
2202 fn trace_drop(self: Pin<&mut Self>, context: TraceCategoryContext) {
2203 let this = self.project();
2204 let name_ref = context.register_string_literal(this.name);
2205 let flow_id = this.flow_id.get_or_insert_with(Id::new);
2206 let duration_start = zx::BootTicks::get();
2207 context.write_flow_end(zx::BootTicks::get(), name_ref, *flow_id, &[]);
2208 this.args.push(ArgValue::of("state", "dropped"));
2209 context.write_duration(name_ref, duration_start, &this.args);
2210 this.args.pop();
2211 }
2212}
2213
2214impl<Fut: Future> Future for TraceFuture<'_, Fut> {
2215 type Output = Fut::Output;
2216 fn poll(self: Pin<&mut Self>, cx: &mut std::task::Context<'_>) -> Poll<Fut::Output> {
2217 if let Some(context) = TraceCategoryContext::acquire(self.as_ref().get_ref().category) {
2218 self.trace_poll(context, cx)
2219 } else {
2220 self.project().future.poll(cx)
2221 }
2222 }
2223}
2224
2225#[pin_project::pinned_drop]
2226impl<Fut: Future> PinnedDrop for TraceFuture<'_, Fut> {
2227 fn drop(self: Pin<&mut Self>) {
2228 if let Some(context) = TraceCategoryContext::acquire(self.as_ref().get_ref().category) {
2229 self.trace_drop(context);
2230 }
2231 }
2232}
2233
2234#[cfg(test)]
2235mod test {
2236 use super::{trim_to_last_char_boundary, Id};
2237
2238 #[test]
2239 fn trim_to_last_char_boundary_trims_to_last_character_boundary() {
2240 assert_eq!(b"x", trim_to_last_char_boundary("x", 5));
2241 assert_eq!(b"x", trim_to_last_char_boundary("x", 1));
2242 assert_eq!(b"", trim_to_last_char_boundary("x", 0));
2243 assert_eq!(b"xxxxx", trim_to_last_char_boundary("xxxxx", 6));
2244 assert_eq!(b"xxxxx", trim_to_last_char_boundary("xxxxx", 5));
2245 assert_eq!(b"xxxx", trim_to_last_char_boundary("xxxxx", 4));
2246
2247 assert_eq!("💩".as_bytes(), trim_to_last_char_boundary("💩", 5));
2248 assert_eq!("💩".as_bytes(), trim_to_last_char_boundary("💩", 4));
2249 assert_eq!(b"", trim_to_last_char_boundary("💩", 3));
2250 }
2251
2252 #[test]
2256 fn test_id_new() {
2257 assert_ne!(Id::new(), Id::new());
2258 }
2259}