use bitflags::*;
use core::arch::asm;
bitflags! {
pub struct Cr0: usize {
const CR0_ENABLE_PAGING = 1 << 31;
const CR0_CACHE_DISABLE = 1 << 30;
const CR0_NOT_WRITE_THROUGH = 1 << 29;
const CR0_ALIGNMENT_MASK = 1 << 18;
const CR0_WRITE_PROTECT = 1 << 16;
const CR0_NUMERIC_ERROR = 1 << 5;
const CR0_EXTENSION_TYPE = 1 << 4;
const CR0_TASK_SWITCHED = 1 << 3;
const CR0_EMULATE_COPROCESSOR = 1 << 2;
const CR0_MONITOR_COPROCESSOR = 1 << 1;
const CR0_PROTECTED_MODE = 1 << 0;
}
}
bitflags! {
pub struct Cr4: usize {
const CR4_ENABLE_PROTECTION_KEY = 1 << 22;
const CR4_ENABLE_SMAP = 1 << 21;
const CR4_ENABLE_SMEP = 1 << 20;
const CR4_ENABLE_OS_XSAVE = 1 << 18;
const CR4_ENABLE_PCID = 1 << 17;
const CR4_ENABLE_FSGSBASE = 1 << 16;
const CR4_ENABLE_SMX = 1 << 14;
const CR4_ENABLE_VMX = 1 << 13;
const CR4_ENABLE_LA57 = 1 << 12;
const CR4_ENABLE_UMIP = 1 << 11;
const CR4_UNMASKED_SSE = 1 << 10;
const CR4_ENABLE_SSE = 1 << 9;
const CR4_ENABLE_PPMC = 1 << 8;
const CR4_ENABLE_GLOBAL_PAGES = 1 << 7;
const CR4_ENABLE_MACHINE_CHECK = 1 << 6;
const CR4_ENABLE_PAE = 1 << 5;
const CR4_ENABLE_PSE = 1 << 4;
const CR4_DEBUGGING_EXTENSIONS = 1 << 3;
const CR4_TIME_STAMP_DISABLE = 1 << 2;
const CR4_VIRTUAL_INTERRUPTS = 1 << 1;
const CR4_ENABLE_VME = 1 << 0;
}
}
bitflags! {
pub struct Xcr0: u64 {
const XCR0_PKRU_STATE = 1 << 9;
const XCR0_HI16_ZMM_STATE = 1 << 7;
const XCR0_ZMM_HI256_STATE = 1 << 6;
const XCR0_OPMASK_STATE = 1 << 5;
const XCR0_BNDCSR_STATE = 1 << 4;
const XCR0_BNDREG_STATE = 1 << 3;
const XCR0_AVX_STATE = 1 << 2;
const XCR0_SSE_STATE = 1 << 1;
const XCR0_FPU_MMX_STATE = 1 << 0;
}
}
pub fn cr0() -> Cr0 {
let ret: usize;
unsafe { asm!("mov %cr0, {0}", out(reg) ret, options(att_syntax)) };
Cr0::from_bits_truncate(ret)
}
pub fn cr0_write(val: Cr0) {
unsafe { asm!("mov {0}, %cr0", in(reg) val.bits, options(att_syntax)) };
}
pub fn cr2() -> usize {
let ret: usize;
unsafe { asm!("mov %cr2, {0}", out(reg) ret, options(att_syntax)) };
ret
}
pub fn cr2_write(val: u64) {
unsafe { asm!("mov {0}, %cr2", in(reg) val as usize, options(att_syntax)) };
}
pub fn cr3() -> u64 {
let ret: usize;
unsafe { asm!("mov %cr3, {0}", out(reg) ret, options(att_syntax)) };
ret as u64
}
pub fn cr3_write(val: u64) {
unsafe { asm!("mov {0}, %cr3", in(reg) val as usize, options(att_syntax)) };
}
pub fn cr4() -> Cr4 {
let ret: usize;
unsafe { asm!("mov %cr4, {0}", out(reg) ret, options(att_syntax)) };
Cr4::from_bits_truncate(ret)
}
pub fn cr4_write(val: Cr4) {
unsafe { asm!("mov {0}, %cr4", in(reg) val.bits, options(att_syntax)) };
}
pub fn xcr0() -> Xcr0 {
Xcr0::from_bits_truncate(unsafe { _xgetbv(0) })
}
pub fn xcr0_write(val: Xcr0) {
unsafe { _xsetbv(0, val.bits) };
}
#[allow(improper_ctypes)]
unsafe extern "C" {
#[link_name = "llvm.x86.xsave"]
pub fn xsave(p: *mut u8, hi: u32, lo: u32);
#[link_name = "llvm.x86.xrstor"]
pub fn xrstor(p: *const u8, hi: u32, lo: u32);
#[link_name = "llvm.x86.xsetbv"]
pub fn xsetbv(v: u32, hi: u32, lo: u32);
#[link_name = "llvm.x86.xgetbv"]
pub fn xgetbv(v: u32) -> i64;
#[link_name = "llvm.x86.xsaveopt"]
pub fn xsaveopt(p: *mut u8, hi: u32, lo: u32);
#[link_name = "llvm.x86.xsavec"]
pub fn xsavec(p: *mut u8, hi: u32, lo: u32);
#[link_name = "llvm.x86.xsaves"]
pub fn xsaves(p: *mut u8, hi: u32, lo: u32);
#[link_name = "llvm.x86.xrstors"]
pub fn xrstors(p: *const u8, hi: u32, lo: u32);
}
#[inline]
#[target_feature(enable = "xsave")]
pub fn _xsetbv(a: u32, val: u64) {
unsafe { xsetbv(a, (val >> 32) as u32, val as u32) };
}
#[inline]
#[target_feature(enable = "xsave")]
pub fn _xgetbv(xcr_no: u32) -> u64 {
unsafe { xgetbv(xcr_no) as u64 }
}