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
#[cfg(all(target_arch = "x86_64", target_feature = "sse2"))]
use core::arch::x86_64::{__rdtscp, _mm_lfence, _rdtsc};

#[derive(Debug, Clone, Default)]
pub struct Counter;

impl Counter {
    #[allow(dead_code)]
    pub fn new() -> Self {
        Counter {}
    }
}

#[cfg(all(target_arch = "x86_64", target_feature = "sse2"))]
impl Counter {
    pub fn now(&self) -> u64 {
        unsafe {
            _mm_lfence();
            _rdtsc()
        }
    }

    pub fn start(&self) -> u64 {
        unsafe {
            _mm_lfence();
            let result = _rdtsc();
            _mm_lfence();
            result
        }
    }

    pub fn end(&self) -> u64 {
        let mut _aux: u32 = 0;
        unsafe {
            let result = __rdtscp(&mut _aux as *mut _);
            _mm_lfence();
            result
        }
    }
}

#[cfg(not(all(target_arch = "x86_64", target_feature = "sse2")))]
impl Counter {
    pub fn now(&self) -> u64 {
        panic!("can't use counter without TSC support");
    }

    pub fn start(&self) -> u64 {
        panic!("can't use counter without TSC support");
    }

    pub fn end(&self) -> u64 {
        panic!("can't use counter without TSC support");
    }
}