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
//! Low level functions for special x86 instructions. pub mod port; pub mod interrupts; pub mod tables; pub mod tlb; pub mod segmentation; /// Halts the CPU by executing the `hlt` instruction. #[inline(always)] pub unsafe fn halt() { asm!("hlt" :::: "volatile"); } /// Read time stamp counters /// Read the time stamp counter using the `RDTSC` instruction. /// /// 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`. pub fn rdtsc() -> u64 { let low: u32; let high: u32; unsafe { asm!("rdtsc" : "={eax}" (low), "={edx}" (high)); } ((u64::from(high)) << 32) | (u64::from(low)) } /// Read the time stamp counter using the `RDTSCP` instruction. /// /// 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. pub fn rdtscp() -> u64 { let low: u32; let high: u32; unsafe { asm!("rdtscp" : "={eax}" (low), "={edx}" (high) ::: "volatile"); } ((high as u64) << 32) | (low as u64) } // Model specific registers /// Write 64 bits to msr register. pub unsafe fn wrmsr(msr: u32, value: u64) { let low = value as u32; let high = (value >> 32) as u32; asm!("wrmsr" :: "{ecx}" (msr), "{eax}" (low), "{edx}" (high) : "memory" : "volatile" ); } /// Read 64 bits msr register. pub fn rdmsr(msr: u32) -> u64 { let (high, low): (u32, u32); unsafe { asm!("rdmsr" : "={eax}" (low), "={edx}" (high) : "{ecx}" (msr) : "memory" : "volatile"); } ((high as u64) << 32) | (low as u64) }