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
#[cfg(target_arch = "x86")]
use core::arch::x86::_rdtsc;
#[cfg(target_arch = "x86_64")]
use core::arch::x86_64::_rdtsc;
lazy_static::lazy_static! {
static ref CYCLES_PER_SEC: u64 = init_cycles_per_sec();
static ref OFFSET_INSTANT: std::time::Instant = std::time::Instant::now();
}
#[inline]
pub(crate) fn real_time_ns() -> u64 {
std::time::SystemTime::now()
.duration_since(std::time::SystemTime::UNIX_EPOCH)
.expect("SystemTime before UNIX EPOCH!")
.as_nanos() as u64
}
#[cfg(any(target_arch = "x86", target_arch = "x86_64"))]
#[inline]
pub(crate) fn monotonic_cycles() -> u64 {
unsafe { _rdtsc() }
}
#[cfg(not(any(target_arch = "x86", target_arch = "x86_64")))]
#[inline]
pub(crate) fn monotonic_cycles() -> u64 {
(*OFFSET_INSTANT).elapsed().as_nanos() as u64
}
#[inline]
pub fn cycles_per_sec() -> u64 {
*CYCLES_PER_SEC
}
#[cfg(any(target_arch = "x86", target_arch = "x86_64"))]
fn init_cycles_per_sec() -> u64 {
let mut cycles_per_sec;
let mut old_cycles = 0.0;
loop {
let time1 = std::time::Instant::now();
let start_cycles = unsafe { _rdtsc() };
loop {
let duration = time1.elapsed();
let stop_cycles = unsafe { _rdtsc() };
let micros = duration.as_micros();
if micros > 10_000 {
cycles_per_sec = 1_000_000.0 * (stop_cycles - start_cycles) as f64 / micros as f64;
break;
}
}
let delta = cycles_per_sec / 100_000.0;
if old_cycles > (cycles_per_sec - delta) && old_cycles < (cycles_per_sec + delta) {
break;
}
old_cycles = cycles_per_sec;
}
cycles_per_sec.round() as u64
}
#[cfg(not(any(target_arch = "x86", target_arch = "x86_64")))]
fn init_cycles_per_sec() -> u64 {
1_000_000_000
}