76 lines
1.7 KiB
C++
76 lines
1.7 KiB
C++
#pragma once
|
|
|
|
#include <BAN/Vector.h>
|
|
#include <kernel/InterruptController.h>
|
|
#include <kernel/Lock/SpinLock.h>
|
|
#include <kernel/Memory/Types.h>
|
|
|
|
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<void> reserve_irq(uint8_t irq) override;
|
|
virtual BAN::Optional<uint8_t> get_free_irq() override;
|
|
|
|
virtual void initialize_multiprocessor() override;
|
|
virtual void send_ipi(ProcessorID target) override;
|
|
virtual void broadcast_ipi() override;
|
|
virtual void enable() override;
|
|
|
|
void initialize_timer();
|
|
|
|
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:
|
|
static constexpr uint8_t m_irq_count = IRQ_MSI_BASE - IRQ_VECTOR_BASE;
|
|
|
|
SpinLock m_lock;
|
|
BAN::Vector<Processor> m_processors;
|
|
Kernel::paddr_t m_local_apic_paddr = 0;
|
|
Kernel::vaddr_t m_local_apic_vaddr = 0;
|
|
BAN::Vector<IOAPIC> m_io_apics;
|
|
uint8_t m_irq_overrides[0x100] {};
|
|
uint8_t m_reserved_gsis[m_irq_count / 8] {};
|
|
uint64_t m_lapic_timer_frequency_hz { 0 };
|
|
};
|
|
|
|
}
|