aboutsummaryrefslogtreecommitdiff
path: root/kernel/src/test_support/page_mapper.cpp
blob: 9236deb60fc0859b955642a0425ed9c30fd693e1 (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
#include "kernel/test_support/page_mapper.hpp"

#include "kapi/memory.hpp"

#include <kstd/units>

#include <cstddef>
#include <format>
#include <stdexcept>

namespace kernel::tests
{

  page_mapper::page_mapper(kstd::units::bytes physical_size, kstd::units::bytes virtual_size)
      : memory{physical_size, virtual_size}
  {}

  auto page_mapper::map(kapi::memory::page page, kapi::memory::frame frame, flags) -> std::byte *
  {
    auto result = page_mappings.insert({page.number(), frame});
    if (!result.second)
    {
      auto error = std::format("Page {} was already mapped!", page.number());
      throw std::invalid_argument{error};
    }

    auto page_address = page.start_address();
    auto virtual_base = memory.virtual_base();
    auto virtual_end = virtual_base + memory.virtual_size();

    if (page_address >= virtual_base && page_address < virtual_end)
    {
      auto virtual_target = static_cast<std::byte *>(page_address);
      auto physical_offset = frame.number() * kapi::memory::frame::size;
      return memory.map(kapi::memory::page::size, virtual_target, physical_offset.value);
    }
    else if (page_address >= kapi::memory::mmio_base)
    {
      throw std::runtime_error("MMIO mapping not yet supported in testing!");
    }
    else if (page_address >= kapi::memory::higher_half_direct_map_base)
    {
      auto offset = frame.number() * kapi::memory::frame::size;
      return memory.physical_base() + offset;
    }

    return nullptr;
  }

  auto page_mapper::unmap(kapi::memory::page page) -> void
  {
    if (!try_unmap(page))
    {
      auto error = std::format("Page {} was never mapped!", page.number());
      throw std::invalid_argument{error};
    }
  }

  auto page_mapper::try_unmap(kapi::memory::page page) noexcept -> bool
  {
    if (page_mappings.contains(page.number()))
    {
      page_mappings.erase(page.number());
      return true;
    }

    return false;
  }

}  // namespace kernel::tests