/* * \brief Multiboot 2 handling * \author Alexander Boettcher * \date 2017-08-11 */ /* * Copyright (C) 2017 Genode Labs GmbH * * This file is part of the Genode OS framework, which is distributed * under the terms of the GNU Affero General Public License version 3. */ #ifndef _SRC__BOOTSTRAP__SPEC__X86_64__MULTIBOOT2_H_ #define _SRC__BOOTSTRAP__SPEC__X86_64__MULTIBOOT2_H_ /* base includes */ #include namespace Genode { class Multiboot2_info; } class Genode::Multiboot2_info : Mmio { private: struct Size : Register <0x0, 32> { }; struct Tag : Genode::Mmio { enum { LOG2_SIZE = 3 }; struct Type : Register <0x00, 32> { enum { END = 0, MEMORY = 6, FRAMEBUFFER = 8, EFI_SYSTEM_TABLE_64 = 12, ACPI_RSDP_V1 = 14, ACPI_RSDP_V2 = 15, }; }; struct Size : Register <0x04, 32> { }; Tag(addr_t addr) : Mmio(addr) { } }; struct Efi_system_table_64 : Tag { struct Pointer : Register <0x08, 64> { }; Efi_system_table_64(addr_t addr) : Tag(addr) { } }; public: enum { MAGIC = 0x36d76289UL }; struct Memory : Genode::Mmio { enum { SIZE = 3 * 8 }; struct Addr : Register <0x00, 64> { }; struct Size : Register <0x08, 64> { }; struct Type : Register <0x10, 32> { enum { MEMORY = 1 }; }; Memory(addr_t mmap = 0) : Mmio(mmap) { } }; Multiboot2_info(addr_t mbi) : Mmio(mbi) { } template void for_each_tag(FUNC_MEM mem_fn, FUNC_ACPI acpi_fn, FUNC_FB fb_fn, FUNC_SYSTAB64 systab64_fn) { addr_t const size = read(); for (addr_t tag_addr = base() + (1UL << Tag::LOG2_SIZE); tag_addr < base() + size;) { Tag tag(tag_addr); if (tag.read() == Tag::Type::END) return; if (tag.read() == Tag::Type::EFI_SYSTEM_TABLE_64) { Efi_system_table_64 const est(tag_addr); systab64_fn(est.read()); } if (tag.read() == Tag::Type::MEMORY) { addr_t mem_start = tag_addr + (1UL << Tag::LOG2_SIZE) + 8; addr_t const mem_end = tag_addr + tag.read(); for (; mem_start < mem_end; mem_start += Memory::SIZE) { Memory mem(mem_start); mem_fn(mem); } } if (tag.read() == Tag::Type::ACPI_RSDP_V1 || tag.read() == Tag::Type::ACPI_RSDP_V2) { size_t const sizeof_tag = 1UL << Tag::LOG2_SIZE; addr_t const rsdp_addr = tag_addr + sizeof_tag; Hw::Acpi_rsdp * rsdp = reinterpret_cast(rsdp_addr); if (tag.read() - sizeof_tag == 20) { /* ACPI RSDP v1 is 20 byte solely */ Hw::Acpi_rsdp rsdp_v1; memset (&rsdp_v1, 0, sizeof(rsdp_v1)); memcpy (&rsdp_v1, rsdp, 20); acpi_fn(rsdp_v1); } if (sizeof(*rsdp) <= tag.read() - sizeof_tag) acpi_fn(*rsdp); } if (tag.read() == Tag::Type::FRAMEBUFFER) { size_t const sizeof_tag = 1UL << Tag::LOG2_SIZE; addr_t const fb_addr = tag_addr + sizeof_tag; Hw::Framebuffer const * fb = reinterpret_cast(fb_addr); if (sizeof(*fb) <= tag.read() - sizeof_tag) fb_fn(*fb); } tag_addr += align_addr(tag.read(), Tag::LOG2_SIZE); } } }; #endif /* _SRC__BOOTSTRAP__SPEC__X86_64__MULTIBOOT2_H_ */