/* * Copyright (c) 2020-2021, Liav A. * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include #include #include #include #include namespace Kernel::ACPI { class ACPISysFSDirectory : public SysFSDirectory { public: virtual StringView name() const override { return "acpi"sv; } static NonnullRefPtr must_create(FirmwareSysFSDirectory& firmware_directory); private: void find_tables_and_register_them_as_components(); explicit ACPISysFSDirectory(FirmwareSysFSDirectory& firmware_directory); }; class ACPISysFSComponent : public SysFSComponent { public: static NonnullRefPtr create(StringView name, PhysicalAddress, size_t table_size); virtual StringView name() const override { return m_table_name->view(); } virtual ErrorOr read_bytes(off_t, size_t, UserOrKernelBuffer&, OpenFileDescription*) const override; protected: ErrorOr> try_to_generate_buffer() const; ACPISysFSComponent(NonnullOwnPtr table_name, PhysicalAddress, size_t table_size); PhysicalAddress m_paddr; size_t m_length; NonnullOwnPtr m_table_name; }; class Parser final : public IRQHandler { public: static Parser* the(); static void must_initialize(PhysicalAddress rsdp, PhysicalAddress fadt, u8 irq_number); virtual StringView purpose() const override { return "ACPI Parser"sv; } virtual bool handle_irq(RegisterState const&) override; Optional find_table(StringView signature); void try_acpi_reboot(); bool can_reboot(); void try_acpi_shutdown(); bool can_shutdown() { return false; } void enable_aml_parsing(); PhysicalAddress rsdp() const { return m_rsdp; } PhysicalAddress main_system_description_table() const { return m_main_system_description_table; } bool is_xsdt_supported() const { return m_xsdt_supported; } void enumerate_static_tables(Function); virtual bool have_8042() const { return m_x86_specific_flags.keyboard_8042; } FADTFlags::HardwareFeatures const& hardware_features() const { return m_hardware_flags; } FADTFlags::x86_Specific_Flags const& x86_specific_flags() const { return m_x86_specific_flags; } ~Parser() = default; private: Parser(PhysicalAddress rsdp, PhysicalAddress fadt, u8 irq_number); void locate_static_data(); void locate_main_system_description_table(); void initialize_main_system_description_table(); size_t get_table_size(PhysicalAddress); u8 get_table_revision(PhysicalAddress); void process_fadt_data(); bool validate_reset_register(Memory::TypedMapping const&); void access_generic_address(Structures::GenericAddressStructure const&, u32 value); PhysicalAddress m_rsdp; PhysicalAddress m_main_system_description_table; Vector m_sdt_pointers; PhysicalAddress m_fadt; bool m_xsdt_supported { false }; bool m_can_process_bytecode { false }; FADTFlags::HardwareFeatures m_hardware_flags; FADTFlags::x86_Specific_Flags m_x86_specific_flags; }; }