#include "arch/devices/local_apic.hpp" #include "kapi/devices.hpp" #include "kapi/memory.hpp" #include #include #include namespace arch::devices { namespace { constexpr auto lapic_sivr_register = 0x00F0uz; constexpr auto lapic_enable_bit = 0x100u; constexpr auto spurious_interrupt_vector = 0xFFu; } // namespace local_apic::local_apic(std::size_t major, std::size_t minor, std::uint64_t hardware_id, kapi::memory::physical_address base, bool is_bsp) : kapi::devices::device{major, minor, "lapic"} , m_hardware_id{hardware_id} , m_base{base} , m_is_bsp{is_bsp} {} auto local_apic::init() -> bool { auto static shared_virtual_base = kapi::memory::allocate_mmio_region(1); auto static is_mapped = false; if (!is_mapped) { if (!kapi::memory::map_mmio_region(shared_virtual_base, m_base, kapi::memory::page_mapper::flags::writable)) { kstd::println("[x86_64:DEV] LAPIC {} MMIO mapping failed!", m_hardware_id); return false; } is_mapped = true; } m_virtual_base = shared_virtual_base; if (m_is_bsp) { write_register(lapic_sivr_register, lapic_enable_bit | spurious_interrupt_vector); kstd::println("[x86_64:DEV] LAPIC {} initialized. {:#018x}@{:#018x}", m_hardware_id, m_base, m_virtual_base); } else { kstd::println("[x86_64:DEV] LAPIC {} is not on the BSP, deferring intialization.", m_hardware_id); } return true; } auto local_apic::read_register(std::size_t offset) const -> std::uint32_t { auto reg = static_cast(m_virtual_base + offset); return *reg; } auto local_apic::write_register(std::size_t offset, std::uint32_t value) -> void { auto reg = static_cast(m_virtual_base + offset); *reg = value; } } // namespace arch::devices