#pragma once #include #include #include #include namespace Kernel { class APIC final : public InterruptController { public: virtual void eoi(uint8_t) override; virtual void enable_irq(uint8_t) override; virtual bool is_in_service(uint8_t) override; virtual BAN::ErrorOr reserve_irq(uint8_t irq) override; virtual BAN::Optional get_free_irq() override; virtual void initialize_multiprocessor() override; virtual void broadcast_ipi() override; virtual void enable() override; private: uint32_t read_from_local_apic(ptrdiff_t); void write_to_local_apic(ptrdiff_t, uint32_t); private: ~APIC() { ASSERT_NOT_REACHED(); } static APIC* create(); friend class InterruptController; private: struct Processor { enum Flags : uint8_t { Enabled = 1, OnlineCapable = 2, }; uint8_t processor_id; uint8_t apic_id; uint8_t flags; }; struct IOAPIC { uint8_t id; Kernel::paddr_t paddr; Kernel::vaddr_t vaddr; uint32_t gsi_base; uint8_t max_redirs; uint32_t read(uint8_t offset); void write(uint8_t offset, uint32_t data); }; private: SpinLock m_lock; BAN::Vector m_processors; Kernel::paddr_t m_local_apic_paddr = 0; Kernel::vaddr_t m_local_apic_vaddr = 0; BAN::Vector m_io_apics; uint8_t m_irq_overrides[0x100] {}; uint8_t m_reserved_gsis[0x100 / 8] {}; }; }