blob: e51cf081c784b4eb045665cff0db5e9ab3afd96f (
plain) (
blame)
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
|
//! Functions to read time stamp counters on x86.
use crate::arch::{__rdtscp, _rdtsc};
/// Read the time stamp counter.
///
/// The RDTSC instruction is not a serializing instruction.
/// It does not necessarily wait until all previous instructions
/// have been executed before reading the counter. Similarly,
/// subsequent instructions may begin execution before the
/// read operation is performed. If software requires RDTSC to be
/// executed only after all previous instructions have completed locally,
/// it can either use RDTSCP or execute the sequence LFENCE;RDTSC.
///
/// # Safety
/// * Causes a GP fault if the TSD flag in register CR4 is set and the CPL
/// is greater than 0.
pub unsafe fn rdtsc() -> u64 {
_rdtsc() as u64
}
/// Read the time stamp counter.
///
/// The RDTSCP instruction waits until all previous instructions
/// have been executed before reading the counter.
/// However, subsequent instructions may begin execution
/// before the read operation is performed.
///
/// Volatile is used here because the function may be used to act as
/// an instruction barrier.
///
/// # Safety
/// * Causes a GP fault if the TSD flag in register CR4 is set and the
/// CPL is greater than 0.
pub unsafe fn rdtscp() -> u64 {
let mut _aux = 0;
__rdtscp(&mut _aux)
}
#[cfg(all(test, feature = "utest"))]
mod test {
use super::*;
#[test]
fn check_rdtsc() {
let cpuid = crate::cpuid::CpuId::new();
let has_tsc = cpuid
.get_feature_info()
.map_or(false, |finfo| finfo.has_tsc());
if has_tsc {
unsafe {
assert!(rdtsc() > 0, "rdtsc returned 0, unlikely!");
}
}
}
#[test]
fn check_rdtscp() {
let cpuid = crate::cpuid::CpuId::new();
let has_tsc = cpuid
.get_feature_info()
.map_or(false, |finfo| finfo.has_tsc());
if has_tsc {
unsafe {
assert!(rdtscp() > 0, "rdtscp returned 0, unlikely!");
}
}
}
}
|