1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
|
use smallvec::SmallVec;
use crate::{Profiler, StringComponent, StringId};
/// Event IDs are strings conforming to the following grammar:
///
/// ```ignore
/// <event_id> = <label> {<argument>}
/// <label> = <text>
/// <argument> = '\x1E' <text>
/// <text> = regex([[[:^cntrl:]][[:space:]]]+) // Anything but ASCII control characters except for whitespace.
/// ```
///
/// This means there's always a "label", followed by an optional list of
/// arguments. Future versions may support other optional suffixes (with a tag
/// other than '\x11' after the '\x1E' separator), such as a "category".
/// The byte used to separate arguments from the label and each other.
pub const SEPARATOR_BYTE: &str = "\x1E";
/// An `EventId` is a `StringId` with the additional guarantee that the
/// corresponding string conforms to the event_id grammar.
#[derive(Clone, Copy, Eq, PartialEq, Hash, Debug)]
#[repr(C)]
pub struct EventId(StringId);
impl EventId {
pub const INVALID: EventId = EventId(StringId::INVALID);
#[inline]
pub fn to_string_id(self) -> StringId {
self.0
}
#[inline]
pub fn as_u32(self) -> u32 {
self.0.as_u32()
}
#[inline]
pub fn from_label(label: StringId) -> EventId {
EventId(label)
}
#[inline]
pub fn from_virtual(virtual_id: StringId) -> EventId {
EventId(virtual_id)
}
/// Create an EventId from a raw u32 value. Only used internally for
/// deserialization.
#[inline]
pub fn from_u32(raw_id: u32) -> EventId {
EventId(StringId::new(raw_id))
}
}
pub struct EventIdBuilder<'p> {
profiler: &'p Profiler,
}
impl<'p> EventIdBuilder<'p> {
pub fn new(profiler: &Profiler) -> EventIdBuilder<'_> {
EventIdBuilder { profiler }
}
#[inline]
pub fn from_label(&self, label: StringId) -> EventId {
// Just forward the string ID, a single identifier is a valid event_id
EventId::from_label(label)
}
pub fn from_label_and_arg(&self, label: StringId, arg: StringId) -> EventId {
EventId(self.profiler.alloc_string(&[
// Label
StringComponent::Ref(label),
// Seperator and start tag for arg
StringComponent::Value(SEPARATOR_BYTE),
// Arg string id
StringComponent::Ref(arg),
]))
}
pub fn from_label_and_args(&self, label: StringId, args: &[StringId]) -> EventId {
// Store up to 7 components on the stack: 1 label + 3 arguments + 3 argument separators
let mut parts = SmallVec::<[StringComponent<'_>; 7]>::with_capacity(1 + args.len() * 2);
parts.push(StringComponent::Ref(label));
for arg in args {
parts.push(StringComponent::Value(SEPARATOR_BYTE));
parts.push(StringComponent::Ref(*arg));
}
EventId(self.profiler.alloc_string(&parts[..]))
}
}
|