use bit_field::BitField;
use super::*;
use crate::msr::{
rdmsr, wrmsr, IA32_APIC_BASE, IA32_TSC_DEADLINE, IA32_X2APIC_APICID, IA32_X2APIC_EOI,
IA32_X2APIC_ESR, IA32_X2APIC_ICR, IA32_X2APIC_LDR, IA32_X2APIC_LVT_LINT0,
IA32_X2APIC_LVT_TIMER, IA32_X2APIC_SELF_IPI, IA32_X2APIC_SIVR, IA32_X2APIC_VERSION,
};
#[derive(Debug)]
pub struct X2APIC {
base: u64,
}
impl Default for X2APIC {
fn default() -> Self {
X2APIC { base: 0x0 }
}
}
impl X2APIC {
pub const fn new() -> Self {
X2APIC { base: 0x0 }
}
pub fn attach(&mut self) {
unsafe {
self.base = rdmsr(IA32_APIC_BASE);
self.base.set_bit(10, true); self.base.set_bit(11, true); wrmsr(IA32_APIC_BASE, self.base);
let svr: u64 = 1 << 8 | 15;
wrmsr(IA32_X2APIC_SIVR, svr);
let lint0 = 1 << 16 | (1 << 15) | (0b111 << 8) | 0x20;
wrmsr(IA32_X2APIC_LVT_LINT0, lint0);
let _esr = rdmsr(IA32_X2APIC_ESR);
}
}
pub fn detach(&mut self) {
unsafe {
self.base = rdmsr(IA32_APIC_BASE);
self.base.set_bit(10, false); self.base.set_bit(11, false); wrmsr(IA32_APIC_BASE, self.base);
}
}
pub unsafe fn send_self_ipi(&self, vector: u64) {
wrmsr(IA32_X2APIC_SELF_IPI, vector);
}
}
impl ApicControl for X2APIC {
fn bsp(&self) -> bool {
(self.base & (1 << 8)) > 0
}
fn id(&self) -> u32 {
unsafe { rdmsr(IA32_X2APIC_APICID) as u32 }
}
fn logical_id(&self) -> u32 {
unsafe { rdmsr(IA32_X2APIC_LDR) as u32 }
}
fn version(&self) -> u32 {
unsafe { rdmsr(IA32_X2APIC_VERSION) as u32 }
}
fn tsc_enable(&mut self, vector: u8) {
unsafe {
wrmsr(IA32_TSC_DEADLINE, 0);
let mut lvt: u64 = rdmsr(IA32_X2APIC_LVT_TIMER);
lvt &= !0xff;
lvt |= vector as u64;
lvt.set_bit(16, false);
lvt.set_bit(17, false);
lvt.set_bit(18, true);
wrmsr(IA32_X2APIC_LVT_TIMER, lvt);
}
}
fn tsc_set(&self, value: u64) {
unsafe {
crate::fence::mfence();
wrmsr(IA32_TSC_DEADLINE, value);
}
}
fn eoi(&mut self) {
unsafe {
wrmsr(IA32_X2APIC_EOI, 0);
}
}
unsafe fn ipi_init(&mut self, core: ApicId) {
let icr = Icr::for_x2apic(
0,
core,
DestinationShorthand::NoShorthand,
DeliveryMode::Init,
DestinationMode::Physical,
DeliveryStatus::Idle,
Level::Assert,
TriggerMode::Level,
);
self.send_ipi(icr);
}
unsafe fn ipi_init_deassert(&mut self) {
let icr = Icr::for_x2apic(
0,
ApicId::X2Apic(0),
DestinationShorthand::AllIncludingSelf,
DeliveryMode::Init,
DestinationMode::Physical,
DeliveryStatus::Idle,
Level::Deassert,
TriggerMode::Level,
);
self.send_ipi(icr);
}
unsafe fn ipi_startup(&mut self, core: ApicId, start_page: u8) {
let icr = Icr::for_x2apic(
start_page,
core,
DestinationShorthand::NoShorthand,
DeliveryMode::StartUp,
DestinationMode::Physical,
DeliveryStatus::Idle,
Level::Assert,
TriggerMode::Edge,
);
self.send_ipi(icr);
}
unsafe fn send_ipi(&mut self, icr: Icr) {
wrmsr(IA32_X2APIC_ESR, 0);
wrmsr(IA32_X2APIC_ESR, 0);
wrmsr(IA32_X2APIC_ICR, icr.0);
loop {
let icr = rdmsr(IA32_X2APIC_ICR);
if (icr >> 12 & 0x1) == 0 {
break;
}
if rdmsr(IA32_X2APIC_ESR) > 0 {
break;
}
}
}
}