2020-01-18 11:38:21 +03:00
|
|
|
/*
|
2021-03-10 00:44:04 +03:00
|
|
|
* Copyright (c) 2018-2021, Andreas Kling <kling@serenityos.org>
|
2020-01-18 11:38:21 +03:00
|
|
|
*
|
2021-04-22 11:24:48 +03:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-01-18 11:38:21 +03:00
|
|
|
*/
|
|
|
|
|
2018-10-18 00:13:55 +03:00
|
|
|
#include <AK/Assertions.h>
|
2020-03-08 14:33:14 +03:00
|
|
|
#include <AK/Memory.h>
|
2020-03-23 15:45:10 +03:00
|
|
|
#include <AK/StringView.h>
|
2021-07-19 18:54:51 +03:00
|
|
|
#include <Kernel/BootInfo.h>
|
2020-05-06 18:40:06 +03:00
|
|
|
#include <Kernel/CMOS.h>
|
2019-06-07 12:43:58 +03:00
|
|
|
#include <Kernel/FileSystem/Inode.h>
|
2020-08-25 01:38:20 +03:00
|
|
|
#include <Kernel/Heap/kmalloc.h>
|
2021-08-06 11:45:34 +03:00
|
|
|
#include <Kernel/Memory/AnonymousVMObject.h>
|
|
|
|
#include <Kernel/Memory/MemoryManager.h>
|
|
|
|
#include <Kernel/Memory/PageDirectory.h>
|
|
|
|
#include <Kernel/Memory/PhysicalRegion.h>
|
|
|
|
#include <Kernel/Memory/SharedInodeVMObject.h>
|
2019-06-09 12:48:58 +03:00
|
|
|
#include <Kernel/Multiboot.h>
|
2021-06-23 22:54:41 +03:00
|
|
|
#include <Kernel/Panic.h>
|
2020-05-06 18:40:06 +03:00
|
|
|
#include <Kernel/Process.h>
|
2021-06-22 18:40:16 +03:00
|
|
|
#include <Kernel/Sections.h>
|
2020-09-18 10:49:51 +03:00
|
|
|
#include <Kernel/StdLib.h>
|
2018-10-18 00:13:55 +03:00
|
|
|
|
2021-07-22 23:11:17 +03:00
|
|
|
extern u8 start_of_kernel_image[];
|
|
|
|
extern u8 end_of_kernel_image[];
|
|
|
|
extern u8 start_of_kernel_text[];
|
|
|
|
extern u8 start_of_kernel_data[];
|
|
|
|
extern u8 end_of_kernel_bss[];
|
|
|
|
extern u8 start_of_ro_after_init[];
|
|
|
|
extern u8 end_of_ro_after_init[];
|
|
|
|
extern u8 start_of_unmap_after_init[];
|
|
|
|
extern u8 end_of_unmap_after_init[];
|
|
|
|
extern u8 start_of_kernel_ksyms[];
|
|
|
|
extern u8 end_of_kernel_ksyms[];
|
2020-02-16 03:27:42 +03:00
|
|
|
|
2021-01-20 19:49:55 +03:00
|
|
|
extern multiboot_module_entry_t multiboot_copy_boot_modules_array[16];
|
|
|
|
extern size_t multiboot_copy_boot_modules_count;
|
|
|
|
|
|
|
|
// Treat the super pages as logically separate from .bss
|
|
|
|
__attribute__((section(".super_pages"))) static u8 super_pages[1 * MiB];
|
|
|
|
|
2021-08-06 14:49:36 +03:00
|
|
|
namespace Kernel::Memory {
|
2020-02-16 03:27:42 +03:00
|
|
|
|
2021-08-07 22:34:11 +03:00
|
|
|
// NOTE: We can NOT use Singleton for this class, because
|
2020-08-25 04:35:19 +03:00
|
|
|
// MemoryManager::initialize is called *before* global constructors are
|
2021-08-07 22:34:11 +03:00
|
|
|
// run. If we do, then Singleton would get re-initialized, causing
|
2020-08-25 04:35:19 +03:00
|
|
|
// the memory manager to be initialized twice!
|
2020-08-22 18:53:34 +03:00
|
|
|
static MemoryManager* s_the;
|
2020-07-06 16:27:22 +03:00
|
|
|
RecursiveSpinLock s_mm_lock;
|
2018-10-18 00:13:55 +03:00
|
|
|
|
2021-08-06 14:49:36 +03:00
|
|
|
MemoryManager& MemoryManager::the()
|
2018-10-18 00:13:55 +03:00
|
|
|
{
|
|
|
|
return *s_the;
|
|
|
|
}
|
|
|
|
|
2020-08-30 01:41:30 +03:00
|
|
|
bool MemoryManager::is_initialized()
|
|
|
|
{
|
|
|
|
return s_the != nullptr;
|
|
|
|
}
|
|
|
|
|
2021-02-19 20:41:50 +03:00
|
|
|
UNMAP_AFTER_INIT MemoryManager::MemoryManager()
|
2018-10-18 00:13:55 +03:00
|
|
|
{
|
2021-07-08 04:50:05 +03:00
|
|
|
s_the = this;
|
|
|
|
|
2020-07-06 18:11:52 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2020-01-17 23:03:52 +03:00
|
|
|
parse_memory_map();
|
2020-02-10 22:00:32 +03:00
|
|
|
write_cr3(kernel_page_directory().cr3());
|
2020-01-18 00:07:20 +03:00
|
|
|
protect_kernel_image();
|
2020-02-15 15:12:02 +03:00
|
|
|
|
2020-09-05 06:12:25 +03:00
|
|
|
// We're temporarily "committing" to two pages that we need to allocate below
|
2021-08-04 23:49:13 +03:00
|
|
|
auto committed_pages = commit_user_physical_pages(2);
|
2020-09-05 06:12:25 +03:00
|
|
|
|
2021-08-04 23:49:13 +03:00
|
|
|
m_shared_zero_page = committed_pages->take_one();
|
2020-09-05 06:12:25 +03:00
|
|
|
|
|
|
|
// We're wasting a page here, we just need a special tag (physical
|
|
|
|
// address) so that we know when we need to lazily allocate a page
|
|
|
|
// that we should be drawing this page from the committed pool rather
|
|
|
|
// than potentially failing if no pages are available anymore.
|
|
|
|
// By using a tag we don't have to query the VMObject for every page
|
|
|
|
// whether it was committed or not
|
2021-08-04 23:49:13 +03:00
|
|
|
m_lazy_committed_page = committed_pages->take_one();
|
2020-01-18 01:56:13 +03:00
|
|
|
}
|
2020-01-18 00:07:20 +03:00
|
|
|
|
2021-02-19 20:41:50 +03:00
|
|
|
UNMAP_AFTER_INIT MemoryManager::~MemoryManager()
|
2020-01-18 01:56:13 +03:00
|
|
|
{
|
2020-01-17 23:03:52 +03:00
|
|
|
}
|
|
|
|
|
2021-02-19 20:41:50 +03:00
|
|
|
UNMAP_AFTER_INIT void MemoryManager::protect_kernel_image()
|
2020-01-18 00:07:20 +03:00
|
|
|
{
|
2020-11-01 02:19:18 +03:00
|
|
|
ScopedSpinLock page_lock(kernel_page_directory().get_lock());
|
2020-01-18 00:07:20 +03:00
|
|
|
// Disable writing to the kernel text and rodata segments.
|
2021-07-22 23:11:17 +03:00
|
|
|
for (auto i = start_of_kernel_text; i < start_of_kernel_data; i += PAGE_SIZE) {
|
2020-09-02 01:10:54 +03:00
|
|
|
auto& pte = *ensure_pte(kernel_page_directory(), VirtualAddress(i));
|
2020-01-18 00:07:20 +03:00
|
|
|
pte.set_writable(false);
|
|
|
|
}
|
2020-07-03 19:23:09 +03:00
|
|
|
if (Processor::current().has_feature(CPUFeature::NX)) {
|
2021-01-20 19:49:55 +03:00
|
|
|
// Disable execution of the kernel data, bss and heap segments.
|
2021-07-22 23:11:17 +03:00
|
|
|
for (auto i = start_of_kernel_data; i < end_of_kernel_image; i += PAGE_SIZE) {
|
2020-09-02 01:10:54 +03:00
|
|
|
auto& pte = *ensure_pte(kernel_page_directory(), VirtualAddress(i));
|
2020-08-25 01:38:20 +03:00
|
|
|
pte.set_execute_disabled(true);
|
|
|
|
}
|
2020-01-18 00:07:20 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-19 20:41:50 +03:00
|
|
|
UNMAP_AFTER_INIT void MemoryManager::protect_readonly_after_init_memory()
|
2021-02-14 19:35:07 +03:00
|
|
|
{
|
|
|
|
ScopedSpinLock page_lock(kernel_page_directory().get_lock());
|
2021-08-09 02:26:02 +03:00
|
|
|
ScopedSpinLock mm_lock(s_mm_lock);
|
2021-02-14 19:35:07 +03:00
|
|
|
// Disable writing to the .ro_after_init section
|
|
|
|
for (auto i = (FlatPtr)&start_of_ro_after_init; i < (FlatPtr)&end_of_ro_after_init; i += PAGE_SIZE) {
|
|
|
|
auto& pte = *ensure_pte(kernel_page_directory(), VirtualAddress(i));
|
|
|
|
pte.set_writable(false);
|
|
|
|
flush_tlb(&kernel_page_directory(), VirtualAddress(i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-16 10:50:34 +03:00
|
|
|
void MemoryManager::unmap_text_after_init()
|
2021-02-19 20:21:54 +03:00
|
|
|
{
|
|
|
|
ScopedSpinLock page_lock(kernel_page_directory().get_lock());
|
2021-08-09 02:26:02 +03:00
|
|
|
ScopedSpinLock mm_lock(s_mm_lock);
|
2021-02-19 20:21:54 +03:00
|
|
|
|
|
|
|
auto start = page_round_down((FlatPtr)&start_of_unmap_after_init);
|
|
|
|
auto end = page_round_up((FlatPtr)&end_of_unmap_after_init);
|
|
|
|
|
|
|
|
// Unmap the entire .unmap_after_init section
|
|
|
|
for (auto i = start; i < end; i += PAGE_SIZE) {
|
|
|
|
auto& pte = *ensure_pte(kernel_page_directory(), VirtualAddress(i));
|
|
|
|
pte.clear();
|
|
|
|
flush_tlb(&kernel_page_directory(), VirtualAddress(i));
|
|
|
|
}
|
|
|
|
|
|
|
|
dmesgln("Unmapped {} KiB of kernel text after init! :^)", (end - start) / KiB);
|
2021-07-16 10:50:34 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void MemoryManager::unmap_ksyms_after_init()
|
|
|
|
{
|
|
|
|
ScopedSpinLock mm_lock(s_mm_lock);
|
|
|
|
ScopedSpinLock page_lock(kernel_page_directory().get_lock());
|
|
|
|
|
2021-07-22 23:11:17 +03:00
|
|
|
auto start = page_round_down((FlatPtr)start_of_kernel_ksyms);
|
|
|
|
auto end = page_round_up((FlatPtr)end_of_kernel_ksyms);
|
2021-07-16 10:50:34 +03:00
|
|
|
|
|
|
|
// Unmap the entire .ksyms section
|
|
|
|
for (auto i = start; i < end; i += PAGE_SIZE) {
|
|
|
|
auto& pte = *ensure_pte(kernel_page_directory(), VirtualAddress(i));
|
|
|
|
pte.clear();
|
|
|
|
flush_tlb(&kernel_page_directory(), VirtualAddress(i));
|
|
|
|
}
|
|
|
|
|
|
|
|
dmesgln("Unmapped {} KiB of kernel symbols after init! :^)", (end - start) / KiB);
|
2021-02-19 20:21:54 +03:00
|
|
|
}
|
|
|
|
|
2021-02-19 20:41:50 +03:00
|
|
|
UNMAP_AFTER_INIT void MemoryManager::register_reserved_ranges()
|
2021-01-29 15:03:25 +03:00
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(!m_physical_memory_ranges.is_empty());
|
2021-08-07 00:42:53 +03:00
|
|
|
ContiguousReservedMemoryRange range;
|
2021-01-29 15:03:25 +03:00
|
|
|
for (auto& current_range : m_physical_memory_ranges) {
|
2021-08-07 00:42:53 +03:00
|
|
|
if (current_range.type != PhysicalMemoryRangeType::Reserved) {
|
2021-01-29 15:03:25 +03:00
|
|
|
if (range.start.is_null())
|
|
|
|
continue;
|
2021-08-07 00:42:53 +03:00
|
|
|
m_reserved_memory_ranges.append(ContiguousReservedMemoryRange { range.start, current_range.start.get() - range.start.get() });
|
2021-01-29 15:03:25 +03:00
|
|
|
range.start.set((FlatPtr) nullptr);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!range.start.is_null()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
range.start = current_range.start;
|
|
|
|
}
|
2021-08-07 00:42:53 +03:00
|
|
|
if (m_physical_memory_ranges.last().type != PhysicalMemoryRangeType::Reserved)
|
2021-01-29 15:03:25 +03:00
|
|
|
return;
|
|
|
|
if (range.start.is_null())
|
|
|
|
return;
|
2021-08-07 00:42:53 +03:00
|
|
|
m_reserved_memory_ranges.append(ContiguousReservedMemoryRange { range.start, m_physical_memory_ranges.last().start.get() + m_physical_memory_ranges.last().length - range.start.get() });
|
2021-01-29 15:03:25 +03:00
|
|
|
}
|
|
|
|
|
2021-08-06 14:54:48 +03:00
|
|
|
bool MemoryManager::is_allowed_to_mmap_to_userspace(PhysicalAddress start_address, VirtualRange const& range) const
|
2021-01-29 15:03:25 +03:00
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(!m_reserved_memory_ranges.is_empty());
|
2021-01-29 15:03:25 +03:00
|
|
|
for (auto& current_range : m_reserved_memory_ranges) {
|
|
|
|
if (!(current_range.start <= start_address))
|
|
|
|
continue;
|
|
|
|
if (!(current_range.start.offset(current_range.length) > start_address))
|
|
|
|
continue;
|
2021-02-12 18:20:21 +03:00
|
|
|
if (current_range.length < range.size())
|
2021-01-29 15:03:25 +03:00
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-02-19 20:41:50 +03:00
|
|
|
UNMAP_AFTER_INIT void MemoryManager::parse_memory_map()
|
2020-01-17 23:03:52 +03:00
|
|
|
{
|
2021-01-19 21:13:03 +03:00
|
|
|
// Register used memory regions that we know of.
|
|
|
|
m_used_memory_ranges.ensure_capacity(4);
|
2021-08-07 00:42:53 +03:00
|
|
|
m_used_memory_ranges.append(UsedMemoryRange { UsedMemoryRangeType::LowMemory, PhysicalAddress(0x00000000), PhysicalAddress(1 * MiB) });
|
|
|
|
m_used_memory_ranges.append(UsedMemoryRange { UsedMemoryRangeType::Prekernel, start_of_prekernel_image, end_of_prekernel_image });
|
|
|
|
m_used_memory_ranges.append(UsedMemoryRange { UsedMemoryRangeType::Kernel, PhysicalAddress(virtual_to_low_physical((FlatPtr)start_of_kernel_image)), PhysicalAddress(page_round_up(virtual_to_low_physical((FlatPtr)end_of_kernel_image))) });
|
2020-08-25 01:38:20 +03:00
|
|
|
|
2021-07-26 16:09:36 +03:00
|
|
|
if (multiboot_flags & 0x4) {
|
2021-01-19 21:13:03 +03:00
|
|
|
auto* bootmods_start = multiboot_copy_boot_modules_array;
|
|
|
|
auto* bootmods_end = bootmods_start + multiboot_copy_boot_modules_count;
|
2019-06-11 14:13:02 +03:00
|
|
|
|
2021-01-19 21:13:03 +03:00
|
|
|
for (auto* bootmod = bootmods_start; bootmod < bootmods_end; bootmod++) {
|
2021-08-07 00:42:53 +03:00
|
|
|
m_used_memory_ranges.append(UsedMemoryRange { UsedMemoryRangeType::BootModule, PhysicalAddress(bootmod->start), PhysicalAddress(bootmod->end) });
|
2021-01-19 21:13:03 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-26 16:09:36 +03:00
|
|
|
auto* mmap_begin = multiboot_memory_map;
|
|
|
|
auto* mmap_end = multiboot_memory_map + multiboot_memory_map_count;
|
2021-01-19 21:13:03 +03:00
|
|
|
|
2021-08-06 14:54:48 +03:00
|
|
|
struct ContiguousPhysicalVirtualRange {
|
2021-07-13 19:22:49 +03:00
|
|
|
PhysicalAddress lower;
|
|
|
|
PhysicalAddress upper;
|
|
|
|
};
|
|
|
|
|
2021-08-06 14:54:48 +03:00
|
|
|
Vector<ContiguousPhysicalVirtualRange> contiguous_physical_ranges;
|
2021-07-13 19:22:49 +03:00
|
|
|
|
2021-01-19 21:13:03 +03:00
|
|
|
for (auto* mmap = mmap_begin; mmap < mmap_end; mmap++) {
|
Kernel: Fix UB caused by taking a reference to a packed struct's member
Taking a reference or a pointer to a value that's not aligned properly
is undefined behavior. While `[[gnu::packed]]` ensures that reads from
and writes to fields of packed structs is a safe operation, the
information about the reduced alignment is lost when creating pointers
to these values.
Weirdly enough, GCC's undefined behavior sanitizer doesn't flag these,
even though the doc of `-Waddress-of-packed-member` says that it usually
leads to UB. In contrast, x86_64 Clang does flag these, which renders
the 64-bit kernel unable to boot.
For now, the `address-of-packed-member` warning will only be enabled in
the kernel, as it is absolutely crucial there because of KUBSAN, but
might get excessively noisy for the userland in the future.
Also note that we can't append to `CMAKE_CXX_FLAGS` like we do for other
flags in the kernel, because flags added via `add_compile_options` come
after these, so the `-Wno-address-of-packed-member` in the root would
cancel it out.
2021-08-01 21:30:43 +03:00
|
|
|
// We have to copy these onto the stack, because we take a reference to these when printing them out,
|
|
|
|
// and doing so on a packed struct field is UB.
|
|
|
|
auto address = mmap->addr;
|
2021-07-07 06:35:15 +03:00
|
|
|
auto length = mmap->len;
|
Kernel: Fix UB caused by taking a reference to a packed struct's member
Taking a reference or a pointer to a value that's not aligned properly
is undefined behavior. While `[[gnu::packed]]` ensures that reads from
and writes to fields of packed structs is a safe operation, the
information about the reduced alignment is lost when creating pointers
to these values.
Weirdly enough, GCC's undefined behavior sanitizer doesn't flag these,
even though the doc of `-Waddress-of-packed-member` says that it usually
leads to UB. In contrast, x86_64 Clang does flag these, which renders
the 64-bit kernel unable to boot.
For now, the `address-of-packed-member` warning will only be enabled in
the kernel, as it is absolutely crucial there because of KUBSAN, but
might get excessively noisy for the userland in the future.
Also note that we can't append to `CMAKE_CXX_FLAGS` like we do for other
flags in the kernel, because flags added via `add_compile_options` come
after these, so the `-Wno-address-of-packed-member` in the root would
cancel it out.
2021-08-01 21:30:43 +03:00
|
|
|
ArmedScopeGuard write_back_guard = [&]() {
|
|
|
|
mmap->addr = address;
|
|
|
|
mmap->len = length;
|
|
|
|
};
|
|
|
|
|
|
|
|
dmesgln("MM: Multiboot mmap: address={:p}, length={}, type={}", address, length, mmap->type);
|
|
|
|
|
|
|
|
auto start_address = PhysicalAddress(address);
|
2021-01-29 15:03:25 +03:00
|
|
|
switch (mmap->type) {
|
|
|
|
case (MULTIBOOT_MEMORY_AVAILABLE):
|
2021-08-07 00:42:53 +03:00
|
|
|
m_physical_memory_ranges.append(PhysicalMemoryRange { PhysicalMemoryRangeType::Usable, start_address, length });
|
2021-01-29 15:03:25 +03:00
|
|
|
break;
|
|
|
|
case (MULTIBOOT_MEMORY_RESERVED):
|
2021-08-07 00:42:53 +03:00
|
|
|
m_physical_memory_ranges.append(PhysicalMemoryRange { PhysicalMemoryRangeType::Reserved, start_address, length });
|
2021-01-29 15:03:25 +03:00
|
|
|
break;
|
|
|
|
case (MULTIBOOT_MEMORY_ACPI_RECLAIMABLE):
|
2021-08-07 00:42:53 +03:00
|
|
|
m_physical_memory_ranges.append(PhysicalMemoryRange { PhysicalMemoryRangeType::ACPI_Reclaimable, start_address, length });
|
2021-01-29 15:03:25 +03:00
|
|
|
break;
|
|
|
|
case (MULTIBOOT_MEMORY_NVS):
|
2021-08-07 00:42:53 +03:00
|
|
|
m_physical_memory_ranges.append(PhysicalMemoryRange { PhysicalMemoryRangeType::ACPI_NVS, start_address, length });
|
2021-01-29 15:03:25 +03:00
|
|
|
break;
|
|
|
|
case (MULTIBOOT_MEMORY_BADRAM):
|
2021-03-10 00:44:04 +03:00
|
|
|
dmesgln("MM: Warning, detected bad memory range!");
|
2021-08-07 00:42:53 +03:00
|
|
|
m_physical_memory_ranges.append(PhysicalMemoryRange { PhysicalMemoryRangeType::BadMemory, start_address, length });
|
2021-01-29 15:03:25 +03:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
dbgln("MM: Unknown range!");
|
2021-08-07 00:42:53 +03:00
|
|
|
m_physical_memory_ranges.append(PhysicalMemoryRange { PhysicalMemoryRangeType::Unknown, start_address, length });
|
2021-01-29 15:03:25 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-01-19 21:13:03 +03:00
|
|
|
if (mmap->type != MULTIBOOT_MEMORY_AVAILABLE)
|
2019-11-23 19:27:09 +03:00
|
|
|
continue;
|
|
|
|
|
2021-01-19 21:13:03 +03:00
|
|
|
// Fix up unaligned memory regions.
|
Kernel: Fix UB caused by taking a reference to a packed struct's member
Taking a reference or a pointer to a value that's not aligned properly
is undefined behavior. While `[[gnu::packed]]` ensures that reads from
and writes to fields of packed structs is a safe operation, the
information about the reduced alignment is lost when creating pointers
to these values.
Weirdly enough, GCC's undefined behavior sanitizer doesn't flag these,
even though the doc of `-Waddress-of-packed-member` says that it usually
leads to UB. In contrast, x86_64 Clang does flag these, which renders
the 64-bit kernel unable to boot.
For now, the `address-of-packed-member` warning will only be enabled in
the kernel, as it is absolutely crucial there because of KUBSAN, but
might get excessively noisy for the userland in the future.
Also note that we can't append to `CMAKE_CXX_FLAGS` like we do for other
flags in the kernel, because flags added via `add_compile_options` come
after these, so the `-Wno-address-of-packed-member` in the root would
cancel it out.
2021-08-01 21:30:43 +03:00
|
|
|
auto diff = (FlatPtr)address % PAGE_SIZE;
|
2019-09-28 09:22:50 +03:00
|
|
|
if (diff != 0) {
|
Kernel: Fix UB caused by taking a reference to a packed struct's member
Taking a reference or a pointer to a value that's not aligned properly
is undefined behavior. While `[[gnu::packed]]` ensures that reads from
and writes to fields of packed structs is a safe operation, the
information about the reduced alignment is lost when creating pointers
to these values.
Weirdly enough, GCC's undefined behavior sanitizer doesn't flag these,
even though the doc of `-Waddress-of-packed-member` says that it usually
leads to UB. In contrast, x86_64 Clang does flag these, which renders
the 64-bit kernel unable to boot.
For now, the `address-of-packed-member` warning will only be enabled in
the kernel, as it is absolutely crucial there because of KUBSAN, but
might get excessively noisy for the userland in the future.
Also note that we can't append to `CMAKE_CXX_FLAGS` like we do for other
flags in the kernel, because flags added via `add_compile_options` come
after these, so the `-Wno-address-of-packed-member` in the root would
cancel it out.
2021-08-01 21:30:43 +03:00
|
|
|
dmesgln("MM: Got an unaligned physical_region from the bootloader; correcting {:p} by {} bytes", address, diff);
|
2019-09-28 09:22:50 +03:00
|
|
|
diff = PAGE_SIZE - diff;
|
Kernel: Fix UB caused by taking a reference to a packed struct's member
Taking a reference or a pointer to a value that's not aligned properly
is undefined behavior. While `[[gnu::packed]]` ensures that reads from
and writes to fields of packed structs is a safe operation, the
information about the reduced alignment is lost when creating pointers
to these values.
Weirdly enough, GCC's undefined behavior sanitizer doesn't flag these,
even though the doc of `-Waddress-of-packed-member` says that it usually
leads to UB. In contrast, x86_64 Clang does flag these, which renders
the 64-bit kernel unable to boot.
For now, the `address-of-packed-member` warning will only be enabled in
the kernel, as it is absolutely crucial there because of KUBSAN, but
might get excessively noisy for the userland in the future.
Also note that we can't append to `CMAKE_CXX_FLAGS` like we do for other
flags in the kernel, because flags added via `add_compile_options` come
after these, so the `-Wno-address-of-packed-member` in the root would
cancel it out.
2021-08-01 21:30:43 +03:00
|
|
|
address += diff;
|
|
|
|
length -= diff;
|
2019-09-28 09:22:50 +03:00
|
|
|
}
|
Kernel: Fix UB caused by taking a reference to a packed struct's member
Taking a reference or a pointer to a value that's not aligned properly
is undefined behavior. While `[[gnu::packed]]` ensures that reads from
and writes to fields of packed structs is a safe operation, the
information about the reduced alignment is lost when creating pointers
to these values.
Weirdly enough, GCC's undefined behavior sanitizer doesn't flag these,
even though the doc of `-Waddress-of-packed-member` says that it usually
leads to UB. In contrast, x86_64 Clang does flag these, which renders
the 64-bit kernel unable to boot.
For now, the `address-of-packed-member` warning will only be enabled in
the kernel, as it is absolutely crucial there because of KUBSAN, but
might get excessively noisy for the userland in the future.
Also note that we can't append to `CMAKE_CXX_FLAGS` like we do for other
flags in the kernel, because flags added via `add_compile_options` come
after these, so the `-Wno-address-of-packed-member` in the root would
cancel it out.
2021-08-01 21:30:43 +03:00
|
|
|
if ((length % PAGE_SIZE) != 0) {
|
|
|
|
dmesgln("MM: Got an unaligned physical_region from the bootloader; correcting length {} by {} bytes", length, length % PAGE_SIZE);
|
|
|
|
length -= length % PAGE_SIZE;
|
2019-09-28 09:22:50 +03:00
|
|
|
}
|
Kernel: Fix UB caused by taking a reference to a packed struct's member
Taking a reference or a pointer to a value that's not aligned properly
is undefined behavior. While `[[gnu::packed]]` ensures that reads from
and writes to fields of packed structs is a safe operation, the
information about the reduced alignment is lost when creating pointers
to these values.
Weirdly enough, GCC's undefined behavior sanitizer doesn't flag these,
even though the doc of `-Waddress-of-packed-member` says that it usually
leads to UB. In contrast, x86_64 Clang does flag these, which renders
the 64-bit kernel unable to boot.
For now, the `address-of-packed-member` warning will only be enabled in
the kernel, as it is absolutely crucial there because of KUBSAN, but
might get excessively noisy for the userland in the future.
Also note that we can't append to `CMAKE_CXX_FLAGS` like we do for other
flags in the kernel, because flags added via `add_compile_options` come
after these, so the `-Wno-address-of-packed-member` in the root would
cancel it out.
2021-08-01 21:30:43 +03:00
|
|
|
if (length < PAGE_SIZE) {
|
|
|
|
dmesgln("MM: Memory physical_region from bootloader is too small; we want >= {} bytes, but got {} bytes", PAGE_SIZE, length);
|
2019-09-28 09:22:50 +03:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
Kernel: Fix UB caused by taking a reference to a packed struct's member
Taking a reference or a pointer to a value that's not aligned properly
is undefined behavior. While `[[gnu::packed]]` ensures that reads from
and writes to fields of packed structs is a safe operation, the
information about the reduced alignment is lost when creating pointers
to these values.
Weirdly enough, GCC's undefined behavior sanitizer doesn't flag these,
even though the doc of `-Waddress-of-packed-member` says that it usually
leads to UB. In contrast, x86_64 Clang does flag these, which renders
the 64-bit kernel unable to boot.
For now, the `address-of-packed-member` warning will only be enabled in
the kernel, as it is absolutely crucial there because of KUBSAN, but
might get excessively noisy for the userland in the future.
Also note that we can't append to `CMAKE_CXX_FLAGS` like we do for other
flags in the kernel, because flags added via `add_compile_options` come
after these, so the `-Wno-address-of-packed-member` in the root would
cancel it out.
2021-08-01 21:30:43 +03:00
|
|
|
for (PhysicalSize page_base = address; page_base <= (address + length); page_base += PAGE_SIZE) {
|
2019-06-11 14:13:02 +03:00
|
|
|
auto addr = PhysicalAddress(page_base);
|
|
|
|
|
2021-01-20 19:49:55 +03:00
|
|
|
// Skip used memory ranges.
|
|
|
|
bool should_skip = false;
|
2021-02-15 14:56:39 +03:00
|
|
|
for (auto& used_range : m_used_memory_ranges) {
|
2021-01-20 19:49:55 +03:00
|
|
|
if (addr.get() >= used_range.start.get() && addr.get() <= used_range.end.get()) {
|
|
|
|
should_skip = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (should_skip)
|
2020-08-25 01:38:20 +03:00
|
|
|
continue;
|
|
|
|
|
2021-07-13 19:22:49 +03:00
|
|
|
if (contiguous_physical_ranges.is_empty() || contiguous_physical_ranges.last().upper.offset(PAGE_SIZE) != addr) {
|
2021-08-06 14:54:48 +03:00
|
|
|
contiguous_physical_ranges.append(ContiguousPhysicalVirtualRange {
|
2021-07-13 19:22:49 +03:00
|
|
|
.lower = addr,
|
|
|
|
.upper = addr,
|
|
|
|
});
|
2019-11-23 19:27:09 +03:00
|
|
|
} else {
|
2021-07-13 19:22:49 +03:00
|
|
|
contiguous_physical_ranges.last().upper = addr;
|
2019-06-09 12:48:58 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-12-31 16:58:03 +03:00
|
|
|
|
2021-07-13 19:22:49 +03:00
|
|
|
for (auto& range : contiguous_physical_ranges) {
|
|
|
|
m_user_physical_regions.append(PhysicalRegion::try_create(range.lower, range.upper).release_nonnull());
|
|
|
|
}
|
|
|
|
|
2021-07-19 10:29:07 +03:00
|
|
|
// Super pages are guaranteed to be in the first 16MB of physical memory
|
|
|
|
VERIFY(virtual_to_low_physical((FlatPtr)super_pages) + sizeof(super_pages) < 0x1000000);
|
|
|
|
|
2021-02-12 18:20:21 +03:00
|
|
|
// Append statically-allocated super physical physical_region.
|
2021-08-04 20:29:02 +03:00
|
|
|
m_super_physical_region = PhysicalRegion::try_create(
|
2021-01-20 19:49:55 +03:00
|
|
|
PhysicalAddress(virtual_to_low_physical(FlatPtr(super_pages))),
|
2021-08-04 20:29:02 +03:00
|
|
|
PhysicalAddress(virtual_to_low_physical(FlatPtr(super_pages + sizeof(super_pages)))));
|
|
|
|
VERIFY(m_super_physical_region);
|
2021-01-20 19:49:55 +03:00
|
|
|
|
2021-08-04 20:29:02 +03:00
|
|
|
m_system_memory_info.super_physical_pages += m_super_physical_region->size();
|
2019-06-11 14:13:02 +03:00
|
|
|
|
2021-07-08 04:50:05 +03:00
|
|
|
for (auto& region : m_user_physical_regions)
|
2021-07-13 19:46:09 +03:00
|
|
|
m_system_memory_info.user_physical_pages += region.size();
|
2021-07-08 04:50:05 +03:00
|
|
|
|
|
|
|
register_reserved_ranges();
|
|
|
|
for (auto& range : m_reserved_memory_ranges) {
|
|
|
|
dmesgln("MM: Contiguous reserved range from {}, length is {}", range.start, range.length);
|
2020-08-25 01:38:20 +03:00
|
|
|
}
|
2020-05-08 23:15:02 +03:00
|
|
|
|
2021-07-08 04:50:05 +03:00
|
|
|
initialize_physical_pages();
|
|
|
|
|
2021-07-07 06:35:15 +03:00
|
|
|
VERIFY(m_system_memory_info.super_physical_pages > 0);
|
|
|
|
VERIFY(m_system_memory_info.user_physical_pages > 0);
|
2020-09-05 06:12:25 +03:00
|
|
|
|
|
|
|
// We start out with no committed pages
|
2021-07-07 06:35:15 +03:00
|
|
|
m_system_memory_info.user_physical_pages_uncommitted = m_system_memory_info.user_physical_pages;
|
|
|
|
|
2021-07-08 04:50:05 +03:00
|
|
|
for (auto& used_range : m_used_memory_ranges) {
|
2021-08-07 00:42:53 +03:00
|
|
|
dmesgln("MM: {} range @ {} - {} (size {:#x})", UserMemoryRangeTypeNames[to_underlying(used_range.type)], used_range.start, used_range.end.offset(-1), used_range.end.as_ptr() - used_range.start.as_ptr());
|
2021-07-08 04:50:05 +03:00
|
|
|
}
|
|
|
|
|
2021-08-04 20:29:02 +03:00
|
|
|
dmesgln("MM: Super physical region: {} - {} (size {:#x})", m_super_physical_region->lower(), m_super_physical_region->upper().offset(-1), PAGE_SIZE * m_super_physical_region->size());
|
|
|
|
m_super_physical_region->initialize_zones();
|
2021-07-08 04:50:05 +03:00
|
|
|
|
2021-07-12 23:52:17 +03:00
|
|
|
for (auto& region : m_user_physical_regions) {
|
2021-07-22 02:30:24 +03:00
|
|
|
dmesgln("MM: User physical region: {} - {} (size {:#x})", region.lower(), region.upper().offset(-1), PAGE_SIZE * region.size());
|
2021-07-12 23:52:17 +03:00
|
|
|
region.initialize_zones();
|
|
|
|
}
|
2021-07-08 04:50:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
UNMAP_AFTER_INIT void MemoryManager::initialize_physical_pages()
|
|
|
|
{
|
|
|
|
// We assume that the physical page range is contiguous and doesn't contain huge gaps!
|
|
|
|
PhysicalAddress highest_physical_address;
|
|
|
|
for (auto& range : m_used_memory_ranges) {
|
|
|
|
if (range.end.get() > highest_physical_address.get())
|
|
|
|
highest_physical_address = range.end;
|
|
|
|
}
|
|
|
|
for (auto& region : m_physical_memory_ranges) {
|
|
|
|
auto range_end = PhysicalAddress(region.start).offset(region.length);
|
|
|
|
if (range_end.get() > highest_physical_address.get())
|
|
|
|
highest_physical_address = range_end;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Calculate how many total physical pages the array will have
|
|
|
|
m_physical_page_entries_count = PhysicalAddress::physical_page_index(highest_physical_address.get()) + 1;
|
|
|
|
VERIFY(m_physical_page_entries_count != 0);
|
|
|
|
VERIFY(!Checked<decltype(m_physical_page_entries_count)>::multiplication_would_overflow(m_physical_page_entries_count, sizeof(PhysicalPageEntry)));
|
|
|
|
|
|
|
|
// Calculate how many bytes the array will consume
|
|
|
|
auto physical_page_array_size = m_physical_page_entries_count * sizeof(PhysicalPageEntry);
|
|
|
|
auto physical_page_array_pages = page_round_up(physical_page_array_size) / PAGE_SIZE;
|
|
|
|
VERIFY(physical_page_array_pages * PAGE_SIZE >= physical_page_array_size);
|
|
|
|
|
|
|
|
// Calculate how many page tables we will need to be able to map them all
|
|
|
|
auto needed_page_table_count = (physical_page_array_pages + 512 - 1) / 512;
|
|
|
|
|
|
|
|
auto physical_page_array_pages_and_page_tables_count = physical_page_array_pages + needed_page_table_count;
|
|
|
|
|
|
|
|
// Now that we know how much memory we need for a contiguous array of PhysicalPage instances, find a memory region that can fit it
|
2021-07-11 15:29:02 +03:00
|
|
|
PhysicalRegion* found_region { nullptr };
|
2021-07-12 23:52:17 +03:00
|
|
|
Optional<size_t> found_region_index;
|
|
|
|
for (size_t i = 0; i < m_user_physical_regions.size(); ++i) {
|
|
|
|
auto& region = m_user_physical_regions[i];
|
2021-07-08 04:50:05 +03:00
|
|
|
if (region.size() >= physical_page_array_pages_and_page_tables_count) {
|
2021-07-11 15:29:02 +03:00
|
|
|
found_region = ®ion;
|
2021-07-12 23:52:17 +03:00
|
|
|
found_region_index = i;
|
2021-07-08 04:50:05 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!found_region) {
|
|
|
|
dmesgln("MM: Need {} bytes for physical page management, but no memory region is large enough!", physical_page_array_pages_and_page_tables_count);
|
|
|
|
VERIFY_NOT_REACHED();
|
|
|
|
}
|
|
|
|
|
|
|
|
VERIFY(m_system_memory_info.user_physical_pages >= physical_page_array_pages_and_page_tables_count);
|
|
|
|
m_system_memory_info.user_physical_pages -= physical_page_array_pages_and_page_tables_count;
|
|
|
|
|
|
|
|
if (found_region->size() == physical_page_array_pages_and_page_tables_count) {
|
|
|
|
// We're stealing the entire region
|
2021-07-12 23:52:17 +03:00
|
|
|
m_physical_pages_region = m_user_physical_regions.take(*found_region_index);
|
2021-07-08 04:50:05 +03:00
|
|
|
} else {
|
2021-07-12 23:52:17 +03:00
|
|
|
m_physical_pages_region = found_region->try_take_pages_from_beginning(physical_page_array_pages_and_page_tables_count);
|
2021-01-29 15:03:25 +03:00
|
|
|
}
|
2021-08-07 00:42:53 +03:00
|
|
|
m_used_memory_ranges.append({ UsedMemoryRangeType::PhysicalPages, m_physical_pages_region->lower(), m_physical_pages_region->upper() });
|
2021-07-08 04:50:05 +03:00
|
|
|
|
|
|
|
// Create the bare page directory. This is not a fully constructed page directory and merely contains the allocators!
|
2021-08-05 19:58:33 +03:00
|
|
|
m_kernel_page_directory = PageDirectory::must_create_kernel_page_directory();
|
2021-07-08 04:50:05 +03:00
|
|
|
|
|
|
|
// Allocate a virtual address range for our array
|
|
|
|
auto range = m_kernel_page_directory->range_allocator().allocate_anywhere(physical_page_array_pages * PAGE_SIZE);
|
|
|
|
if (!range.has_value()) {
|
|
|
|
dmesgln("MM: Could not allocate {} bytes to map physical page array!", physical_page_array_pages * PAGE_SIZE);
|
|
|
|
VERIFY_NOT_REACHED();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now that we have our special m_physical_pages_region region with enough pages to hold the entire array
|
|
|
|
// try to map the entire region into kernel space so we always have it
|
|
|
|
// We can't use ensure_pte here because it would try to allocate a PhysicalPage and we don't have the array
|
|
|
|
// mapped yet so we can't create them
|
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
|
|
|
|
|
|
|
// Create page tables at the beginning of m_physical_pages_region, followed by the PhysicalPageEntry array
|
|
|
|
auto page_tables_base = m_physical_pages_region->lower();
|
|
|
|
auto physical_page_array_base = page_tables_base.offset(needed_page_table_count * PAGE_SIZE);
|
|
|
|
auto physical_page_array_current_page = physical_page_array_base.get();
|
|
|
|
auto virtual_page_array_base = range.value().base().get();
|
|
|
|
auto virtual_page_array_current_page = virtual_page_array_base;
|
|
|
|
for (size_t pt_index = 0; pt_index < needed_page_table_count; pt_index++) {
|
|
|
|
auto virtual_page_base_for_this_pt = virtual_page_array_current_page;
|
|
|
|
auto pt_paddr = page_tables_base.offset(pt_index * PAGE_SIZE);
|
|
|
|
auto* pt = reinterpret_cast<PageTableEntry*>(quickmap_page(pt_paddr));
|
|
|
|
__builtin_memset(pt, 0, PAGE_SIZE);
|
|
|
|
for (size_t pte_index = 0; pte_index < PAGE_SIZE / sizeof(PageTableEntry); pte_index++) {
|
|
|
|
auto& pte = pt[pte_index];
|
|
|
|
pte.set_physical_page_base(physical_page_array_current_page);
|
|
|
|
pte.set_user_allowed(false);
|
|
|
|
pte.set_writable(true);
|
|
|
|
if (Processor::current().has_feature(CPUFeature::NX))
|
|
|
|
pte.set_execute_disabled(false);
|
|
|
|
pte.set_global(true);
|
|
|
|
pte.set_present(true);
|
|
|
|
|
|
|
|
physical_page_array_current_page += PAGE_SIZE;
|
|
|
|
virtual_page_array_current_page += PAGE_SIZE;
|
|
|
|
}
|
|
|
|
unquickmap_page();
|
|
|
|
|
|
|
|
// Hook the page table into the kernel page directory
|
|
|
|
u32 page_directory_index = (virtual_page_base_for_this_pt >> 21) & 0x1ff;
|
2021-07-19 19:24:15 +03:00
|
|
|
auto* pd = reinterpret_cast<PageDirectoryEntry*>(quickmap_page(boot_pd_kernel));
|
2021-07-08 04:50:05 +03:00
|
|
|
PageDirectoryEntry& pde = pd[page_directory_index];
|
|
|
|
|
|
|
|
VERIFY(!pde.is_present()); // Nothing should be using this PD yet
|
|
|
|
|
|
|
|
// We can't use ensure_pte quite yet!
|
|
|
|
pde.set_page_table_base(pt_paddr.get());
|
|
|
|
pde.set_user_allowed(false);
|
|
|
|
pde.set_present(true);
|
|
|
|
pde.set_writable(true);
|
|
|
|
pde.set_global(true);
|
|
|
|
|
|
|
|
unquickmap_page();
|
|
|
|
|
|
|
|
flush_tlb_local(VirtualAddress(virtual_page_base_for_this_pt));
|
|
|
|
}
|
|
|
|
|
|
|
|
// We now have the entire PhysicalPageEntry array mapped!
|
|
|
|
m_physical_page_entries = (PhysicalPageEntry*)range.value().base().get();
|
|
|
|
for (size_t i = 0; i < m_physical_page_entries_count; i++)
|
|
|
|
new (&m_physical_page_entries[i]) PageTableEntry();
|
|
|
|
|
|
|
|
// Now we should be able to allocate PhysicalPage instances,
|
|
|
|
// so finish setting up the kernel page directory
|
|
|
|
m_kernel_page_directory->allocate_kernel_directory();
|
|
|
|
|
|
|
|
// Now create legit PhysicalPage objects for the page tables we created, so that
|
|
|
|
// we can put them into kernel_page_directory().m_page_tables
|
|
|
|
auto& kernel_page_tables = kernel_page_directory().m_page_tables;
|
|
|
|
virtual_page_array_current_page = virtual_page_array_base;
|
|
|
|
for (size_t pt_index = 0; pt_index < needed_page_table_count; pt_index++) {
|
|
|
|
VERIFY(virtual_page_array_current_page <= range.value().end().get());
|
|
|
|
auto pt_paddr = page_tables_base.offset(pt_index * PAGE_SIZE);
|
|
|
|
auto physical_page_index = PhysicalAddress::physical_page_index(pt_paddr.get());
|
|
|
|
auto& physical_page_entry = m_physical_page_entries[physical_page_index];
|
2021-07-17 14:30:47 +03:00
|
|
|
auto physical_page = adopt_ref(*new (&physical_page_entry.allocated.physical_page) PhysicalPage(MayReturnToFreeList::No));
|
2021-07-08 04:50:05 +03:00
|
|
|
auto result = kernel_page_tables.set(virtual_page_array_current_page & ~0x1fffff, move(physical_page));
|
|
|
|
VERIFY(result == AK::HashSetResult::InsertedNewEntry);
|
|
|
|
|
2021-07-13 22:20:24 +03:00
|
|
|
virtual_page_array_current_page += (PAGE_SIZE / sizeof(PageTableEntry)) * PAGE_SIZE;
|
2021-07-08 04:50:05 +03:00
|
|
|
}
|
|
|
|
|
2021-07-10 03:17:28 +03:00
|
|
|
dmesgln("MM: Physical page entries: {}", range.value());
|
2021-07-08 04:50:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
PhysicalPageEntry& MemoryManager::get_physical_page_entry(PhysicalAddress physical_address)
|
|
|
|
{
|
|
|
|
VERIFY(m_physical_page_entries);
|
|
|
|
auto physical_page_entry_index = PhysicalAddress::physical_page_index(physical_address.get());
|
|
|
|
VERIFY(physical_page_entry_index < m_physical_page_entries_count);
|
|
|
|
return m_physical_page_entries[physical_page_entry_index];
|
|
|
|
}
|
|
|
|
|
|
|
|
PhysicalAddress MemoryManager::get_physical_address(PhysicalPage const& physical_page)
|
|
|
|
{
|
2021-07-12 23:52:17 +03:00
|
|
|
PhysicalPageEntry const& physical_page_entry = *reinterpret_cast<PhysicalPageEntry const*>((u8 const*)&physical_page - __builtin_offsetof(PhysicalPageEntry, allocated.physical_page));
|
2021-07-08 04:50:05 +03:00
|
|
|
VERIFY(m_physical_page_entries);
|
|
|
|
size_t physical_page_entry_index = &physical_page_entry - m_physical_page_entries;
|
|
|
|
VERIFY(physical_page_entry_index < m_physical_page_entries_count);
|
|
|
|
return PhysicalAddress((PhysicalPtr)physical_page_entry_index * PAGE_SIZE);
|
2018-10-18 00:13:55 +03:00
|
|
|
}
|
|
|
|
|
2020-11-01 02:19:18 +03:00
|
|
|
PageTableEntry* MemoryManager::pte(PageDirectory& page_directory, VirtualAddress vaddr)
|
2020-02-21 15:05:39 +03:00
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY_INTERRUPTS_DISABLED();
|
|
|
|
VERIFY(s_mm_lock.own_lock());
|
|
|
|
VERIFY(page_directory.get_lock().own_lock());
|
2021-07-17 03:42:59 +03:00
|
|
|
u32 page_directory_table_index = (vaddr.get() >> 30) & 0x1ff;
|
2020-02-21 15:05:39 +03:00
|
|
|
u32 page_directory_index = (vaddr.get() >> 21) & 0x1ff;
|
|
|
|
u32 page_table_index = (vaddr.get() >> 12) & 0x1ff;
|
|
|
|
|
|
|
|
auto* pd = quickmap_pd(const_cast<PageDirectory&>(page_directory), page_directory_table_index);
|
2021-07-14 14:31:21 +03:00
|
|
|
PageDirectoryEntry const& pde = pd[page_directory_index];
|
2020-02-21 15:05:39 +03:00
|
|
|
if (!pde.is_present())
|
|
|
|
return nullptr;
|
|
|
|
|
2020-03-08 12:36:51 +03:00
|
|
|
return &quickmap_pt(PhysicalAddress((FlatPtr)pde.page_table_base()))[page_table_index];
|
2020-02-21 15:05:39 +03:00
|
|
|
}
|
|
|
|
|
2020-09-02 01:10:54 +03:00
|
|
|
PageTableEntry* MemoryManager::ensure_pte(PageDirectory& page_directory, VirtualAddress vaddr)
|
2018-10-18 00:13:55 +03:00
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY_INTERRUPTS_DISABLED();
|
|
|
|
VERIFY(s_mm_lock.own_lock());
|
|
|
|
VERIFY(page_directory.get_lock().own_lock());
|
2021-07-17 03:42:59 +03:00
|
|
|
u32 page_directory_table_index = (vaddr.get() >> 30) & 0x1ff;
|
2019-12-25 13:22:16 +03:00
|
|
|
u32 page_directory_index = (vaddr.get() >> 21) & 0x1ff;
|
|
|
|
u32 page_table_index = (vaddr.get() >> 12) & 0x1ff;
|
2018-10-18 00:13:55 +03:00
|
|
|
|
2020-01-17 21:59:20 +03:00
|
|
|
auto* pd = quickmap_pd(page_directory, page_directory_table_index);
|
|
|
|
PageDirectoryEntry& pde = pd[page_directory_index];
|
2018-12-03 03:38:22 +03:00
|
|
|
if (!pde.is_present()) {
|
2020-09-01 22:40:34 +03:00
|
|
|
bool did_purge = false;
|
|
|
|
auto page_table = allocate_user_physical_page(ShouldZeroFill::Yes, &did_purge);
|
2020-09-02 01:10:54 +03:00
|
|
|
if (!page_table) {
|
2021-01-10 18:21:56 +03:00
|
|
|
dbgln("MM: Unable to allocate page table to map {}", vaddr);
|
2020-09-02 01:10:54 +03:00
|
|
|
return nullptr;
|
|
|
|
}
|
2020-09-01 22:40:34 +03:00
|
|
|
if (did_purge) {
|
|
|
|
// If any memory had to be purged, ensure_pte may have been called as part
|
|
|
|
// of the purging process. So we need to re-map the pd in this case to ensure
|
|
|
|
// we're writing to the correct underlying physical page
|
|
|
|
pd = quickmap_pd(page_directory, page_directory_table_index);
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(&pde == &pd[page_directory_index]); // Sanity check
|
2020-09-18 10:49:51 +03:00
|
|
|
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(!pde.is_present()); // Should have not changed
|
2020-09-01 22:40:34 +03:00
|
|
|
}
|
2020-01-17 22:57:32 +03:00
|
|
|
pde.set_page_table_base(page_table->paddr().get());
|
|
|
|
pde.set_user_allowed(true);
|
|
|
|
pde.set_present(true);
|
|
|
|
pde.set_writable(true);
|
|
|
|
pde.set_global(&page_directory == m_kernel_page_directory.ptr());
|
2020-08-28 06:29:17 +03:00
|
|
|
// Use page_directory_table_index and page_directory_index as key
|
|
|
|
// This allows us to release the page table entry when no longer needed
|
2021-08-15 14:18:43 +03:00
|
|
|
auto result = page_directory.m_page_tables.set(vaddr.get() & ~(FlatPtr)0x1fffff, page_table.release_nonnull());
|
2021-07-18 03:34:57 +03:00
|
|
|
// If you're hitting this VERIFY on x86_64 chances are a 64-bit pointer was truncated somewhere
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(result == AK::HashSetResult::InsertedNewEntry);
|
2018-10-18 00:13:55 +03:00
|
|
|
}
|
2020-01-17 21:59:20 +03:00
|
|
|
|
2020-09-02 01:10:54 +03:00
|
|
|
return &quickmap_pt(PhysicalAddress((FlatPtr)pde.page_table_base()))[page_table_index];
|
2018-10-18 00:13:55 +03:00
|
|
|
}
|
|
|
|
|
2020-08-28 06:29:17 +03:00
|
|
|
void MemoryManager::release_pte(PageDirectory& page_directory, VirtualAddress vaddr, bool is_last_release)
|
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY_INTERRUPTS_DISABLED();
|
|
|
|
VERIFY(s_mm_lock.own_lock());
|
|
|
|
VERIFY(page_directory.get_lock().own_lock());
|
2021-07-17 03:42:59 +03:00
|
|
|
u32 page_directory_table_index = (vaddr.get() >> 30) & 0x1ff;
|
2020-08-28 06:29:17 +03:00
|
|
|
u32 page_directory_index = (vaddr.get() >> 21) & 0x1ff;
|
|
|
|
u32 page_table_index = (vaddr.get() >> 12) & 0x1ff;
|
|
|
|
|
|
|
|
auto* pd = quickmap_pd(page_directory, page_directory_table_index);
|
|
|
|
PageDirectoryEntry& pde = pd[page_directory_index];
|
|
|
|
if (pde.is_present()) {
|
|
|
|
auto* page_table = quickmap_pt(PhysicalAddress((FlatPtr)pde.page_table_base()));
|
|
|
|
auto& pte = page_table[page_table_index];
|
|
|
|
pte.clear();
|
|
|
|
|
|
|
|
if (is_last_release || page_table_index == 0x1ff) {
|
|
|
|
// If this is the last PTE in a region or the last PTE in a page table then
|
|
|
|
// check if we can also release the page table
|
|
|
|
bool all_clear = true;
|
|
|
|
for (u32 i = 0; i <= 0x1ff; i++) {
|
|
|
|
if (!page_table[i].is_null()) {
|
|
|
|
all_clear = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (all_clear) {
|
|
|
|
pde.clear();
|
|
|
|
|
|
|
|
auto result = page_directory.m_page_tables.remove(vaddr.get() & ~0x1fffff);
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(result);
|
2020-08-28 06:29:17 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-19 20:41:50 +03:00
|
|
|
UNMAP_AFTER_INIT void MemoryManager::initialize(u32 cpu)
|
2018-10-18 00:13:55 +03:00
|
|
|
{
|
2021-07-27 15:30:26 +03:00
|
|
|
ProcessorSpecific<MemoryManagerData>::initialize();
|
2020-06-29 01:04:35 +03:00
|
|
|
|
2020-08-30 01:41:30 +03:00
|
|
|
if (cpu == 0) {
|
2021-07-08 04:50:05 +03:00
|
|
|
new MemoryManager;
|
2020-08-30 01:41:30 +03:00
|
|
|
kmalloc_enable_expand();
|
|
|
|
}
|
2018-10-18 00:13:55 +03:00
|
|
|
}
|
|
|
|
|
2019-08-06 11:31:20 +03:00
|
|
|
Region* MemoryManager::kernel_region_from_vaddr(VirtualAddress vaddr)
|
2018-11-05 15:48:07 +03:00
|
|
|
{
|
2020-07-06 16:27:22 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2019-08-08 14:40:58 +03:00
|
|
|
for (auto& region : MM.m_kernel_regions) {
|
|
|
|
if (region.contains(vaddr))
|
|
|
|
return ®ion;
|
2019-05-14 12:51:00 +03:00
|
|
|
}
|
2019-08-06 11:31:20 +03:00
|
|
|
return nullptr;
|
|
|
|
}
|
2019-05-14 12:51:00 +03:00
|
|
|
|
2021-08-06 14:57:39 +03:00
|
|
|
Region* MemoryManager::find_user_region_from_vaddr_no_lock(AddressSpace& space, VirtualAddress vaddr)
|
2021-07-18 19:31:28 +03:00
|
|
|
{
|
|
|
|
VERIFY(space.get_lock().own_lock());
|
|
|
|
return space.find_region_containing({ vaddr, 1 });
|
|
|
|
}
|
|
|
|
|
2021-08-06 14:57:39 +03:00
|
|
|
Region* MemoryManager::find_user_region_from_vaddr(AddressSpace& space, VirtualAddress vaddr)
|
2019-08-06 11:31:20 +03:00
|
|
|
{
|
2021-02-08 17:45:40 +03:00
|
|
|
ScopedSpinLock lock(space.get_lock());
|
2021-07-18 19:31:28 +03:00
|
|
|
return find_user_region_from_vaddr_no_lock(space, vaddr);
|
|
|
|
}
|
|
|
|
|
2021-08-06 14:57:39 +03:00
|
|
|
void MemoryManager::validate_syscall_preconditions(AddressSpace& space, RegisterState const& regs)
|
2021-07-18 19:31:28 +03:00
|
|
|
{
|
|
|
|
// We take the space lock once here and then use the no_lock variants
|
|
|
|
// to avoid excessive spinlock recursion in this extemely common path.
|
|
|
|
ScopedSpinLock lock(space.get_lock());
|
|
|
|
|
|
|
|
auto unlock_and_handle_crash = [&lock, ®s](const char* description, int signal) {
|
|
|
|
lock.unlock();
|
|
|
|
handle_crash(regs, description, signal);
|
|
|
|
};
|
|
|
|
|
|
|
|
{
|
|
|
|
VirtualAddress userspace_sp = VirtualAddress { regs.userspace_sp() };
|
|
|
|
if (!MM.validate_user_stack_no_lock(space, userspace_sp)) {
|
|
|
|
dbgln("Invalid stack pointer: {:p}", userspace_sp);
|
|
|
|
unlock_and_handle_crash("Bad stack on syscall entry", SIGSTKFLT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
VirtualAddress ip = VirtualAddress { regs.ip() };
|
|
|
|
auto* calling_region = MM.find_user_region_from_vaddr_no_lock(space, ip);
|
|
|
|
if (!calling_region) {
|
|
|
|
dbgln("Syscall from {:p} which has no associated region", ip);
|
|
|
|
unlock_and_handle_crash("Syscall from unknown region", SIGSEGV);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (calling_region->is_writable()) {
|
|
|
|
dbgln("Syscall from writable memory at {:p}", ip);
|
|
|
|
unlock_and_handle_crash("Syscall from writable memory", SIGSEGV);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (space.enforces_syscall_regions() && !calling_region->is_syscall_region()) {
|
|
|
|
dbgln("Syscall from non-syscall region");
|
|
|
|
unlock_and_handle_crash("Syscall from non-syscall region", SIGSEGV);
|
|
|
|
}
|
|
|
|
}
|
2019-01-25 03:39:15 +03:00
|
|
|
}
|
|
|
|
|
2020-07-31 00:52:28 +03:00
|
|
|
Region* MemoryManager::find_region_from_vaddr(VirtualAddress vaddr)
|
2019-08-06 12:19:16 +03:00
|
|
|
{
|
|
|
|
if (auto* region = kernel_region_from_vaddr(vaddr))
|
|
|
|
return region;
|
2020-02-10 22:00:32 +03:00
|
|
|
auto page_directory = PageDirectory::find_by_cr3(read_cr3());
|
2019-08-06 12:19:16 +03:00
|
|
|
if (!page_directory)
|
|
|
|
return nullptr;
|
2021-08-06 14:59:22 +03:00
|
|
|
VERIFY(page_directory->address_space());
|
|
|
|
return find_user_region_from_vaddr(*page_directory->address_space(), vaddr);
|
2019-08-06 12:19:16 +03:00
|
|
|
}
|
|
|
|
|
2021-07-14 14:31:21 +03:00
|
|
|
PageFaultResponse MemoryManager::handle_page_fault(PageFault const& fault)
|
2018-10-18 14:05:00 +03:00
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY_INTERRUPTS_DISABLED();
|
2020-06-27 22:42:28 +03:00
|
|
|
if (Processor::current().in_irq()) {
|
2021-01-10 18:21:56 +03:00
|
|
|
dbgln("CPU[{}] BUG! Page fault while handling IRQ! code={}, vaddr={}, irq level: {}",
|
2021-01-27 06:44:01 +03:00
|
|
|
Processor::id(), fault.code(), fault.vaddr(), Processor::current().in_irq());
|
2020-02-23 12:46:16 +03:00
|
|
|
dump_kernel_regions();
|
2020-06-27 22:42:28 +03:00
|
|
|
return PageFaultResponse::ShouldCrash;
|
2020-02-23 12:46:16 +03:00
|
|
|
}
|
2021-03-10 00:44:04 +03:00
|
|
|
dbgln_if(PAGE_FAULT_DEBUG, "MM: CPU[{}] handle_page_fault({:#04x}) at {}", Processor::id(), fault.code(), fault.vaddr());
|
2020-07-31 00:52:28 +03:00
|
|
|
auto* region = find_region_from_vaddr(fault.vaddr());
|
2018-11-08 17:39:26 +03:00
|
|
|
if (!region) {
|
|
|
|
return PageFaultResponse::ShouldCrash;
|
|
|
|
}
|
2021-07-23 03:40:16 +03:00
|
|
|
return region->handle_fault(fault);
|
2018-10-18 14:05:00 +03:00
|
|
|
}
|
|
|
|
|
2021-07-13 17:10:48 +03:00
|
|
|
OwnPtr<Region> MemoryManager::allocate_contiguous_kernel_region(size_t size, StringView name, Region::Access access, Region::Cacheable cacheable)
|
2020-03-07 20:24:41 +03:00
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(!(size % PAGE_SIZE));
|
2021-08-09 02:26:02 +03:00
|
|
|
ScopedSpinLock lock(kernel_page_directory().get_lock());
|
2020-03-07 20:24:41 +03:00
|
|
|
auto range = kernel_page_directory().range_allocator().allocate_anywhere(size);
|
2021-01-27 23:01:45 +03:00
|
|
|
if (!range.has_value())
|
2021-01-11 02:29:28 +03:00
|
|
|
return {};
|
2021-08-15 12:07:59 +03:00
|
|
|
auto maybe_vmobject = AnonymousVMObject::try_create_physically_contiguous_with_size(size);
|
|
|
|
if (maybe_vmobject.is_error()) {
|
2021-05-28 13:56:42 +03:00
|
|
|
kernel_page_directory().range_allocator().deallocate(range.value());
|
2021-08-15 12:07:59 +03:00
|
|
|
// FIXME: Would be nice to be able to return a KResultOr from here.
|
2021-05-28 13:56:42 +03:00
|
|
|
return {};
|
|
|
|
}
|
2021-08-15 12:07:59 +03:00
|
|
|
return allocate_kernel_region_with_vmobject(range.value(), maybe_vmobject.release_value(), name, access, cacheable);
|
2020-03-07 20:24:41 +03:00
|
|
|
}
|
|
|
|
|
2021-05-28 10:33:14 +03:00
|
|
|
OwnPtr<Region> MemoryManager::allocate_kernel_region(size_t size, StringView name, Region::Access access, AllocationStrategy strategy, Region::Cacheable cacheable)
|
2019-05-14 12:51:00 +03:00
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(!(size % PAGE_SIZE));
|
2021-08-15 12:07:59 +03:00
|
|
|
auto maybe_vm_object = AnonymousVMObject::try_create_with_size(size, strategy);
|
|
|
|
if (maybe_vm_object.is_error())
|
2021-05-18 11:45:05 +03:00
|
|
|
return {};
|
2021-08-09 02:26:02 +03:00
|
|
|
ScopedSpinLock lock(kernel_page_directory().get_lock());
|
2019-05-20 05:46:29 +03:00
|
|
|
auto range = kernel_page_directory().range_allocator().allocate_anywhere(size);
|
2021-01-27 23:01:45 +03:00
|
|
|
if (!range.has_value())
|
2021-01-11 02:29:28 +03:00
|
|
|
return {};
|
2021-08-15 12:07:59 +03:00
|
|
|
return allocate_kernel_region_with_vmobject(range.value(), maybe_vm_object.release_value(), name, access, cacheable);
|
2019-05-14 12:51:00 +03:00
|
|
|
}
|
|
|
|
|
2021-05-28 10:33:14 +03:00
|
|
|
OwnPtr<Region> MemoryManager::allocate_kernel_region(PhysicalAddress paddr, size_t size, StringView name, Region::Access access, Region::Cacheable cacheable)
|
2019-07-19 18:01:16 +03:00
|
|
|
{
|
2021-08-15 12:07:59 +03:00
|
|
|
auto maybe_vm_object = AnonymousVMObject::try_create_for_physical_range(paddr, size);
|
|
|
|
if (maybe_vm_object.is_error())
|
2021-05-18 11:45:05 +03:00
|
|
|
return {};
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(!(size % PAGE_SIZE));
|
2021-08-09 02:26:02 +03:00
|
|
|
ScopedSpinLock lock(kernel_page_directory().get_lock());
|
2020-01-10 00:29:31 +03:00
|
|
|
auto range = kernel_page_directory().range_allocator().allocate_anywhere(size);
|
2021-01-27 23:01:45 +03:00
|
|
|
if (!range.has_value())
|
2021-01-11 02:29:28 +03:00
|
|
|
return {};
|
2021-08-15 12:07:59 +03:00
|
|
|
return allocate_kernel_region_with_vmobject(range.value(), maybe_vm_object.release_value(), name, access, cacheable);
|
2019-07-19 18:01:16 +03:00
|
|
|
}
|
|
|
|
|
2021-08-06 14:54:48 +03:00
|
|
|
OwnPtr<Region> MemoryManager::allocate_kernel_region_with_vmobject(VirtualRange const& range, VMObject& vmobject, StringView name, Region::Access access, Region::Cacheable cacheable)
|
2019-12-16 00:21:28 +03:00
|
|
|
{
|
2021-08-15 13:11:05 +03:00
|
|
|
auto maybe_region = Region::try_create_kernel_only(range, vmobject, 0, KString::try_create(name), access, cacheable);
|
|
|
|
if (maybe_region.is_error())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
auto region = maybe_region.release_value();
|
|
|
|
region->map(kernel_page_directory());
|
2019-12-16 00:21:28 +03:00
|
|
|
return region;
|
|
|
|
}
|
|
|
|
|
2021-05-28 10:33:14 +03:00
|
|
|
OwnPtr<Region> MemoryManager::allocate_kernel_region_with_vmobject(VMObject& vmobject, size_t size, StringView name, Region::Access access, Region::Cacheable cacheable)
|
2020-03-01 17:55:27 +03:00
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(!(size % PAGE_SIZE));
|
2021-08-09 02:26:02 +03:00
|
|
|
ScopedSpinLock lock(kernel_page_directory().get_lock());
|
2020-03-01 17:55:27 +03:00
|
|
|
auto range = kernel_page_directory().range_allocator().allocate_anywhere(size);
|
2021-01-27 23:01:45 +03:00
|
|
|
if (!range.has_value())
|
2021-01-11 02:29:28 +03:00
|
|
|
return {};
|
2021-05-28 10:33:14 +03:00
|
|
|
return allocate_kernel_region_with_vmobject(range.value(), vmobject, name, access, cacheable);
|
2020-03-01 17:55:27 +03:00
|
|
|
}
|
|
|
|
|
2021-08-04 23:49:13 +03:00
|
|
|
Optional<CommittedPhysicalPageSet> MemoryManager::commit_user_physical_pages(size_t page_count)
|
2020-09-05 06:12:25 +03:00
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(page_count > 0);
|
2020-09-05 06:12:25 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2021-07-07 06:35:15 +03:00
|
|
|
if (m_system_memory_info.user_physical_pages_uncommitted < page_count)
|
2021-08-04 23:49:13 +03:00
|
|
|
return {};
|
2020-09-05 06:12:25 +03:00
|
|
|
|
2021-07-07 06:35:15 +03:00
|
|
|
m_system_memory_info.user_physical_pages_uncommitted -= page_count;
|
|
|
|
m_system_memory_info.user_physical_pages_committed += page_count;
|
2021-08-04 23:49:13 +03:00
|
|
|
return CommittedPhysicalPageSet { {}, page_count };
|
2020-09-05 06:12:25 +03:00
|
|
|
}
|
|
|
|
|
2021-08-04 23:49:13 +03:00
|
|
|
void MemoryManager::uncommit_user_physical_pages(Badge<CommittedPhysicalPageSet>, size_t page_count)
|
2020-09-05 06:12:25 +03:00
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(page_count > 0);
|
2021-08-04 23:49:13 +03:00
|
|
|
|
2020-09-05 06:12:25 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2021-07-07 06:35:15 +03:00
|
|
|
VERIFY(m_system_memory_info.user_physical_pages_committed >= page_count);
|
2020-09-05 06:12:25 +03:00
|
|
|
|
2021-07-07 06:35:15 +03:00
|
|
|
m_system_memory_info.user_physical_pages_uncommitted += page_count;
|
|
|
|
m_system_memory_info.user_physical_pages_committed -= page_count;
|
2020-09-05 06:12:25 +03:00
|
|
|
}
|
|
|
|
|
2021-07-12 00:12:32 +03:00
|
|
|
void MemoryManager::deallocate_physical_page(PhysicalAddress paddr)
|
2019-06-11 14:13:02 +03:00
|
|
|
{
|
2020-07-06 16:27:22 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2021-07-12 00:12:32 +03:00
|
|
|
|
|
|
|
// Are we returning a user page?
|
2019-06-11 14:13:02 +03:00
|
|
|
for (auto& region : m_user_physical_regions) {
|
2021-07-08 05:28:51 +03:00
|
|
|
if (!region.contains(paddr))
|
2019-06-11 14:13:02 +03:00
|
|
|
continue;
|
|
|
|
|
2021-07-08 05:28:51 +03:00
|
|
|
region.return_page(paddr);
|
2021-07-07 06:35:15 +03:00
|
|
|
--m_system_memory_info.user_physical_pages_used;
|
2019-06-11 14:13:02 +03:00
|
|
|
|
2020-09-05 06:12:25 +03:00
|
|
|
// Always return pages to the uncommitted pool. Pages that were
|
|
|
|
// committed and allocated are only freed upon request. Once
|
|
|
|
// returned there is no guarantee being able to get them back.
|
2021-07-07 06:35:15 +03:00
|
|
|
++m_system_memory_info.user_physical_pages_uncommitted;
|
2019-06-11 14:13:02 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-07-12 00:12:32 +03:00
|
|
|
// If it's not a user page, it should be a supervisor page.
|
2021-08-04 20:29:02 +03:00
|
|
|
if (!m_super_physical_region->contains(paddr))
|
|
|
|
PANIC("MM: deallocate_user_physical_page couldn't figure out region for page @ {}", paddr);
|
2021-07-12 00:12:32 +03:00
|
|
|
|
2021-08-04 20:29:02 +03:00
|
|
|
m_super_physical_region->return_page(paddr);
|
|
|
|
--m_system_memory_info.super_physical_pages_used;
|
2019-06-11 14:13:02 +03:00
|
|
|
}
|
|
|
|
|
2020-09-05 06:12:25 +03:00
|
|
|
RefPtr<PhysicalPage> MemoryManager::find_free_user_physical_page(bool committed)
|
2018-11-08 14:59:16 +03:00
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(s_mm_lock.is_locked());
|
2019-06-27 14:34:28 +03:00
|
|
|
RefPtr<PhysicalPage> page;
|
2020-09-05 06:12:25 +03:00
|
|
|
if (committed) {
|
|
|
|
// Draw from the committed pages pool. We should always have these pages available
|
2021-07-07 06:35:15 +03:00
|
|
|
VERIFY(m_system_memory_info.user_physical_pages_committed > 0);
|
|
|
|
m_system_memory_info.user_physical_pages_committed--;
|
2020-09-05 06:12:25 +03:00
|
|
|
} else {
|
|
|
|
// We need to make sure we don't touch pages that we have committed to
|
2021-07-07 06:35:15 +03:00
|
|
|
if (m_system_memory_info.user_physical_pages_uncommitted == 0)
|
2020-09-05 06:12:25 +03:00
|
|
|
return {};
|
2021-07-07 06:35:15 +03:00
|
|
|
m_system_memory_info.user_physical_pages_uncommitted--;
|
2020-09-05 06:12:25 +03:00
|
|
|
}
|
2019-06-11 14:13:02 +03:00
|
|
|
for (auto& region : m_user_physical_regions) {
|
2021-07-12 00:12:32 +03:00
|
|
|
page = region.take_free_page();
|
2020-09-05 06:12:25 +03:00
|
|
|
if (!page.is_null()) {
|
2021-07-07 06:35:15 +03:00
|
|
|
++m_system_memory_info.user_physical_pages_used;
|
2019-11-09 00:39:29 +03:00
|
|
|
break;
|
2020-09-05 06:12:25 +03:00
|
|
|
}
|
2019-06-11 14:13:02 +03:00
|
|
|
}
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(!committed || !page.is_null());
|
2019-11-09 00:39:29 +03:00
|
|
|
return page;
|
|
|
|
}
|
|
|
|
|
2021-08-05 20:58:09 +03:00
|
|
|
NonnullRefPtr<PhysicalPage> MemoryManager::allocate_committed_user_physical_page(Badge<CommittedPhysicalPageSet>, ShouldZeroFill should_zero_fill)
|
2020-09-05 06:12:25 +03:00
|
|
|
{
|
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
|
|
|
auto page = find_free_user_physical_page(true);
|
|
|
|
if (should_zero_fill == ShouldZeroFill::Yes) {
|
|
|
|
auto* ptr = quickmap_page(*page);
|
|
|
|
memset(ptr, 0, PAGE_SIZE);
|
|
|
|
unquickmap_page();
|
|
|
|
}
|
|
|
|
return page.release_nonnull();
|
|
|
|
}
|
|
|
|
|
2020-09-01 22:40:34 +03:00
|
|
|
RefPtr<PhysicalPage> MemoryManager::allocate_user_physical_page(ShouldZeroFill should_zero_fill, bool* did_purge)
|
2019-11-09 00:39:29 +03:00
|
|
|
{
|
2020-07-06 16:27:22 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2020-09-05 06:12:25 +03:00
|
|
|
auto page = find_free_user_physical_page(false);
|
2020-09-01 22:40:34 +03:00
|
|
|
bool purged_pages = false;
|
2019-06-11 14:13:02 +03:00
|
|
|
|
|
|
|
if (!page) {
|
2020-05-08 23:15:02 +03:00
|
|
|
// We didn't have a single free physical page. Let's try to free something up!
|
|
|
|
// First, we look for a purgeable VMObject in the volatile state.
|
2021-01-01 17:32:06 +03:00
|
|
|
for_each_vmobject([&](auto& vmobject) {
|
2020-09-06 00:52:14 +03:00
|
|
|
if (!vmobject.is_anonymous())
|
2021-01-01 17:32:06 +03:00
|
|
|
return IterationDecision::Continue;
|
2021-07-25 03:11:32 +03:00
|
|
|
auto& anonymous_vmobject = static_cast<AnonymousVMObject&>(vmobject);
|
|
|
|
if (!anonymous_vmobject.is_purgeable() || !anonymous_vmobject.is_volatile())
|
|
|
|
return IterationDecision::Continue;
|
|
|
|
if (auto purged_page_count = anonymous_vmobject.purge()) {
|
2021-02-12 18:20:21 +03:00
|
|
|
dbgln("MM: Purge saved the day! Purged {} pages from AnonymousVMObject", purged_page_count);
|
2020-09-05 06:12:25 +03:00
|
|
|
page = find_free_user_physical_page(false);
|
2020-09-01 22:40:34 +03:00
|
|
|
purged_pages = true;
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(page);
|
2020-05-08 23:10:47 +03:00
|
|
|
return IterationDecision::Break;
|
2019-12-26 13:45:36 +03:00
|
|
|
}
|
|
|
|
return IterationDecision::Continue;
|
|
|
|
});
|
|
|
|
if (!page) {
|
2021-02-12 18:20:21 +03:00
|
|
|
dmesgln("MM: no user physical pages available");
|
2019-12-26 13:45:36 +03:00
|
|
|
return {};
|
|
|
|
}
|
2019-02-20 17:34:55 +03:00
|
|
|
}
|
2019-06-11 14:13:02 +03:00
|
|
|
|
2019-01-31 05:57:06 +03:00
|
|
|
if (should_zero_fill == ShouldZeroFill::Yes) {
|
2020-01-20 15:06:14 +03:00
|
|
|
auto* ptr = quickmap_page(*page);
|
2020-01-10 08:59:01 +03:00
|
|
|
memset(ptr, 0, PAGE_SIZE);
|
2019-01-31 05:57:06 +03:00
|
|
|
unquickmap_page();
|
|
|
|
}
|
2019-06-11 14:13:02 +03:00
|
|
|
|
2020-09-01 22:40:34 +03:00
|
|
|
if (did_purge)
|
|
|
|
*did_purge = purged_pages;
|
2019-06-11 14:13:02 +03:00
|
|
|
return page;
|
|
|
|
}
|
|
|
|
|
2021-07-13 17:10:48 +03:00
|
|
|
NonnullRefPtrVector<PhysicalPage> MemoryManager::allocate_contiguous_supervisor_physical_pages(size_t size)
|
2020-03-07 20:24:41 +03:00
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(!(size % PAGE_SIZE));
|
2020-07-06 16:27:22 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2021-03-04 19:50:05 +03:00
|
|
|
size_t count = ceil_div(size, static_cast<size_t>(PAGE_SIZE));
|
2021-08-04 20:29:02 +03:00
|
|
|
auto physical_pages = m_super_physical_region->take_contiguous_free_pages(count);
|
2020-03-07 20:24:41 +03:00
|
|
|
|
|
|
|
if (physical_pages.is_empty()) {
|
2021-02-12 18:20:21 +03:00
|
|
|
dmesgln("MM: no super physical pages available");
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-03-07 20:24:41 +03:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-05-08 21:15:01 +03:00
|
|
|
auto cleanup_region = MM.allocate_kernel_region(physical_pages[0].paddr(), PAGE_SIZE * count, "MemoryManager Allocation Sanitization", Region::Access::Read | Region::Access::Write);
|
2020-03-07 20:24:41 +03:00
|
|
|
fast_u32_fill((u32*)cleanup_region->vaddr().as_ptr(), 0, (PAGE_SIZE * count) / sizeof(u32));
|
2021-07-07 06:35:15 +03:00
|
|
|
m_system_memory_info.super_physical_pages_used += count;
|
2020-03-07 20:24:41 +03:00
|
|
|
return physical_pages;
|
|
|
|
}
|
|
|
|
|
2019-06-21 19:37:47 +03:00
|
|
|
RefPtr<PhysicalPage> MemoryManager::allocate_supervisor_physical_page()
|
2018-11-01 13:30:48 +03:00
|
|
|
{
|
2020-07-06 16:27:22 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2021-08-04 20:29:02 +03:00
|
|
|
auto page = m_super_physical_region->take_free_page();
|
2019-06-11 14:13:02 +03:00
|
|
|
|
|
|
|
if (!page) {
|
2021-02-12 18:20:21 +03:00
|
|
|
dmesgln("MM: no super physical pages available");
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY_NOT_REACHED();
|
2019-06-07 12:43:58 +03:00
|
|
|
return {};
|
2019-02-20 17:34:55 +03:00
|
|
|
}
|
2019-06-11 14:13:02 +03:00
|
|
|
|
2021-07-26 16:10:51 +03:00
|
|
|
fast_u32_fill((u32*)page->paddr().offset(physical_to_virtual_offset).as_ptr(), 0, PAGE_SIZE / sizeof(u32));
|
2021-07-07 06:35:15 +03:00
|
|
|
++m_system_memory_info.super_physical_pages_used;
|
2019-06-11 14:13:02 +03:00
|
|
|
return page;
|
2018-11-01 13:30:48 +03:00
|
|
|
}
|
|
|
|
|
2018-11-01 15:15:46 +03:00
|
|
|
void MemoryManager::enter_process_paging_scope(Process& process)
|
2021-02-08 17:45:40 +03:00
|
|
|
{
|
2021-08-06 14:59:22 +03:00
|
|
|
enter_space(process.address_space());
|
2021-02-08 17:45:40 +03:00
|
|
|
}
|
|
|
|
|
2021-08-06 14:57:39 +03:00
|
|
|
void MemoryManager::enter_space(AddressSpace& space)
|
2018-11-01 13:30:48 +03:00
|
|
|
{
|
2020-06-29 00:34:31 +03:00
|
|
|
auto current_thread = Thread::current();
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(current_thread != nullptr);
|
2020-07-06 16:27:22 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2019-11-27 14:40:42 +03:00
|
|
|
|
2021-06-26 20:57:16 +03:00
|
|
|
current_thread->regs().cr3 = space.page_directory().cr3();
|
2021-02-08 17:45:40 +03:00
|
|
|
write_cr3(space.page_directory().cr3());
|
2018-11-01 13:30:48 +03:00
|
|
|
}
|
|
|
|
|
2020-07-06 16:27:22 +03:00
|
|
|
void MemoryManager::flush_tlb_local(VirtualAddress vaddr, size_t page_count)
|
2018-10-23 12:03:56 +03:00
|
|
|
{
|
2020-07-06 16:27:22 +03:00
|
|
|
Processor::flush_tlb_local(vaddr, page_count);
|
2018-10-23 12:03:56 +03:00
|
|
|
}
|
|
|
|
|
2021-07-14 14:31:21 +03:00
|
|
|
void MemoryManager::flush_tlb(PageDirectory const* page_directory, VirtualAddress vaddr, size_t page_count)
|
2018-10-23 16:53:11 +03:00
|
|
|
{
|
2021-01-02 22:27:38 +03:00
|
|
|
Processor::flush_tlb(page_directory, vaddr, page_count);
|
2018-10-23 16:53:11 +03:00
|
|
|
}
|
|
|
|
|
2020-01-17 21:59:20 +03:00
|
|
|
PageDirectoryEntry* MemoryManager::quickmap_pd(PageDirectory& directory, size_t pdpt_index)
|
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(s_mm_lock.own_lock());
|
2020-11-01 02:19:50 +03:00
|
|
|
auto& mm_data = get_data();
|
2021-07-19 19:24:15 +03:00
|
|
|
auto& pte = boot_pd_kernel_pt1023[(KERNEL_QUICKMAP_PD - KERNEL_PT1024_BASE) / PAGE_SIZE];
|
2020-01-17 21:59:20 +03:00
|
|
|
auto pd_paddr = directory.m_directory_pages[pdpt_index]->paddr();
|
2021-07-07 06:35:15 +03:00
|
|
|
if (pte.physical_page_base() != pd_paddr.get()) {
|
2020-01-17 21:59:20 +03:00
|
|
|
pte.set_physical_page_base(pd_paddr.get());
|
|
|
|
pte.set_present(true);
|
|
|
|
pte.set_writable(true);
|
|
|
|
pte.set_user_allowed(false);
|
2020-07-06 18:11:52 +03:00
|
|
|
// Because we must continue to hold the MM lock while we use this
|
|
|
|
// mapping, it is sufficient to only flush on the current CPU. Other
|
|
|
|
// CPUs trying to use this API must wait on the MM lock anyway
|
2021-07-08 04:50:05 +03:00
|
|
|
flush_tlb_local(VirtualAddress(KERNEL_QUICKMAP_PD));
|
2020-11-01 02:19:50 +03:00
|
|
|
} else {
|
|
|
|
// Even though we don't allow this to be called concurrently, it's
|
|
|
|
// possible that this PD was mapped on a different CPU and we don't
|
|
|
|
// broadcast the flush. If so, we still need to flush the TLB.
|
|
|
|
if (mm_data.m_last_quickmap_pd != pd_paddr)
|
2021-07-08 04:50:05 +03:00
|
|
|
flush_tlb_local(VirtualAddress(KERNEL_QUICKMAP_PD));
|
2020-01-17 21:59:20 +03:00
|
|
|
}
|
2020-11-01 02:19:50 +03:00
|
|
|
mm_data.m_last_quickmap_pd = pd_paddr;
|
2021-07-08 04:50:05 +03:00
|
|
|
return (PageDirectoryEntry*)KERNEL_QUICKMAP_PD;
|
2020-01-17 21:59:20 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
PageTableEntry* MemoryManager::quickmap_pt(PhysicalAddress pt_paddr)
|
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(s_mm_lock.own_lock());
|
2020-11-01 02:19:50 +03:00
|
|
|
auto& mm_data = get_data();
|
2021-07-18 15:47:32 +03:00
|
|
|
auto& pte = ((PageTableEntry*)boot_pd_kernel_pt1023)[(KERNEL_QUICKMAP_PT - KERNEL_PT1024_BASE) / PAGE_SIZE];
|
2021-07-07 06:35:15 +03:00
|
|
|
if (pte.physical_page_base() != pt_paddr.get()) {
|
2020-01-17 21:59:20 +03:00
|
|
|
pte.set_physical_page_base(pt_paddr.get());
|
|
|
|
pte.set_present(true);
|
|
|
|
pte.set_writable(true);
|
|
|
|
pte.set_user_allowed(false);
|
2020-07-06 18:11:52 +03:00
|
|
|
// Because we must continue to hold the MM lock while we use this
|
|
|
|
// mapping, it is sufficient to only flush on the current CPU. Other
|
|
|
|
// CPUs trying to use this API must wait on the MM lock anyway
|
2021-07-08 04:50:05 +03:00
|
|
|
flush_tlb_local(VirtualAddress(KERNEL_QUICKMAP_PT));
|
2020-11-01 02:19:50 +03:00
|
|
|
} else {
|
|
|
|
// Even though we don't allow this to be called concurrently, it's
|
|
|
|
// possible that this PT was mapped on a different CPU and we don't
|
|
|
|
// broadcast the flush. If so, we still need to flush the TLB.
|
|
|
|
if (mm_data.m_last_quickmap_pt != pt_paddr)
|
2021-07-08 04:50:05 +03:00
|
|
|
flush_tlb_local(VirtualAddress(KERNEL_QUICKMAP_PT));
|
2020-01-17 21:59:20 +03:00
|
|
|
}
|
2020-11-01 02:19:50 +03:00
|
|
|
mm_data.m_last_quickmap_pt = pt_paddr;
|
2021-07-08 04:50:05 +03:00
|
|
|
return (PageTableEntry*)KERNEL_QUICKMAP_PT;
|
2020-01-17 21:59:20 +03:00
|
|
|
}
|
|
|
|
|
2021-07-08 04:50:05 +03:00
|
|
|
u8* MemoryManager::quickmap_page(PhysicalAddress const& physical_address)
|
2018-11-05 15:48:07 +03:00
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY_INTERRUPTS_DISABLED();
|
2020-06-29 01:04:35 +03:00
|
|
|
auto& mm_data = get_data();
|
2020-07-03 14:19:50 +03:00
|
|
|
mm_data.m_quickmap_prev_flags = mm_data.m_quickmap_in_use.lock();
|
2020-07-06 16:27:22 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2020-01-18 00:18:56 +03:00
|
|
|
|
2021-07-08 04:50:05 +03:00
|
|
|
VirtualAddress vaddr(KERNEL_QUICKMAP_PER_CPU_BASE + Processor::id() * PAGE_SIZE);
|
|
|
|
u32 pte_idx = (vaddr.get() - KERNEL_PT1024_BASE) / PAGE_SIZE;
|
2020-06-29 01:04:35 +03:00
|
|
|
|
2021-07-18 15:47:32 +03:00
|
|
|
auto& pte = ((PageTableEntry*)boot_pd_kernel_pt1023)[pte_idx];
|
2021-07-08 04:50:05 +03:00
|
|
|
if (pte.physical_page_base() != physical_address.get()) {
|
|
|
|
pte.set_physical_page_base(physical_address.get());
|
2020-01-18 00:18:56 +03:00
|
|
|
pte.set_present(true);
|
|
|
|
pte.set_writable(true);
|
|
|
|
pte.set_user_allowed(false);
|
2020-07-06 18:11:52 +03:00
|
|
|
flush_tlb_local(vaddr);
|
2020-01-18 00:18:56 +03:00
|
|
|
}
|
2020-06-29 01:04:35 +03:00
|
|
|
return vaddr.as_ptr();
|
2018-11-05 15:48:07 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void MemoryManager::unquickmap_page()
|
|
|
|
{
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY_INTERRUPTS_DISABLED();
|
2020-07-06 16:27:22 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2020-06-29 01:04:35 +03:00
|
|
|
auto& mm_data = get_data();
|
2021-02-23 22:42:32 +03:00
|
|
|
VERIFY(mm_data.m_quickmap_in_use.is_locked());
|
2021-07-08 04:50:05 +03:00
|
|
|
VirtualAddress vaddr(KERNEL_QUICKMAP_PER_CPU_BASE + Processor::id() * PAGE_SIZE);
|
|
|
|
u32 pte_idx = (vaddr.get() - KERNEL_PT1024_BASE) / PAGE_SIZE;
|
2021-07-18 15:47:32 +03:00
|
|
|
auto& pte = ((PageTableEntry*)boot_pd_kernel_pt1023)[pte_idx];
|
2020-02-08 14:49:00 +03:00
|
|
|
pte.clear();
|
2020-07-06 18:11:52 +03:00
|
|
|
flush_tlb_local(vaddr);
|
2020-07-03 14:19:50 +03:00
|
|
|
mm_data.m_quickmap_in_use.unlock(mm_data.m_quickmap_prev_flags);
|
2018-11-05 15:48:07 +03:00
|
|
|
}
|
|
|
|
|
2021-08-06 14:57:39 +03:00
|
|
|
bool MemoryManager::validate_user_stack_no_lock(AddressSpace& space, VirtualAddress vaddr) const
|
2019-11-17 14:11:43 +03:00
|
|
|
{
|
2021-07-18 18:53:37 +03:00
|
|
|
VERIFY(space.get_lock().own_lock());
|
|
|
|
|
2019-12-31 05:43:24 +03:00
|
|
|
if (!is_user_address(vaddr))
|
|
|
|
return false;
|
2021-07-18 18:53:37 +03:00
|
|
|
|
2021-07-18 19:31:28 +03:00
|
|
|
auto* region = find_user_region_from_vaddr_no_lock(space, vaddr);
|
2021-02-14 03:25:22 +03:00
|
|
|
return region && region->is_user() && region->is_stack();
|
2019-11-17 14:11:43 +03:00
|
|
|
}
|
|
|
|
|
2021-08-06 14:57:39 +03:00
|
|
|
bool MemoryManager::validate_user_stack(AddressSpace& space, VirtualAddress vaddr) const
|
2021-07-18 18:53:37 +03:00
|
|
|
{
|
|
|
|
ScopedSpinLock lock(space.get_lock());
|
|
|
|
return validate_user_stack_no_lock(space, vaddr);
|
|
|
|
}
|
|
|
|
|
2018-11-09 00:24:02 +03:00
|
|
|
void MemoryManager::register_region(Region& region)
|
|
|
|
{
|
2020-07-06 16:27:22 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2020-06-02 07:55:09 +03:00
|
|
|
if (region.is_kernel())
|
2021-05-26 12:47:47 +03:00
|
|
|
m_kernel_regions.append(region);
|
2018-11-09 00:24:02 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void MemoryManager::unregister_region(Region& region)
|
|
|
|
{
|
2020-07-06 16:27:22 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2020-06-02 07:55:09 +03:00
|
|
|
if (region.is_kernel())
|
2021-05-26 12:47:47 +03:00
|
|
|
m_kernel_regions.remove(region);
|
2018-11-09 00:24:02 +03:00
|
|
|
}
|
2018-11-09 03:25:31 +03:00
|
|
|
|
2020-01-18 10:34:28 +03:00
|
|
|
void MemoryManager::dump_kernel_regions()
|
|
|
|
{
|
2021-02-12 18:20:21 +03:00
|
|
|
dbgln("Kernel regions:");
|
2021-07-22 02:21:39 +03:00
|
|
|
#if ARCH(I386)
|
|
|
|
auto addr_padding = "";
|
|
|
|
#else
|
|
|
|
auto addr_padding = " ";
|
|
|
|
#endif
|
|
|
|
dbgln("BEGIN{} END{} SIZE{} ACCESS NAME",
|
|
|
|
addr_padding, addr_padding, addr_padding);
|
2020-07-06 16:27:22 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2021-02-12 18:20:21 +03:00
|
|
|
for (auto& region : m_kernel_regions) {
|
2021-07-21 20:53:38 +03:00
|
|
|
dbgln("{:p} -- {:p} {:p} {:c}{:c}{:c}{:c}{:c}{:c} {}",
|
2021-02-12 18:33:58 +03:00
|
|
|
region.vaddr().get(),
|
|
|
|
region.vaddr().offset(region.size() - 1).get(),
|
|
|
|
region.size(),
|
2021-02-12 18:20:21 +03:00
|
|
|
region.is_readable() ? 'R' : ' ',
|
|
|
|
region.is_writable() ? 'W' : ' ',
|
|
|
|
region.is_executable() ? 'X' : ' ',
|
|
|
|
region.is_shared() ? 'S' : ' ',
|
|
|
|
region.is_stack() ? 'T' : ' ',
|
|
|
|
region.is_syscall_region() ? 'C' : ' ',
|
|
|
|
region.name());
|
2020-01-18 10:34:28 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-11 15:03:40 +03:00
|
|
|
void MemoryManager::set_page_writable_direct(VirtualAddress vaddr, bool writable)
|
|
|
|
{
|
|
|
|
ScopedSpinLock page_lock(kernel_page_directory().get_lock());
|
2021-08-09 02:26:02 +03:00
|
|
|
ScopedSpinLock lock(s_mm_lock);
|
2021-03-11 15:03:40 +03:00
|
|
|
auto* pte = ensure_pte(kernel_page_directory(), vaddr);
|
|
|
|
VERIFY(pte);
|
|
|
|
if (pte->is_writable() == writable)
|
|
|
|
return;
|
|
|
|
pte->set_writable(writable);
|
|
|
|
flush_tlb(&kernel_page_directory(), vaddr);
|
|
|
|
}
|
|
|
|
|
2021-08-04 23:49:13 +03:00
|
|
|
CommittedPhysicalPageSet::~CommittedPhysicalPageSet()
|
|
|
|
{
|
|
|
|
if (m_page_count)
|
|
|
|
MM.uncommit_user_physical_pages({}, m_page_count);
|
|
|
|
}
|
|
|
|
|
|
|
|
NonnullRefPtr<PhysicalPage> CommittedPhysicalPageSet::take_one()
|
|
|
|
{
|
|
|
|
VERIFY(m_page_count > 0);
|
|
|
|
--m_page_count;
|
2021-08-05 20:58:09 +03:00
|
|
|
return MM.allocate_committed_user_physical_page({}, MemoryManager::ShouldZeroFill::Yes);
|
2021-08-04 23:49:13 +03:00
|
|
|
}
|
|
|
|
|
2021-08-05 18:14:13 +03:00
|
|
|
void CommittedPhysicalPageSet::uncommit_one()
|
|
|
|
{
|
|
|
|
VERIFY(m_page_count > 0);
|
|
|
|
--m_page_count;
|
|
|
|
MM.uncommit_user_physical_pages({}, 1);
|
|
|
|
}
|
|
|
|
|
2020-02-16 03:27:42 +03:00
|
|
|
}
|