blob: 787bcff30c1f2cb646a953483666d8ba84fb3478 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
|
#include "kapi/acpi.hpp"
#include "kapi/memory.hpp"
#include "kapi/system.hpp"
#include "kernel/acpi/manager.hpp"
#include <kstd/units>
#include <algorithm>
#include <atomic>
#include <bit>
#include <cstddef>
#include <cstdint>
#include <optional>
#include <span>
#include <string_view>
namespace kapi::acpi
{
namespace
{
auto constinit manager = std::optional<kernel::acpi::manager>{};
} // namespace
auto root_system_description_pointer::oem_id() const noexcept -> std::string_view
{
return {m_oem_id.data(), m_oem_id.size()};
}
auto root_system_description_pointer::revision() const noexcept -> std::uint8_t
{
return m_revision;
}
auto root_system_description_pointer::signature() const noexcept -> std::string_view
{
return {m_signature.data(), m_signature.size()};
}
auto root_system_description_pointer::table_address() const noexcept -> memory::physical_address
{
auto raw = std::bit_cast<std::uint32_t>(m_rsdt_address);
return memory::physical_address{static_cast<std::uintptr_t>(raw)};
}
auto root_system_description_pointer::validate() const noexcept -> bool
{
return validate_checksum({reinterpret_cast<std::byte const *>(this), sizeof(root_system_description_pointer)});
}
auto extended_system_description_pointer::length() const noexcept -> kstd::units::bytes
{
return kstd::units::bytes{m_length};
}
auto extended_system_description_pointer::table_address() const noexcept -> memory::physical_address
{
return memory::physical_address{std::bit_cast<std::uintptr_t>(m_xsdt_address)};
}
auto extended_system_description_pointer::validate() const noexcept -> bool
{
return validate_checksum({reinterpret_cast<std::byte const *>(this), m_length});
}
auto init(root_system_description_pointer const & sdp) -> bool
{
auto static constinit initialized = std::atomic_flag{};
if (initialized.test_and_set())
{
system::panic("[OS::ACPI] The ACPI manager has already been initialized!");
}
manager.emplace(sdp);
return manager->load_tables();
}
auto validate_checksum(std::span<std::byte const> data) -> bool
{
auto sum = std::ranges::fold_left(data, std::uint8_t{}, [](auto acc, auto byte) {
return static_cast<std::uint8_t>(acc + static_cast<std::uint8_t>(byte));
});
return sum == 0;
}
}; // namespace kapi::acpi
|