Skip to content

Commit

Permalink
x86_64: enable APIC and interrupts and add test code for system timer
Browse files Browse the repository at this point in the history
  • Loading branch information
Qix- committed Sep 13, 2024
1 parent 1eb9209 commit 7492f4b
Show file tree
Hide file tree
Showing 4 changed files with 490 additions and 2 deletions.
16 changes: 14 additions & 2 deletions oro-arch-x86_64/src/init.rs
Original file line number Diff line number Diff line change
Expand Up @@ -55,14 +55,26 @@ pub unsafe fn initialize_primary(pat: OffsetTranslator, pfa: crate::Pfa) {
/// (i.e. boot, or powerdown/subsequent bringup).
pub unsafe fn boot(lapic: Lapic) -> ! {
// SAFETY(qix-): THIS MUST ABSOLUTELY BE FIRST.
let _kernel = crate::Kernel::initialize_for_core(
let kernel = crate::Kernel::initialize_for_core(
KERNEL_STATE.assume_init_ref(),
crate::CoreState { lapic },
)
.expect("failed to initialize kernel");

crate::interrupt::install_idt();

dbg!("boot");

crate::asm::hang();
// XXX DEBUG
let id = kernel.core().lapic.id();
loop {
dbg!(
"DBG: timer counter: {id}: {}",
crate::interrupt::TIM_COUNT.load(core::sync::atomic::Ordering::Relaxed)
);

kernel.core().lapic.set_timer_initial_count(1_000_000);

crate::asm::halt_once();
}
}
187 changes: 187 additions & 0 deletions oro-arch-x86_64/src/interrupt.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,187 @@
//! Interrupt handling for x86_64 architecture.
use crate::lapic::{ApicSvr, ApicTimerConfig, ApicTimerMode};

/// A single IDT (Interrupt Descriptor Table) entry.
#[derive(Debug, Clone, Copy, Default)]
#[repr(C, packed)]
struct IdtEntry {
/// The lower 16 bits of the ISR (Interrupt Service Routine) address.
isr_low: u16,
/// The code segment selector used when handling the interrupt.
kernel_cs: u16,
/// The IST (Interrupt Stack Table) index.
ist: u8,
/// Attributes and flags for the IDT entry, including type.
attributes: u8,
/// The middle 16 bits of the ISR address.
isr_mid: u16,
/// The higher 32 bits of the ISR address.
isr_high: u32,
/// Reserved.
_reserved: u32,
}

impl IdtEntry {
/// Creates a new, empty IDT entry.
pub const fn new() -> Self {
Self {
isr_low: 0,
kernel_cs: 0,
ist: 0,
attributes: 0,
isr_mid: 0,
isr_high: 0,
_reserved: 0,
}
}

/// Sets the ISR address for the IDT entry.
///
/// # Safety
/// Caller must ensure that the given address is
/// a real function that is suitable for handling
/// the interrupt.
pub const unsafe fn with_isr_raw(mut self, isr: u64) -> Self {
self.isr_low = isr as u16;
self.isr_mid = (isr >> 16) as u16;
self.isr_high = (isr >> 32) as u32;

self
}

/// Sets the ISR handler as a function pointer.
pub fn with_isr(self, isr: unsafe extern "C" fn() -> !) -> Self {
unsafe { self.with_isr_raw(isr as usize as u64) }
}

/// Sets the code segment selector for the IDT entry.
///
/// There is no configurable index for this; it's always
/// set to 0x08 (the kernel code segment).
pub const fn with_kernel_cs(mut self) -> Self {
self.kernel_cs = 0x08;
self
}

/// Sets the attributes for the IDT entry.
pub const fn with_attributes(mut self, attributes: u8) -> Self {
self.attributes = attributes;
self
}
}

/// The IDT (Interrupt Descriptor Table) for the kernel.
static mut IDT: Aligned16<[IdtEntry; 256]> = Aligned16([IdtEntry::new(); 256]);

/// XXX DEBUG
pub static TIM_COUNT: core::sync::atomic::AtomicU64 = core::sync::atomic::AtomicU64::new(0);

/// The ISR (Interrupt Service Routine) for the system timer.
#[no_mangle]
unsafe extern "C" fn isr_sys_timer_rust() {
TIM_COUNT.fetch_add(1, core::sync::atomic::Ordering::Relaxed);
crate::Kernel::get().core().lapic.eoi();
}

/// The ISR (Interrupt Service Routine) trampoline stub for the system timer.
#[naked]
unsafe extern "C" fn isr_sys_timer() -> ! {
core::arch::asm!(
"cli",
"call isr_sys_timer_rust",
"sti",
"iretq",
options(noreturn)
);
}

/// The ISR (Interrupt Service Routine) for the APIC spurious interrupt.
#[no_mangle]
unsafe extern "C" fn isr_apic_svr_rust() {
crate::Kernel::get().core().lapic.eoi();
}

/// The ISR (Interrupt Service Routine) trampoline stub for the APIC spurious interrupt.
#[naked]
unsafe extern "C" fn isr_apic_svr() -> ! {
core::arch::asm!(
"cli",
"call isr_apic_svr_rust",
"sti",
"iretq",
options(noreturn)
);
}

/// Aligns a `T` value to 16 bytes.
#[repr(C, align(16))]
struct Aligned16<T: Sized>(pub T);

/// The vector for the main system timer interrupt.
const TIMER_VECTOR: u8 = 32;
/// The vector for the APIC spurious interrupt.
const APIC_SVR_VECTOR: u8 = 255;

/// Installs the IDT (Interrupt Descriptor Table) for the kernel
/// and enables interrupts.
///
/// # Safety
/// Modifies global state, and must be called only once.
///
/// The kernel MUST be fully initialized before calling this function.
pub unsafe fn install_idt() {
// Get the LAPIC.
let lapic = &crate::Kernel::get().core().lapic;

/// The IDTR (Interrupt Descriptor Table Register) structure,
/// read in by the `lidt` instruction.
#[repr(C, packed)]
struct Idtr {
/// How long the IDT is in bytes, minus 1.
limit: u16,
/// The base address of the IDT.
base: *const IdtEntry,
}

#[allow(static_mut_refs)]
let idtr = Idtr {
limit: (core::mem::size_of_val(&IDT) - 1) as u16,
base: IDT.0.as_ptr(),
};

core::arch::asm!(
"lidt [{}]",
"sti",
in(reg) &idtr,
options(nostack, preserves_flags)
);

// Set up the main system timer.
IDT.0[usize::from(TIMER_VECTOR)] = IdtEntry::new()
.with_kernel_cs()
.with_attributes(0x8E)
.with_isr(isr_sys_timer);

lapic.set_timer_divider(crate::lapic::ApicTimerDivideBy::Div128);

// Note: this also enables the timer interrupts
lapic.configure_timer(
ApicTimerConfig::new()
.with_vector(TIMER_VECTOR)
.with_mode(ApicTimerMode::OneShot),
);

// Set up the APIC spurious interrupt.
// This also enables the APIC if it isn't already.
IDT.0[usize::from(APIC_SVR_VECTOR)] = IdtEntry::new()
.with_kernel_cs()
.with_attributes(0x8E)
.with_isr(isr_apic_svr);

lapic.set_spurious_vector(
ApicSvr::new()
.with_vector(APIC_SVR_VECTOR)
.with_software_enable(),
);
}
Loading

0 comments on commit 7492f4b

Please sign in to comment.