use core::fmt;
use lazyinit::LazyInit;
use x86_64::instructions::tables::{lgdt, load_tss};
use x86_64::registers::segmentation::{Segment, SegmentSelector, CS};
use x86_64::structures::gdt::{Descriptor, DescriptorFlags};
use x86_64::structures::{tss::TaskStateSegment, DescriptorTablePointer};
use x86_64::{addr::VirtAddr, PrivilegeLevel};
#[no_mangle]
#[percpu::def_percpu]
static TSS: TaskStateSegment = TaskStateSegment::new();
#[percpu::def_percpu]
static GDT: LazyInit<GdtStruct> = LazyInit::new();
#[repr(align(16))]
pub struct GdtStruct {
table: [u64; 16],
}
impl GdtStruct {
pub const KCODE32_SELECTOR: SegmentSelector = SegmentSelector::new(1, PrivilegeLevel::Ring0);
pub const KCODE64_SELECTOR: SegmentSelector = SegmentSelector::new(2, PrivilegeLevel::Ring0);
pub const KDATA_SELECTOR: SegmentSelector = SegmentSelector::new(3, PrivilegeLevel::Ring0);
pub const UCODE32_SELECTOR: SegmentSelector = SegmentSelector::new(4, PrivilegeLevel::Ring3);
pub const UDATA_SELECTOR: SegmentSelector = SegmentSelector::new(5, PrivilegeLevel::Ring3);
pub const UCODE64_SELECTOR: SegmentSelector = SegmentSelector::new(6, PrivilegeLevel::Ring3);
pub const TSS_SELECTOR: SegmentSelector = SegmentSelector::new(7, PrivilegeLevel::Ring0);
pub fn new(tss: &'static TaskStateSegment) -> Self {
let mut table = [0; 16];
table[1] = DescriptorFlags::KERNEL_CODE32.bits(); table[2] = DescriptorFlags::KERNEL_CODE64.bits(); table[3] = DescriptorFlags::KERNEL_DATA.bits(); table[4] = DescriptorFlags::USER_CODE32.bits(); table[5] = DescriptorFlags::USER_DATA.bits(); table[6] = DescriptorFlags::USER_CODE64.bits(); if let Descriptor::SystemSegment(low, high) = Descriptor::tss_segment(tss) {
table[7] = low;
table[8] = high;
}
Self { table }
}
pub fn pointer(&self) -> DescriptorTablePointer {
DescriptorTablePointer {
base: VirtAddr::new(self.table.as_ptr() as u64),
limit: (core::mem::size_of_val(&self.table) - 1) as u16,
}
}
pub unsafe fn load(&'static self) {
lgdt(&self.pointer());
CS::set_reg(Self::KCODE64_SELECTOR);
}
pub unsafe fn load_tss(&'static self) {
load_tss(Self::TSS_SELECTOR);
}
}
impl fmt::Debug for GdtStruct {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.debug_struct("GdtStruct")
.field("pointer", &self.pointer())
.field("table", &self.table)
.finish()
}
}
pub fn init_gdt() {
unsafe {
let gdt = GDT.current_ref_raw();
gdt.init_once(GdtStruct::new(TSS.current_ref_raw()));
gdt.load();
gdt.load_tss();
}
}
pub fn tss_get_rsp0() -> memory_addr::VirtAddr {
let tss = unsafe { TSS.current_ref_raw() };
memory_addr::VirtAddr::from(tss.privilege_stack_table[0].as_u64() as usize)
}
pub unsafe fn tss_set_rsp0(rsp0: memory_addr::VirtAddr) {
let tss = unsafe { TSS.current_ref_mut_raw() };
tss.privilege_stack_table[0] = VirtAddr::new_truncate(rsp0.as_usize() as u64);
}