diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/bits64/mod.rs | 15 | ||||
-rw-r--r-- | src/bits64/paging.rs | 1 | ||||
-rw-r--r-- | src/shared/control_regs.rs | 36 | ||||
-rw-r--r-- | src/shared/dtables.rs | 24 |
4 files changed, 41 insertions, 35 deletions
diff --git a/src/bits64/mod.rs b/src/bits64/mod.rs index a5dd078..1fcb17e 100644 --- a/src/bits64/mod.rs +++ b/src/bits64/mod.rs @@ -15,21 +15,6 @@ macro_rules! check_flag { ) } -macro_rules! is_bit_set { - ($field:expr, $bit:expr) => ( - $field & (1 << $bit) > 0 - ) -} - -macro_rules! check_bit_fn { - ($doc:meta, $fun:ident, $field:ident, $bit:expr) => ( - #[$doc] - pub fn $fun(&self) -> bool { - is_bit_set!(self.$field, $bit) - } - ) -} - pub mod time; pub mod irq; pub mod paging; diff --git a/src/bits64/paging.rs b/src/bits64/paging.rs index 47ac8ed..8858de8 100644 --- a/src/bits64/paging.rs +++ b/src/bits64/paging.rs @@ -74,6 +74,7 @@ pub type PD = [PDEntry; 512]; pub type PT = [PTEntry; 512]; /// Given virtual address calculate corresponding entry in PML4. +#[inline] pub fn pml4_index(addr: VAddr) -> usize { (addr.as_usize() >> 39) & 0b111111111 } diff --git a/src/shared/control_regs.rs b/src/shared/control_regs.rs index 306e52f..1a1f6a3 100644 --- a/src/shared/control_regs.rs +++ b/src/shared/control_regs.rs @@ -1,5 +1,5 @@ //! Functions to read and write control registers. -//! See Intel Vol. 3a Section 2.5, especially Figure 2.6. +//! See Intel Vol. 3a Section 2.5, especially Figure 2-7. bitflags! { pub flags Cr0: usize { @@ -19,12 +19,15 @@ bitflags! { bitflags! { pub flags 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_UMIP = 1 << 11, const CR4_UNMASKED_SSE = 1 << 10, const CR4_ENABLE_SSE = 1 << 9, const CR4_ENABLE_PPMC = 1 << 8, @@ -39,6 +42,20 @@ bitflags! { } } +bitflags! { + pub flags 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, + } +} + /// Read cr0 pub unsafe fn cr0() -> Cr0 { @@ -82,3 +99,20 @@ pub unsafe fn cr4() -> Cr4 { pub unsafe fn cr4_write(val: Cr4) { asm!("mov $0, %cr4" :: "r" (val.bits) : "memory"); } + +/// Read Extended Control Register XCR0. +/// Only supported if CR4_ENABLE_OS_XSAVE is set. +pub unsafe fn xcr0() -> Xcr0 { + let high: u32; + let low: u32; + asm!("xgetbv" : "={eax}"(low), "={edx}"(high) : "{ecx}"(0)); + Xcr0::from_bits_truncate((high as u64) << 32 | low as u64) +} + +/// Write to Extended Control Register XCR0. +/// Only supported if CR4_ENABLE_OS_XSAVE is set. +pub unsafe fn xcr0_write(val: Xcr0) { + let high: u32 = (val.bits >> 32) as u32; + let low: u32 = val.bits as u32; + asm!("xsetbv" :: "{eax}"(low), "{ecx}"(0), "{edx}"(high)); +} diff --git a/src/shared/dtables.rs b/src/shared/dtables.rs index c5337fb..5ec2117 100644 --- a/src/shared/dtables.rs +++ b/src/shared/dtables.rs @@ -17,8 +17,11 @@ pub struct DescriptorTablePointer<Entry> { } impl<T> DescriptorTablePointer<T> { - fn new(slice: &[T]) -> Self { - let len = slice.len() * size_of::<T>(); + pub fn new(slice: &[T]) -> Self { + // GDT, LDT, and IDT all expect the limit to be set to "one less". + // See Intel 3a, Section 3.5.1 "Segment Descriptor Tables" and + // Section 6.10 "Interrupt Descriptor Table (IDT)". + let len = slice.len() * size_of::<T>() - 1; assert!(len < 0x10000); DescriptorTablePointer { base: slice.as_ptr(), @@ -27,23 +30,6 @@ impl<T> DescriptorTablePointer<T> { } } -impl DescriptorTablePointer<SegmentDescriptor> { - pub fn new_gdtp(gdt: &[SegmentDescriptor]) -> Self { - let mut p = Self::new(gdt); - p.limit -= 1; - p - } - pub fn new_ldtp(ldt: &[SegmentDescriptor]) -> Self { - Self::new(ldt) - } -} - -impl DescriptorTablePointer<IdtEntry> { - pub fn new_idtp(idt: &[IdtEntry]) -> Self { - Self::new(idt) - } -} - /// Load GDT table. pub unsafe fn lgdt(gdt: &DescriptorTablePointer<SegmentDescriptor>) { |