blob: 2746a7113a302517dc64597dc08421f6d1300dbf (
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
|
#include "arch/memory/main.hpp"
#include "arch/exception_handling/assert.hpp"
#include "arch/kernel/cpu/control_register.hpp"
#include "arch/kernel/cpu/msr.hpp"
#include "arch/memory/allocator/area_frame_allocator.hpp"
#include "arch/memory/allocator/concept.hpp"
#include "arch/memory/heap/global_heap_allocator.hpp"
#include "arch/memory/paging/active_page_table.hpp"
#include "arch/memory/paging/kernel_mapper.hpp"
#include <optional>
namespace teachos::arch::memory
{
namespace
{
static std::optional<allocator::area_frame_allocator> frame_allocator;
auto create_frame_allocator(multiboot::memory_information const & memory_information)
-> allocator::area_frame_allocator &
{
frame_allocator.emplace(memory_information);
return frame_allocator.value();
}
auto get_frame_allocator() -> allocator::area_frame_allocator &
{
exception_handling::assert(frame_allocator.has_value(),
"[Initialization] Frame allocator has not been created yet");
return frame_allocator.value();
}
} // namespace
auto remap_heap(std::size_t heap_start, std::size_t heap_size, paging::entry::bitset additional_flags = {}) -> void
{
decltype(auto) allocator = get_frame_allocator();
decltype(auto) active_table = paging::active_page_table::create_or_get();
auto const start_page = paging::virtual_page::containing_address(heap_start);
auto const end_page = ++(paging::virtual_page::containing_address(heap_start + heap_size - 1));
paging::page_container::iterator const begin{start_page};
paging::page_container::iterator const end{end_page};
paging::page_container const pages{begin, end};
constexpr auto base_flags = paging::entry::WRITABLE;
auto const flags = base_flags | additional_flags;
for (auto const & page : pages)
{
active_table.map_page_to_next_free_frame(allocator, page, flags);
}
}
auto initialize_memory_management() -> void
{
static bool has_been_called = false;
arch::exception_handling::assert(!has_been_called,
"[Initialization] Memory management has already been initialized");
has_been_called = true;
auto const memory_information = multiboot::read_multiboot2();
decltype(auto) allocator = create_frame_allocator(memory_information);
kernel::cpu::set_cr0_bit(kernel::cpu::cr0_flags::WRITE_PROTECT);
kernel::cpu::set_efer_bit(kernel::cpu::efer_flags::NXE);
paging::kernel_mapper kernel(allocator, memory_information);
kernel.remap_kernel();
video::vga::text::write("Kernel remapping successful", video::vga::text::common_attributes::green_on_black);
video::vga::text::newline();
remap_heap(heap::KERNEL_HEAP_START, heap::KERNEL_HEAP_SIZE);
video::vga::text::write("Heap remapping successful", video::vga::text::common_attributes::green_on_black);
video::vga::text::newline();
}
} // namespace teachos::arch::memory
|