/* * Copyright (c) 2020, Liav A. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #pragma once #include #include #include #include #include #include #include #include #include #include namespace Kernel { class ISAInterruptOverrideMetadata { public: ISAInterruptOverrideMetadata(u8 bus, u8 source, u32 global_system_interrupt, u16 flags) : m_bus(bus) , m_source(source) , m_global_system_interrupt(global_system_interrupt) , m_flags(flags) { } u8 bus() const { return m_bus; } u8 source() const { return m_source; } u32 gsi() const { return m_global_system_interrupt; } u16 flags() const { return m_flags; } private: const u8 m_bus; const u8 m_source; const u32 m_global_system_interrupt; const u16 m_flags; }; class InterruptManagement { public: static InterruptManagement& the(); static void initialize(); static bool initialized(); static u8 acquire_mapped_interrupt_number(u8 original_irq); static u8 acquire_irq_number(u8 mapped_interrupt_vector); virtual void switch_to_pic_mode(); virtual void switch_to_ioapic_mode(); bool smp_enabled() const { return m_smp_enabled; } RefPtr get_responsible_irq_controller(u8 interrupt_vector); const Vector& isa_overrides() const { return m_isa_interrupt_overrides; } u8 get_mapped_interrupt_vector(u8 original_irq); u8 get_irq_vector(u8 mapped_interrupt_vector); void enumerate_interrupt_handlers(Function); IRQController& get_interrupt_controller(int index); private: InterruptManagement(); PhysicalAddress search_for_madt(); void locate_apic_data(); bool m_smp_enabled { false }; Vector> m_interrupt_controllers; Vector m_isa_interrupt_overrides; Vector m_pci_interrupt_overrides; PhysicalAddress m_madt; }; }