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
use {
crate::*,
std::{
convert::TryInto,
time::{Duration, Instant},
},
};
pub const WARMUP_ITERATIONS: usize = 2;
pub const ESTIMATE_ITERATIONS: u32 = 5;
pub const OPTIMAL_DURATION_NS: u128 = Duration::from_secs(2).as_nanos();
pub const MINIMAL_ITERATIONS: u32 = 50;
#[derive(Debug, Clone)]
pub struct TaskBench {
pub name: String,
pub measure: Option<TaskMeasure>,
}
impl TaskBench {
pub(crate) fn new(name: String) -> Self {
Self {
name,
measure: None,
}
}
pub(crate) fn diff_with(&self, old_bench: &Bench) -> Option<TaskBenchDiff> {
old_bench.tasks.iter()
.find(|tb| tb.name == self.name)
.and_then(|old_tb| old_tb.measure)
.and_then(|old_mes| {
self.measure.map(|new_mes| TaskBenchDiff::new(old_mes, new_mes))
})
}
pub fn iter<M, R>(&mut self, mut measured: M)
where
M: FnMut() -> R,
{
if self.measure.is_some() {
eprintln!("bench already used - please fix your benchmark");
return;
}
for _ in 0..WARMUP_ITERATIONS {
measured();
}
let start = Instant::now();
for _ in 0..ESTIMATE_ITERATIONS {
measured();
}
let estimate_ns = start.elapsed().as_nanos();
let iterations = ((OPTIMAL_DURATION_NS * ESTIMATE_ITERATIONS as u128) / estimate_ns)
.try_into()
.unwrap_or(MINIMAL_ITERATIONS)
.max(MINIMAL_ITERATIONS);
let start = Instant::now();
for _ in 0..iterations {
measured();
}
let total_duration = start.elapsed();
self.measure = Some(TaskMeasure {
iterations,
total_duration,
});
}
}