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
mod alloc;
mod any;
mod r#box;
mod cell;
mod collection;
mod marker;
mod option;
mod path;
mod primitive;
mod ptr;
mod remote;
mod result;
mod slice;
mod string;
mod sync;
pub use alloc::*;
pub use any::*;
pub use cell::*;
pub use collection::*;
pub use marker::*;
pub use option::*;
pub use path::*;
pub use primitive::*;
pub use ptr::*;
pub use r#box::*;
pub use remote::*;
pub use result::*;
pub use slice::*;
pub use string::*;
pub use sync::*;
pub const POINTER_BYTE_SIZE: usize = if cfg!(target_pointer_width = "16") {
2
} else if cfg!(target_pointer_width = "32") {
4
} else {
8
};
pub trait MemoryUsageTracker {
fn track(&mut self, address: *const ()) -> bool;
}
impl MemoryUsageTracker for std::collections::BTreeSet<*const ()> {
fn track(&mut self, address: *const ()) -> bool {
self.insert(address)
}
}
impl MemoryUsageTracker for std::collections::HashSet<*const ()> {
fn track(&mut self, address: *const ()) -> bool {
self.insert(address)
}
}
pub trait MemoryUsage {
fn size_of_val(&self, tracker: &mut dyn MemoryUsageTracker) -> usize;
}
#[macro_export]
macro_rules! assert_size_of_val_eq {
($value:expr, $expected:expr $(,)*) => {
assert_size_of_val_eq!($value, $expected, &mut std::collections::BTreeSet::new());
};
($value:expr, $expected:expr, $tracker:expr $(,)*) => {
assert_eq!(
$crate::MemoryUsage::size_of_val(&$value, $tracker),
$expected
);
};
}