2021-02-08 15:45:40 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
|
2021-02-24 16:44:00 +01:00
|
|
|
* Copyright (c) 2021, Leon Albrecht <leon2002.la@gmail.com>
|
2021-02-08 15:45:40 +01:00
|
|
|
*
|
2021-04-22 01:24:48 -07:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2021-02-08 15:45:40 +01:00
|
|
|
*/
|
|
|
|
|
2022-04-02 23:47:27 +01:00
|
|
|
#include <Kernel/Arch/CPU.h>
|
2021-08-22 01:37:17 +02:00
|
|
|
#include <Kernel/Locking/Spinlock.h>
|
2021-08-06 13:57:39 +02:00
|
|
|
#include <Kernel/Memory/AddressSpace.h>
|
2021-08-06 10:45:34 +02:00
|
|
|
#include <Kernel/Memory/AnonymousVMObject.h>
|
|
|
|
#include <Kernel/Memory/InodeVMObject.h>
|
|
|
|
#include <Kernel/Memory/MemoryManager.h>
|
2021-05-28 11:03:21 +02:00
|
|
|
#include <Kernel/PerformanceManager.h>
|
2021-02-08 15:45:40 +01:00
|
|
|
#include <Kernel/Process.h>
|
2022-01-29 13:08:37 +01:00
|
|
|
#include <Kernel/Scheduler.h>
|
2021-02-08 15:45:40 +01:00
|
|
|
|
2021-08-06 13:49:36 +02:00
|
|
|
namespace Kernel::Memory {
|
2021-02-08 15:45:40 +01:00
|
|
|
|
2021-11-08 00:51:39 +01:00
|
|
|
ErrorOr<NonnullOwnPtr<AddressSpace>> AddressSpace::try_create(AddressSpace const* parent)
|
2021-02-08 15:45:40 +01:00
|
|
|
{
|
2021-09-05 15:13:20 +02:00
|
|
|
auto page_directory = TRY(PageDirectory::try_create_for_userspace(parent ? &parent->page_directory().range_allocator() : nullptr));
|
|
|
|
auto space = TRY(adopt_nonnull_own_or_enomem(new (nothrow) AddressSpace(page_directory)));
|
2021-02-08 15:45:40 +01:00
|
|
|
space->page_directory().set_space({}, *space);
|
|
|
|
return space;
|
|
|
|
}
|
|
|
|
|
2021-08-07 21:32:30 +02:00
|
|
|
AddressSpace::AddressSpace(NonnullRefPtr<PageDirectory> page_directory)
|
|
|
|
: m_page_directory(move(page_directory))
|
2021-02-08 15:45:40 +01:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-03-16 13:15:15 -06:00
|
|
|
AddressSpace::~AddressSpace() = default;
|
2021-02-08 15:45:40 +01:00
|
|
|
|
2021-11-08 00:51:39 +01:00
|
|
|
ErrorOr<void> AddressSpace::unmap_mmap_range(VirtualAddress addr, size_t size)
|
2021-05-28 11:03:21 +02:00
|
|
|
{
|
|
|
|
if (!size)
|
|
|
|
return EINVAL;
|
|
|
|
|
2021-09-05 14:40:06 +02:00
|
|
|
auto range_to_unmap = TRY(VirtualRange::expand_to_page_boundaries(addr.get(), size));
|
2021-05-28 11:03:21 +02:00
|
|
|
|
|
|
|
if (!is_user_range(range_to_unmap))
|
|
|
|
return EFAULT;
|
|
|
|
|
|
|
|
if (auto* whole_region = find_region_from_range(range_to_unmap)) {
|
|
|
|
if (!whole_region->is_mmap())
|
|
|
|
return EPERM;
|
|
|
|
|
2021-08-19 22:45:07 +03:00
|
|
|
PerformanceManager::add_unmap_perf_event(Process::current(), whole_region->range());
|
2021-05-28 11:03:21 +02:00
|
|
|
|
2021-07-17 06:07:02 -07:00
|
|
|
deallocate_region(*whole_region);
|
2021-11-08 00:51:39 +01:00
|
|
|
return {};
|
2021-05-28 11:03:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (auto* old_region = find_region_containing(range_to_unmap)) {
|
|
|
|
if (!old_region->is_mmap())
|
|
|
|
return EPERM;
|
|
|
|
|
|
|
|
// Remove the old region from our regions tree, since were going to add another region
|
2021-07-27 15:04:04 +02:00
|
|
|
// with the exact same start address, but don't deallocate it yet.
|
2021-05-28 11:03:21 +02:00
|
|
|
auto region = take_region(*old_region);
|
|
|
|
|
|
|
|
// We manually unmap the old region here, specifying that we *don't* want the VM deallocated.
|
2021-08-06 21:45:05 +02:00
|
|
|
region->unmap(Region::ShouldDeallocateVirtualRange::No);
|
2021-05-28 11:03:21 +02:00
|
|
|
|
2021-09-05 14:40:06 +02:00
|
|
|
auto new_regions = TRY(try_split_region_around_range(*region, range_to_unmap));
|
2021-05-28 11:03:21 +02:00
|
|
|
|
|
|
|
// Instead we give back the unwanted VM manually.
|
|
|
|
page_directory().range_allocator().deallocate(range_to_unmap);
|
|
|
|
|
|
|
|
// And finally we map the new region(s) using our page directory (they were just allocated and don't have one).
|
|
|
|
for (auto* new_region : new_regions) {
|
2021-08-24 12:53:47 -07:00
|
|
|
// TODO: Ideally we should do this in a way that can be rolled back on failure, as failing here
|
|
|
|
// leaves the caller in an undefined state.
|
2021-09-06 12:52:23 +02:00
|
|
|
TRY(new_region->map(page_directory()));
|
2021-05-28 11:03:21 +02:00
|
|
|
}
|
|
|
|
|
2021-08-19 22:45:07 +03:00
|
|
|
PerformanceManager::add_unmap_perf_event(Process::current(), range_to_unmap);
|
2021-05-28 11:03:21 +02:00
|
|
|
|
2021-11-08 00:51:39 +01:00
|
|
|
return {};
|
2021-05-28 11:03:21 +02:00
|
|
|
}
|
|
|
|
|
2021-07-27 15:04:04 +02:00
|
|
|
// Try again while checking multiple regions at a time.
|
2022-01-25 15:33:48 +02:00
|
|
|
auto const& regions = TRY(find_regions_intersecting(range_to_unmap));
|
2021-07-30 01:17:59 -07:00
|
|
|
if (regions.is_empty())
|
2021-11-08 00:51:39 +01:00
|
|
|
return {};
|
2021-05-28 11:03:21 +02:00
|
|
|
|
2021-07-27 15:04:04 +02:00
|
|
|
// Check if any of the regions is not mmap'ed, to not accidentally
|
|
|
|
// error out with just half a region map left.
|
2021-05-28 11:03:21 +02:00
|
|
|
for (auto* region : regions) {
|
|
|
|
if (!region->is_mmap())
|
|
|
|
return EPERM;
|
|
|
|
}
|
|
|
|
|
|
|
|
Vector<Region*, 2> new_regions;
|
|
|
|
|
|
|
|
for (auto* old_region : regions) {
|
2021-07-27 15:04:04 +02:00
|
|
|
// If it's a full match we can remove the entire old region.
|
2021-05-28 11:03:21 +02:00
|
|
|
if (old_region->range().intersect(range_to_unmap).size() == old_region->size()) {
|
2021-07-17 06:07:02 -07:00
|
|
|
deallocate_region(*old_region);
|
2021-05-28 11:03:21 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove the old region from our regions tree, since were going to add another region
|
2021-07-27 15:04:04 +02:00
|
|
|
// with the exact same start address, but don't deallocate it yet.
|
2021-05-28 11:03:21 +02:00
|
|
|
auto region = take_region(*old_region);
|
|
|
|
|
|
|
|
// We manually unmap the old region here, specifying that we *don't* want the VM deallocated.
|
2021-08-06 21:45:05 +02:00
|
|
|
region->unmap(Region::ShouldDeallocateVirtualRange::No);
|
2021-05-28 11:03:21 +02:00
|
|
|
|
2021-07-27 15:04:04 +02:00
|
|
|
// Otherwise, split the regions and collect them for future mapping.
|
2021-09-05 14:40:06 +02:00
|
|
|
auto split_regions = TRY(try_split_region_around_range(*region, range_to_unmap));
|
2021-11-10 11:55:37 +01:00
|
|
|
TRY(new_regions.try_extend(split_regions));
|
2021-05-28 11:03:21 +02:00
|
|
|
}
|
2021-07-27 15:04:04 +02:00
|
|
|
|
|
|
|
// Give back any unwanted VM to the range allocator.
|
2021-05-28 11:03:21 +02:00
|
|
|
page_directory().range_allocator().deallocate(range_to_unmap);
|
2021-07-27 15:04:04 +02:00
|
|
|
|
|
|
|
// And finally map the new region(s) into our page directory.
|
2021-05-28 11:03:21 +02:00
|
|
|
for (auto* new_region : new_regions) {
|
2021-08-24 12:53:47 -07:00
|
|
|
// TODO: Ideally we should do this in a way that can be rolled back on failure, as failing here
|
|
|
|
// leaves the caller in an undefined state.
|
2021-09-06 12:52:23 +02:00
|
|
|
TRY(new_region->map(page_directory()));
|
2021-05-28 11:03:21 +02:00
|
|
|
}
|
|
|
|
|
2021-08-19 22:45:07 +03:00
|
|
|
PerformanceManager::add_unmap_perf_event(Process::current(), range_to_unmap);
|
2021-05-28 11:03:21 +02:00
|
|
|
|
2021-11-08 00:51:39 +01:00
|
|
|
return {};
|
2021-05-28 11:03:21 +02:00
|
|
|
}
|
|
|
|
|
2021-11-08 00:51:39 +01:00
|
|
|
ErrorOr<VirtualRange> AddressSpace::try_allocate_range(VirtualAddress vaddr, size_t size, size_t alignment)
|
2021-02-08 15:45:40 +01:00
|
|
|
{
|
|
|
|
vaddr.mask(PAGE_MASK);
|
2021-12-24 11:22:11 -03:00
|
|
|
size = TRY(page_round_up(size));
|
2021-02-08 15:45:40 +01:00
|
|
|
if (vaddr.is_null())
|
2021-09-05 23:12:16 +02:00
|
|
|
return page_directory().range_allocator().try_allocate_anywhere(size, alignment);
|
|
|
|
return page_directory().range_allocator().try_allocate_specific(vaddr, size);
|
2021-02-08 15:45:40 +01:00
|
|
|
}
|
|
|
|
|
2021-11-08 00:51:39 +01:00
|
|
|
ErrorOr<Region*> AddressSpace::try_allocate_split_region(Region const& source_region, VirtualRange const& range, size_t offset_in_vmobject)
|
2021-02-08 15:45:40 +01:00
|
|
|
{
|
2021-09-06 19:24:54 +02:00
|
|
|
OwnPtr<KString> region_name;
|
|
|
|
if (!source_region.name().is_null())
|
|
|
|
region_name = TRY(KString::try_create(source_region.name()));
|
|
|
|
|
2021-09-06 01:59:30 +02:00
|
|
|
auto new_region = TRY(Region::try_create_user_accessible(
|
2021-09-06 19:24:54 +02:00
|
|
|
range, source_region.vmobject(), offset_in_vmobject, move(region_name), source_region.access(), source_region.is_cacheable() ? Region::Cacheable::Yes : Region::Cacheable::No, source_region.is_shared()));
|
2022-02-10 19:55:10 +02:00
|
|
|
new_region->set_syscall_region(source_region.is_syscall_region());
|
|
|
|
new_region->set_mmap(source_region.is_mmap());
|
|
|
|
new_region->set_stack(source_region.is_stack());
|
2021-02-08 15:45:40 +01:00
|
|
|
size_t page_offset_in_source_region = (offset_in_vmobject - source_region.offset_in_vmobject()) / PAGE_SIZE;
|
2022-02-10 19:55:10 +02:00
|
|
|
for (size_t i = 0; i < new_region->page_count(); ++i) {
|
2021-02-08 15:45:40 +01:00
|
|
|
if (source_region.should_cow(page_offset_in_source_region + i))
|
2022-02-10 19:55:10 +02:00
|
|
|
TRY(new_region->set_should_cow(i, true));
|
2021-02-08 15:45:40 +01:00
|
|
|
}
|
2022-02-10 19:55:10 +02:00
|
|
|
return add_region(move(new_region));
|
2021-02-08 15:45:40 +01:00
|
|
|
}
|
|
|
|
|
2021-11-08 00:51:39 +01:00
|
|
|
ErrorOr<Region*> AddressSpace::allocate_region(VirtualRange const& range, StringView name, int prot, AllocationStrategy strategy)
|
2021-02-08 15:45:40 +01:00
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(range.is_valid());
|
2021-09-06 19:24:54 +02:00
|
|
|
OwnPtr<KString> region_name;
|
|
|
|
if (!name.is_null())
|
|
|
|
region_name = TRY(KString::try_create(name));
|
2021-09-06 01:59:30 +02:00
|
|
|
auto vmobject = TRY(AnonymousVMObject::try_create_with_size(range.size(), strategy));
|
2021-09-06 19:24:54 +02:00
|
|
|
auto region = TRY(Region::try_create_user_accessible(range, move(vmobject), 0, move(region_name), prot_to_region_access_flags(prot), Region::Cacheable::Yes, false));
|
2022-01-12 21:47:23 +01:00
|
|
|
TRY(region->map(page_directory(), ShouldFlushTLB::No));
|
2021-09-06 02:02:06 +02:00
|
|
|
return add_region(move(region));
|
2021-02-08 15:45:40 +01:00
|
|
|
}
|
|
|
|
|
2021-11-08 00:51:39 +01:00
|
|
|
ErrorOr<Region*> AddressSpace::allocate_region_with_vmobject(VirtualRange const& range, NonnullRefPtr<VMObject> vmobject, size_t offset_in_vmobject, StringView name, int prot, bool shared)
|
2021-02-08 15:45:40 +01:00
|
|
|
{
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(range.is_valid());
|
2021-02-08 15:45:40 +01:00
|
|
|
size_t end_in_vmobject = offset_in_vmobject + range.size();
|
|
|
|
if (end_in_vmobject <= offset_in_vmobject) {
|
|
|
|
dbgln("allocate_region_with_vmobject: Overflow (offset + size)");
|
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
if (offset_in_vmobject >= vmobject->size()) {
|
|
|
|
dbgln("allocate_region_with_vmobject: Attempt to allocate a region with an offset past the end of its VMObject.");
|
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
if (end_in_vmobject > vmobject->size()) {
|
|
|
|
dbgln("allocate_region_with_vmobject: Attempt to allocate a region with an end past the end of its VMObject.");
|
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
offset_in_vmobject &= PAGE_MASK;
|
2021-09-06 19:24:54 +02:00
|
|
|
OwnPtr<KString> region_name;
|
|
|
|
if (!name.is_null())
|
|
|
|
region_name = TRY(KString::try_create(name));
|
|
|
|
auto region = TRY(Region::try_create_user_accessible(range, move(vmobject), offset_in_vmobject, move(region_name), prot_to_region_access_flags(prot), Region::Cacheable::Yes, shared));
|
2022-01-15 17:37:17 +01:00
|
|
|
if (prot == PROT_NONE) {
|
|
|
|
// For PROT_NONE mappings, we don't have to set up any page table mappings.
|
|
|
|
// We do still need to attach the region to the page_directory though.
|
|
|
|
SpinlockLocker mm_locker(s_mm_lock);
|
2022-02-10 19:55:10 +02:00
|
|
|
region->set_page_directory(page_directory());
|
2022-01-15 17:37:17 +01:00
|
|
|
} else {
|
2022-02-10 19:55:10 +02:00
|
|
|
TRY(region->map(page_directory(), ShouldFlushTLB::No));
|
2022-01-15 17:37:17 +01:00
|
|
|
}
|
2022-02-10 19:55:10 +02:00
|
|
|
return add_region(move(region));
|
2021-02-08 15:45:40 +01:00
|
|
|
}
|
|
|
|
|
2021-08-06 13:57:39 +02:00
|
|
|
void AddressSpace::deallocate_region(Region& region)
|
2021-02-08 15:45:40 +01:00
|
|
|
{
|
2021-12-02 12:52:09 +00:00
|
|
|
(void)take_region(region);
|
2021-04-07 02:17:05 +03:00
|
|
|
}
|
|
|
|
|
2021-08-06 13:57:39 +02:00
|
|
|
NonnullOwnPtr<Region> AddressSpace::take_region(Region& region)
|
2021-04-07 02:17:05 +03:00
|
|
|
{
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker lock(m_lock);
|
2021-07-17 06:07:02 -07:00
|
|
|
auto found_region = m_regions.unsafe_remove(region.vaddr().get());
|
|
|
|
VERIFY(found_region.ptr() == ®ion);
|
|
|
|
return found_region;
|
2021-02-08 15:45:40 +01:00
|
|
|
}
|
|
|
|
|
2021-08-06 13:57:39 +02:00
|
|
|
Region* AddressSpace::find_region_from_range(VirtualRange const& range)
|
2021-02-08 15:45:40 +01:00
|
|
|
{
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker lock(m_lock);
|
2021-12-28 19:54:05 +01:00
|
|
|
auto* found_region = m_regions.find(range.base().get());
|
2021-04-07 02:20:29 +03:00
|
|
|
if (!found_region)
|
|
|
|
return nullptr;
|
|
|
|
auto& region = *found_region;
|
2021-12-24 11:22:11 -03:00
|
|
|
auto rounded_range_size = page_round_up(range.size());
|
|
|
|
if (rounded_range_size.is_error() || region->size() != rounded_range_size.value())
|
2021-04-07 02:20:29 +03:00
|
|
|
return nullptr;
|
|
|
|
return region;
|
2021-02-08 15:45:40 +01:00
|
|
|
}
|
|
|
|
|
2021-08-06 13:57:39 +02:00
|
|
|
Region* AddressSpace::find_region_containing(VirtualRange const& range)
|
2021-02-08 15:45:40 +01:00
|
|
|
{
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker lock(m_lock);
|
2021-12-28 19:54:05 +01:00
|
|
|
auto* candidate = m_regions.find_largest_not_above(range.base().get());
|
2021-04-07 02:20:29 +03:00
|
|
|
if (!candidate)
|
|
|
|
return nullptr;
|
|
|
|
return (*candidate)->range().contains(range) ? candidate->ptr() : nullptr;
|
2021-02-08 15:45:40 +01:00
|
|
|
}
|
|
|
|
|
2022-01-25 15:33:48 +02:00
|
|
|
ErrorOr<Vector<Region*>> AddressSpace::find_regions_intersecting(VirtualRange const& range)
|
2021-02-24 16:44:00 +01:00
|
|
|
{
|
|
|
|
Vector<Region*> regions = {};
|
|
|
|
size_t total_size_collected = 0;
|
|
|
|
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker lock(m_lock);
|
2021-02-24 16:44:00 +01:00
|
|
|
|
2021-12-28 19:54:05 +01:00
|
|
|
auto* found_region = m_regions.find_largest_not_above(range.base().get());
|
2021-04-07 02:20:29 +03:00
|
|
|
if (!found_region)
|
|
|
|
return regions;
|
|
|
|
for (auto iter = m_regions.begin_from((*found_region)->vaddr().get()); !iter.is_end(); ++iter) {
|
2022-04-01 20:58:27 +03:00
|
|
|
auto const& iter_range = (*iter)->range();
|
2021-09-15 23:50:46 -07:00
|
|
|
if (iter_range.base() < range.end() && iter_range.end() > range.base()) {
|
2022-01-25 15:33:48 +02:00
|
|
|
TRY(regions.try_append(*iter));
|
2021-04-07 02:20:29 +03:00
|
|
|
|
2021-09-15 23:50:46 -07:00
|
|
|
total_size_collected += (*iter)->size() - iter_range.intersect(range).size();
|
2021-02-24 16:44:00 +01:00
|
|
|
if (total_size_collected == range.size())
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return regions;
|
|
|
|
}
|
|
|
|
|
2021-11-08 00:51:39 +01:00
|
|
|
ErrorOr<Region*> AddressSpace::add_region(NonnullOwnPtr<Region> region)
|
2021-02-08 15:45:40 +01:00
|
|
|
{
|
|
|
|
auto* ptr = region.ptr();
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker lock(m_lock);
|
2021-11-17 15:44:58 +01:00
|
|
|
TRY(m_regions.try_insert(region->vaddr().get(), move(region)));
|
2021-09-06 02:02:06 +02:00
|
|
|
return ptr;
|
2021-02-08 15:45:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Carve out a virtual address range from a region and return the two regions on either side
|
2022-04-01 20:58:27 +03:00
|
|
|
ErrorOr<Vector<Region*, 2>> AddressSpace::try_split_region_around_range(Region const& source_region, VirtualRange const& desired_range)
|
2021-02-08 15:45:40 +01:00
|
|
|
{
|
2021-08-06 13:54:48 +02:00
|
|
|
VirtualRange old_region_range = source_region.range();
|
2021-02-08 15:45:40 +01:00
|
|
|
auto remaining_ranges_after_unmap = old_region_range.carve(desired_range);
|
|
|
|
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(!remaining_ranges_after_unmap.is_empty());
|
2021-11-08 00:51:39 +01:00
|
|
|
auto try_make_replacement_region = [&](VirtualRange const& new_range) -> ErrorOr<Region*> {
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(old_region_range.contains(new_range));
|
2021-02-08 15:45:40 +01:00
|
|
|
size_t new_range_offset_in_vmobject = source_region.offset_in_vmobject() + (new_range.base().get() - old_region_range.base().get());
|
2021-07-11 18:51:54 +02:00
|
|
|
return try_allocate_split_region(source_region, new_range, new_range_offset_in_vmobject);
|
2021-02-08 15:45:40 +01:00
|
|
|
};
|
|
|
|
Vector<Region*, 2> new_regions;
|
|
|
|
for (auto& new_range : remaining_ranges_after_unmap) {
|
2021-12-28 19:54:05 +01:00
|
|
|
auto* new_region = TRY(try_make_replacement_region(new_range));
|
2021-09-05 14:40:06 +02:00
|
|
|
new_regions.unchecked_append(new_region);
|
2021-02-08 15:45:40 +01:00
|
|
|
}
|
|
|
|
return new_regions;
|
|
|
|
}
|
|
|
|
|
2021-08-06 13:57:39 +02:00
|
|
|
void AddressSpace::dump_regions()
|
2021-02-08 15:45:40 +01:00
|
|
|
{
|
2021-02-12 16:33:58 +01:00
|
|
|
dbgln("Process regions:");
|
2021-07-22 01:21:39 +02:00
|
|
|
#if ARCH(I386)
|
2021-12-28 19:54:05 +01:00
|
|
|
char const* addr_padding = "";
|
2021-07-22 01:21:39 +02:00
|
|
|
#else
|
2021-12-28 19:54:05 +01:00
|
|
|
char const* addr_padding = " ";
|
2021-07-22 01:21:39 +02:00
|
|
|
#endif
|
|
|
|
dbgln("BEGIN{} END{} SIZE{} ACCESS NAME",
|
|
|
|
addr_padding, addr_padding, addr_padding);
|
2021-02-08 15:45:40 +01:00
|
|
|
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker lock(m_lock);
|
2021-02-08 15:45:40 +01:00
|
|
|
|
2021-12-28 19:54:05 +01:00
|
|
|
for (auto const& sorted_region : m_regions) {
|
|
|
|
auto const& region = *sorted_region;
|
2021-07-21 19:53:38 +02:00
|
|
|
dbgln("{:p} -- {:p} {:p} {:c}{:c}{:c}{:c}{:c}{:c} {}", region.vaddr().get(), region.vaddr().offset(region.size() - 1).get(), region.size(),
|
2021-02-08 15:45:40 +01: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());
|
|
|
|
}
|
|
|
|
MM.dump_kernel_regions();
|
|
|
|
}
|
|
|
|
|
2021-08-06 13:57:39 +02:00
|
|
|
void AddressSpace::remove_all_regions(Badge<Process>)
|
2021-02-08 15:45:40 +01:00
|
|
|
{
|
2022-01-12 13:48:43 +01:00
|
|
|
VERIFY(Thread::current() == g_finalizer);
|
2022-01-12 14:32:21 +01:00
|
|
|
SpinlockLocker locker(m_lock);
|
2022-01-30 16:07:59 +01:00
|
|
|
{
|
|
|
|
SpinlockLocker pd_locker(m_page_directory->get_lock());
|
|
|
|
SpinlockLocker mm_locker(s_mm_lock);
|
|
|
|
for (auto& region : m_regions)
|
|
|
|
(*region).unmap_with_locks_held(Region::ShouldDeallocateVirtualRange::No, ShouldFlushTLB::No, pd_locker, mm_locker);
|
|
|
|
}
|
2021-02-08 15:45:40 +01:00
|
|
|
m_regions.clear();
|
|
|
|
}
|
|
|
|
|
2021-08-06 13:57:39 +02:00
|
|
|
size_t AddressSpace::amount_dirty_private() const
|
2021-02-08 22:11:10 +01:00
|
|
|
{
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker lock(m_lock);
|
2021-02-08 22:11:10 +01:00
|
|
|
// FIXME: This gets a bit more complicated for Regions sharing the same underlying VMObject.
|
|
|
|
// The main issue I'm thinking of is when the VMObject has physical pages that none of the Regions are mapping.
|
|
|
|
// That's probably a situation that needs to be looked at in general.
|
|
|
|
size_t amount = 0;
|
2021-12-28 19:54:05 +01:00
|
|
|
for (auto const& region : m_regions) {
|
2021-04-07 02:20:29 +03:00
|
|
|
if (!region->is_shared())
|
|
|
|
amount += region->amount_dirty();
|
2021-02-08 22:11:10 +01:00
|
|
|
}
|
|
|
|
return amount;
|
|
|
|
}
|
|
|
|
|
2022-01-25 15:02:37 +02:00
|
|
|
ErrorOr<size_t> AddressSpace::amount_clean_inode() const
|
2021-02-08 22:11:10 +01:00
|
|
|
{
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker lock(m_lock);
|
2022-04-01 20:58:27 +03:00
|
|
|
HashTable<InodeVMObject const*> vmobjects;
|
2021-12-28 19:54:05 +01:00
|
|
|
for (auto const& region : m_regions) {
|
2021-04-07 02:20:29 +03:00
|
|
|
if (region->vmobject().is_inode())
|
2022-04-01 20:58:27 +03:00
|
|
|
TRY(vmobjects.try_set(&static_cast<InodeVMObject const&>(region->vmobject())));
|
2021-02-08 22:11:10 +01:00
|
|
|
}
|
|
|
|
size_t amount = 0;
|
|
|
|
for (auto& vmobject : vmobjects)
|
|
|
|
amount += vmobject->amount_clean();
|
|
|
|
return amount;
|
|
|
|
}
|
|
|
|
|
2021-08-06 13:57:39 +02:00
|
|
|
size_t AddressSpace::amount_virtual() const
|
2021-02-08 22:11:10 +01:00
|
|
|
{
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker lock(m_lock);
|
2021-02-08 22:18:26 +01:00
|
|
|
size_t amount = 0;
|
2021-12-28 19:54:05 +01:00
|
|
|
for (auto const& region : m_regions) {
|
2021-04-07 02:20:29 +03:00
|
|
|
amount += region->size();
|
2021-02-08 22:11:10 +01:00
|
|
|
}
|
|
|
|
return amount;
|
|
|
|
}
|
|
|
|
|
2021-08-06 13:57:39 +02:00
|
|
|
size_t AddressSpace::amount_resident() const
|
2021-02-08 22:11:10 +01:00
|
|
|
{
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker lock(m_lock);
|
2021-02-08 22:11:10 +01:00
|
|
|
// FIXME: This will double count if multiple regions use the same physical page.
|
|
|
|
size_t amount = 0;
|
2021-12-28 19:54:05 +01:00
|
|
|
for (auto const& region : m_regions) {
|
2021-04-07 02:20:29 +03:00
|
|
|
amount += region->amount_resident();
|
2021-02-08 22:11:10 +01:00
|
|
|
}
|
|
|
|
return amount;
|
|
|
|
}
|
|
|
|
|
2021-08-06 13:57:39 +02:00
|
|
|
size_t AddressSpace::amount_shared() const
|
2021-02-08 22:11:10 +01:00
|
|
|
{
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker lock(m_lock);
|
2021-02-08 22:11:10 +01:00
|
|
|
// FIXME: This will double count if multiple regions use the same physical page.
|
|
|
|
// FIXME: It doesn't work at the moment, since it relies on PhysicalPage ref counts,
|
|
|
|
// and each PhysicalPage is only reffed by its VMObject. This needs to be refactored
|
|
|
|
// so that every Region contributes +1 ref to each of its PhysicalPages.
|
|
|
|
size_t amount = 0;
|
2021-12-28 19:54:05 +01:00
|
|
|
for (auto const& region : m_regions) {
|
2021-04-07 02:20:29 +03:00
|
|
|
amount += region->amount_shared();
|
2021-02-08 22:11:10 +01:00
|
|
|
}
|
|
|
|
return amount;
|
|
|
|
}
|
|
|
|
|
2021-08-06 13:57:39 +02:00
|
|
|
size_t AddressSpace::amount_purgeable_volatile() const
|
2021-02-08 22:11:10 +01:00
|
|
|
{
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker lock(m_lock);
|
2021-02-08 22:18:26 +01:00
|
|
|
size_t amount = 0;
|
2021-12-28 19:54:05 +01:00
|
|
|
for (auto const& region : m_regions) {
|
2021-07-25 01:46:44 +02:00
|
|
|
if (!region->vmobject().is_anonymous())
|
|
|
|
continue;
|
|
|
|
auto const& vmobject = static_cast<AnonymousVMObject const&>(region->vmobject());
|
|
|
|
if (vmobject.is_purgeable() && vmobject.is_volatile())
|
2021-04-07 02:20:29 +03:00
|
|
|
amount += region->amount_resident();
|
2021-02-08 22:11:10 +01:00
|
|
|
}
|
|
|
|
return amount;
|
|
|
|
}
|
|
|
|
|
2021-08-06 13:57:39 +02:00
|
|
|
size_t AddressSpace::amount_purgeable_nonvolatile() const
|
2021-02-08 22:11:10 +01:00
|
|
|
{
|
2021-08-22 01:49:22 +02:00
|
|
|
SpinlockLocker lock(m_lock);
|
2021-02-08 22:18:26 +01:00
|
|
|
size_t amount = 0;
|
2021-12-28 19:54:05 +01:00
|
|
|
for (auto const& region : m_regions) {
|
2021-07-25 01:46:44 +02:00
|
|
|
if (!region->vmobject().is_anonymous())
|
|
|
|
continue;
|
|
|
|
auto const& vmobject = static_cast<AnonymousVMObject const&>(region->vmobject());
|
|
|
|
if (vmobject.is_purgeable() && !vmobject.is_volatile())
|
2021-04-07 02:20:29 +03:00
|
|
|
amount += region->amount_resident();
|
2021-02-08 22:11:10 +01:00
|
|
|
}
|
|
|
|
return amount;
|
|
|
|
}
|
|
|
|
|
2021-02-08 15:45:40 +01:00
|
|
|
}
|