mirror of
https://github.com/Atmosphere-NX/Atmosphere.git
synced 2024-12-12 16:01:15 +01:00
270 lines
12 KiB
C++
270 lines
12 KiB
C++
/*
|
|
* Copyright (c) 2018-2020 Atmosphère-NX
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms and conditions of the GNU General Public License,
|
|
* version 2, as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
* more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#include <mesosphere.hpp>
|
|
|
|
namespace ams::kern {
|
|
|
|
namespace {
|
|
|
|
class KMemoryRegionAllocator {
|
|
NON_COPYABLE(KMemoryRegionAllocator);
|
|
NON_MOVEABLE(KMemoryRegionAllocator);
|
|
public:
|
|
static constexpr size_t MaxMemoryRegions = 1000;
|
|
private:
|
|
KMemoryRegion region_heap[MaxMemoryRegions];
|
|
size_t num_regions;
|
|
public:
|
|
constexpr ALWAYS_INLINE KMemoryRegionAllocator() : region_heap(), num_regions() { /* ... */ }
|
|
public:
|
|
template<typename... Args>
|
|
ALWAYS_INLINE KMemoryRegion *Allocate(Args&&... args) {
|
|
/* Ensure we stay within the bounds of our heap. */
|
|
MESOSPHERE_INIT_ABORT_UNLESS(this->num_regions < MaxMemoryRegions);
|
|
|
|
/* Create the new region. */
|
|
KMemoryRegion *region = std::addressof(this->region_heap[this->num_regions++]);
|
|
new (region) KMemoryRegion(std::forward<Args>(args)...);
|
|
|
|
return region;
|
|
|
|
return &this->region_heap[this->num_regions++];
|
|
}
|
|
};
|
|
|
|
constinit KMemoryRegionAllocator g_memory_region_allocator;
|
|
|
|
template<typename... Args>
|
|
ALWAYS_INLINE KMemoryRegion *AllocateRegion(Args&&... args) {
|
|
return g_memory_region_allocator.Allocate(std::forward<Args>(args)...);
|
|
}
|
|
|
|
|
|
}
|
|
|
|
void KMemoryRegionTree::InsertDirectly(uintptr_t address, size_t size, u32 attr, u32 type_id) {
|
|
this->insert(*AllocateRegion(address, size, attr, type_id));
|
|
}
|
|
|
|
bool KMemoryRegionTree::Insert(uintptr_t address, size_t size, u32 type_id, u32 new_attr, u32 old_attr) {
|
|
/* Locate the memory region that contains the address. */
|
|
KMemoryRegion *found = this->FindModifiable(address);
|
|
|
|
/* We require that the old attr is correct. */
|
|
if (found->GetAttributes() != old_attr) {
|
|
return false;
|
|
}
|
|
|
|
/* We further require that the region can be split from the old region. */
|
|
const uintptr_t inserted_region_end = address + size;
|
|
const uintptr_t inserted_region_last = inserted_region_end - 1;
|
|
if (found->GetLastAddress() < inserted_region_last) {
|
|
return false;
|
|
}
|
|
|
|
/* Further, we require that the type id is a valid transformation. */
|
|
if (!found->CanDerive(type_id)) {
|
|
return false;
|
|
}
|
|
|
|
/* Cache information from the region before we remove it. */
|
|
const uintptr_t old_address = found->GetAddress();
|
|
const size_t old_size = found->GetSize();
|
|
const uintptr_t old_end = old_address + old_size;
|
|
const uintptr_t old_last = old_end - 1;
|
|
const uintptr_t old_pair = found->GetPairAddress();
|
|
const u32 old_type = found->GetType();
|
|
|
|
/* Erase the existing region from the tree. */
|
|
this->erase(this->iterator_to(*found));
|
|
|
|
/* Insert the new region into the tree. */
|
|
const uintptr_t new_pair = (old_pair != std::numeric_limits<uintptr_t>::max()) ? old_pair + (address - old_address) : old_pair;
|
|
if (old_address == address) {
|
|
/* Reuse the old object for the new region, if we can. */
|
|
found->Reset(address, size, new_pair, new_attr, type_id);
|
|
this->insert(*found);
|
|
} else {
|
|
/* If we can't re-use, adjust the old region. */
|
|
found->Reset(old_address, address - old_address, old_pair, old_attr, old_type);
|
|
this->insert(*found);
|
|
|
|
/* Insert a new region for the split. */
|
|
this->insert(*AllocateRegion(address, size, new_pair, new_attr, type_id));
|
|
}
|
|
|
|
/* If we need to insert a region after the region, do so. */
|
|
if (old_last != inserted_region_last) {
|
|
const uintptr_t after_pair = (old_pair != std::numeric_limits<uintptr_t>::max()) ? old_pair + (inserted_region_end - old_address) : old_pair;
|
|
this->insert(*AllocateRegion(inserted_region_end, old_end - inserted_region_end, after_pair, old_attr, old_type));
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
KVirtualAddress KMemoryRegionTree::GetRandomAlignedRegion(size_t size, size_t alignment, u32 type_id) {
|
|
/* We want to find the total extents of the type id. */
|
|
const auto extents = this->GetDerivedRegionExtents(type_id);
|
|
|
|
/* Ensure that our alignment is correct. */
|
|
MESOSPHERE_INIT_ABORT_UNLESS(util::IsAligned(extents.GetAddress(), alignment));
|
|
|
|
const uintptr_t first_address = extents.GetAddress();
|
|
const uintptr_t last_address = extents.GetLastAddress();
|
|
|
|
while (true) {
|
|
const uintptr_t candidate = util::AlignDown(KSystemControl::Init::GenerateRandomRange(first_address, last_address), alignment);
|
|
|
|
/* Ensure that the candidate doesn't overflow with the size. */
|
|
if (!(candidate < candidate + size)) {
|
|
continue;
|
|
}
|
|
|
|
const uintptr_t candidate_last = candidate + size - 1;
|
|
|
|
/* Ensure that the candidate fits within the region. */
|
|
if (candidate_last > last_address) {
|
|
continue;
|
|
}
|
|
|
|
/* Locate the candidate region, and ensure it fits and has the correct type id. */
|
|
if (const auto &candidate_region = *this->Find(candidate); !(candidate_last <= candidate_region.GetLastAddress() && candidate_region.GetType() == type_id)) {
|
|
continue;
|
|
}
|
|
|
|
return candidate;
|
|
}
|
|
}
|
|
|
|
void KMemoryLayout::InitializeLinearMemoryRegionTrees(KPhysicalAddress aligned_linear_phys_start, KVirtualAddress linear_virtual_start) {
|
|
/* Set static differences. */
|
|
s_linear_phys_to_virt_diff = GetInteger(linear_virtual_start) - GetInteger(aligned_linear_phys_start);
|
|
s_linear_virt_to_phys_diff = GetInteger(aligned_linear_phys_start) - GetInteger(linear_virtual_start);
|
|
|
|
/* Initialize linear trees. */
|
|
for (auto ®ion : GetPhysicalMemoryRegionTree()) {
|
|
if (region.HasTypeAttribute(KMemoryRegionAttr_LinearMapped)) {
|
|
GetPhysicalLinearMemoryRegionTree().InsertDirectly(region.GetAddress(), region.GetSize(), region.GetAttributes(), region.GetType());
|
|
}
|
|
}
|
|
|
|
for (auto ®ion : GetVirtualMemoryRegionTree()) {
|
|
if (region.IsDerivedFrom(KMemoryRegionType_Dram)) {
|
|
GetVirtualLinearMemoryRegionTree().InsertDirectly(region.GetAddress(), region.GetSize(), region.GetAttributes(), region.GetType());
|
|
}
|
|
}
|
|
}
|
|
|
|
size_t KMemoryLayout::GetResourceRegionSizeForInit() {
|
|
/* Calculate resource region size based on whether we allow extra threads. */
|
|
const bool use_extra_resources = KSystemControl::Init::ShouldIncreaseThreadResourceLimit();
|
|
size_t resource_region_size = KernelResourceSize + (use_extra_resources ? KernelSlabHeapAdditionalSize : 0);
|
|
|
|
/* 10.0.0 reduced the slab heap gaps by 64K. */
|
|
if (kern::GetTargetFirmware() < ams::TargetFirmware_10_0_0) {
|
|
resource_region_size += (KernelSlabHeapGapsSizeDeprecated - KernelSlabHeapGapsSize);
|
|
}
|
|
|
|
return resource_region_size;
|
|
}
|
|
|
|
namespace init {
|
|
|
|
namespace {
|
|
|
|
constexpr PageTableEntry KernelRwDataAttribute(PageTableEntry::Permission_KernelRW, PageTableEntry::PageAttribute_NormalMemory, PageTableEntry::Shareable_InnerShareable, PageTableEntry::MappingFlag_Mapped);
|
|
|
|
constexpr size_t CoreLocalRegionAlign = PageSize;
|
|
constexpr size_t CoreLocalRegionSize = PageSize * (1 + cpu::NumCores);
|
|
constexpr size_t CoreLocalRegionSizeWithGuards = CoreLocalRegionSize + 2 * PageSize;
|
|
constexpr size_t CoreLocalRegionBoundsAlign = 1_GB;
|
|
static_assert(CoreLocalRegionSize == sizeof(KCoreLocalRegion));
|
|
|
|
KVirtualAddress GetCoreLocalRegionVirtualAddress() {
|
|
while (true) {
|
|
const uintptr_t candidate_start = GetInteger(KMemoryLayout::GetVirtualMemoryRegionTree().GetRandomAlignedRegion(CoreLocalRegionSizeWithGuards, CoreLocalRegionAlign, KMemoryRegionType_None));
|
|
const uintptr_t candidate_end = candidate_start + CoreLocalRegionSizeWithGuards;
|
|
const uintptr_t candidate_last = candidate_end - 1;
|
|
|
|
const auto &containing_region = *KMemoryLayout::GetVirtualMemoryRegionTree().Find(candidate_start);
|
|
|
|
if (candidate_last > containing_region.GetLastAddress()) {
|
|
continue;
|
|
}
|
|
|
|
if (containing_region.GetType() != KMemoryRegionType_None) {
|
|
continue;
|
|
}
|
|
|
|
if (util::AlignDown(candidate_start, CoreLocalRegionBoundsAlign) != util::AlignDown(candidate_last, CoreLocalRegionBoundsAlign)) {
|
|
continue;
|
|
}
|
|
|
|
if (containing_region.GetAddress() > util::AlignDown(candidate_start, CoreLocalRegionBoundsAlign)) {
|
|
continue;
|
|
}
|
|
|
|
if (util::AlignUp(candidate_last, CoreLocalRegionBoundsAlign) - 1 > containing_region.GetLastAddress()) {
|
|
continue;
|
|
}
|
|
|
|
return candidate_start + PageSize;
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void SetupCoreLocalRegionMemoryRegions(KInitialPageTable &page_table, KInitialPageAllocator &page_allocator) {
|
|
/* NOTE: Nintendo passes page table here to use num_l1_entries; we don't use this at present. */
|
|
MESOSPHERE_UNUSED(page_table);
|
|
|
|
/* Get the virtual address of the core local reigon. */
|
|
const KVirtualAddress core_local_virt_start = GetCoreLocalRegionVirtualAddress();
|
|
MESOSPHERE_INIT_ABORT_UNLESS(KMemoryLayout::GetVirtualMemoryRegionTree().Insert(GetInteger(core_local_virt_start), CoreLocalRegionSize, KMemoryRegionType_CoreLocalRegion));
|
|
|
|
/* Allocate a page for each core. */
|
|
KPhysicalAddress core_local_region_start_phys[cpu::NumCores] = {};
|
|
for (size_t i = 0; i < cpu::NumCores; i++) {
|
|
core_local_region_start_phys[i] = page_allocator.Allocate();
|
|
}
|
|
|
|
/* Allocate an l1 page table for each core. */
|
|
KPhysicalAddress core_l1_ttbr1_phys[cpu::NumCores] = {};
|
|
core_l1_ttbr1_phys[0] = util::AlignDown(cpu::GetTtbr1El1(), PageSize);
|
|
for (size_t i = 1; i < cpu::NumCores; i++) {
|
|
core_l1_ttbr1_phys[i] = page_allocator.Allocate();
|
|
std::memcpy(reinterpret_cast<void *>(GetInteger(core_l1_ttbr1_phys[i])), reinterpret_cast<void *>(GetInteger(core_l1_ttbr1_phys[0])), PageSize);
|
|
}
|
|
|
|
/* Use the l1 page table for each core to map the core local region for each core. */
|
|
for (size_t i = 0; i < cpu::NumCores; i++) {
|
|
KInitialPageTable temp_pt(core_l1_ttbr1_phys[i], KInitialPageTable::NoClear{});
|
|
temp_pt.Map(core_local_virt_start, PageSize, core_local_region_start_phys[i], KernelRwDataAttribute, page_allocator);
|
|
for (size_t j = 0; j < cpu::NumCores; j++) {
|
|
temp_pt.Map(core_local_virt_start + (j + 1) * PageSize, PageSize, core_local_region_start_phys[j], KernelRwDataAttribute, page_allocator);
|
|
}
|
|
|
|
/* Setup the InitArguments. */
|
|
SetInitArguments(static_cast<s32>(i), core_local_region_start_phys[i], GetInteger(core_l1_ttbr1_phys[i]));
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
}
|