mirror of
https://github.com/yuzu-mirror/yuzu.git
synced 2024-11-19 00:37:34 +01:00
Merge pull request #9173 from bunnei/kern-update-15
Kernel: Various updates for FW 15.0.x
This commit is contained in:
commit
4c198bbf06
@ -190,11 +190,13 @@ add_library(core STATIC
|
|||||||
hle/kernel/k_code_memory.h
|
hle/kernel/k_code_memory.h
|
||||||
hle/kernel/k_condition_variable.cpp
|
hle/kernel/k_condition_variable.cpp
|
||||||
hle/kernel/k_condition_variable.h
|
hle/kernel/k_condition_variable.h
|
||||||
|
hle/kernel/k_debug.h
|
||||||
hle/kernel/k_dynamic_page_manager.h
|
hle/kernel/k_dynamic_page_manager.h
|
||||||
hle/kernel/k_dynamic_resource_manager.h
|
hle/kernel/k_dynamic_resource_manager.h
|
||||||
hle/kernel/k_dynamic_slab_heap.h
|
hle/kernel/k_dynamic_slab_heap.h
|
||||||
hle/kernel/k_event.cpp
|
hle/kernel/k_event.cpp
|
||||||
hle/kernel/k_event.h
|
hle/kernel/k_event.h
|
||||||
|
hle/kernel/k_event_info.h
|
||||||
hle/kernel/k_handle_table.cpp
|
hle/kernel/k_handle_table.cpp
|
||||||
hle/kernel/k_handle_table.h
|
hle/kernel/k_handle_table.h
|
||||||
hle/kernel/k_interrupt_manager.cpp
|
hle/kernel/k_interrupt_manager.cpp
|
||||||
@ -222,6 +224,8 @@ add_library(core STATIC
|
|||||||
hle/kernel/k_page_group.h
|
hle/kernel/k_page_group.h
|
||||||
hle/kernel/k_page_table.cpp
|
hle/kernel/k_page_table.cpp
|
||||||
hle/kernel/k_page_table.h
|
hle/kernel/k_page_table.h
|
||||||
|
hle/kernel/k_page_table_manager.h
|
||||||
|
hle/kernel/k_page_table_slab_heap.h
|
||||||
hle/kernel/k_port.cpp
|
hle/kernel/k_port.cpp
|
||||||
hle/kernel/k_port.h
|
hle/kernel/k_port.h
|
||||||
hle/kernel/k_priority_queue.h
|
hle/kernel/k_priority_queue.h
|
||||||
@ -254,6 +258,8 @@ add_library(core STATIC
|
|||||||
hle/kernel/k_synchronization_object.cpp
|
hle/kernel/k_synchronization_object.cpp
|
||||||
hle/kernel/k_synchronization_object.h
|
hle/kernel/k_synchronization_object.h
|
||||||
hle/kernel/k_system_control.h
|
hle/kernel/k_system_control.h
|
||||||
|
hle/kernel/k_system_resource.cpp
|
||||||
|
hle/kernel/k_system_resource.h
|
||||||
hle/kernel/k_thread.cpp
|
hle/kernel/k_thread.cpp
|
||||||
hle/kernel/k_thread.h
|
hle/kernel/k_thread.h
|
||||||
hle/kernel/k_thread_local_page.cpp
|
hle/kernel/k_thread_local_page.cpp
|
||||||
|
@ -8,6 +8,10 @@
|
|||||||
namespace Kernel::Board::Nintendo::Nx {
|
namespace Kernel::Board::Nintendo::Nx {
|
||||||
|
|
||||||
class KSystemControl {
|
class KSystemControl {
|
||||||
|
public:
|
||||||
|
// This can be overridden as needed.
|
||||||
|
static constexpr size_t SecureAppletMemorySize = 4 * 1024 * 1024; // 4_MB
|
||||||
|
|
||||||
public:
|
public:
|
||||||
class Init {
|
class Init {
|
||||||
public:
|
public:
|
||||||
|
@ -10,7 +10,9 @@
|
|||||||
#include "core/hardware_properties.h"
|
#include "core/hardware_properties.h"
|
||||||
#include "core/hle/kernel/init/init_slab_setup.h"
|
#include "core/hle/kernel/init/init_slab_setup.h"
|
||||||
#include "core/hle/kernel/k_code_memory.h"
|
#include "core/hle/kernel/k_code_memory.h"
|
||||||
|
#include "core/hle/kernel/k_debug.h"
|
||||||
#include "core/hle/kernel/k_event.h"
|
#include "core/hle/kernel/k_event.h"
|
||||||
|
#include "core/hle/kernel/k_event_info.h"
|
||||||
#include "core/hle/kernel/k_memory_layout.h"
|
#include "core/hle/kernel/k_memory_layout.h"
|
||||||
#include "core/hle/kernel/k_memory_manager.h"
|
#include "core/hle/kernel/k_memory_manager.h"
|
||||||
#include "core/hle/kernel/k_page_buffer.h"
|
#include "core/hle/kernel/k_page_buffer.h"
|
||||||
@ -22,6 +24,7 @@
|
|||||||
#include "core/hle/kernel/k_shared_memory.h"
|
#include "core/hle/kernel/k_shared_memory.h"
|
||||||
#include "core/hle/kernel/k_shared_memory_info.h"
|
#include "core/hle/kernel/k_shared_memory_info.h"
|
||||||
#include "core/hle/kernel/k_system_control.h"
|
#include "core/hle/kernel/k_system_control.h"
|
||||||
|
#include "core/hle/kernel/k_system_resource.h"
|
||||||
#include "core/hle/kernel/k_thread.h"
|
#include "core/hle/kernel/k_thread.h"
|
||||||
#include "core/hle/kernel/k_thread_local_page.h"
|
#include "core/hle/kernel/k_thread_local_page.h"
|
||||||
#include "core/hle/kernel/k_transfer_memory.h"
|
#include "core/hle/kernel/k_transfer_memory.h"
|
||||||
@ -44,7 +47,10 @@ namespace Kernel::Init {
|
|||||||
HANDLER(KThreadLocalPage, \
|
HANDLER(KThreadLocalPage, \
|
||||||
(SLAB_COUNT(KProcess) + (SLAB_COUNT(KProcess) + SLAB_COUNT(KThread)) / 8), \
|
(SLAB_COUNT(KProcess) + (SLAB_COUNT(KProcess) + SLAB_COUNT(KThread)) / 8), \
|
||||||
##__VA_ARGS__) \
|
##__VA_ARGS__) \
|
||||||
HANDLER(KResourceLimit, (SLAB_COUNT(KResourceLimit)), ##__VA_ARGS__)
|
HANDLER(KResourceLimit, (SLAB_COUNT(KResourceLimit)), ##__VA_ARGS__) \
|
||||||
|
HANDLER(KEventInfo, (SLAB_COUNT(KThread) + SLAB_COUNT(KDebug)), ##__VA_ARGS__) \
|
||||||
|
HANDLER(KDebug, (SLAB_COUNT(KDebug)), ##__VA_ARGS__) \
|
||||||
|
HANDLER(KSecureSystemResource, (SLAB_COUNT(KProcess)), ##__VA_ARGS__)
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
@ -73,8 +79,20 @@ constexpr size_t SlabCountKResourceLimit = 5;
|
|||||||
constexpr size_t SlabCountKDebug = Core::Hardware::NUM_CPU_CORES;
|
constexpr size_t SlabCountKDebug = Core::Hardware::NUM_CPU_CORES;
|
||||||
constexpr size_t SlabCountKIoPool = 1;
|
constexpr size_t SlabCountKIoPool = 1;
|
||||||
constexpr size_t SlabCountKIoRegion = 6;
|
constexpr size_t SlabCountKIoRegion = 6;
|
||||||
|
constexpr size_t SlabcountKSessionRequestMappings = 40;
|
||||||
|
|
||||||
constexpr size_t SlabCountExtraKThread = 160;
|
constexpr size_t SlabCountExtraKThread = (1024 + 256 + 256) - SlabCountKThread;
|
||||||
|
|
||||||
|
namespace test {
|
||||||
|
|
||||||
|
static_assert(KernelPageBufferHeapSize ==
|
||||||
|
2 * PageSize + (SlabCountKProcess + SlabCountKThread +
|
||||||
|
(SlabCountKProcess + SlabCountKThread) / 8) *
|
||||||
|
PageSize);
|
||||||
|
static_assert(KernelPageBufferAdditionalSize ==
|
||||||
|
(SlabCountExtraKThread + (SlabCountExtraKThread / 8)) * PageSize);
|
||||||
|
|
||||||
|
} // namespace test
|
||||||
|
|
||||||
/// Helper function to translate from the slab virtual address to the reserved location in physical
|
/// Helper function to translate from the slab virtual address to the reserved location in physical
|
||||||
/// memory.
|
/// memory.
|
||||||
@ -109,7 +127,7 @@ VAddr InitializeSlabHeap(Core::System& system, KMemoryLayout& memory_layout, VAd
|
|||||||
}
|
}
|
||||||
|
|
||||||
size_t CalculateSlabHeapGapSize() {
|
size_t CalculateSlabHeapGapSize() {
|
||||||
constexpr size_t KernelSlabHeapGapSize = 2_MiB - 296_KiB;
|
constexpr size_t KernelSlabHeapGapSize = 2_MiB - 320_KiB;
|
||||||
static_assert(KernelSlabHeapGapSize <= KernelSlabHeapGapsSizeMax);
|
static_assert(KernelSlabHeapGapSize <= KernelSlabHeapGapsSizeMax);
|
||||||
return KernelSlabHeapGapSize;
|
return KernelSlabHeapGapSize;
|
||||||
}
|
}
|
||||||
@ -134,6 +152,7 @@ KSlabResourceCounts KSlabResourceCounts::CreateDefault() {
|
|||||||
.num_KDebug = SlabCountKDebug,
|
.num_KDebug = SlabCountKDebug,
|
||||||
.num_KIoPool = SlabCountKIoPool,
|
.num_KIoPool = SlabCountKIoPool,
|
||||||
.num_KIoRegion = SlabCountKIoRegion,
|
.num_KIoRegion = SlabCountKIoRegion,
|
||||||
|
.num_KSessionRequestMappings = SlabcountKSessionRequestMappings,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -164,29 +183,6 @@ size_t CalculateTotalSlabHeapSize(const KernelCore& kernel) {
|
|||||||
return size;
|
return size;
|
||||||
}
|
}
|
||||||
|
|
||||||
void InitializeKPageBufferSlabHeap(Core::System& system) {
|
|
||||||
auto& kernel = system.Kernel();
|
|
||||||
|
|
||||||
const auto& counts = kernel.SlabResourceCounts();
|
|
||||||
const size_t num_pages =
|
|
||||||
counts.num_KProcess + counts.num_KThread + (counts.num_KProcess + counts.num_KThread) / 8;
|
|
||||||
const size_t slab_size = num_pages * PageSize;
|
|
||||||
|
|
||||||
// Reserve memory from the system resource limit.
|
|
||||||
ASSERT(kernel.GetSystemResourceLimit()->Reserve(LimitableResource::PhysicalMemory, slab_size));
|
|
||||||
|
|
||||||
// Allocate memory for the slab.
|
|
||||||
constexpr auto AllocateOption = KMemoryManager::EncodeOption(
|
|
||||||
KMemoryManager::Pool::System, KMemoryManager::Direction::FromFront);
|
|
||||||
const PAddr slab_address =
|
|
||||||
kernel.MemoryManager().AllocateAndOpenContinuous(num_pages, 1, AllocateOption);
|
|
||||||
ASSERT(slab_address != 0);
|
|
||||||
|
|
||||||
// Initialize the slabheap.
|
|
||||||
KPageBuffer::InitializeSlabHeap(kernel, system.DeviceMemory().GetPointer<void>(slab_address),
|
|
||||||
slab_size);
|
|
||||||
}
|
|
||||||
|
|
||||||
void InitializeSlabHeaps(Core::System& system, KMemoryLayout& memory_layout) {
|
void InitializeSlabHeaps(Core::System& system, KMemoryLayout& memory_layout) {
|
||||||
auto& kernel = system.Kernel();
|
auto& kernel = system.Kernel();
|
||||||
|
|
||||||
@ -258,3 +254,29 @@ void InitializeSlabHeaps(Core::System& system, KMemoryLayout& memory_layout) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
} // namespace Kernel::Init
|
} // namespace Kernel::Init
|
||||||
|
|
||||||
|
namespace Kernel {
|
||||||
|
|
||||||
|
void KPageBufferSlabHeap::Initialize(Core::System& system) {
|
||||||
|
auto& kernel = system.Kernel();
|
||||||
|
const auto& counts = kernel.SlabResourceCounts();
|
||||||
|
const size_t num_pages =
|
||||||
|
counts.num_KProcess + counts.num_KThread + (counts.num_KProcess + counts.num_KThread) / 8;
|
||||||
|
const size_t slab_size = num_pages * PageSize;
|
||||||
|
|
||||||
|
// Reserve memory from the system resource limit.
|
||||||
|
ASSERT(kernel.GetSystemResourceLimit()->Reserve(LimitableResource::PhysicalMemory, slab_size));
|
||||||
|
|
||||||
|
// Allocate memory for the slab.
|
||||||
|
constexpr auto AllocateOption = KMemoryManager::EncodeOption(
|
||||||
|
KMemoryManager::Pool::System, KMemoryManager::Direction::FromFront);
|
||||||
|
const PAddr slab_address =
|
||||||
|
kernel.MemoryManager().AllocateAndOpenContinuous(num_pages, 1, AllocateOption);
|
||||||
|
ASSERT(slab_address != 0);
|
||||||
|
|
||||||
|
// Initialize the slabheap.
|
||||||
|
KPageBuffer::InitializeSlabHeap(kernel, system.DeviceMemory().GetPointer<void>(slab_address),
|
||||||
|
slab_size);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Kernel
|
||||||
|
@ -33,11 +33,11 @@ struct KSlabResourceCounts {
|
|||||||
size_t num_KDebug;
|
size_t num_KDebug;
|
||||||
size_t num_KIoPool;
|
size_t num_KIoPool;
|
||||||
size_t num_KIoRegion;
|
size_t num_KIoRegion;
|
||||||
|
size_t num_KSessionRequestMappings;
|
||||||
};
|
};
|
||||||
|
|
||||||
void InitializeSlabResourceCounts(KernelCore& kernel);
|
void InitializeSlabResourceCounts(KernelCore& kernel);
|
||||||
size_t CalculateTotalSlabHeapSize(const KernelCore& kernel);
|
size_t CalculateTotalSlabHeapSize(const KernelCore& kernel);
|
||||||
void InitializeKPageBufferSlabHeap(Core::System& system);
|
|
||||||
void InitializeSlabHeaps(Core::System& system, KMemoryLayout& memory_layout);
|
void InitializeSlabHeaps(Core::System& system, KMemoryLayout& memory_layout);
|
||||||
|
|
||||||
} // namespace Kernel::Init
|
} // namespace Kernel::Init
|
||||||
|
@ -16,6 +16,7 @@
|
|||||||
#include "core/hle/kernel/k_session.h"
|
#include "core/hle/kernel/k_session.h"
|
||||||
#include "core/hle/kernel/k_shared_memory.h"
|
#include "core/hle/kernel/k_shared_memory.h"
|
||||||
#include "core/hle/kernel/k_synchronization_object.h"
|
#include "core/hle/kernel/k_synchronization_object.h"
|
||||||
|
#include "core/hle/kernel/k_system_resource.h"
|
||||||
#include "core/hle/kernel/k_thread.h"
|
#include "core/hle/kernel/k_thread.h"
|
||||||
#include "core/hle/kernel/k_transfer_memory.h"
|
#include "core/hle/kernel/k_transfer_memory.h"
|
||||||
|
|
||||||
@ -119,4 +120,6 @@ static_assert(std::is_final_v<KTransferMemory> && std::is_base_of_v<KAutoObject,
|
|||||||
// static_assert(std::is_final_v<KCodeMemory> &&
|
// static_assert(std::is_final_v<KCodeMemory> &&
|
||||||
// std::is_base_of_v<KAutoObject, KCodeMemory>);
|
// std::is_base_of_v<KAutoObject, KCodeMemory>);
|
||||||
|
|
||||||
|
static_assert(std::is_base_of_v<KAutoObject, KSystemResource>);
|
||||||
|
|
||||||
} // namespace Kernel
|
} // namespace Kernel
|
||||||
|
@ -10,6 +10,8 @@ namespace Kernel {
|
|||||||
|
|
||||||
class KAutoObject;
|
class KAutoObject;
|
||||||
|
|
||||||
|
class KSystemResource;
|
||||||
|
|
||||||
class KClassTokenGenerator {
|
class KClassTokenGenerator {
|
||||||
public:
|
public:
|
||||||
using TokenBaseType = u16;
|
using TokenBaseType = u16;
|
||||||
@ -58,7 +60,7 @@ private:
|
|||||||
if constexpr (std::is_same<T, KAutoObject>::value) {
|
if constexpr (std::is_same<T, KAutoObject>::value) {
|
||||||
static_assert(T::ObjectType == ObjectType::KAutoObject);
|
static_assert(T::ObjectType == ObjectType::KAutoObject);
|
||||||
return 0;
|
return 0;
|
||||||
} else if constexpr (!std::is_final<T>::value) {
|
} else if constexpr (!std::is_final<T>::value && !std::same_as<T, KSystemResource>) {
|
||||||
static_assert(ObjectType::BaseClassesStart <= T::ObjectType &&
|
static_assert(ObjectType::BaseClassesStart <= T::ObjectType &&
|
||||||
T::ObjectType < ObjectType::BaseClassesEnd);
|
T::ObjectType < ObjectType::BaseClassesEnd);
|
||||||
constexpr auto ClassIndex = static_cast<TokenBaseType>(T::ObjectType) -
|
constexpr auto ClassIndex = static_cast<TokenBaseType>(T::ObjectType) -
|
||||||
@ -108,6 +110,8 @@ public:
|
|||||||
KSessionRequest,
|
KSessionRequest,
|
||||||
KCodeMemory,
|
KCodeMemory,
|
||||||
|
|
||||||
|
KSystemResource,
|
||||||
|
|
||||||
// NOTE: True order for these has not been determined yet.
|
// NOTE: True order for these has not been determined yet.
|
||||||
KAlpha,
|
KAlpha,
|
||||||
KBeta,
|
KBeta,
|
||||||
|
20
src/core/hle/kernel/k_debug.h
Normal file
20
src/core/hle/kernel/k_debug.h
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "core/hle/kernel/k_auto_object.h"
|
||||||
|
#include "core/hle/kernel/slab_helpers.h"
|
||||||
|
|
||||||
|
namespace Kernel {
|
||||||
|
|
||||||
|
class KDebug final : public KAutoObjectWithSlabHeapAndContainer<KDebug, KAutoObjectWithList> {
|
||||||
|
KERNEL_AUTOOBJECT_TRAITS(KDebug, KAutoObject);
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit KDebug(KernelCore& kernel_) : KAutoObjectWithSlabHeapAndContainer{kernel_} {}
|
||||||
|
|
||||||
|
static void PostDestroy([[maybe_unused]] uintptr_t arg) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Kernel
|
@ -3,6 +3,8 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
#include "common/alignment.h"
|
#include "common/alignment.h"
|
||||||
#include "common/common_types.h"
|
#include "common/common_types.h"
|
||||||
#include "core/hle/kernel/k_page_bitmap.h"
|
#include "core/hle/kernel/k_page_bitmap.h"
|
||||||
@ -33,28 +35,36 @@ public:
|
|||||||
return reinterpret_cast<T*>(m_backing_memory.data() + (addr - m_address));
|
return reinterpret_cast<T*>(m_backing_memory.data() + (addr - m_address));
|
||||||
}
|
}
|
||||||
|
|
||||||
Result Initialize(VAddr addr, size_t sz) {
|
Result Initialize(VAddr memory, size_t size, size_t align) {
|
||||||
// We need to have positive size.
|
// We need to have positive size.
|
||||||
R_UNLESS(sz > 0, ResultOutOfMemory);
|
R_UNLESS(size > 0, ResultOutOfMemory);
|
||||||
m_backing_memory.resize(sz);
|
m_backing_memory.resize(size);
|
||||||
|
|
||||||
// Calculate management overhead.
|
// Set addresses.
|
||||||
const size_t management_size =
|
m_address = memory;
|
||||||
KPageBitmap::CalculateManagementOverheadSize(sz / sizeof(PageBuffer));
|
m_aligned_address = Common::AlignDown(memory, align);
|
||||||
const size_t allocatable_size = sz - management_size;
|
|
||||||
|
// Calculate extents.
|
||||||
|
const size_t managed_size = m_address + size - m_aligned_address;
|
||||||
|
const size_t overhead_size = Common::AlignUp(
|
||||||
|
KPageBitmap::CalculateManagementOverheadSize(managed_size / sizeof(PageBuffer)),
|
||||||
|
sizeof(PageBuffer));
|
||||||
|
R_UNLESS(overhead_size < size, ResultOutOfMemory);
|
||||||
|
|
||||||
// Set tracking fields.
|
// Set tracking fields.
|
||||||
m_address = addr;
|
m_size = Common::AlignDown(size - overhead_size, sizeof(PageBuffer));
|
||||||
m_size = Common::AlignDown(allocatable_size, sizeof(PageBuffer));
|
m_count = m_size / sizeof(PageBuffer);
|
||||||
m_count = allocatable_size / sizeof(PageBuffer);
|
|
||||||
R_UNLESS(m_count > 0, ResultOutOfMemory);
|
|
||||||
|
|
||||||
// Clear the management region.
|
// Clear the management region.
|
||||||
u64* management_ptr = GetPointer<u64>(m_address + allocatable_size);
|
u64* management_ptr = GetPointer<u64>(m_address + size - overhead_size);
|
||||||
std::memset(management_ptr, 0, management_size);
|
std::memset(management_ptr, 0, overhead_size);
|
||||||
|
|
||||||
// Initialize the bitmap.
|
// Initialize the bitmap.
|
||||||
m_page_bitmap.Initialize(management_ptr, m_count);
|
const size_t allocatable_region_size =
|
||||||
|
(m_address + size - overhead_size) - m_aligned_address;
|
||||||
|
ASSERT(allocatable_region_size >= sizeof(PageBuffer));
|
||||||
|
|
||||||
|
m_page_bitmap.Initialize(management_ptr, allocatable_region_size / sizeof(PageBuffer));
|
||||||
|
|
||||||
// Free the pages to the bitmap.
|
// Free the pages to the bitmap.
|
||||||
for (size_t i = 0; i < m_count; i++) {
|
for (size_t i = 0; i < m_count; i++) {
|
||||||
@ -62,7 +72,8 @@ public:
|
|||||||
std::memset(GetPointer<PageBuffer>(m_address) + i, 0, PageSize);
|
std::memset(GetPointer<PageBuffer>(m_address) + i, 0, PageSize);
|
||||||
|
|
||||||
// Set the bit for the free page.
|
// Set the bit for the free page.
|
||||||
m_page_bitmap.SetBit(i);
|
m_page_bitmap.SetBit((m_address + (i * sizeof(PageBuffer)) - m_aligned_address) /
|
||||||
|
sizeof(PageBuffer));
|
||||||
}
|
}
|
||||||
|
|
||||||
R_SUCCEED();
|
R_SUCCEED();
|
||||||
@ -101,7 +112,28 @@ public:
|
|||||||
m_page_bitmap.ClearBit(offset);
|
m_page_bitmap.ClearBit(offset);
|
||||||
m_peak = std::max(m_peak, (++m_used));
|
m_peak = std::max(m_peak, (++m_used));
|
||||||
|
|
||||||
return GetPointer<PageBuffer>(m_address) + offset;
|
return GetPointer<PageBuffer>(m_aligned_address) + offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
PageBuffer* Allocate(size_t count) {
|
||||||
|
// Take the lock.
|
||||||
|
// TODO(bunnei): We should disable interrupts here via KScopedInterruptDisable.
|
||||||
|
KScopedSpinLock lk(m_lock);
|
||||||
|
|
||||||
|
// Find a random free block.
|
||||||
|
s64 soffset = m_page_bitmap.FindFreeRange(count);
|
||||||
|
if (soffset < 0) [[likely]] {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
const size_t offset = static_cast<size_t>(soffset);
|
||||||
|
|
||||||
|
// Update our tracking.
|
||||||
|
m_page_bitmap.ClearRange(offset, count);
|
||||||
|
m_used += count;
|
||||||
|
m_peak = std::max(m_peak, m_used);
|
||||||
|
|
||||||
|
return GetPointer<PageBuffer>(m_aligned_address) + offset;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Free(PageBuffer* pb) {
|
void Free(PageBuffer* pb) {
|
||||||
@ -113,7 +145,7 @@ public:
|
|||||||
KScopedSpinLock lk(m_lock);
|
KScopedSpinLock lk(m_lock);
|
||||||
|
|
||||||
// Set the bit for the free page.
|
// Set the bit for the free page.
|
||||||
size_t offset = (reinterpret_cast<uintptr_t>(pb) - m_address) / sizeof(PageBuffer);
|
size_t offset = (reinterpret_cast<uintptr_t>(pb) - m_aligned_address) / sizeof(PageBuffer);
|
||||||
m_page_bitmap.SetBit(offset);
|
m_page_bitmap.SetBit(offset);
|
||||||
|
|
||||||
// Decrement our used count.
|
// Decrement our used count.
|
||||||
@ -127,6 +159,7 @@ private:
|
|||||||
size_t m_peak{};
|
size_t m_peak{};
|
||||||
size_t m_count{};
|
size_t m_count{};
|
||||||
VAddr m_address{};
|
VAddr m_address{};
|
||||||
|
VAddr m_aligned_address{};
|
||||||
size_t m_size{};
|
size_t m_size{};
|
||||||
|
|
||||||
// TODO(bunnei): Back by host memory until we emulate kernel virtual address space.
|
// TODO(bunnei): Back by host memory until we emulate kernel virtual address space.
|
||||||
|
@ -6,6 +6,7 @@
|
|||||||
#include "common/common_funcs.h"
|
#include "common/common_funcs.h"
|
||||||
#include "core/hle/kernel/k_dynamic_slab_heap.h"
|
#include "core/hle/kernel/k_dynamic_slab_heap.h"
|
||||||
#include "core/hle/kernel/k_memory_block.h"
|
#include "core/hle/kernel/k_memory_block.h"
|
||||||
|
#include "core/hle/kernel/k_page_group.h"
|
||||||
|
|
||||||
namespace Kernel {
|
namespace Kernel {
|
||||||
|
|
||||||
@ -51,8 +52,10 @@ private:
|
|||||||
DynamicSlabType* m_slab_heap{};
|
DynamicSlabType* m_slab_heap{};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class KBlockInfoManager : public KDynamicResourceManager<KBlockInfo> {};
|
||||||
class KMemoryBlockSlabManager : public KDynamicResourceManager<KMemoryBlock> {};
|
class KMemoryBlockSlabManager : public KDynamicResourceManager<KMemoryBlock> {};
|
||||||
|
|
||||||
|
using KBlockInfoSlabHeap = typename KBlockInfoManager::DynamicSlabType;
|
||||||
using KMemoryBlockSlabHeap = typename KMemoryBlockSlabManager::DynamicSlabType;
|
using KMemoryBlockSlabHeap = typename KMemoryBlockSlabManager::DynamicSlabType;
|
||||||
|
|
||||||
} // namespace Kernel
|
} // namespace Kernel
|
||||||
|
64
src/core/hle/kernel/k_event_info.h
Normal file
64
src/core/hle/kernel/k_event_info.h
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
|
||||||
|
#include <boost/intrusive/list.hpp>
|
||||||
|
|
||||||
|
#include "core/hle/kernel/slab_helpers.h"
|
||||||
|
#include "core/hle/kernel/svc_types.h"
|
||||||
|
|
||||||
|
namespace Kernel {
|
||||||
|
|
||||||
|
class KEventInfo : public KSlabAllocated<KEventInfo>, public boost::intrusive::list_base_hook<> {
|
||||||
|
public:
|
||||||
|
struct InfoCreateThread {
|
||||||
|
u32 thread_id{};
|
||||||
|
uintptr_t tls_address{};
|
||||||
|
};
|
||||||
|
|
||||||
|
struct InfoExitProcess {
|
||||||
|
Svc::ProcessExitReason reason{};
|
||||||
|
};
|
||||||
|
|
||||||
|
struct InfoExitThread {
|
||||||
|
Svc::ThreadExitReason reason{};
|
||||||
|
};
|
||||||
|
|
||||||
|
struct InfoException {
|
||||||
|
Svc::DebugException exception_type{};
|
||||||
|
s32 exception_data_count{};
|
||||||
|
uintptr_t exception_address{};
|
||||||
|
std::array<uintptr_t, 4> exception_data{};
|
||||||
|
};
|
||||||
|
|
||||||
|
struct InfoSystemCall {
|
||||||
|
s64 tick{};
|
||||||
|
s32 id{};
|
||||||
|
};
|
||||||
|
|
||||||
|
public:
|
||||||
|
KEventInfo() = default;
|
||||||
|
~KEventInfo() = default;
|
||||||
|
|
||||||
|
public:
|
||||||
|
Svc::DebugEvent event{};
|
||||||
|
u32 thread_id{};
|
||||||
|
u32 flags{};
|
||||||
|
bool is_attached{};
|
||||||
|
bool continue_flag{};
|
||||||
|
bool ignore_continue{};
|
||||||
|
bool close_once{};
|
||||||
|
union {
|
||||||
|
InfoCreateThread create_thread;
|
||||||
|
InfoExitProcess exit_process;
|
||||||
|
InfoExitThread exit_thread;
|
||||||
|
InfoException exception;
|
||||||
|
InfoSystemCall system_call;
|
||||||
|
} info{};
|
||||||
|
KThread* debug_thread{};
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Kernel
|
@ -5,14 +5,11 @@
|
|||||||
|
|
||||||
namespace Kernel {
|
namespace Kernel {
|
||||||
|
|
||||||
KHandleTable::KHandleTable(KernelCore& kernel_) : kernel{kernel_} {}
|
|
||||||
KHandleTable::~KHandleTable() = default;
|
|
||||||
|
|
||||||
Result KHandleTable::Finalize() {
|
Result KHandleTable::Finalize() {
|
||||||
// Get the table and clear our record of it.
|
// Get the table and clear our record of it.
|
||||||
u16 saved_table_size = 0;
|
u16 saved_table_size = 0;
|
||||||
{
|
{
|
||||||
KScopedDisableDispatch dd(kernel);
|
KScopedDisableDispatch dd{m_kernel};
|
||||||
KScopedSpinLock lk(m_lock);
|
KScopedSpinLock lk(m_lock);
|
||||||
|
|
||||||
std::swap(m_table_size, saved_table_size);
|
std::swap(m_table_size, saved_table_size);
|
||||||
@ -25,28 +22,28 @@ Result KHandleTable::Finalize() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return ResultSuccess;
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool KHandleTable::Remove(Handle handle) {
|
bool KHandleTable::Remove(Handle handle) {
|
||||||
// Don't allow removal of a pseudo-handle.
|
// Don't allow removal of a pseudo-handle.
|
||||||
if (Svc::IsPseudoHandle(handle)) {
|
if (Svc::IsPseudoHandle(handle)) [[unlikely]] {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handles must not have reserved bits set.
|
// Handles must not have reserved bits set.
|
||||||
const auto handle_pack = HandlePack(handle);
|
const auto handle_pack = HandlePack(handle);
|
||||||
if (handle_pack.reserved != 0) {
|
if (handle_pack.reserved != 0) [[unlikely]] {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Find the object and free the entry.
|
// Find the object and free the entry.
|
||||||
KAutoObject* obj = nullptr;
|
KAutoObject* obj = nullptr;
|
||||||
{
|
{
|
||||||
KScopedDisableDispatch dd(kernel);
|
KScopedDisableDispatch dd{m_kernel};
|
||||||
KScopedSpinLock lk(m_lock);
|
KScopedSpinLock lk(m_lock);
|
||||||
|
|
||||||
if (this->IsValidHandle(handle)) {
|
if (this->IsValidHandle(handle)) [[likely]] {
|
||||||
const auto index = handle_pack.index;
|
const auto index = handle_pack.index;
|
||||||
|
|
||||||
obj = m_objects[index];
|
obj = m_objects[index];
|
||||||
@ -57,13 +54,13 @@ bool KHandleTable::Remove(Handle handle) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Close the object.
|
// Close the object.
|
||||||
kernel.UnregisterInUseObject(obj);
|
m_kernel.UnregisterInUseObject(obj);
|
||||||
obj->Close();
|
obj->Close();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
Result KHandleTable::Add(Handle* out_handle, KAutoObject* obj) {
|
Result KHandleTable::Add(Handle* out_handle, KAutoObject* obj) {
|
||||||
KScopedDisableDispatch dd(kernel);
|
KScopedDisableDispatch dd{m_kernel};
|
||||||
KScopedSpinLock lk(m_lock);
|
KScopedSpinLock lk(m_lock);
|
||||||
|
|
||||||
// Never exceed our capacity.
|
// Never exceed our capacity.
|
||||||
@ -82,22 +79,22 @@ Result KHandleTable::Add(Handle* out_handle, KAutoObject* obj) {
|
|||||||
*out_handle = EncodeHandle(static_cast<u16>(index), linear_id);
|
*out_handle = EncodeHandle(static_cast<u16>(index), linear_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
return ResultSuccess;
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
Result KHandleTable::Reserve(Handle* out_handle) {
|
Result KHandleTable::Reserve(Handle* out_handle) {
|
||||||
KScopedDisableDispatch dd(kernel);
|
KScopedDisableDispatch dd{m_kernel};
|
||||||
KScopedSpinLock lk(m_lock);
|
KScopedSpinLock lk(m_lock);
|
||||||
|
|
||||||
// Never exceed our capacity.
|
// Never exceed our capacity.
|
||||||
R_UNLESS(m_count < m_table_size, ResultOutOfHandles);
|
R_UNLESS(m_count < m_table_size, ResultOutOfHandles);
|
||||||
|
|
||||||
*out_handle = EncodeHandle(static_cast<u16>(this->AllocateEntry()), this->AllocateLinearId());
|
*out_handle = EncodeHandle(static_cast<u16>(this->AllocateEntry()), this->AllocateLinearId());
|
||||||
return ResultSuccess;
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
void KHandleTable::Unreserve(Handle handle) {
|
void KHandleTable::Unreserve(Handle handle) {
|
||||||
KScopedDisableDispatch dd(kernel);
|
KScopedDisableDispatch dd{m_kernel};
|
||||||
KScopedSpinLock lk(m_lock);
|
KScopedSpinLock lk(m_lock);
|
||||||
|
|
||||||
// Unpack the handle.
|
// Unpack the handle.
|
||||||
@ -108,7 +105,7 @@ void KHandleTable::Unreserve(Handle handle) {
|
|||||||
ASSERT(reserved == 0);
|
ASSERT(reserved == 0);
|
||||||
ASSERT(linear_id != 0);
|
ASSERT(linear_id != 0);
|
||||||
|
|
||||||
if (index < m_table_size) {
|
if (index < m_table_size) [[likely]] {
|
||||||
// NOTE: This code does not check the linear id.
|
// NOTE: This code does not check the linear id.
|
||||||
ASSERT(m_objects[index] == nullptr);
|
ASSERT(m_objects[index] == nullptr);
|
||||||
this->FreeEntry(index);
|
this->FreeEntry(index);
|
||||||
@ -116,7 +113,7 @@ void KHandleTable::Unreserve(Handle handle) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void KHandleTable::Register(Handle handle, KAutoObject* obj) {
|
void KHandleTable::Register(Handle handle, KAutoObject* obj) {
|
||||||
KScopedDisableDispatch dd(kernel);
|
KScopedDisableDispatch dd{m_kernel};
|
||||||
KScopedSpinLock lk(m_lock);
|
KScopedSpinLock lk(m_lock);
|
||||||
|
|
||||||
// Unpack the handle.
|
// Unpack the handle.
|
||||||
@ -127,7 +124,7 @@ void KHandleTable::Register(Handle handle, KAutoObject* obj) {
|
|||||||
ASSERT(reserved == 0);
|
ASSERT(reserved == 0);
|
||||||
ASSERT(linear_id != 0);
|
ASSERT(linear_id != 0);
|
||||||
|
|
||||||
if (index < m_table_size) {
|
if (index < m_table_size) [[likely]] {
|
||||||
// Set the entry.
|
// Set the entry.
|
||||||
ASSERT(m_objects[index] == nullptr);
|
ASSERT(m_objects[index] == nullptr);
|
||||||
|
|
||||||
|
@ -21,33 +21,38 @@ namespace Kernel {
|
|||||||
class KernelCore;
|
class KernelCore;
|
||||||
|
|
||||||
class KHandleTable {
|
class KHandleTable {
|
||||||
public:
|
|
||||||
YUZU_NON_COPYABLE(KHandleTable);
|
YUZU_NON_COPYABLE(KHandleTable);
|
||||||
YUZU_NON_MOVEABLE(KHandleTable);
|
YUZU_NON_MOVEABLE(KHandleTable);
|
||||||
|
|
||||||
|
public:
|
||||||
static constexpr size_t MaxTableSize = 1024;
|
static constexpr size_t MaxTableSize = 1024;
|
||||||
|
|
||||||
explicit KHandleTable(KernelCore& kernel_);
|
public:
|
||||||
~KHandleTable();
|
explicit KHandleTable(KernelCore& kernel) : m_kernel(kernel) {}
|
||||||
|
|
||||||
Result Initialize(s32 size) {
|
Result Initialize(s32 size) {
|
||||||
|
// Check that the table size is valid.
|
||||||
R_UNLESS(size <= static_cast<s32>(MaxTableSize), ResultOutOfMemory);
|
R_UNLESS(size <= static_cast<s32>(MaxTableSize), ResultOutOfMemory);
|
||||||
|
|
||||||
|
// Lock.
|
||||||
|
KScopedDisableDispatch dd{m_kernel};
|
||||||
|
KScopedSpinLock lk(m_lock);
|
||||||
|
|
||||||
// Initialize all fields.
|
// Initialize all fields.
|
||||||
m_max_count = 0;
|
m_max_count = 0;
|
||||||
m_table_size = static_cast<u16>((size <= 0) ? MaxTableSize : size);
|
m_table_size = static_cast<s16>((size <= 0) ? MaxTableSize : size);
|
||||||
m_next_linear_id = MinLinearId;
|
m_next_linear_id = MinLinearId;
|
||||||
m_count = 0;
|
m_count = 0;
|
||||||
m_free_head_index = -1;
|
m_free_head_index = -1;
|
||||||
|
|
||||||
// Free all entries.
|
// Free all entries.
|
||||||
for (s16 i = 0; i < static_cast<s16>(m_table_size); ++i) {
|
for (s32 i = 0; i < static_cast<s32>(m_table_size); ++i) {
|
||||||
m_objects[i] = nullptr;
|
m_objects[i] = nullptr;
|
||||||
m_entry_infos[i].next_free_index = i - 1;
|
m_entry_infos[i].next_free_index = static_cast<s16>(i - 1);
|
||||||
m_free_head_index = i;
|
m_free_head_index = i;
|
||||||
}
|
}
|
||||||
|
|
||||||
return ResultSuccess;
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t GetTableSize() const {
|
size_t GetTableSize() const {
|
||||||
@ -66,13 +71,13 @@ public:
|
|||||||
template <typename T = KAutoObject>
|
template <typename T = KAutoObject>
|
||||||
KScopedAutoObject<T> GetObjectWithoutPseudoHandle(Handle handle) const {
|
KScopedAutoObject<T> GetObjectWithoutPseudoHandle(Handle handle) const {
|
||||||
// Lock and look up in table.
|
// Lock and look up in table.
|
||||||
KScopedDisableDispatch dd(kernel);
|
KScopedDisableDispatch dd{m_kernel};
|
||||||
KScopedSpinLock lk(m_lock);
|
KScopedSpinLock lk(m_lock);
|
||||||
|
|
||||||
if constexpr (std::is_same_v<T, KAutoObject>) {
|
if constexpr (std::is_same_v<T, KAutoObject>) {
|
||||||
return this->GetObjectImpl(handle);
|
return this->GetObjectImpl(handle);
|
||||||
} else {
|
} else {
|
||||||
if (auto* obj = this->GetObjectImpl(handle); obj != nullptr) {
|
if (auto* obj = this->GetObjectImpl(handle); obj != nullptr) [[likely]] {
|
||||||
return obj->DynamicCast<T*>();
|
return obj->DynamicCast<T*>();
|
||||||
} else {
|
} else {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
@ -85,13 +90,13 @@ public:
|
|||||||
// Handle pseudo-handles.
|
// Handle pseudo-handles.
|
||||||
if constexpr (std::derived_from<KProcess, T>) {
|
if constexpr (std::derived_from<KProcess, T>) {
|
||||||
if (handle == Svc::PseudoHandle::CurrentProcess) {
|
if (handle == Svc::PseudoHandle::CurrentProcess) {
|
||||||
auto* const cur_process = kernel.CurrentProcess();
|
auto* const cur_process = m_kernel.CurrentProcess();
|
||||||
ASSERT(cur_process != nullptr);
|
ASSERT(cur_process != nullptr);
|
||||||
return cur_process;
|
return cur_process;
|
||||||
}
|
}
|
||||||
} else if constexpr (std::derived_from<KThread, T>) {
|
} else if constexpr (std::derived_from<KThread, T>) {
|
||||||
if (handle == Svc::PseudoHandle::CurrentThread) {
|
if (handle == Svc::PseudoHandle::CurrentThread) {
|
||||||
auto* const cur_thread = GetCurrentThreadPointer(kernel);
|
auto* const cur_thread = GetCurrentThreadPointer(m_kernel);
|
||||||
ASSERT(cur_thread != nullptr);
|
ASSERT(cur_thread != nullptr);
|
||||||
return cur_thread;
|
return cur_thread;
|
||||||
}
|
}
|
||||||
@ -100,6 +105,37 @@ public:
|
|||||||
return this->template GetObjectWithoutPseudoHandle<T>(handle);
|
return this->template GetObjectWithoutPseudoHandle<T>(handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
KScopedAutoObject<KAutoObject> GetObjectForIpcWithoutPseudoHandle(Handle handle) const {
|
||||||
|
// Lock and look up in table.
|
||||||
|
KScopedDisableDispatch dd{m_kernel};
|
||||||
|
KScopedSpinLock lk(m_lock);
|
||||||
|
|
||||||
|
return this->GetObjectImpl(handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
KScopedAutoObject<KAutoObject> GetObjectForIpc(Handle handle, KThread* cur_thread) const {
|
||||||
|
// Handle pseudo-handles.
|
||||||
|
ASSERT(cur_thread != nullptr);
|
||||||
|
if (handle == Svc::PseudoHandle::CurrentProcess) {
|
||||||
|
auto* const cur_process =
|
||||||
|
static_cast<KAutoObject*>(static_cast<void*>(cur_thread->GetOwnerProcess()));
|
||||||
|
ASSERT(cur_process != nullptr);
|
||||||
|
return cur_process;
|
||||||
|
}
|
||||||
|
if (handle == Svc::PseudoHandle::CurrentThread) {
|
||||||
|
return static_cast<KAutoObject*>(cur_thread);
|
||||||
|
}
|
||||||
|
|
||||||
|
return GetObjectForIpcWithoutPseudoHandle(handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
KScopedAutoObject<KAutoObject> GetObjectByIndex(Handle* out_handle, size_t index) const {
|
||||||
|
KScopedDisableDispatch dd{m_kernel};
|
||||||
|
KScopedSpinLock lk(m_lock);
|
||||||
|
|
||||||
|
return this->GetObjectByIndexImpl(out_handle, index);
|
||||||
|
}
|
||||||
|
|
||||||
Result Reserve(Handle* out_handle);
|
Result Reserve(Handle* out_handle);
|
||||||
void Unreserve(Handle handle);
|
void Unreserve(Handle handle);
|
||||||
|
|
||||||
@ -112,7 +148,7 @@ public:
|
|||||||
size_t num_opened;
|
size_t num_opened;
|
||||||
{
|
{
|
||||||
// Lock the table.
|
// Lock the table.
|
||||||
KScopedDisableDispatch dd(kernel);
|
KScopedDisableDispatch dd{m_kernel};
|
||||||
KScopedSpinLock lk(m_lock);
|
KScopedSpinLock lk(m_lock);
|
||||||
for (num_opened = 0; num_opened < num_handles; num_opened++) {
|
for (num_opened = 0; num_opened < num_handles; num_opened++) {
|
||||||
// Get the current handle.
|
// Get the current handle.
|
||||||
@ -120,13 +156,13 @@ public:
|
|||||||
|
|
||||||
// Get the object for the current handle.
|
// Get the object for the current handle.
|
||||||
KAutoObject* cur_object = this->GetObjectImpl(cur_handle);
|
KAutoObject* cur_object = this->GetObjectImpl(cur_handle);
|
||||||
if (cur_object == nullptr) {
|
if (cur_object == nullptr) [[unlikely]] {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Cast the current object to the desired type.
|
// Cast the current object to the desired type.
|
||||||
T* cur_t = cur_object->DynamicCast<T*>();
|
T* cur_t = cur_object->DynamicCast<T*>();
|
||||||
if (cur_t == nullptr) {
|
if (cur_t == nullptr) [[unlikely]] {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -137,7 +173,7 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
// If we converted every object, succeed.
|
// If we converted every object, succeed.
|
||||||
if (num_opened == num_handles) {
|
if (num_opened == num_handles) [[likely]] {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -191,21 +227,21 @@ private:
|
|||||||
ASSERT(reserved == 0);
|
ASSERT(reserved == 0);
|
||||||
|
|
||||||
// Validate our indexing information.
|
// Validate our indexing information.
|
||||||
if (raw_value == 0) {
|
if (raw_value == 0) [[unlikely]] {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (linear_id == 0) {
|
if (linear_id == 0) [[unlikely]] {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (index >= m_table_size) {
|
if (index >= m_table_size) [[unlikely]] {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check that there's an object, and our serial id is correct.
|
// Check that there's an object, and our serial id is correct.
|
||||||
if (m_objects[index] == nullptr) {
|
if (m_objects[index] == nullptr) [[unlikely]] {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (m_entry_infos[index].GetLinearId() != linear_id) {
|
if (m_entry_infos[index].GetLinearId() != linear_id) [[unlikely]] {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -215,11 +251,11 @@ private:
|
|||||||
KAutoObject* GetObjectImpl(Handle handle) const {
|
KAutoObject* GetObjectImpl(Handle handle) const {
|
||||||
// Handles must not have reserved bits set.
|
// Handles must not have reserved bits set.
|
||||||
const auto handle_pack = HandlePack(handle);
|
const auto handle_pack = HandlePack(handle);
|
||||||
if (handle_pack.reserved != 0) {
|
if (handle_pack.reserved != 0) [[unlikely]] {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->IsValidHandle(handle)) {
|
if (this->IsValidHandle(handle)) [[likely]] {
|
||||||
return m_objects[handle_pack.index];
|
return m_objects[handle_pack.index];
|
||||||
} else {
|
} else {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
@ -227,9 +263,8 @@ private:
|
|||||||
}
|
}
|
||||||
|
|
||||||
KAutoObject* GetObjectByIndexImpl(Handle* out_handle, size_t index) const {
|
KAutoObject* GetObjectByIndexImpl(Handle* out_handle, size_t index) const {
|
||||||
|
|
||||||
// Index must be in bounds.
|
// Index must be in bounds.
|
||||||
if (index >= m_table_size) {
|
if (index >= m_table_size) [[unlikely]] {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -244,18 +279,15 @@ private:
|
|||||||
|
|
||||||
private:
|
private:
|
||||||
union HandlePack {
|
union HandlePack {
|
||||||
HandlePack() = default;
|
constexpr HandlePack() = default;
|
||||||
HandlePack(Handle handle) : raw{static_cast<u32>(handle)} {}
|
constexpr HandlePack(Handle handle) : raw{static_cast<u32>(handle)} {}
|
||||||
|
|
||||||
u32 raw;
|
u32 raw{};
|
||||||
BitField<0, 15, u32> index;
|
BitField<0, 15, u32> index;
|
||||||
BitField<15, 15, u32> linear_id;
|
BitField<15, 15, u32> linear_id;
|
||||||
BitField<30, 2, u32> reserved;
|
BitField<30, 2, u32> reserved;
|
||||||
};
|
};
|
||||||
|
|
||||||
static constexpr u16 MinLinearId = 1;
|
|
||||||
static constexpr u16 MaxLinearId = 0x7FFF;
|
|
||||||
|
|
||||||
static constexpr Handle EncodeHandle(u16 index, u16 linear_id) {
|
static constexpr Handle EncodeHandle(u16 index, u16 linear_id) {
|
||||||
HandlePack handle{};
|
HandlePack handle{};
|
||||||
handle.index.Assign(index);
|
handle.index.Assign(index);
|
||||||
@ -264,6 +296,10 @@ private:
|
|||||||
return handle.raw;
|
return handle.raw;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
static constexpr u16 MinLinearId = 1;
|
||||||
|
static constexpr u16 MaxLinearId = 0x7FFF;
|
||||||
|
|
||||||
union EntryInfo {
|
union EntryInfo {
|
||||||
u16 linear_id;
|
u16 linear_id;
|
||||||
s16 next_free_index;
|
s16 next_free_index;
|
||||||
@ -271,21 +307,21 @@ private:
|
|||||||
constexpr u16 GetLinearId() const {
|
constexpr u16 GetLinearId() const {
|
||||||
return linear_id;
|
return linear_id;
|
||||||
}
|
}
|
||||||
constexpr s16 GetNextFreeIndex() const {
|
constexpr s32 GetNextFreeIndex() const {
|
||||||
return next_free_index;
|
return next_free_index;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
KernelCore& m_kernel;
|
||||||
std::array<EntryInfo, MaxTableSize> m_entry_infos{};
|
std::array<EntryInfo, MaxTableSize> m_entry_infos{};
|
||||||
std::array<KAutoObject*, MaxTableSize> m_objects{};
|
std::array<KAutoObject*, MaxTableSize> m_objects{};
|
||||||
s32 m_free_head_index{-1};
|
mutable KSpinLock m_lock;
|
||||||
|
s32 m_free_head_index{};
|
||||||
u16 m_table_size{};
|
u16 m_table_size{};
|
||||||
u16 m_max_count{};
|
u16 m_max_count{};
|
||||||
u16 m_next_linear_id{MinLinearId};
|
u16 m_next_linear_id{};
|
||||||
u16 m_count{};
|
u16 m_count{};
|
||||||
mutable KSpinLock m_lock;
|
|
||||||
KernelCore& kernel;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Kernel
|
} // namespace Kernel
|
||||||
|
@ -35,26 +35,32 @@ enum class KMemoryState : u32 {
|
|||||||
FlagCanMapProcess = (1 << 23),
|
FlagCanMapProcess = (1 << 23),
|
||||||
FlagCanChangeAttribute = (1 << 24),
|
FlagCanChangeAttribute = (1 << 24),
|
||||||
FlagCanCodeMemory = (1 << 25),
|
FlagCanCodeMemory = (1 << 25),
|
||||||
|
FlagLinearMapped = (1 << 26),
|
||||||
|
|
||||||
FlagsData = FlagCanReprotect | FlagCanUseIpc | FlagCanUseNonDeviceIpc | FlagCanUseNonSecureIpc |
|
FlagsData = FlagCanReprotect | FlagCanUseIpc | FlagCanUseNonDeviceIpc | FlagCanUseNonSecureIpc |
|
||||||
FlagMapped | FlagCanAlias | FlagCanTransfer | FlagCanQueryPhysical |
|
FlagMapped | FlagCanAlias | FlagCanTransfer | FlagCanQueryPhysical |
|
||||||
FlagCanDeviceMap | FlagCanAlignedDeviceMap | FlagCanIpcUserBuffer |
|
FlagCanDeviceMap | FlagCanAlignedDeviceMap | FlagCanIpcUserBuffer |
|
||||||
FlagReferenceCounted | FlagCanChangeAttribute,
|
FlagReferenceCounted | FlagCanChangeAttribute | FlagLinearMapped,
|
||||||
|
|
||||||
FlagsCode = FlagCanDebug | FlagCanUseIpc | FlagCanUseNonDeviceIpc | FlagCanUseNonSecureIpc |
|
FlagsCode = FlagCanDebug | FlagCanUseIpc | FlagCanUseNonDeviceIpc | FlagCanUseNonSecureIpc |
|
||||||
FlagMapped | FlagCode | FlagCanQueryPhysical | FlagCanDeviceMap |
|
FlagMapped | FlagCode | FlagCanQueryPhysical | FlagCanDeviceMap |
|
||||||
FlagCanAlignedDeviceMap | FlagReferenceCounted,
|
FlagCanAlignedDeviceMap | FlagReferenceCounted | FlagLinearMapped,
|
||||||
|
|
||||||
FlagsMisc = FlagMapped | FlagReferenceCounted | FlagCanQueryPhysical | FlagCanDeviceMap,
|
FlagsMisc = FlagMapped | FlagReferenceCounted | FlagCanQueryPhysical | FlagCanDeviceMap |
|
||||||
|
FlagLinearMapped,
|
||||||
|
|
||||||
Free = static_cast<u32>(Svc::MemoryState::Free),
|
Free = static_cast<u32>(Svc::MemoryState::Free),
|
||||||
Io = static_cast<u32>(Svc::MemoryState::Io) | FlagMapped,
|
Io = static_cast<u32>(Svc::MemoryState::Io) | FlagMapped | FlagCanDeviceMap |
|
||||||
|
FlagCanAlignedDeviceMap,
|
||||||
Static = static_cast<u32>(Svc::MemoryState::Static) | FlagMapped | FlagCanQueryPhysical,
|
Static = static_cast<u32>(Svc::MemoryState::Static) | FlagMapped | FlagCanQueryPhysical,
|
||||||
Code = static_cast<u32>(Svc::MemoryState::Code) | FlagsCode | FlagCanMapProcess,
|
Code = static_cast<u32>(Svc::MemoryState::Code) | FlagsCode | FlagCanMapProcess,
|
||||||
CodeData = static_cast<u32>(Svc::MemoryState::CodeData) | FlagsData | FlagCanMapProcess |
|
CodeData = static_cast<u32>(Svc::MemoryState::CodeData) | FlagsData | FlagCanMapProcess |
|
||||||
FlagCanCodeMemory,
|
FlagCanCodeMemory,
|
||||||
Shared = static_cast<u32>(Svc::MemoryState::Shared) | FlagMapped | FlagReferenceCounted,
|
|
||||||
Normal = static_cast<u32>(Svc::MemoryState::Normal) | FlagsData | FlagCanCodeMemory,
|
Normal = static_cast<u32>(Svc::MemoryState::Normal) | FlagsData | FlagCanCodeMemory,
|
||||||
|
Shared = static_cast<u32>(Svc::MemoryState::Shared) | FlagMapped | FlagReferenceCounted |
|
||||||
|
FlagLinearMapped,
|
||||||
|
|
||||||
|
// Alias was removed after 1.0.0.
|
||||||
|
|
||||||
AliasCode = static_cast<u32>(Svc::MemoryState::AliasCode) | FlagsCode | FlagCanMapProcess |
|
AliasCode = static_cast<u32>(Svc::MemoryState::AliasCode) | FlagsCode | FlagCanMapProcess |
|
||||||
FlagCanCodeAlias,
|
FlagCanCodeAlias,
|
||||||
@ -67,18 +73,18 @@ enum class KMemoryState : u32 {
|
|||||||
Stack = static_cast<u32>(Svc::MemoryState::Stack) | FlagsMisc | FlagCanAlignedDeviceMap |
|
Stack = static_cast<u32>(Svc::MemoryState::Stack) | FlagsMisc | FlagCanAlignedDeviceMap |
|
||||||
FlagCanUseIpc | FlagCanUseNonSecureIpc | FlagCanUseNonDeviceIpc,
|
FlagCanUseIpc | FlagCanUseNonSecureIpc | FlagCanUseNonDeviceIpc,
|
||||||
|
|
||||||
ThreadLocal =
|
ThreadLocal = static_cast<u32>(Svc::MemoryState::ThreadLocal) | FlagMapped | FlagLinearMapped,
|
||||||
static_cast<u32>(Svc::MemoryState::ThreadLocal) | FlagMapped | FlagReferenceCounted,
|
|
||||||
|
|
||||||
Transfered = static_cast<u32>(Svc::MemoryState::Transferred) | FlagsMisc |
|
Transfered = static_cast<u32>(Svc::MemoryState::Transfered) | FlagsMisc |
|
||||||
FlagCanAlignedDeviceMap | FlagCanChangeAttribute | FlagCanUseIpc |
|
FlagCanAlignedDeviceMap | FlagCanChangeAttribute | FlagCanUseIpc |
|
||||||
FlagCanUseNonSecureIpc | FlagCanUseNonDeviceIpc,
|
FlagCanUseNonSecureIpc | FlagCanUseNonDeviceIpc,
|
||||||
|
|
||||||
SharedTransfered = static_cast<u32>(Svc::MemoryState::SharedTransferred) | FlagsMisc |
|
SharedTransfered = static_cast<u32>(Svc::MemoryState::SharedTransfered) | FlagsMisc |
|
||||||
FlagCanAlignedDeviceMap | FlagCanUseNonSecureIpc | FlagCanUseNonDeviceIpc,
|
FlagCanAlignedDeviceMap | FlagCanUseNonSecureIpc | FlagCanUseNonDeviceIpc,
|
||||||
|
|
||||||
SharedCode = static_cast<u32>(Svc::MemoryState::SharedCode) | FlagMapped |
|
SharedCode = static_cast<u32>(Svc::MemoryState::SharedCode) | FlagMapped |
|
||||||
FlagReferenceCounted | FlagCanUseNonSecureIpc | FlagCanUseNonDeviceIpc,
|
FlagReferenceCounted | FlagLinearMapped | FlagCanUseNonSecureIpc |
|
||||||
|
FlagCanUseNonDeviceIpc,
|
||||||
|
|
||||||
Inaccessible = static_cast<u32>(Svc::MemoryState::Inaccessible),
|
Inaccessible = static_cast<u32>(Svc::MemoryState::Inaccessible),
|
||||||
|
|
||||||
@ -91,69 +97,69 @@ enum class KMemoryState : u32 {
|
|||||||
Kernel = static_cast<u32>(Svc::MemoryState::Kernel) | FlagMapped,
|
Kernel = static_cast<u32>(Svc::MemoryState::Kernel) | FlagMapped,
|
||||||
|
|
||||||
GeneratedCode = static_cast<u32>(Svc::MemoryState::GeneratedCode) | FlagMapped |
|
GeneratedCode = static_cast<u32>(Svc::MemoryState::GeneratedCode) | FlagMapped |
|
||||||
FlagReferenceCounted | FlagCanDebug,
|
FlagReferenceCounted | FlagCanDebug | FlagLinearMapped,
|
||||||
CodeOut = static_cast<u32>(Svc::MemoryState::CodeOut) | FlagMapped | FlagReferenceCounted,
|
CodeOut = static_cast<u32>(Svc::MemoryState::CodeOut) | FlagMapped | FlagReferenceCounted |
|
||||||
|
FlagLinearMapped,
|
||||||
|
|
||||||
Coverage = static_cast<u32>(Svc::MemoryState::Coverage) | FlagMapped,
|
Coverage = static_cast<u32>(Svc::MemoryState::Coverage) | FlagMapped,
|
||||||
|
|
||||||
|
Insecure = static_cast<u32>(Svc::MemoryState::Insecure) | FlagMapped | FlagReferenceCounted |
|
||||||
|
FlagLinearMapped | FlagCanChangeAttribute | FlagCanDeviceMap |
|
||||||
|
FlagCanAlignedDeviceMap | FlagCanUseNonSecureIpc | FlagCanUseNonDeviceIpc,
|
||||||
};
|
};
|
||||||
DECLARE_ENUM_FLAG_OPERATORS(KMemoryState);
|
DECLARE_ENUM_FLAG_OPERATORS(KMemoryState);
|
||||||
|
|
||||||
static_assert(static_cast<u32>(KMemoryState::Free) == 0x00000000);
|
static_assert(static_cast<u32>(KMemoryState::Free) == 0x00000000);
|
||||||
static_assert(static_cast<u32>(KMemoryState::Io) == 0x00002001);
|
static_assert(static_cast<u32>(KMemoryState::Io) == 0x00182001);
|
||||||
static_assert(static_cast<u32>(KMemoryState::Static) == 0x00042002);
|
static_assert(static_cast<u32>(KMemoryState::Static) == 0x00042002);
|
||||||
static_assert(static_cast<u32>(KMemoryState::Code) == 0x00DC7E03);
|
static_assert(static_cast<u32>(KMemoryState::Code) == 0x04DC7E03);
|
||||||
static_assert(static_cast<u32>(KMemoryState::CodeData) == 0x03FEBD04);
|
static_assert(static_cast<u32>(KMemoryState::CodeData) == 0x07FEBD04);
|
||||||
static_assert(static_cast<u32>(KMemoryState::Normal) == 0x037EBD05);
|
static_assert(static_cast<u32>(KMemoryState::Normal) == 0x077EBD05);
|
||||||
static_assert(static_cast<u32>(KMemoryState::Shared) == 0x00402006);
|
static_assert(static_cast<u32>(KMemoryState::Shared) == 0x04402006);
|
||||||
static_assert(static_cast<u32>(KMemoryState::AliasCode) == 0x00DD7E08);
|
|
||||||
static_assert(static_cast<u32>(KMemoryState::AliasCodeData) == 0x03FFBD09);
|
static_assert(static_cast<u32>(KMemoryState::AliasCode) == 0x04DD7E08);
|
||||||
static_assert(static_cast<u32>(KMemoryState::Ipc) == 0x005C3C0A);
|
static_assert(static_cast<u32>(KMemoryState::AliasCodeData) == 0x07FFBD09);
|
||||||
static_assert(static_cast<u32>(KMemoryState::Stack) == 0x005C3C0B);
|
static_assert(static_cast<u32>(KMemoryState::Ipc) == 0x045C3C0A);
|
||||||
static_assert(static_cast<u32>(KMemoryState::ThreadLocal) == 0x0040200C);
|
static_assert(static_cast<u32>(KMemoryState::Stack) == 0x045C3C0B);
|
||||||
static_assert(static_cast<u32>(KMemoryState::Transfered) == 0x015C3C0D);
|
static_assert(static_cast<u32>(KMemoryState::ThreadLocal) == 0x0400200C);
|
||||||
static_assert(static_cast<u32>(KMemoryState::SharedTransfered) == 0x005C380E);
|
static_assert(static_cast<u32>(KMemoryState::Transfered) == 0x055C3C0D);
|
||||||
static_assert(static_cast<u32>(KMemoryState::SharedCode) == 0x0040380F);
|
static_assert(static_cast<u32>(KMemoryState::SharedTransfered) == 0x045C380E);
|
||||||
|
static_assert(static_cast<u32>(KMemoryState::SharedCode) == 0x0440380F);
|
||||||
static_assert(static_cast<u32>(KMemoryState::Inaccessible) == 0x00000010);
|
static_assert(static_cast<u32>(KMemoryState::Inaccessible) == 0x00000010);
|
||||||
static_assert(static_cast<u32>(KMemoryState::NonSecureIpc) == 0x005C3811);
|
static_assert(static_cast<u32>(KMemoryState::NonSecureIpc) == 0x045C3811);
|
||||||
static_assert(static_cast<u32>(KMemoryState::NonDeviceIpc) == 0x004C2812);
|
static_assert(static_cast<u32>(KMemoryState::NonDeviceIpc) == 0x044C2812);
|
||||||
static_assert(static_cast<u32>(KMemoryState::Kernel) == 0x00002013);
|
static_assert(static_cast<u32>(KMemoryState::Kernel) == 0x00002013);
|
||||||
static_assert(static_cast<u32>(KMemoryState::GeneratedCode) == 0x00402214);
|
static_assert(static_cast<u32>(KMemoryState::GeneratedCode) == 0x04402214);
|
||||||
static_assert(static_cast<u32>(KMemoryState::CodeOut) == 0x00402015);
|
static_assert(static_cast<u32>(KMemoryState::CodeOut) == 0x04402015);
|
||||||
static_assert(static_cast<u32>(KMemoryState::Coverage) == 0x00002016);
|
static_assert(static_cast<u32>(KMemoryState::Coverage) == 0x00002016);
|
||||||
|
static_assert(static_cast<u32>(KMemoryState::Insecure) == 0x05583817);
|
||||||
|
|
||||||
enum class KMemoryPermission : u8 {
|
enum class KMemoryPermission : u8 {
|
||||||
None = 0,
|
None = 0,
|
||||||
All = static_cast<u8>(~None),
|
All = static_cast<u8>(~None),
|
||||||
|
|
||||||
Read = 1 << 0,
|
|
||||||
Write = 1 << 1,
|
|
||||||
Execute = 1 << 2,
|
|
||||||
|
|
||||||
ReadAndWrite = Read | Write,
|
|
||||||
ReadAndExecute = Read | Execute,
|
|
||||||
|
|
||||||
UserMask = static_cast<u8>(Svc::MemoryPermission::Read | Svc::MemoryPermission::Write |
|
|
||||||
Svc::MemoryPermission::Execute),
|
|
||||||
|
|
||||||
KernelShift = 3,
|
KernelShift = 3,
|
||||||
|
|
||||||
KernelRead = Read << KernelShift,
|
KernelRead = static_cast<u8>(Svc::MemoryPermission::Read) << KernelShift,
|
||||||
KernelWrite = Write << KernelShift,
|
KernelWrite = static_cast<u8>(Svc::MemoryPermission::Write) << KernelShift,
|
||||||
KernelExecute = Execute << KernelShift,
|
KernelExecute = static_cast<u8>(Svc::MemoryPermission::Execute) << KernelShift,
|
||||||
|
|
||||||
NotMapped = (1 << (2 * KernelShift)),
|
NotMapped = (1 << (2 * KernelShift)),
|
||||||
|
|
||||||
KernelReadWrite = KernelRead | KernelWrite,
|
KernelReadWrite = KernelRead | KernelWrite,
|
||||||
KernelReadExecute = KernelRead | KernelExecute,
|
KernelReadExecute = KernelRead | KernelExecute,
|
||||||
|
|
||||||
UserRead = Read | KernelRead,
|
UserRead = static_cast<u8>(Svc::MemoryPermission::Read) | KernelRead,
|
||||||
UserWrite = Write | KernelWrite,
|
UserWrite = static_cast<u8>(Svc::MemoryPermission::Write) | KernelWrite,
|
||||||
UserExecute = Execute,
|
UserExecute = static_cast<u8>(Svc::MemoryPermission::Execute),
|
||||||
|
|
||||||
UserReadWrite = UserRead | UserWrite,
|
UserReadWrite = UserRead | UserWrite,
|
||||||
UserReadExecute = UserRead | UserExecute,
|
UserReadExecute = UserRead | UserExecute,
|
||||||
|
|
||||||
IpcLockChangeMask = NotMapped | UserReadWrite
|
UserMask = static_cast<u8>(Svc::MemoryPermission::Read | Svc::MemoryPermission::Write |
|
||||||
|
Svc::MemoryPermission::Execute),
|
||||||
|
|
||||||
|
IpcLockChangeMask = NotMapped | UserReadWrite,
|
||||||
};
|
};
|
||||||
DECLARE_ENUM_FLAG_OPERATORS(KMemoryPermission);
|
DECLARE_ENUM_FLAG_OPERATORS(KMemoryPermission);
|
||||||
|
|
||||||
@ -468,6 +474,7 @@ public:
|
|||||||
|
|
||||||
constexpr void UpdateDeviceDisableMergeStateForShareLeft(
|
constexpr void UpdateDeviceDisableMergeStateForShareLeft(
|
||||||
[[maybe_unused]] KMemoryPermission new_perm, bool left, [[maybe_unused]] bool right) {
|
[[maybe_unused]] KMemoryPermission new_perm, bool left, [[maybe_unused]] bool right) {
|
||||||
|
// New permission/right aren't used.
|
||||||
if (left) {
|
if (left) {
|
||||||
m_disable_merge_attribute = static_cast<KMemoryBlockDisableMergeAttribute>(
|
m_disable_merge_attribute = static_cast<KMemoryBlockDisableMergeAttribute>(
|
||||||
m_disable_merge_attribute | KMemoryBlockDisableMergeAttribute::DeviceLeft);
|
m_disable_merge_attribute | KMemoryBlockDisableMergeAttribute::DeviceLeft);
|
||||||
@ -478,6 +485,7 @@ public:
|
|||||||
|
|
||||||
constexpr void UpdateDeviceDisableMergeStateForShareRight(
|
constexpr void UpdateDeviceDisableMergeStateForShareRight(
|
||||||
[[maybe_unused]] KMemoryPermission new_perm, [[maybe_unused]] bool left, bool right) {
|
[[maybe_unused]] KMemoryPermission new_perm, [[maybe_unused]] bool left, bool right) {
|
||||||
|
// New permission/left aren't used.
|
||||||
if (right) {
|
if (right) {
|
||||||
m_disable_merge_attribute = static_cast<KMemoryBlockDisableMergeAttribute>(
|
m_disable_merge_attribute = static_cast<KMemoryBlockDisableMergeAttribute>(
|
||||||
m_disable_merge_attribute | KMemoryBlockDisableMergeAttribute::DeviceRight);
|
m_disable_merge_attribute | KMemoryBlockDisableMergeAttribute::DeviceRight);
|
||||||
@ -494,6 +502,8 @@ public:
|
|||||||
|
|
||||||
constexpr void ShareToDevice([[maybe_unused]] KMemoryPermission new_perm, bool left,
|
constexpr void ShareToDevice([[maybe_unused]] KMemoryPermission new_perm, bool left,
|
||||||
bool right) {
|
bool right) {
|
||||||
|
// New permission isn't used.
|
||||||
|
|
||||||
// We must either be shared or have a zero lock count.
|
// We must either be shared or have a zero lock count.
|
||||||
ASSERT((m_attribute & KMemoryAttribute::DeviceShared) == KMemoryAttribute::DeviceShared ||
|
ASSERT((m_attribute & KMemoryAttribute::DeviceShared) == KMemoryAttribute::DeviceShared ||
|
||||||
m_device_use_count == 0);
|
m_device_use_count == 0);
|
||||||
@ -509,6 +519,7 @@ public:
|
|||||||
|
|
||||||
constexpr void UpdateDeviceDisableMergeStateForUnshareLeft(
|
constexpr void UpdateDeviceDisableMergeStateForUnshareLeft(
|
||||||
[[maybe_unused]] KMemoryPermission new_perm, bool left, [[maybe_unused]] bool right) {
|
[[maybe_unused]] KMemoryPermission new_perm, bool left, [[maybe_unused]] bool right) {
|
||||||
|
// New permission/right aren't used.
|
||||||
|
|
||||||
if (left) {
|
if (left) {
|
||||||
if (!m_device_disable_merge_left_count) {
|
if (!m_device_disable_merge_left_count) {
|
||||||
@ -528,6 +539,8 @@ public:
|
|||||||
|
|
||||||
constexpr void UpdateDeviceDisableMergeStateForUnshareRight(
|
constexpr void UpdateDeviceDisableMergeStateForUnshareRight(
|
||||||
[[maybe_unused]] KMemoryPermission new_perm, [[maybe_unused]] bool left, bool right) {
|
[[maybe_unused]] KMemoryPermission new_perm, [[maybe_unused]] bool left, bool right) {
|
||||||
|
// New permission/left aren't used.
|
||||||
|
|
||||||
if (right) {
|
if (right) {
|
||||||
const u16 old_device_disable_merge_right_count = m_device_disable_merge_right_count--;
|
const u16 old_device_disable_merge_right_count = m_device_disable_merge_right_count--;
|
||||||
ASSERT(old_device_disable_merge_right_count > 0);
|
ASSERT(old_device_disable_merge_right_count > 0);
|
||||||
@ -546,6 +559,8 @@ public:
|
|||||||
|
|
||||||
constexpr void UnshareToDevice([[maybe_unused]] KMemoryPermission new_perm, bool left,
|
constexpr void UnshareToDevice([[maybe_unused]] KMemoryPermission new_perm, bool left,
|
||||||
bool right) {
|
bool right) {
|
||||||
|
// New permission isn't used.
|
||||||
|
|
||||||
// We must be shared.
|
// We must be shared.
|
||||||
ASSERT((m_attribute & KMemoryAttribute::DeviceShared) == KMemoryAttribute::DeviceShared);
|
ASSERT((m_attribute & KMemoryAttribute::DeviceShared) == KMemoryAttribute::DeviceShared);
|
||||||
|
|
||||||
@ -563,6 +578,7 @@ public:
|
|||||||
|
|
||||||
constexpr void UnshareToDeviceRight([[maybe_unused]] KMemoryPermission new_perm, bool left,
|
constexpr void UnshareToDeviceRight([[maybe_unused]] KMemoryPermission new_perm, bool left,
|
||||||
bool right) {
|
bool right) {
|
||||||
|
// New permission isn't used.
|
||||||
|
|
||||||
// We must be shared.
|
// We must be shared.
|
||||||
ASSERT((m_attribute & KMemoryAttribute::DeviceShared) == KMemoryAttribute::DeviceShared);
|
ASSERT((m_attribute & KMemoryAttribute::DeviceShared) == KMemoryAttribute::DeviceShared);
|
||||||
@ -613,6 +629,8 @@ public:
|
|||||||
|
|
||||||
constexpr void UnlockForIpc([[maybe_unused]] KMemoryPermission new_perm, bool left,
|
constexpr void UnlockForIpc([[maybe_unused]] KMemoryPermission new_perm, bool left,
|
||||||
[[maybe_unused]] bool right) {
|
[[maybe_unused]] bool right) {
|
||||||
|
// New permission isn't used.
|
||||||
|
|
||||||
// We must be locked.
|
// We must be locked.
|
||||||
ASSERT((m_attribute & KMemoryAttribute::IpcLocked) == KMemoryAttribute::IpcLocked);
|
ASSERT((m_attribute & KMemoryAttribute::IpcLocked) == KMemoryAttribute::IpcLocked);
|
||||||
|
|
||||||
|
@ -153,13 +153,9 @@ void KMemoryLayout::InitializeLinearMemoryRegionTrees(PAddr aligned_linear_phys_
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t KMemoryLayout::GetResourceRegionSizeForInit() {
|
size_t KMemoryLayout::GetResourceRegionSizeForInit(bool use_extra_resource) {
|
||||||
// Calculate resource region size based on whether we allow extra threads.
|
return KernelResourceSize + KSystemControl::SecureAppletMemorySize +
|
||||||
const bool use_extra_resources = KSystemControl::Init::ShouldIncreaseThreadResourceLimit();
|
(use_extra_resource ? KernelSlabHeapAdditionalSize + KernelPageBufferAdditionalSize : 0);
|
||||||
size_t resource_region_size =
|
|
||||||
KernelResourceSize + (use_extra_resources ? KernelSlabHeapAdditionalSize : 0);
|
|
||||||
|
|
||||||
return resource_region_size;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace Kernel
|
} // namespace Kernel
|
||||||
|
@ -60,10 +60,12 @@ constexpr std::size_t KernelSlabHeapGapsSizeMax = 2_MiB - 64_KiB;
|
|||||||
constexpr std::size_t KernelSlabHeapSize = KernelSlabHeapDataSize + KernelSlabHeapGapsSizeMax;
|
constexpr std::size_t KernelSlabHeapSize = KernelSlabHeapDataSize + KernelSlabHeapGapsSizeMax;
|
||||||
|
|
||||||
// NOTE: This is calculated from KThread slab counts, assuming KThread size <= 0x860.
|
// NOTE: This is calculated from KThread slab counts, assuming KThread size <= 0x860.
|
||||||
constexpr std::size_t KernelSlabHeapAdditionalSize = 0x68000;
|
constexpr size_t KernelPageBufferHeapSize = 0x3E0000;
|
||||||
|
constexpr size_t KernelSlabHeapAdditionalSize = 0x148000;
|
||||||
|
constexpr size_t KernelPageBufferAdditionalSize = 0x33C000;
|
||||||
|
|
||||||
constexpr std::size_t KernelResourceSize =
|
constexpr std::size_t KernelResourceSize = KernelPageTableHeapSize + KernelInitialPageHeapSize +
|
||||||
KernelPageTableHeapSize + KernelInitialPageHeapSize + KernelSlabHeapSize;
|
KernelSlabHeapSize + KernelPageBufferHeapSize;
|
||||||
|
|
||||||
constexpr bool IsKernelAddressKey(VAddr key) {
|
constexpr bool IsKernelAddressKey(VAddr key) {
|
||||||
return KernelVirtualAddressSpaceBase <= key && key <= KernelVirtualAddressSpaceLast;
|
return KernelVirtualAddressSpaceBase <= key && key <= KernelVirtualAddressSpaceLast;
|
||||||
@ -168,6 +170,11 @@ public:
|
|||||||
KMemoryRegionType_VirtualDramKernelTraceBuffer));
|
KMemoryRegionType_VirtualDramKernelTraceBuffer));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const KMemoryRegion& GetSecureAppletMemoryRegion() {
|
||||||
|
return Dereference(GetVirtualMemoryRegionTree().FindByType(
|
||||||
|
KMemoryRegionType_VirtualDramKernelSecureAppletMemory));
|
||||||
|
}
|
||||||
|
|
||||||
const KMemoryRegion& GetVirtualLinearRegion(VAddr address) const {
|
const KMemoryRegion& GetVirtualLinearRegion(VAddr address) const {
|
||||||
return Dereference(FindVirtualLinear(address));
|
return Dereference(FindVirtualLinear(address));
|
||||||
}
|
}
|
||||||
@ -229,7 +236,7 @@ public:
|
|||||||
|
|
||||||
void InitializeLinearMemoryRegionTrees(PAddr aligned_linear_phys_start,
|
void InitializeLinearMemoryRegionTrees(PAddr aligned_linear_phys_start,
|
||||||
VAddr linear_virtual_start);
|
VAddr linear_virtual_start);
|
||||||
static size_t GetResourceRegionSizeForInit();
|
static size_t GetResourceRegionSizeForInit(bool use_extra_resource);
|
||||||
|
|
||||||
auto GetKernelRegionExtents() const {
|
auto GetKernelRegionExtents() const {
|
||||||
return GetVirtualMemoryRegionTree().GetDerivedRegionExtents(KMemoryRegionType_Kernel);
|
return GetVirtualMemoryRegionTree().GetDerivedRegionExtents(KMemoryRegionType_Kernel);
|
||||||
@ -279,6 +286,10 @@ public:
|
|||||||
return GetPhysicalMemoryRegionTree().GetDerivedRegionExtents(
|
return GetPhysicalMemoryRegionTree().GetDerivedRegionExtents(
|
||||||
KMemoryRegionType_DramKernelSlab);
|
KMemoryRegionType_DramKernelSlab);
|
||||||
}
|
}
|
||||||
|
auto GetKernelSecureAppletMemoryRegionPhysicalExtents() {
|
||||||
|
return GetPhysicalMemoryRegionTree().GetDerivedRegionExtents(
|
||||||
|
KMemoryRegionType_DramKernelSecureAppletMemory);
|
||||||
|
}
|
||||||
auto GetKernelPageTableHeapRegionPhysicalExtents() const {
|
auto GetKernelPageTableHeapRegionPhysicalExtents() const {
|
||||||
return GetPhysicalMemoryRegionTree().GetDerivedRegionExtents(
|
return GetPhysicalMemoryRegionTree().GetDerivedRegionExtents(
|
||||||
KMemoryRegionType_DramKernelPtHeap);
|
KMemoryRegionType_DramKernelPtHeap);
|
||||||
|
@ -29,43 +29,44 @@ constexpr KMemoryManager::Pool GetPoolFromMemoryRegionType(u32 type) {
|
|||||||
} else if ((type | KMemoryRegionType_DramSystemNonSecurePool) == type) {
|
} else if ((type | KMemoryRegionType_DramSystemNonSecurePool) == type) {
|
||||||
return KMemoryManager::Pool::SystemNonSecure;
|
return KMemoryManager::Pool::SystemNonSecure;
|
||||||
} else {
|
} else {
|
||||||
ASSERT_MSG(false, "InvalidMemoryRegionType for conversion to Pool");
|
UNREACHABLE_MSG("InvalidMemoryRegionType for conversion to Pool");
|
||||||
return {};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
KMemoryManager::KMemoryManager(Core::System& system_)
|
KMemoryManager::KMemoryManager(Core::System& system)
|
||||||
: system{system_}, pool_locks{
|
: m_system{system}, m_memory_layout{system.Kernel().MemoryLayout()},
|
||||||
KLightLock{system_.Kernel()},
|
m_pool_locks{
|
||||||
KLightLock{system_.Kernel()},
|
KLightLock{system.Kernel()},
|
||||||
KLightLock{system_.Kernel()},
|
KLightLock{system.Kernel()},
|
||||||
KLightLock{system_.Kernel()},
|
KLightLock{system.Kernel()},
|
||||||
|
KLightLock{system.Kernel()},
|
||||||
} {}
|
} {}
|
||||||
|
|
||||||
void KMemoryManager::Initialize(VAddr management_region, size_t management_region_size) {
|
void KMemoryManager::Initialize(VAddr management_region, size_t management_region_size) {
|
||||||
|
|
||||||
// Clear the management region to zero.
|
// Clear the management region to zero.
|
||||||
const VAddr management_region_end = management_region + management_region_size;
|
const VAddr management_region_end = management_region + management_region_size;
|
||||||
|
// std::memset(GetVoidPointer(management_region), 0, management_region_size);
|
||||||
|
|
||||||
// Reset our manager count.
|
// Reset our manager count.
|
||||||
num_managers = 0;
|
m_num_managers = 0;
|
||||||
|
|
||||||
// Traverse the virtual memory layout tree, initializing each manager as appropriate.
|
// Traverse the virtual memory layout tree, initializing each manager as appropriate.
|
||||||
while (num_managers != MaxManagerCount) {
|
while (m_num_managers != MaxManagerCount) {
|
||||||
// Locate the region that should initialize the current manager.
|
// Locate the region that should initialize the current manager.
|
||||||
PAddr region_address = 0;
|
PAddr region_address = 0;
|
||||||
size_t region_size = 0;
|
size_t region_size = 0;
|
||||||
Pool region_pool = Pool::Count;
|
Pool region_pool = Pool::Count;
|
||||||
for (const auto& it : system.Kernel().MemoryLayout().GetPhysicalMemoryRegionTree()) {
|
for (const auto& it : m_system.Kernel().MemoryLayout().GetPhysicalMemoryRegionTree()) {
|
||||||
// We only care about regions that we need to create managers for.
|
// We only care about regions that we need to create managers for.
|
||||||
if (!it.IsDerivedFrom(KMemoryRegionType_DramUserPool)) {
|
if (!it.IsDerivedFrom(KMemoryRegionType_DramUserPool)) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// We want to initialize the managers in order.
|
// We want to initialize the managers in order.
|
||||||
if (it.GetAttributes() != num_managers) {
|
if (it.GetAttributes() != m_num_managers) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -97,8 +98,8 @@ void KMemoryManager::Initialize(VAddr management_region, size_t management_regio
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Initialize a new manager for the region.
|
// Initialize a new manager for the region.
|
||||||
Impl* manager = std::addressof(managers[num_managers++]);
|
Impl* manager = std::addressof(m_managers[m_num_managers++]);
|
||||||
ASSERT(num_managers <= managers.size());
|
ASSERT(m_num_managers <= m_managers.size());
|
||||||
|
|
||||||
const size_t cur_size = manager->Initialize(region_address, region_size, management_region,
|
const size_t cur_size = manager->Initialize(region_address, region_size, management_region,
|
||||||
management_region_end, region_pool);
|
management_region_end, region_pool);
|
||||||
@ -107,13 +108,13 @@ void KMemoryManager::Initialize(VAddr management_region, size_t management_regio
|
|||||||
|
|
||||||
// Insert the manager into the pool list.
|
// Insert the manager into the pool list.
|
||||||
const auto region_pool_index = static_cast<u32>(region_pool);
|
const auto region_pool_index = static_cast<u32>(region_pool);
|
||||||
if (pool_managers_tail[region_pool_index] == nullptr) {
|
if (m_pool_managers_tail[region_pool_index] == nullptr) {
|
||||||
pool_managers_head[region_pool_index] = manager;
|
m_pool_managers_head[region_pool_index] = manager;
|
||||||
} else {
|
} else {
|
||||||
pool_managers_tail[region_pool_index]->SetNext(manager);
|
m_pool_managers_tail[region_pool_index]->SetNext(manager);
|
||||||
manager->SetPrev(pool_managers_tail[region_pool_index]);
|
manager->SetPrev(m_pool_managers_tail[region_pool_index]);
|
||||||
}
|
}
|
||||||
pool_managers_tail[region_pool_index] = manager;
|
m_pool_managers_tail[region_pool_index] = manager;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Free each region to its corresponding heap.
|
// Free each region to its corresponding heap.
|
||||||
@ -121,11 +122,10 @@ void KMemoryManager::Initialize(VAddr management_region, size_t management_regio
|
|||||||
const PAddr ini_start = GetInitialProcessBinaryPhysicalAddress();
|
const PAddr ini_start = GetInitialProcessBinaryPhysicalAddress();
|
||||||
const PAddr ini_end = ini_start + InitialProcessBinarySizeMax;
|
const PAddr ini_end = ini_start + InitialProcessBinarySizeMax;
|
||||||
const PAddr ini_last = ini_end - 1;
|
const PAddr ini_last = ini_end - 1;
|
||||||
for (const auto& it : system.Kernel().MemoryLayout().GetPhysicalMemoryRegionTree()) {
|
for (const auto& it : m_system.Kernel().MemoryLayout().GetPhysicalMemoryRegionTree()) {
|
||||||
if (it.IsDerivedFrom(KMemoryRegionType_DramUserPool)) {
|
if (it.IsDerivedFrom(KMemoryRegionType_DramUserPool)) {
|
||||||
// Get the manager for the region.
|
// Get the manager for the region.
|
||||||
auto index = it.GetAttributes();
|
auto& manager = m_managers[it.GetAttributes()];
|
||||||
auto& manager = managers[index];
|
|
||||||
|
|
||||||
const PAddr cur_start = it.GetAddress();
|
const PAddr cur_start = it.GetAddress();
|
||||||
const PAddr cur_last = it.GetLastAddress();
|
const PAddr cur_last = it.GetLastAddress();
|
||||||
@ -162,11 +162,19 @@ void KMemoryManager::Initialize(VAddr management_region, size_t management_regio
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Update the used size for all managers.
|
// Update the used size for all managers.
|
||||||
for (size_t i = 0; i < num_managers; ++i) {
|
for (size_t i = 0; i < m_num_managers; ++i) {
|
||||||
managers[i].SetInitialUsedHeapSize(reserved_sizes[i]);
|
m_managers[i].SetInitialUsedHeapSize(reserved_sizes[i]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Result KMemoryManager::InitializeOptimizedMemory(u64 process_id, Pool pool) {
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
|
||||||
|
void KMemoryManager::FinalizeOptimizedMemory(u64 process_id, Pool pool) {
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
|
||||||
PAddr KMemoryManager::AllocateAndOpenContinuous(size_t num_pages, size_t align_pages, u32 option) {
|
PAddr KMemoryManager::AllocateAndOpenContinuous(size_t num_pages, size_t align_pages, u32 option) {
|
||||||
// Early return if we're allocating no pages.
|
// Early return if we're allocating no pages.
|
||||||
if (num_pages == 0) {
|
if (num_pages == 0) {
|
||||||
@ -175,7 +183,7 @@ PAddr KMemoryManager::AllocateAndOpenContinuous(size_t num_pages, size_t align_p
|
|||||||
|
|
||||||
// Lock the pool that we're allocating from.
|
// Lock the pool that we're allocating from.
|
||||||
const auto [pool, dir] = DecodeOption(option);
|
const auto [pool, dir] = DecodeOption(option);
|
||||||
KScopedLightLock lk(pool_locks[static_cast<std::size_t>(pool)]);
|
KScopedLightLock lk(m_pool_locks[static_cast<std::size_t>(pool)]);
|
||||||
|
|
||||||
// Choose a heap based on our page size request.
|
// Choose a heap based on our page size request.
|
||||||
const s32 heap_index = KPageHeap::GetAlignedBlockIndex(num_pages, align_pages);
|
const s32 heap_index = KPageHeap::GetAlignedBlockIndex(num_pages, align_pages);
|
||||||
@ -185,7 +193,7 @@ PAddr KMemoryManager::AllocateAndOpenContinuous(size_t num_pages, size_t align_p
|
|||||||
PAddr allocated_block = 0;
|
PAddr allocated_block = 0;
|
||||||
for (chosen_manager = this->GetFirstManager(pool, dir); chosen_manager != nullptr;
|
for (chosen_manager = this->GetFirstManager(pool, dir); chosen_manager != nullptr;
|
||||||
chosen_manager = this->GetNextManager(chosen_manager, dir)) {
|
chosen_manager = this->GetNextManager(chosen_manager, dir)) {
|
||||||
allocated_block = chosen_manager->AllocateBlock(heap_index, true);
|
allocated_block = chosen_manager->AllocateAligned(heap_index, num_pages, align_pages);
|
||||||
if (allocated_block != 0) {
|
if (allocated_block != 0) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -196,10 +204,9 @@ PAddr KMemoryManager::AllocateAndOpenContinuous(size_t num_pages, size_t align_p
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we allocated more than we need, free some.
|
// Maintain the optimized memory bitmap, if we should.
|
||||||
const size_t allocated_pages = KPageHeap::GetBlockNumPages(heap_index);
|
if (m_has_optimized_process[static_cast<size_t>(pool)]) {
|
||||||
if (allocated_pages > num_pages) {
|
UNIMPLEMENTED();
|
||||||
chosen_manager->Free(allocated_block + num_pages * PageSize, allocated_pages - num_pages);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Open the first reference to the pages.
|
// Open the first reference to the pages.
|
||||||
@ -209,20 +216,21 @@ PAddr KMemoryManager::AllocateAndOpenContinuous(size_t num_pages, size_t align_p
|
|||||||
}
|
}
|
||||||
|
|
||||||
Result KMemoryManager::AllocatePageGroupImpl(KPageGroup* out, size_t num_pages, Pool pool,
|
Result KMemoryManager::AllocatePageGroupImpl(KPageGroup* out, size_t num_pages, Pool pool,
|
||||||
Direction dir, bool random) {
|
Direction dir, bool unoptimized, bool random) {
|
||||||
// Choose a heap based on our page size request.
|
// Choose a heap based on our page size request.
|
||||||
const s32 heap_index = KPageHeap::GetBlockIndex(num_pages);
|
const s32 heap_index = KPageHeap::GetBlockIndex(num_pages);
|
||||||
R_UNLESS(0 <= heap_index, ResultOutOfMemory);
|
R_UNLESS(0 <= heap_index, ResultOutOfMemory);
|
||||||
|
|
||||||
// Ensure that we don't leave anything un-freed.
|
// Ensure that we don't leave anything un-freed.
|
||||||
auto group_guard = SCOPE_GUARD({
|
ON_RESULT_FAILURE {
|
||||||
for (const auto& it : out->Nodes()) {
|
for (const auto& it : out->Nodes()) {
|
||||||
auto& manager = this->GetManager(system.Kernel().MemoryLayout(), it.GetAddress());
|
auto& manager = this->GetManager(it.GetAddress());
|
||||||
const size_t num_pages_to_free =
|
const size_t node_num_pages =
|
||||||
std::min(it.GetNumPages(), (manager.GetEndAddress() - it.GetAddress()) / PageSize);
|
std::min(it.GetNumPages(), (manager.GetEndAddress() - it.GetAddress()) / PageSize);
|
||||||
manager.Free(it.GetAddress(), num_pages_to_free);
|
manager.Free(it.GetAddress(), node_num_pages);
|
||||||
}
|
}
|
||||||
});
|
out->Finalize();
|
||||||
|
};
|
||||||
|
|
||||||
// Keep allocating until we've allocated all our pages.
|
// Keep allocating until we've allocated all our pages.
|
||||||
for (s32 index = heap_index; index >= 0 && num_pages > 0; index--) {
|
for (s32 index = heap_index; index >= 0 && num_pages > 0; index--) {
|
||||||
@ -236,12 +244,17 @@ Result KMemoryManager::AllocatePageGroupImpl(KPageGroup* out, size_t num_pages,
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Safely add it to our group.
|
// Ensure we don't leak the block if we fail.
|
||||||
{
|
ON_RESULT_FAILURE_2 {
|
||||||
auto block_guard =
|
cur_manager->Free(allocated_block, pages_per_alloc);
|
||||||
SCOPE_GUARD({ cur_manager->Free(allocated_block, pages_per_alloc); });
|
};
|
||||||
|
|
||||||
|
// Add the block to our group.
|
||||||
R_TRY(out->AddBlock(allocated_block, pages_per_alloc));
|
R_TRY(out->AddBlock(allocated_block, pages_per_alloc));
|
||||||
block_guard.Cancel();
|
|
||||||
|
// Maintain the optimized memory bitmap, if we should.
|
||||||
|
if (unoptimized) {
|
||||||
|
UNIMPLEMENTED();
|
||||||
}
|
}
|
||||||
|
|
||||||
num_pages -= pages_per_alloc;
|
num_pages -= pages_per_alloc;
|
||||||
@ -253,8 +266,7 @@ Result KMemoryManager::AllocatePageGroupImpl(KPageGroup* out, size_t num_pages,
|
|||||||
R_UNLESS(num_pages == 0, ResultOutOfMemory);
|
R_UNLESS(num_pages == 0, ResultOutOfMemory);
|
||||||
|
|
||||||
// We succeeded!
|
// We succeeded!
|
||||||
group_guard.Cancel();
|
R_SUCCEED();
|
||||||
return ResultSuccess;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Result KMemoryManager::AllocateAndOpen(KPageGroup* out, size_t num_pages, u32 option) {
|
Result KMemoryManager::AllocateAndOpen(KPageGroup* out, size_t num_pages, u32 option) {
|
||||||
@ -266,10 +278,11 @@ Result KMemoryManager::AllocateAndOpen(KPageGroup* out, size_t num_pages, u32 op
|
|||||||
|
|
||||||
// Lock the pool that we're allocating from.
|
// Lock the pool that we're allocating from.
|
||||||
const auto [pool, dir] = DecodeOption(option);
|
const auto [pool, dir] = DecodeOption(option);
|
||||||
KScopedLightLock lk(pool_locks[static_cast<size_t>(pool)]);
|
KScopedLightLock lk(m_pool_locks[static_cast<size_t>(pool)]);
|
||||||
|
|
||||||
// Allocate the page group.
|
// Allocate the page group.
|
||||||
R_TRY(this->AllocatePageGroupImpl(out, num_pages, pool, dir, false));
|
R_TRY(this->AllocatePageGroupImpl(out, num_pages, pool, dir,
|
||||||
|
m_has_optimized_process[static_cast<size_t>(pool)], true));
|
||||||
|
|
||||||
// Open the first reference to the pages.
|
// Open the first reference to the pages.
|
||||||
for (const auto& block : out->Nodes()) {
|
for (const auto& block : out->Nodes()) {
|
||||||
@ -277,7 +290,7 @@ Result KMemoryManager::AllocateAndOpen(KPageGroup* out, size_t num_pages, u32 op
|
|||||||
size_t remaining_pages = block.GetNumPages();
|
size_t remaining_pages = block.GetNumPages();
|
||||||
while (remaining_pages > 0) {
|
while (remaining_pages > 0) {
|
||||||
// Get the manager for the current address.
|
// Get the manager for the current address.
|
||||||
auto& manager = this->GetManager(system.Kernel().MemoryLayout(), cur_address);
|
auto& manager = this->GetManager(cur_address);
|
||||||
|
|
||||||
// Process part or all of the block.
|
// Process part or all of the block.
|
||||||
const size_t cur_pages =
|
const size_t cur_pages =
|
||||||
@ -290,10 +303,10 @@ Result KMemoryManager::AllocateAndOpen(KPageGroup* out, size_t num_pages, u32 op
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return ResultSuccess;
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
Result KMemoryManager::AllocateAndOpenForProcess(KPageGroup* out, size_t num_pages, u32 option,
|
Result KMemoryManager::AllocateForProcess(KPageGroup* out, size_t num_pages, u32 option,
|
||||||
u64 process_id, u8 fill_pattern) {
|
u64 process_id, u8 fill_pattern) {
|
||||||
ASSERT(out != nullptr);
|
ASSERT(out != nullptr);
|
||||||
ASSERT(out->GetNumPages() == 0);
|
ASSERT(out->GetNumPages() == 0);
|
||||||
@ -302,25 +315,73 @@ Result KMemoryManager::AllocateAndOpenForProcess(KPageGroup* out, size_t num_pag
|
|||||||
const auto [pool, dir] = DecodeOption(option);
|
const auto [pool, dir] = DecodeOption(option);
|
||||||
|
|
||||||
// Allocate the memory.
|
// Allocate the memory.
|
||||||
|
bool optimized;
|
||||||
{
|
{
|
||||||
// Lock the pool that we're allocating from.
|
// Lock the pool that we're allocating from.
|
||||||
KScopedLightLock lk(pool_locks[static_cast<size_t>(pool)]);
|
KScopedLightLock lk(m_pool_locks[static_cast<size_t>(pool)]);
|
||||||
|
|
||||||
|
// Check if we have an optimized process.
|
||||||
|
const bool has_optimized = m_has_optimized_process[static_cast<size_t>(pool)];
|
||||||
|
const bool is_optimized = m_optimized_process_ids[static_cast<size_t>(pool)] == process_id;
|
||||||
|
|
||||||
// Allocate the page group.
|
// Allocate the page group.
|
||||||
R_TRY(this->AllocatePageGroupImpl(out, num_pages, pool, dir, false));
|
R_TRY(this->AllocatePageGroupImpl(out, num_pages, pool, dir, has_optimized && !is_optimized,
|
||||||
|
false));
|
||||||
|
|
||||||
// Open the first reference to the pages.
|
// Set whether we should optimize.
|
||||||
|
optimized = has_optimized && is_optimized;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Perform optimized memory tracking, if we should.
|
||||||
|
if (optimized) {
|
||||||
|
// Iterate over the allocated blocks.
|
||||||
for (const auto& block : out->Nodes()) {
|
for (const auto& block : out->Nodes()) {
|
||||||
PAddr cur_address = block.GetAddress();
|
// Get the block extents.
|
||||||
size_t remaining_pages = block.GetNumPages();
|
const PAddr block_address = block.GetAddress();
|
||||||
|
const size_t block_pages = block.GetNumPages();
|
||||||
|
|
||||||
|
// If it has no pages, we don't need to do anything.
|
||||||
|
if (block_pages == 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fill all the pages that we need to fill.
|
||||||
|
bool any_new = false;
|
||||||
|
{
|
||||||
|
PAddr cur_address = block_address;
|
||||||
|
size_t remaining_pages = block_pages;
|
||||||
while (remaining_pages > 0) {
|
while (remaining_pages > 0) {
|
||||||
// Get the manager for the current address.
|
// Get the manager for the current address.
|
||||||
auto& manager = this->GetManager(system.Kernel().MemoryLayout(), cur_address);
|
auto& manager = this->GetManager(cur_address);
|
||||||
|
|
||||||
// Process part or all of the block.
|
// Process part or all of the block.
|
||||||
const size_t cur_pages =
|
const size_t cur_pages =
|
||||||
std::min(remaining_pages, manager.GetPageOffsetToEnd(cur_address));
|
std::min(remaining_pages, manager.GetPageOffsetToEnd(cur_address));
|
||||||
manager.OpenFirst(cur_address, cur_pages);
|
any_new =
|
||||||
|
manager.ProcessOptimizedAllocation(cur_address, cur_pages, fill_pattern);
|
||||||
|
|
||||||
|
// Advance.
|
||||||
|
cur_address += cur_pages * PageSize;
|
||||||
|
remaining_pages -= cur_pages;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If there are new pages, update tracking for the allocation.
|
||||||
|
if (any_new) {
|
||||||
|
// Update tracking for the allocation.
|
||||||
|
PAddr cur_address = block_address;
|
||||||
|
size_t remaining_pages = block_pages;
|
||||||
|
while (remaining_pages > 0) {
|
||||||
|
// Get the manager for the current address.
|
||||||
|
auto& manager = this->GetManager(cur_address);
|
||||||
|
|
||||||
|
// Lock the pool for the manager.
|
||||||
|
KScopedLightLock lk(m_pool_locks[static_cast<size_t>(manager.GetPool())]);
|
||||||
|
|
||||||
|
// Track some or all of the current pages.
|
||||||
|
const size_t cur_pages =
|
||||||
|
std::min(remaining_pages, manager.GetPageOffsetToEnd(cur_address));
|
||||||
|
manager.TrackOptimizedAllocation(cur_address, cur_pages);
|
||||||
|
|
||||||
// Advance.
|
// Advance.
|
||||||
cur_address += cur_pages * PageSize;
|
cur_address += cur_pages * PageSize;
|
||||||
@ -328,57 +389,15 @@ Result KMemoryManager::AllocateAndOpenForProcess(KPageGroup* out, size_t num_pag
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
// Set all the allocated memory.
|
// Set all the allocated memory.
|
||||||
for (const auto& block : out->Nodes()) {
|
for (const auto& block : out->Nodes()) {
|
||||||
std::memset(system.DeviceMemory().GetPointer<void>(block.GetAddress()), fill_pattern,
|
std::memset(m_system.DeviceMemory().GetPointer<void>(block.GetAddress()), fill_pattern,
|
||||||
block.GetSize());
|
block.GetSize());
|
||||||
}
|
}
|
||||||
|
|
||||||
return ResultSuccess;
|
|
||||||
}
|
|
||||||
|
|
||||||
void KMemoryManager::Open(PAddr address, size_t num_pages) {
|
|
||||||
// Repeatedly open references until we've done so for all pages.
|
|
||||||
while (num_pages) {
|
|
||||||
auto& manager = this->GetManager(system.Kernel().MemoryLayout(), address);
|
|
||||||
const size_t cur_pages = std::min(num_pages, manager.GetPageOffsetToEnd(address));
|
|
||||||
|
|
||||||
{
|
|
||||||
KScopedLightLock lk(pool_locks[static_cast<size_t>(manager.GetPool())]);
|
|
||||||
manager.Open(address, cur_pages);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
num_pages -= cur_pages;
|
R_SUCCEED();
|
||||||
address += cur_pages * PageSize;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void KMemoryManager::Close(PAddr address, size_t num_pages) {
|
|
||||||
// Repeatedly close references until we've done so for all pages.
|
|
||||||
while (num_pages) {
|
|
||||||
auto& manager = this->GetManager(system.Kernel().MemoryLayout(), address);
|
|
||||||
const size_t cur_pages = std::min(num_pages, manager.GetPageOffsetToEnd(address));
|
|
||||||
|
|
||||||
{
|
|
||||||
KScopedLightLock lk(pool_locks[static_cast<size_t>(manager.GetPool())]);
|
|
||||||
manager.Close(address, cur_pages);
|
|
||||||
}
|
|
||||||
|
|
||||||
num_pages -= cur_pages;
|
|
||||||
address += cur_pages * PageSize;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void KMemoryManager::Close(const KPageGroup& pg) {
|
|
||||||
for (const auto& node : pg.Nodes()) {
|
|
||||||
Close(node.GetAddress(), node.GetNumPages());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
void KMemoryManager::Open(const KPageGroup& pg) {
|
|
||||||
for (const auto& node : pg.Nodes()) {
|
|
||||||
Open(node.GetAddress(), node.GetNumPages());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t KMemoryManager::Impl::Initialize(PAddr address, size_t size, VAddr management,
|
size_t KMemoryManager::Impl::Initialize(PAddr address, size_t size, VAddr management,
|
||||||
@ -394,18 +413,31 @@ size_t KMemoryManager::Impl::Initialize(PAddr address, size_t size, VAddr manage
|
|||||||
ASSERT(Common::IsAligned(total_management_size, PageSize));
|
ASSERT(Common::IsAligned(total_management_size, PageSize));
|
||||||
|
|
||||||
// Setup region.
|
// Setup region.
|
||||||
pool = p;
|
m_pool = p;
|
||||||
management_region = management;
|
m_management_region = management;
|
||||||
page_reference_counts.resize(
|
m_page_reference_counts.resize(
|
||||||
Kernel::Board::Nintendo::Nx::KSystemControl::Init::GetIntendedMemorySize() / PageSize);
|
Kernel::Board::Nintendo::Nx::KSystemControl::Init::GetIntendedMemorySize() / PageSize);
|
||||||
ASSERT(Common::IsAligned(management_region, PageSize));
|
ASSERT(Common::IsAligned(m_management_region, PageSize));
|
||||||
|
|
||||||
// Initialize the manager's KPageHeap.
|
// Initialize the manager's KPageHeap.
|
||||||
heap.Initialize(address, size, management + manager_size, page_heap_size);
|
m_heap.Initialize(address, size, management + manager_size, page_heap_size);
|
||||||
|
|
||||||
return total_management_size;
|
return total_management_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void KMemoryManager::Impl::TrackUnoptimizedAllocation(PAddr block, size_t num_pages) {
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
|
||||||
|
void KMemoryManager::Impl::TrackOptimizedAllocation(PAddr block, size_t num_pages) {
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool KMemoryManager::Impl::ProcessOptimizedAllocation(PAddr block, size_t num_pages,
|
||||||
|
u8 fill_pattern) {
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
|
||||||
size_t KMemoryManager::Impl::CalculateManagementOverheadSize(size_t region_size) {
|
size_t KMemoryManager::Impl::CalculateManagementOverheadSize(size_t region_size) {
|
||||||
const size_t ref_count_size = (region_size / PageSize) * sizeof(u16);
|
const size_t ref_count_size = (region_size / PageSize) * sizeof(u16);
|
||||||
const size_t optimize_map_size =
|
const size_t optimize_map_size =
|
||||||
|
@ -21,11 +21,8 @@ namespace Kernel {
|
|||||||
|
|
||||||
class KPageGroup;
|
class KPageGroup;
|
||||||
|
|
||||||
class KMemoryManager final {
|
class KMemoryManager {
|
||||||
public:
|
public:
|
||||||
YUZU_NON_COPYABLE(KMemoryManager);
|
|
||||||
YUZU_NON_MOVEABLE(KMemoryManager);
|
|
||||||
|
|
||||||
enum class Pool : u32 {
|
enum class Pool : u32 {
|
||||||
Application = 0,
|
Application = 0,
|
||||||
Applet = 1,
|
Applet = 1,
|
||||||
@ -45,16 +42,85 @@ public:
|
|||||||
enum class Direction : u32 {
|
enum class Direction : u32 {
|
||||||
FromFront = 0,
|
FromFront = 0,
|
||||||
FromBack = 1,
|
FromBack = 1,
|
||||||
|
|
||||||
Shift = 0,
|
Shift = 0,
|
||||||
Mask = (0xF << Shift),
|
Mask = (0xF << Shift),
|
||||||
};
|
};
|
||||||
|
|
||||||
explicit KMemoryManager(Core::System& system_);
|
static constexpr size_t MaxManagerCount = 10;
|
||||||
|
|
||||||
|
explicit KMemoryManager(Core::System& system);
|
||||||
|
|
||||||
void Initialize(VAddr management_region, size_t management_region_size);
|
void Initialize(VAddr management_region, size_t management_region_size);
|
||||||
|
|
||||||
constexpr size_t GetSize(Pool pool) const {
|
Result InitializeOptimizedMemory(u64 process_id, Pool pool);
|
||||||
|
void FinalizeOptimizedMemory(u64 process_id, Pool pool);
|
||||||
|
|
||||||
|
PAddr AllocateAndOpenContinuous(size_t num_pages, size_t align_pages, u32 option);
|
||||||
|
Result AllocateAndOpen(KPageGroup* out, size_t num_pages, u32 option);
|
||||||
|
Result AllocateForProcess(KPageGroup* out, size_t num_pages, u32 option, u64 process_id,
|
||||||
|
u8 fill_pattern);
|
||||||
|
|
||||||
|
Pool GetPool(PAddr address) const {
|
||||||
|
return this->GetManager(address).GetPool();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Open(PAddr address, size_t num_pages) {
|
||||||
|
// Repeatedly open references until we've done so for all pages.
|
||||||
|
while (num_pages) {
|
||||||
|
auto& manager = this->GetManager(address);
|
||||||
|
const size_t cur_pages = std::min(num_pages, manager.GetPageOffsetToEnd(address));
|
||||||
|
|
||||||
|
{
|
||||||
|
KScopedLightLock lk(m_pool_locks[static_cast<size_t>(manager.GetPool())]);
|
||||||
|
manager.Open(address, cur_pages);
|
||||||
|
}
|
||||||
|
|
||||||
|
num_pages -= cur_pages;
|
||||||
|
address += cur_pages * PageSize;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void OpenFirst(PAddr address, size_t num_pages) {
|
||||||
|
// Repeatedly open references until we've done so for all pages.
|
||||||
|
while (num_pages) {
|
||||||
|
auto& manager = this->GetManager(address);
|
||||||
|
const size_t cur_pages = std::min(num_pages, manager.GetPageOffsetToEnd(address));
|
||||||
|
|
||||||
|
{
|
||||||
|
KScopedLightLock lk(m_pool_locks[static_cast<size_t>(manager.GetPool())]);
|
||||||
|
manager.OpenFirst(address, cur_pages);
|
||||||
|
}
|
||||||
|
|
||||||
|
num_pages -= cur_pages;
|
||||||
|
address += cur_pages * PageSize;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Close(PAddr address, size_t num_pages) {
|
||||||
|
// Repeatedly close references until we've done so for all pages.
|
||||||
|
while (num_pages) {
|
||||||
|
auto& manager = this->GetManager(address);
|
||||||
|
const size_t cur_pages = std::min(num_pages, manager.GetPageOffsetToEnd(address));
|
||||||
|
|
||||||
|
{
|
||||||
|
KScopedLightLock lk(m_pool_locks[static_cast<size_t>(manager.GetPool())]);
|
||||||
|
manager.Close(address, cur_pages);
|
||||||
|
}
|
||||||
|
|
||||||
|
num_pages -= cur_pages;
|
||||||
|
address += cur_pages * PageSize;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t GetSize() {
|
||||||
|
size_t total = 0;
|
||||||
|
for (size_t i = 0; i < m_num_managers; i++) {
|
||||||
|
total += m_managers[i].GetSize();
|
||||||
|
}
|
||||||
|
return total;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t GetSize(Pool pool) {
|
||||||
constexpr Direction GetSizeDirection = Direction::FromFront;
|
constexpr Direction GetSizeDirection = Direction::FromFront;
|
||||||
size_t total = 0;
|
size_t total = 0;
|
||||||
for (auto* manager = this->GetFirstManager(pool, GetSizeDirection); manager != nullptr;
|
for (auto* manager = this->GetFirstManager(pool, GetSizeDirection); manager != nullptr;
|
||||||
@ -64,18 +130,36 @@ public:
|
|||||||
return total;
|
return total;
|
||||||
}
|
}
|
||||||
|
|
||||||
PAddr AllocateAndOpenContinuous(size_t num_pages, size_t align_pages, u32 option);
|
size_t GetFreeSize() {
|
||||||
Result AllocateAndOpen(KPageGroup* out, size_t num_pages, u32 option);
|
size_t total = 0;
|
||||||
Result AllocateAndOpenForProcess(KPageGroup* out, size_t num_pages, u32 option, u64 process_id,
|
for (size_t i = 0; i < m_num_managers; i++) {
|
||||||
u8 fill_pattern);
|
KScopedLightLock lk(m_pool_locks[static_cast<size_t>(m_managers[i].GetPool())]);
|
||||||
|
total += m_managers[i].GetFreeSize();
|
||||||
|
}
|
||||||
|
return total;
|
||||||
|
}
|
||||||
|
|
||||||
static constexpr size_t MaxManagerCount = 10;
|
size_t GetFreeSize(Pool pool) {
|
||||||
|
KScopedLightLock lk(m_pool_locks[static_cast<size_t>(pool)]);
|
||||||
|
|
||||||
void Close(PAddr address, size_t num_pages);
|
constexpr Direction GetSizeDirection = Direction::FromFront;
|
||||||
void Close(const KPageGroup& pg);
|
size_t total = 0;
|
||||||
|
for (auto* manager = this->GetFirstManager(pool, GetSizeDirection); manager != nullptr;
|
||||||
|
manager = this->GetNextManager(manager, GetSizeDirection)) {
|
||||||
|
total += manager->GetFreeSize();
|
||||||
|
}
|
||||||
|
return total;
|
||||||
|
}
|
||||||
|
|
||||||
void Open(PAddr address, size_t num_pages);
|
void DumpFreeList(Pool pool) {
|
||||||
void Open(const KPageGroup& pg);
|
KScopedLightLock lk(m_pool_locks[static_cast<size_t>(pool)]);
|
||||||
|
|
||||||
|
constexpr Direction DumpDirection = Direction::FromFront;
|
||||||
|
for (auto* manager = this->GetFirstManager(pool, DumpDirection); manager != nullptr;
|
||||||
|
manager = this->GetNextManager(manager, DumpDirection)) {
|
||||||
|
manager->DumpFreeList();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
static size_t CalculateManagementOverheadSize(size_t region_size) {
|
static size_t CalculateManagementOverheadSize(size_t region_size) {
|
||||||
@ -88,13 +172,12 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
static constexpr Pool GetPool(u32 option) {
|
static constexpr Pool GetPool(u32 option) {
|
||||||
return static_cast<Pool>((static_cast<u32>(option) & static_cast<u32>(Pool::Mask)) >>
|
return static_cast<Pool>((option & static_cast<u32>(Pool::Mask)) >>
|
||||||
static_cast<u32>(Pool::Shift));
|
static_cast<u32>(Pool::Shift));
|
||||||
}
|
}
|
||||||
|
|
||||||
static constexpr Direction GetDirection(u32 option) {
|
static constexpr Direction GetDirection(u32 option) {
|
||||||
return static_cast<Direction>(
|
return static_cast<Direction>((option & static_cast<u32>(Direction::Mask)) >>
|
||||||
(static_cast<u32>(option) & static_cast<u32>(Direction::Mask)) >>
|
|
||||||
static_cast<u32>(Direction::Shift));
|
static_cast<u32>(Direction::Shift));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -103,74 +186,88 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
class Impl final {
|
class Impl {
|
||||||
public:
|
public:
|
||||||
YUZU_NON_COPYABLE(Impl);
|
static size_t CalculateManagementOverheadSize(size_t region_size);
|
||||||
YUZU_NON_MOVEABLE(Impl);
|
|
||||||
|
|
||||||
|
static constexpr size_t CalculateOptimizedProcessOverheadSize(size_t region_size) {
|
||||||
|
return (Common::AlignUp((region_size / PageSize), Common::BitSize<u64>()) /
|
||||||
|
Common::BitSize<u64>()) *
|
||||||
|
sizeof(u64);
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
Impl() = default;
|
Impl() = default;
|
||||||
~Impl() = default;
|
|
||||||
|
|
||||||
size_t Initialize(PAddr address, size_t size, VAddr management, VAddr management_end,
|
size_t Initialize(PAddr address, size_t size, VAddr management, VAddr management_end,
|
||||||
Pool p);
|
Pool p);
|
||||||
|
|
||||||
VAddr AllocateBlock(s32 index, bool random) {
|
PAddr AllocateBlock(s32 index, bool random) {
|
||||||
return heap.AllocateBlock(index, random);
|
return m_heap.AllocateBlock(index, random);
|
||||||
}
|
}
|
||||||
|
PAddr AllocateAligned(s32 index, size_t num_pages, size_t align_pages) {
|
||||||
void Free(VAddr addr, size_t num_pages) {
|
return m_heap.AllocateAligned(index, num_pages, align_pages);
|
||||||
heap.Free(addr, num_pages);
|
}
|
||||||
|
void Free(PAddr addr, size_t num_pages) {
|
||||||
|
m_heap.Free(addr, num_pages);
|
||||||
}
|
}
|
||||||
|
|
||||||
void SetInitialUsedHeapSize(size_t reserved_size) {
|
void SetInitialUsedHeapSize(size_t reserved_size) {
|
||||||
heap.SetInitialUsedSize(reserved_size);
|
m_heap.SetInitialUsedSize(reserved_size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void InitializeOptimizedMemory() {
|
||||||
|
UNIMPLEMENTED();
|
||||||
|
}
|
||||||
|
|
||||||
|
void TrackUnoptimizedAllocation(PAddr block, size_t num_pages);
|
||||||
|
void TrackOptimizedAllocation(PAddr block, size_t num_pages);
|
||||||
|
|
||||||
|
bool ProcessOptimizedAllocation(PAddr block, size_t num_pages, u8 fill_pattern);
|
||||||
|
|
||||||
constexpr Pool GetPool() const {
|
constexpr Pool GetPool() const {
|
||||||
return pool;
|
return m_pool;
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr size_t GetSize() const {
|
constexpr size_t GetSize() const {
|
||||||
return heap.GetSize();
|
return m_heap.GetSize();
|
||||||
|
}
|
||||||
|
constexpr PAddr GetEndAddress() const {
|
||||||
|
return m_heap.GetEndAddress();
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr VAddr GetAddress() const {
|
size_t GetFreeSize() const {
|
||||||
return heap.GetAddress();
|
return m_heap.GetFreeSize();
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr VAddr GetEndAddress() const {
|
void DumpFreeList() const {
|
||||||
return heap.GetEndAddress();
|
UNIMPLEMENTED();
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr size_t GetPageOffset(PAddr address) const {
|
constexpr size_t GetPageOffset(PAddr address) const {
|
||||||
return heap.GetPageOffset(address);
|
return m_heap.GetPageOffset(address);
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr size_t GetPageOffsetToEnd(PAddr address) const {
|
constexpr size_t GetPageOffsetToEnd(PAddr address) const {
|
||||||
return heap.GetPageOffsetToEnd(address);
|
return m_heap.GetPageOffsetToEnd(address);
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr void SetNext(Impl* n) {
|
constexpr void SetNext(Impl* n) {
|
||||||
next = n;
|
m_next = n;
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr void SetPrev(Impl* n) {
|
constexpr void SetPrev(Impl* n) {
|
||||||
prev = n;
|
m_prev = n;
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr Impl* GetNext() const {
|
constexpr Impl* GetNext() const {
|
||||||
return next;
|
return m_next;
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr Impl* GetPrev() const {
|
constexpr Impl* GetPrev() const {
|
||||||
return prev;
|
return m_prev;
|
||||||
}
|
}
|
||||||
|
|
||||||
void OpenFirst(PAddr address, size_t num_pages) {
|
void OpenFirst(PAddr address, size_t num_pages) {
|
||||||
size_t index = this->GetPageOffset(address);
|
size_t index = this->GetPageOffset(address);
|
||||||
const size_t end = index + num_pages;
|
const size_t end = index + num_pages;
|
||||||
while (index < end) {
|
while (index < end) {
|
||||||
const RefCount ref_count = (++page_reference_counts[index]);
|
const RefCount ref_count = (++m_page_reference_counts[index]);
|
||||||
ASSERT(ref_count == 1);
|
ASSERT(ref_count == 1);
|
||||||
|
|
||||||
index++;
|
index++;
|
||||||
@ -181,7 +278,7 @@ private:
|
|||||||
size_t index = this->GetPageOffset(address);
|
size_t index = this->GetPageOffset(address);
|
||||||
const size_t end = index + num_pages;
|
const size_t end = index + num_pages;
|
||||||
while (index < end) {
|
while (index < end) {
|
||||||
const RefCount ref_count = (++page_reference_counts[index]);
|
const RefCount ref_count = (++m_page_reference_counts[index]);
|
||||||
ASSERT(ref_count > 1);
|
ASSERT(ref_count > 1);
|
||||||
|
|
||||||
index++;
|
index++;
|
||||||
@ -195,8 +292,8 @@ private:
|
|||||||
size_t free_start = 0;
|
size_t free_start = 0;
|
||||||
size_t free_count = 0;
|
size_t free_count = 0;
|
||||||
while (index < end) {
|
while (index < end) {
|
||||||
ASSERT(page_reference_counts[index] > 0);
|
ASSERT(m_page_reference_counts[index] > 0);
|
||||||
const RefCount ref_count = (--page_reference_counts[index]);
|
const RefCount ref_count = (--m_page_reference_counts[index]);
|
||||||
|
|
||||||
// Keep track of how many zero refcounts we see in a row, to minimize calls to free.
|
// Keep track of how many zero refcounts we see in a row, to minimize calls to free.
|
||||||
if (ref_count == 0) {
|
if (ref_count == 0) {
|
||||||
@ -208,7 +305,7 @@ private:
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (free_count > 0) {
|
if (free_count > 0) {
|
||||||
this->Free(heap.GetAddress() + free_start * PageSize, free_count);
|
this->Free(m_heap.GetAddress() + free_start * PageSize, free_count);
|
||||||
free_count = 0;
|
free_count = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -217,44 +314,36 @@ private:
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (free_count > 0) {
|
if (free_count > 0) {
|
||||||
this->Free(heap.GetAddress() + free_start * PageSize, free_count);
|
this->Free(m_heap.GetAddress() + free_start * PageSize, free_count);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static size_t CalculateManagementOverheadSize(size_t region_size);
|
|
||||||
|
|
||||||
static constexpr size_t CalculateOptimizedProcessOverheadSize(size_t region_size) {
|
|
||||||
return (Common::AlignUp((region_size / PageSize), Common::BitSize<u64>()) /
|
|
||||||
Common::BitSize<u64>()) *
|
|
||||||
sizeof(u64);
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
using RefCount = u16;
|
using RefCount = u16;
|
||||||
|
|
||||||
KPageHeap heap;
|
KPageHeap m_heap;
|
||||||
std::vector<RefCount> page_reference_counts;
|
std::vector<RefCount> m_page_reference_counts;
|
||||||
VAddr management_region{};
|
VAddr m_management_region{};
|
||||||
Pool pool{};
|
Pool m_pool{};
|
||||||
Impl* next{};
|
Impl* m_next{};
|
||||||
Impl* prev{};
|
Impl* m_prev{};
|
||||||
};
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Impl& GetManager(const KMemoryLayout& memory_layout, PAddr address) {
|
Impl& GetManager(PAddr address) {
|
||||||
return managers[memory_layout.GetPhysicalLinearRegion(address).GetAttributes()];
|
return m_managers[m_memory_layout.GetPhysicalLinearRegion(address).GetAttributes()];
|
||||||
}
|
}
|
||||||
|
|
||||||
const Impl& GetManager(const KMemoryLayout& memory_layout, PAddr address) const {
|
const Impl& GetManager(PAddr address) const {
|
||||||
return managers[memory_layout.GetPhysicalLinearRegion(address).GetAttributes()];
|
return m_managers[m_memory_layout.GetPhysicalLinearRegion(address).GetAttributes()];
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr Impl* GetFirstManager(Pool pool, Direction dir) const {
|
constexpr Impl* GetFirstManager(Pool pool, Direction dir) {
|
||||||
return dir == Direction::FromBack ? pool_managers_tail[static_cast<size_t>(pool)]
|
return dir == Direction::FromBack ? m_pool_managers_tail[static_cast<size_t>(pool)]
|
||||||
: pool_managers_head[static_cast<size_t>(pool)];
|
: m_pool_managers_head[static_cast<size_t>(pool)];
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr Impl* GetNextManager(Impl* cur, Direction dir) const {
|
constexpr Impl* GetNextManager(Impl* cur, Direction dir) {
|
||||||
if (dir == Direction::FromBack) {
|
if (dir == Direction::FromBack) {
|
||||||
return cur->GetPrev();
|
return cur->GetPrev();
|
||||||
} else {
|
} else {
|
||||||
@ -263,15 +352,21 @@ private:
|
|||||||
}
|
}
|
||||||
|
|
||||||
Result AllocatePageGroupImpl(KPageGroup* out, size_t num_pages, Pool pool, Direction dir,
|
Result AllocatePageGroupImpl(KPageGroup* out, size_t num_pages, Pool pool, Direction dir,
|
||||||
bool random);
|
bool unoptimized, bool random);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Core::System& system;
|
template <typename T>
|
||||||
std::array<KLightLock, static_cast<size_t>(Pool::Count)> pool_locks;
|
using PoolArray = std::array<T, static_cast<size_t>(Pool::Count)>;
|
||||||
std::array<Impl*, MaxManagerCount> pool_managers_head{};
|
|
||||||
std::array<Impl*, MaxManagerCount> pool_managers_tail{};
|
Core::System& m_system;
|
||||||
std::array<Impl, MaxManagerCount> managers;
|
const KMemoryLayout& m_memory_layout;
|
||||||
size_t num_managers{};
|
PoolArray<KLightLock> m_pool_locks;
|
||||||
|
std::array<Impl*, MaxManagerCount> m_pool_managers_head{};
|
||||||
|
std::array<Impl*, MaxManagerCount> m_pool_managers_tail{};
|
||||||
|
std::array<Impl, MaxManagerCount> m_managers;
|
||||||
|
size_t m_num_managers{};
|
||||||
|
PoolArray<u64> m_optimized_process_ids{};
|
||||||
|
PoolArray<bool> m_has_optimized_process{};
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Kernel
|
} // namespace Kernel
|
||||||
|
@ -142,32 +142,38 @@ private:
|
|||||||
|
|
||||||
} // namespace impl
|
} // namespace impl
|
||||||
|
|
||||||
constexpr auto KMemoryRegionType_None = impl::KMemoryRegionTypeValue();
|
constexpr inline auto KMemoryRegionType_None = impl::KMemoryRegionTypeValue();
|
||||||
constexpr auto KMemoryRegionType_Kernel = KMemoryRegionType_None.DeriveInitial(0, 2);
|
|
||||||
constexpr auto KMemoryRegionType_Dram = KMemoryRegionType_None.DeriveInitial(1, 2);
|
constexpr inline auto KMemoryRegionType_Kernel = KMemoryRegionType_None.DeriveInitial(0, 2);
|
||||||
|
constexpr inline auto KMemoryRegionType_Dram = KMemoryRegionType_None.DeriveInitial(1, 2);
|
||||||
static_assert(KMemoryRegionType_Kernel.GetValue() == 0x1);
|
static_assert(KMemoryRegionType_Kernel.GetValue() == 0x1);
|
||||||
static_assert(KMemoryRegionType_Dram.GetValue() == 0x2);
|
static_assert(KMemoryRegionType_Dram.GetValue() == 0x2);
|
||||||
|
|
||||||
constexpr auto KMemoryRegionType_DramKernelBase =
|
// constexpr inline auto KMemoryRegionType_CoreLocalRegion =
|
||||||
|
// KMemoryRegionType_None.DeriveInitial(2).Finalize();
|
||||||
|
// static_assert(KMemoryRegionType_CoreLocalRegion.GetValue() == 0x4);
|
||||||
|
|
||||||
|
constexpr inline auto KMemoryRegionType_DramKernelBase =
|
||||||
KMemoryRegionType_Dram.DeriveSparse(0, 3, 0)
|
KMemoryRegionType_Dram.DeriveSparse(0, 3, 0)
|
||||||
.SetAttribute(KMemoryRegionAttr_NoUserMap)
|
.SetAttribute(KMemoryRegionAttr_NoUserMap)
|
||||||
.SetAttribute(KMemoryRegionAttr_CarveoutProtected);
|
.SetAttribute(KMemoryRegionAttr_CarveoutProtected);
|
||||||
constexpr auto KMemoryRegionType_DramReservedBase = KMemoryRegionType_Dram.DeriveSparse(0, 3, 1);
|
constexpr inline auto KMemoryRegionType_DramReservedBase =
|
||||||
constexpr auto KMemoryRegionType_DramHeapBase =
|
KMemoryRegionType_Dram.DeriveSparse(0, 3, 1);
|
||||||
|
constexpr inline auto KMemoryRegionType_DramHeapBase =
|
||||||
KMemoryRegionType_Dram.DeriveSparse(0, 3, 2).SetAttribute(KMemoryRegionAttr_LinearMapped);
|
KMemoryRegionType_Dram.DeriveSparse(0, 3, 2).SetAttribute(KMemoryRegionAttr_LinearMapped);
|
||||||
static_assert(KMemoryRegionType_DramKernelBase.GetValue() ==
|
static_assert(KMemoryRegionType_DramKernelBase.GetValue() ==
|
||||||
(0xE | KMemoryRegionAttr_CarveoutProtected | KMemoryRegionAttr_NoUserMap));
|
(0xE | KMemoryRegionAttr_CarveoutProtected | KMemoryRegionAttr_NoUserMap));
|
||||||
static_assert(KMemoryRegionType_DramReservedBase.GetValue() == (0x16));
|
static_assert(KMemoryRegionType_DramReservedBase.GetValue() == (0x16));
|
||||||
static_assert(KMemoryRegionType_DramHeapBase.GetValue() == (0x26 | KMemoryRegionAttr_LinearMapped));
|
static_assert(KMemoryRegionType_DramHeapBase.GetValue() == (0x26 | KMemoryRegionAttr_LinearMapped));
|
||||||
|
|
||||||
constexpr auto KMemoryRegionType_DramKernelCode =
|
constexpr inline auto KMemoryRegionType_DramKernelCode =
|
||||||
KMemoryRegionType_DramKernelBase.DeriveSparse(0, 4, 0);
|
KMemoryRegionType_DramKernelBase.DeriveSparse(0, 4, 0);
|
||||||
constexpr auto KMemoryRegionType_DramKernelSlab =
|
constexpr inline auto KMemoryRegionType_DramKernelSlab =
|
||||||
KMemoryRegionType_DramKernelBase.DeriveSparse(0, 4, 1);
|
KMemoryRegionType_DramKernelBase.DeriveSparse(0, 4, 1);
|
||||||
constexpr auto KMemoryRegionType_DramKernelPtHeap =
|
constexpr inline auto KMemoryRegionType_DramKernelPtHeap =
|
||||||
KMemoryRegionType_DramKernelBase.DeriveSparse(0, 4, 2).SetAttribute(
|
KMemoryRegionType_DramKernelBase.DeriveSparse(0, 4, 2).SetAttribute(
|
||||||
KMemoryRegionAttr_LinearMapped);
|
KMemoryRegionAttr_LinearMapped);
|
||||||
constexpr auto KMemoryRegionType_DramKernelInitPt =
|
constexpr inline auto KMemoryRegionType_DramKernelInitPt =
|
||||||
KMemoryRegionType_DramKernelBase.DeriveSparse(0, 4, 3).SetAttribute(
|
KMemoryRegionType_DramKernelBase.DeriveSparse(0, 4, 3).SetAttribute(
|
||||||
KMemoryRegionAttr_LinearMapped);
|
KMemoryRegionAttr_LinearMapped);
|
||||||
static_assert(KMemoryRegionType_DramKernelCode.GetValue() ==
|
static_assert(KMemoryRegionType_DramKernelCode.GetValue() ==
|
||||||
@ -181,32 +187,40 @@ static_assert(KMemoryRegionType_DramKernelInitPt.GetValue() ==
|
|||||||
(0x44E | KMemoryRegionAttr_CarveoutProtected | KMemoryRegionAttr_NoUserMap |
|
(0x44E | KMemoryRegionAttr_CarveoutProtected | KMemoryRegionAttr_NoUserMap |
|
||||||
KMemoryRegionAttr_LinearMapped));
|
KMemoryRegionAttr_LinearMapped));
|
||||||
|
|
||||||
constexpr auto KMemoryRegionType_DramReservedEarly =
|
constexpr inline auto KMemoryRegionType_DramKernelSecureAppletMemory =
|
||||||
|
KMemoryRegionType_DramKernelBase.DeriveSparse(1, 3, 0).SetAttribute(
|
||||||
|
KMemoryRegionAttr_LinearMapped);
|
||||||
|
static_assert(KMemoryRegionType_DramKernelSecureAppletMemory.GetValue() ==
|
||||||
|
(0x18E | KMemoryRegionAttr_CarveoutProtected | KMemoryRegionAttr_NoUserMap |
|
||||||
|
KMemoryRegionAttr_LinearMapped));
|
||||||
|
|
||||||
|
constexpr inline auto KMemoryRegionType_DramReservedEarly =
|
||||||
KMemoryRegionType_DramReservedBase.DeriveAttribute(KMemoryRegionAttr_NoUserMap);
|
KMemoryRegionType_DramReservedBase.DeriveAttribute(KMemoryRegionAttr_NoUserMap);
|
||||||
static_assert(KMemoryRegionType_DramReservedEarly.GetValue() ==
|
static_assert(KMemoryRegionType_DramReservedEarly.GetValue() ==
|
||||||
(0x16 | KMemoryRegionAttr_NoUserMap));
|
(0x16 | KMemoryRegionAttr_NoUserMap));
|
||||||
|
|
||||||
constexpr auto KMemoryRegionType_KernelTraceBuffer =
|
constexpr inline auto KMemoryRegionType_KernelTraceBuffer =
|
||||||
KMemoryRegionType_DramReservedBase.DeriveSparse(0, 3, 0)
|
KMemoryRegionType_DramReservedBase.DeriveSparse(0, 3, 0)
|
||||||
.SetAttribute(KMemoryRegionAttr_LinearMapped)
|
.SetAttribute(KMemoryRegionAttr_LinearMapped)
|
||||||
.SetAttribute(KMemoryRegionAttr_UserReadOnly);
|
.SetAttribute(KMemoryRegionAttr_UserReadOnly);
|
||||||
constexpr auto KMemoryRegionType_OnMemoryBootImage =
|
constexpr inline auto KMemoryRegionType_OnMemoryBootImage =
|
||||||
KMemoryRegionType_DramReservedBase.DeriveSparse(0, 3, 1);
|
KMemoryRegionType_DramReservedBase.DeriveSparse(0, 3, 1);
|
||||||
constexpr auto KMemoryRegionType_DTB = KMemoryRegionType_DramReservedBase.DeriveSparse(0, 3, 2);
|
constexpr inline auto KMemoryRegionType_DTB =
|
||||||
|
KMemoryRegionType_DramReservedBase.DeriveSparse(0, 3, 2);
|
||||||
static_assert(KMemoryRegionType_KernelTraceBuffer.GetValue() ==
|
static_assert(KMemoryRegionType_KernelTraceBuffer.GetValue() ==
|
||||||
(0xD6 | KMemoryRegionAttr_LinearMapped | KMemoryRegionAttr_UserReadOnly));
|
(0xD6 | KMemoryRegionAttr_LinearMapped | KMemoryRegionAttr_UserReadOnly));
|
||||||
static_assert(KMemoryRegionType_OnMemoryBootImage.GetValue() == 0x156);
|
static_assert(KMemoryRegionType_OnMemoryBootImage.GetValue() == 0x156);
|
||||||
static_assert(KMemoryRegionType_DTB.GetValue() == 0x256);
|
static_assert(KMemoryRegionType_DTB.GetValue() == 0x256);
|
||||||
|
|
||||||
constexpr auto KMemoryRegionType_DramPoolPartition =
|
constexpr inline auto KMemoryRegionType_DramPoolPartition =
|
||||||
KMemoryRegionType_DramHeapBase.DeriveAttribute(KMemoryRegionAttr_NoUserMap);
|
KMemoryRegionType_DramHeapBase.DeriveAttribute(KMemoryRegionAttr_NoUserMap);
|
||||||
static_assert(KMemoryRegionType_DramPoolPartition.GetValue() ==
|
static_assert(KMemoryRegionType_DramPoolPartition.GetValue() ==
|
||||||
(0x26 | KMemoryRegionAttr_LinearMapped | KMemoryRegionAttr_NoUserMap));
|
(0x26 | KMemoryRegionAttr_LinearMapped | KMemoryRegionAttr_NoUserMap));
|
||||||
|
|
||||||
constexpr auto KMemoryRegionType_DramPoolManagement =
|
constexpr inline auto KMemoryRegionType_DramPoolManagement =
|
||||||
KMemoryRegionType_DramPoolPartition.DeriveTransition(0, 2).DeriveTransition().SetAttribute(
|
KMemoryRegionType_DramPoolPartition.DeriveTransition(0, 2).DeriveTransition().SetAttribute(
|
||||||
KMemoryRegionAttr_CarveoutProtected);
|
KMemoryRegionAttr_CarveoutProtected);
|
||||||
constexpr auto KMemoryRegionType_DramUserPool =
|
constexpr inline auto KMemoryRegionType_DramUserPool =
|
||||||
KMemoryRegionType_DramPoolPartition.DeriveTransition(1, 2).DeriveTransition();
|
KMemoryRegionType_DramPoolPartition.DeriveTransition(1, 2).DeriveTransition();
|
||||||
static_assert(KMemoryRegionType_DramPoolManagement.GetValue() ==
|
static_assert(KMemoryRegionType_DramPoolManagement.GetValue() ==
|
||||||
(0x166 | KMemoryRegionAttr_LinearMapped | KMemoryRegionAttr_NoUserMap |
|
(0x166 | KMemoryRegionAttr_LinearMapped | KMemoryRegionAttr_NoUserMap |
|
||||||
@ -214,11 +228,13 @@ static_assert(KMemoryRegionType_DramPoolManagement.GetValue() ==
|
|||||||
static_assert(KMemoryRegionType_DramUserPool.GetValue() ==
|
static_assert(KMemoryRegionType_DramUserPool.GetValue() ==
|
||||||
(0x1A6 | KMemoryRegionAttr_LinearMapped | KMemoryRegionAttr_NoUserMap));
|
(0x1A6 | KMemoryRegionAttr_LinearMapped | KMemoryRegionAttr_NoUserMap));
|
||||||
|
|
||||||
constexpr auto KMemoryRegionType_DramApplicationPool = KMemoryRegionType_DramUserPool.Derive(4, 0);
|
constexpr inline auto KMemoryRegionType_DramApplicationPool =
|
||||||
constexpr auto KMemoryRegionType_DramAppletPool = KMemoryRegionType_DramUserPool.Derive(4, 1);
|
KMemoryRegionType_DramUserPool.Derive(4, 0);
|
||||||
constexpr auto KMemoryRegionType_DramSystemNonSecurePool =
|
constexpr inline auto KMemoryRegionType_DramAppletPool =
|
||||||
|
KMemoryRegionType_DramUserPool.Derive(4, 1);
|
||||||
|
constexpr inline auto KMemoryRegionType_DramSystemNonSecurePool =
|
||||||
KMemoryRegionType_DramUserPool.Derive(4, 2);
|
KMemoryRegionType_DramUserPool.Derive(4, 2);
|
||||||
constexpr auto KMemoryRegionType_DramSystemPool =
|
constexpr inline auto KMemoryRegionType_DramSystemPool =
|
||||||
KMemoryRegionType_DramUserPool.Derive(4, 3).SetAttribute(KMemoryRegionAttr_CarveoutProtected);
|
KMemoryRegionType_DramUserPool.Derive(4, 3).SetAttribute(KMemoryRegionAttr_CarveoutProtected);
|
||||||
static_assert(KMemoryRegionType_DramApplicationPool.GetValue() ==
|
static_assert(KMemoryRegionType_DramApplicationPool.GetValue() ==
|
||||||
(0x7A6 | KMemoryRegionAttr_LinearMapped | KMemoryRegionAttr_NoUserMap));
|
(0x7A6 | KMemoryRegionAttr_LinearMapped | KMemoryRegionAttr_NoUserMap));
|
||||||
@ -230,50 +246,55 @@ static_assert(KMemoryRegionType_DramSystemPool.GetValue() ==
|
|||||||
(0x13A6 | KMemoryRegionAttr_LinearMapped | KMemoryRegionAttr_NoUserMap |
|
(0x13A6 | KMemoryRegionAttr_LinearMapped | KMemoryRegionAttr_NoUserMap |
|
||||||
KMemoryRegionAttr_CarveoutProtected));
|
KMemoryRegionAttr_CarveoutProtected));
|
||||||
|
|
||||||
constexpr auto KMemoryRegionType_VirtualDramHeapBase = KMemoryRegionType_Dram.DeriveSparse(1, 3, 0);
|
constexpr inline auto KMemoryRegionType_VirtualDramHeapBase =
|
||||||
constexpr auto KMemoryRegionType_VirtualDramKernelPtHeap =
|
KMemoryRegionType_Dram.DeriveSparse(1, 3, 0);
|
||||||
|
constexpr inline auto KMemoryRegionType_VirtualDramKernelPtHeap =
|
||||||
KMemoryRegionType_Dram.DeriveSparse(1, 3, 1);
|
KMemoryRegionType_Dram.DeriveSparse(1, 3, 1);
|
||||||
constexpr auto KMemoryRegionType_VirtualDramKernelTraceBuffer =
|
constexpr inline auto KMemoryRegionType_VirtualDramKernelTraceBuffer =
|
||||||
KMemoryRegionType_Dram.DeriveSparse(1, 3, 2);
|
KMemoryRegionType_Dram.DeriveSparse(1, 3, 2);
|
||||||
static_assert(KMemoryRegionType_VirtualDramHeapBase.GetValue() == 0x1A);
|
static_assert(KMemoryRegionType_VirtualDramHeapBase.GetValue() == 0x1A);
|
||||||
static_assert(KMemoryRegionType_VirtualDramKernelPtHeap.GetValue() == 0x2A);
|
static_assert(KMemoryRegionType_VirtualDramKernelPtHeap.GetValue() == 0x2A);
|
||||||
static_assert(KMemoryRegionType_VirtualDramKernelTraceBuffer.GetValue() == 0x4A);
|
static_assert(KMemoryRegionType_VirtualDramKernelTraceBuffer.GetValue() == 0x4A);
|
||||||
|
|
||||||
// UNUSED: .DeriveSparse(2, 2, 0);
|
// UNUSED: .DeriveSparse(2, 2, 0);
|
||||||
constexpr auto KMemoryRegionType_VirtualDramUnknownDebug =
|
constexpr inline auto KMemoryRegionType_VirtualDramUnknownDebug =
|
||||||
KMemoryRegionType_Dram.DeriveSparse(2, 2, 1);
|
KMemoryRegionType_Dram.DeriveSparse(2, 2, 1);
|
||||||
static_assert(KMemoryRegionType_VirtualDramUnknownDebug.GetValue() == (0x52));
|
static_assert(KMemoryRegionType_VirtualDramUnknownDebug.GetValue() == (0x52));
|
||||||
|
|
||||||
constexpr auto KMemoryRegionType_VirtualDramKernelInitPt =
|
constexpr inline auto KMemoryRegionType_VirtualDramKernelSecureAppletMemory =
|
||||||
|
KMemoryRegionType_Dram.DeriveSparse(3, 1, 0);
|
||||||
|
static_assert(KMemoryRegionType_VirtualDramKernelSecureAppletMemory.GetValue() == (0x62));
|
||||||
|
|
||||||
|
constexpr inline auto KMemoryRegionType_VirtualDramKernelInitPt =
|
||||||
KMemoryRegionType_VirtualDramHeapBase.Derive(3, 0);
|
KMemoryRegionType_VirtualDramHeapBase.Derive(3, 0);
|
||||||
constexpr auto KMemoryRegionType_VirtualDramPoolManagement =
|
constexpr inline auto KMemoryRegionType_VirtualDramPoolManagement =
|
||||||
KMemoryRegionType_VirtualDramHeapBase.Derive(3, 1);
|
KMemoryRegionType_VirtualDramHeapBase.Derive(3, 1);
|
||||||
constexpr auto KMemoryRegionType_VirtualDramUserPool =
|
constexpr inline auto KMemoryRegionType_VirtualDramUserPool =
|
||||||
KMemoryRegionType_VirtualDramHeapBase.Derive(3, 2);
|
KMemoryRegionType_VirtualDramHeapBase.Derive(3, 2);
|
||||||
static_assert(KMemoryRegionType_VirtualDramKernelInitPt.GetValue() == 0x19A);
|
static_assert(KMemoryRegionType_VirtualDramKernelInitPt.GetValue() == 0x19A);
|
||||||
static_assert(KMemoryRegionType_VirtualDramPoolManagement.GetValue() == 0x29A);
|
static_assert(KMemoryRegionType_VirtualDramPoolManagement.GetValue() == 0x29A);
|
||||||
static_assert(KMemoryRegionType_VirtualDramUserPool.GetValue() == 0x31A);
|
static_assert(KMemoryRegionType_VirtualDramUserPool.GetValue() == 0x31A);
|
||||||
|
|
||||||
// NOTE: For unknown reason, the pools are derived out-of-order here. It's worth eventually trying
|
// NOTE: For unknown reason, the pools are derived out-of-order here.
|
||||||
// to understand why Nintendo made this choice.
|
// It's worth eventually trying to understand why Nintendo made this choice.
|
||||||
// UNUSED: .Derive(6, 0);
|
// UNUSED: .Derive(6, 0);
|
||||||
// UNUSED: .Derive(6, 1);
|
// UNUSED: .Derive(6, 1);
|
||||||
constexpr auto KMemoryRegionType_VirtualDramAppletPool =
|
constexpr inline auto KMemoryRegionType_VirtualDramAppletPool =
|
||||||
KMemoryRegionType_VirtualDramUserPool.Derive(6, 2);
|
KMemoryRegionType_VirtualDramUserPool.Derive(6, 2);
|
||||||
constexpr auto KMemoryRegionType_VirtualDramApplicationPool =
|
constexpr inline auto KMemoryRegionType_VirtualDramApplicationPool =
|
||||||
KMemoryRegionType_VirtualDramUserPool.Derive(6, 3);
|
KMemoryRegionType_VirtualDramUserPool.Derive(6, 3);
|
||||||
constexpr auto KMemoryRegionType_VirtualDramSystemNonSecurePool =
|
constexpr inline auto KMemoryRegionType_VirtualDramSystemNonSecurePool =
|
||||||
KMemoryRegionType_VirtualDramUserPool.Derive(6, 4);
|
KMemoryRegionType_VirtualDramUserPool.Derive(6, 4);
|
||||||
constexpr auto KMemoryRegionType_VirtualDramSystemPool =
|
constexpr inline auto KMemoryRegionType_VirtualDramSystemPool =
|
||||||
KMemoryRegionType_VirtualDramUserPool.Derive(6, 5);
|
KMemoryRegionType_VirtualDramUserPool.Derive(6, 5);
|
||||||
static_assert(KMemoryRegionType_VirtualDramAppletPool.GetValue() == 0x1B1A);
|
static_assert(KMemoryRegionType_VirtualDramAppletPool.GetValue() == 0x1B1A);
|
||||||
static_assert(KMemoryRegionType_VirtualDramApplicationPool.GetValue() == 0x271A);
|
static_assert(KMemoryRegionType_VirtualDramApplicationPool.GetValue() == 0x271A);
|
||||||
static_assert(KMemoryRegionType_VirtualDramSystemNonSecurePool.GetValue() == 0x2B1A);
|
static_assert(KMemoryRegionType_VirtualDramSystemNonSecurePool.GetValue() == 0x2B1A);
|
||||||
static_assert(KMemoryRegionType_VirtualDramSystemPool.GetValue() == 0x331A);
|
static_assert(KMemoryRegionType_VirtualDramSystemPool.GetValue() == 0x331A);
|
||||||
|
|
||||||
constexpr auto KMemoryRegionType_ArchDeviceBase =
|
constexpr inline auto KMemoryRegionType_ArchDeviceBase =
|
||||||
KMemoryRegionType_Kernel.DeriveTransition(0, 1).SetSparseOnly();
|
KMemoryRegionType_Kernel.DeriveTransition(0, 1).SetSparseOnly();
|
||||||
constexpr auto KMemoryRegionType_BoardDeviceBase =
|
constexpr inline auto KMemoryRegionType_BoardDeviceBase =
|
||||||
KMemoryRegionType_Kernel.DeriveTransition(0, 2).SetDenseOnly();
|
KMemoryRegionType_Kernel.DeriveTransition(0, 2).SetDenseOnly();
|
||||||
static_assert(KMemoryRegionType_ArchDeviceBase.GetValue() == 0x5);
|
static_assert(KMemoryRegionType_ArchDeviceBase.GetValue() == 0x5);
|
||||||
static_assert(KMemoryRegionType_BoardDeviceBase.GetValue() == 0x5);
|
static_assert(KMemoryRegionType_BoardDeviceBase.GetValue() == 0x5);
|
||||||
@ -284,7 +305,7 @@ static_assert(KMemoryRegionType_BoardDeviceBase.GetValue() == 0x5);
|
|||||||
#error "Unimplemented"
|
#error "Unimplemented"
|
||||||
#else
|
#else
|
||||||
// Default to no architecture devices.
|
// Default to no architecture devices.
|
||||||
constexpr auto NumArchitectureDeviceRegions = 0;
|
constexpr inline auto NumArchitectureDeviceRegions = 0;
|
||||||
#endif
|
#endif
|
||||||
static_assert(NumArchitectureDeviceRegions >= 0);
|
static_assert(NumArchitectureDeviceRegions >= 0);
|
||||||
|
|
||||||
@ -292,34 +313,35 @@ static_assert(NumArchitectureDeviceRegions >= 0);
|
|||||||
#include "core/hle/kernel/board/nintendo/nx/k_memory_region_device_types.inc"
|
#include "core/hle/kernel/board/nintendo/nx/k_memory_region_device_types.inc"
|
||||||
#else
|
#else
|
||||||
// Default to no board devices.
|
// Default to no board devices.
|
||||||
constexpr auto NumBoardDeviceRegions = 0;
|
constexpr inline auto NumBoardDeviceRegions = 0;
|
||||||
#endif
|
#endif
|
||||||
static_assert(NumBoardDeviceRegions >= 0);
|
static_assert(NumBoardDeviceRegions >= 0);
|
||||||
|
|
||||||
constexpr auto KMemoryRegionType_KernelCode = KMemoryRegionType_Kernel.DeriveSparse(1, 4, 0);
|
constexpr inline auto KMemoryRegionType_KernelCode = KMemoryRegionType_Kernel.DeriveSparse(1, 4, 0);
|
||||||
constexpr auto KMemoryRegionType_KernelStack = KMemoryRegionType_Kernel.DeriveSparse(1, 4, 1);
|
constexpr inline auto KMemoryRegionType_KernelStack =
|
||||||
constexpr auto KMemoryRegionType_KernelMisc = KMemoryRegionType_Kernel.DeriveSparse(1, 4, 2);
|
KMemoryRegionType_Kernel.DeriveSparse(1, 4, 1);
|
||||||
constexpr auto KMemoryRegionType_KernelSlab = KMemoryRegionType_Kernel.DeriveSparse(1, 4, 3);
|
constexpr inline auto KMemoryRegionType_KernelMisc = KMemoryRegionType_Kernel.DeriveSparse(1, 4, 2);
|
||||||
|
constexpr inline auto KMemoryRegionType_KernelSlab = KMemoryRegionType_Kernel.DeriveSparse(1, 4, 3);
|
||||||
static_assert(KMemoryRegionType_KernelCode.GetValue() == 0x19);
|
static_assert(KMemoryRegionType_KernelCode.GetValue() == 0x19);
|
||||||
static_assert(KMemoryRegionType_KernelStack.GetValue() == 0x29);
|
static_assert(KMemoryRegionType_KernelStack.GetValue() == 0x29);
|
||||||
static_assert(KMemoryRegionType_KernelMisc.GetValue() == 0x49);
|
static_assert(KMemoryRegionType_KernelMisc.GetValue() == 0x49);
|
||||||
static_assert(KMemoryRegionType_KernelSlab.GetValue() == 0x89);
|
static_assert(KMemoryRegionType_KernelSlab.GetValue() == 0x89);
|
||||||
|
|
||||||
constexpr auto KMemoryRegionType_KernelMiscDerivedBase =
|
constexpr inline auto KMemoryRegionType_KernelMiscDerivedBase =
|
||||||
KMemoryRegionType_KernelMisc.DeriveTransition();
|
KMemoryRegionType_KernelMisc.DeriveTransition();
|
||||||
static_assert(KMemoryRegionType_KernelMiscDerivedBase.GetValue() == 0x149);
|
static_assert(KMemoryRegionType_KernelMiscDerivedBase.GetValue() == 0x149);
|
||||||
|
|
||||||
// UNUSED: .Derive(7, 0);
|
// UNUSED: .Derive(7, 0);
|
||||||
constexpr auto KMemoryRegionType_KernelMiscMainStack =
|
constexpr inline auto KMemoryRegionType_KernelMiscMainStack =
|
||||||
KMemoryRegionType_KernelMiscDerivedBase.Derive(7, 1);
|
KMemoryRegionType_KernelMiscDerivedBase.Derive(7, 1);
|
||||||
constexpr auto KMemoryRegionType_KernelMiscMappedDevice =
|
constexpr inline auto KMemoryRegionType_KernelMiscMappedDevice =
|
||||||
KMemoryRegionType_KernelMiscDerivedBase.Derive(7, 2);
|
KMemoryRegionType_KernelMiscDerivedBase.Derive(7, 2);
|
||||||
constexpr auto KMemoryRegionType_KernelMiscExceptionStack =
|
constexpr inline auto KMemoryRegionType_KernelMiscExceptionStack =
|
||||||
KMemoryRegionType_KernelMiscDerivedBase.Derive(7, 3);
|
KMemoryRegionType_KernelMiscDerivedBase.Derive(7, 3);
|
||||||
constexpr auto KMemoryRegionType_KernelMiscUnknownDebug =
|
constexpr inline auto KMemoryRegionType_KernelMiscUnknownDebug =
|
||||||
KMemoryRegionType_KernelMiscDerivedBase.Derive(7, 4);
|
KMemoryRegionType_KernelMiscDerivedBase.Derive(7, 4);
|
||||||
// UNUSED: .Derive(7, 5);
|
// UNUSED: .Derive(7, 5);
|
||||||
constexpr auto KMemoryRegionType_KernelMiscIdleStack =
|
constexpr inline auto KMemoryRegionType_KernelMiscIdleStack =
|
||||||
KMemoryRegionType_KernelMiscDerivedBase.Derive(7, 6);
|
KMemoryRegionType_KernelMiscDerivedBase.Derive(7, 6);
|
||||||
static_assert(KMemoryRegionType_KernelMiscMainStack.GetValue() == 0xB49);
|
static_assert(KMemoryRegionType_KernelMiscMainStack.GetValue() == 0xB49);
|
||||||
static_assert(KMemoryRegionType_KernelMiscMappedDevice.GetValue() == 0xD49);
|
static_assert(KMemoryRegionType_KernelMiscMappedDevice.GetValue() == 0xD49);
|
||||||
@ -327,7 +349,8 @@ static_assert(KMemoryRegionType_KernelMiscExceptionStack.GetValue() == 0x1349);
|
|||||||
static_assert(KMemoryRegionType_KernelMiscUnknownDebug.GetValue() == 0x1549);
|
static_assert(KMemoryRegionType_KernelMiscUnknownDebug.GetValue() == 0x1549);
|
||||||
static_assert(KMemoryRegionType_KernelMiscIdleStack.GetValue() == 0x2349);
|
static_assert(KMemoryRegionType_KernelMiscIdleStack.GetValue() == 0x2349);
|
||||||
|
|
||||||
constexpr auto KMemoryRegionType_KernelTemp = KMemoryRegionType_Kernel.Advance(2).Derive(2, 0);
|
constexpr inline auto KMemoryRegionType_KernelTemp =
|
||||||
|
KMemoryRegionType_Kernel.Advance(2).Derive(2, 0);
|
||||||
static_assert(KMemoryRegionType_KernelTemp.GetValue() == 0x31);
|
static_assert(KMemoryRegionType_KernelTemp.GetValue() == 0x31);
|
||||||
|
|
||||||
constexpr KMemoryRegionType GetTypeForVirtualLinearMapping(u32 type_id) {
|
constexpr KMemoryRegionType GetTypeForVirtualLinearMapping(u32 type_id) {
|
||||||
@ -335,6 +358,8 @@ constexpr KMemoryRegionType GetTypeForVirtualLinearMapping(u32 type_id) {
|
|||||||
return KMemoryRegionType_VirtualDramKernelTraceBuffer;
|
return KMemoryRegionType_VirtualDramKernelTraceBuffer;
|
||||||
} else if (KMemoryRegionType_DramKernelPtHeap.IsAncestorOf(type_id)) {
|
} else if (KMemoryRegionType_DramKernelPtHeap.IsAncestorOf(type_id)) {
|
||||||
return KMemoryRegionType_VirtualDramKernelPtHeap;
|
return KMemoryRegionType_VirtualDramKernelPtHeap;
|
||||||
|
} else if (KMemoryRegionType_DramKernelSecureAppletMemory.IsAncestorOf(type_id)) {
|
||||||
|
return KMemoryRegionType_VirtualDramKernelSecureAppletMemory;
|
||||||
} else if ((type_id | KMemoryRegionAttr_ShouldKernelMap) == type_id) {
|
} else if ((type_id | KMemoryRegionAttr_ShouldKernelMap) == type_id) {
|
||||||
return KMemoryRegionType_VirtualDramUnknownDebug;
|
return KMemoryRegionType_VirtualDramUnknownDebug;
|
||||||
} else {
|
} else {
|
||||||
|
@ -16,107 +16,126 @@
|
|||||||
namespace Kernel {
|
namespace Kernel {
|
||||||
|
|
||||||
class KPageBitmap {
|
class KPageBitmap {
|
||||||
private:
|
public:
|
||||||
class RandomBitGenerator {
|
class RandomBitGenerator {
|
||||||
private:
|
|
||||||
Common::TinyMT rng{};
|
|
||||||
u32 entropy{};
|
|
||||||
u32 bits_available{};
|
|
||||||
|
|
||||||
private:
|
|
||||||
void RefreshEntropy() {
|
|
||||||
entropy = rng.GenerateRandomU32();
|
|
||||||
bits_available = static_cast<u32>(Common::BitSize<decltype(entropy)>());
|
|
||||||
}
|
|
||||||
|
|
||||||
bool GenerateRandomBit() {
|
|
||||||
if (bits_available == 0) {
|
|
||||||
this->RefreshEntropy();
|
|
||||||
}
|
|
||||||
|
|
||||||
const bool rnd_bit = (entropy & 1) != 0;
|
|
||||||
entropy >>= 1;
|
|
||||||
--bits_available;
|
|
||||||
return rnd_bit;
|
|
||||||
}
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
RandomBitGenerator() {
|
RandomBitGenerator() {
|
||||||
rng.Initialize(static_cast<u32>(KSystemControl::GenerateRandomU64()));
|
m_rng.Initialize(static_cast<u32>(KSystemControl::GenerateRandomU64()));
|
||||||
}
|
}
|
||||||
|
|
||||||
std::size_t SelectRandomBit(u64 bitmap) {
|
u64 SelectRandomBit(u64 bitmap) {
|
||||||
u64 selected = 0;
|
u64 selected = 0;
|
||||||
|
|
||||||
u64 cur_num_bits = Common::BitSize<decltype(bitmap)>() / 2;
|
for (size_t cur_num_bits = Common::BitSize<decltype(bitmap)>() / 2; cur_num_bits != 0;
|
||||||
u64 cur_mask = (1ULL << cur_num_bits) - 1;
|
cur_num_bits /= 2) {
|
||||||
|
const u64 high = (bitmap >> cur_num_bits);
|
||||||
|
const u64 low = (bitmap & (~(UINT64_C(0xFFFFFFFFFFFFFFFF) << cur_num_bits)));
|
||||||
|
|
||||||
while (cur_num_bits) {
|
// Choose high if we have high and (don't have low or select high randomly).
|
||||||
const u64 low = (bitmap >> 0) & cur_mask;
|
if (high && (low == 0 || this->GenerateRandomBit())) {
|
||||||
const u64 high = (bitmap >> cur_num_bits) & cur_mask;
|
|
||||||
|
|
||||||
bool choose_low;
|
|
||||||
if (high == 0) {
|
|
||||||
// If only low val is set, choose low.
|
|
||||||
choose_low = true;
|
|
||||||
} else if (low == 0) {
|
|
||||||
// If only high val is set, choose high.
|
|
||||||
choose_low = false;
|
|
||||||
} else {
|
|
||||||
// If both are set, choose random.
|
|
||||||
choose_low = this->GenerateRandomBit();
|
|
||||||
}
|
|
||||||
|
|
||||||
// If we chose low, proceed with low.
|
|
||||||
if (choose_low) {
|
|
||||||
bitmap = low;
|
|
||||||
selected += 0;
|
|
||||||
} else {
|
|
||||||
bitmap = high;
|
bitmap = high;
|
||||||
selected += cur_num_bits;
|
selected += cur_num_bits;
|
||||||
|
} else {
|
||||||
|
bitmap = low;
|
||||||
|
selected += 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Proceed.
|
|
||||||
cur_num_bits /= 2;
|
|
||||||
cur_mask >>= cur_num_bits;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return selected;
|
return selected;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
u64 GenerateRandom(u64 max) {
|
||||||
|
// Determine the number of bits we need.
|
||||||
|
const u64 bits_needed = 1 + (Common::BitSize<decltype(max)>() - std::countl_zero(max));
|
||||||
|
|
||||||
|
// Generate a random value of the desired bitwidth.
|
||||||
|
const u64 rnd = this->GenerateRandomBits(static_cast<u32>(bits_needed));
|
||||||
|
|
||||||
|
// Adjust the value to be in range.
|
||||||
|
return rnd - ((rnd / max) * max);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
void RefreshEntropy() {
|
||||||
|
m_entropy = m_rng.GenerateRandomU32();
|
||||||
|
m_bits_available = static_cast<u32>(Common::BitSize<decltype(m_entropy)>());
|
||||||
|
}
|
||||||
|
|
||||||
|
bool GenerateRandomBit() {
|
||||||
|
if (m_bits_available == 0) {
|
||||||
|
this->RefreshEntropy();
|
||||||
|
}
|
||||||
|
|
||||||
|
const bool rnd_bit = (m_entropy & 1) != 0;
|
||||||
|
m_entropy >>= 1;
|
||||||
|
--m_bits_available;
|
||||||
|
return rnd_bit;
|
||||||
|
}
|
||||||
|
|
||||||
|
u64 GenerateRandomBits(u32 num_bits) {
|
||||||
|
u64 result = 0;
|
||||||
|
|
||||||
|
// Iteratively add random bits to our result.
|
||||||
|
while (num_bits > 0) {
|
||||||
|
// Ensure we have random bits to take from.
|
||||||
|
if (m_bits_available == 0) {
|
||||||
|
this->RefreshEntropy();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Determine how many bits to take this round.
|
||||||
|
const auto cur_bits = std::min(num_bits, m_bits_available);
|
||||||
|
|
||||||
|
// Generate mask for our current bits.
|
||||||
|
const u64 mask = (static_cast<u64>(1) << cur_bits) - 1;
|
||||||
|
|
||||||
|
// Add bits to output from our entropy.
|
||||||
|
result <<= cur_bits;
|
||||||
|
result |= (m_entropy & mask);
|
||||||
|
|
||||||
|
// Remove bits from our entropy.
|
||||||
|
m_entropy >>= cur_bits;
|
||||||
|
m_bits_available -= cur_bits;
|
||||||
|
|
||||||
|
// Advance.
|
||||||
|
num_bits -= cur_bits;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
Common::TinyMT m_rng;
|
||||||
|
u32 m_entropy{};
|
||||||
|
u32 m_bits_available{};
|
||||||
};
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
static constexpr std::size_t MaxDepth = 4;
|
static constexpr size_t MaxDepth = 4;
|
||||||
|
|
||||||
private:
|
|
||||||
std::array<u64*, MaxDepth> bit_storages{};
|
|
||||||
RandomBitGenerator rng{};
|
|
||||||
std::size_t num_bits{};
|
|
||||||
std::size_t used_depths{};
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
KPageBitmap() = default;
|
KPageBitmap() = default;
|
||||||
|
|
||||||
constexpr std::size_t GetNumBits() const {
|
constexpr size_t GetNumBits() const {
|
||||||
return num_bits;
|
return m_num_bits;
|
||||||
}
|
}
|
||||||
constexpr s32 GetHighestDepthIndex() const {
|
constexpr s32 GetHighestDepthIndex() const {
|
||||||
return static_cast<s32>(used_depths) - 1;
|
return static_cast<s32>(m_used_depths) - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
u64* Initialize(u64* storage, std::size_t size) {
|
u64* Initialize(u64* storage, size_t size) {
|
||||||
// Initially, everything is un-set.
|
// Initially, everything is un-set.
|
||||||
num_bits = 0;
|
m_num_bits = 0;
|
||||||
|
|
||||||
// Calculate the needed bitmap depth.
|
// Calculate the needed bitmap depth.
|
||||||
used_depths = static_cast<std::size_t>(GetRequiredDepth(size));
|
m_used_depths = static_cast<size_t>(GetRequiredDepth(size));
|
||||||
ASSERT(used_depths <= MaxDepth);
|
ASSERT(m_used_depths <= MaxDepth);
|
||||||
|
|
||||||
// Set the bitmap pointers.
|
// Set the bitmap pointers.
|
||||||
for (s32 depth = this->GetHighestDepthIndex(); depth >= 0; depth--) {
|
for (s32 depth = this->GetHighestDepthIndex(); depth >= 0; depth--) {
|
||||||
bit_storages[depth] = storage;
|
m_bit_storages[depth] = storage;
|
||||||
size = Common::AlignUp(size, Common::BitSize<u64>()) / Common::BitSize<u64>();
|
size = Common::AlignUp(size, Common::BitSize<u64>()) / Common::BitSize<u64>();
|
||||||
storage += size;
|
storage += size;
|
||||||
|
m_end_storages[depth] = storage;
|
||||||
}
|
}
|
||||||
|
|
||||||
return storage;
|
return storage;
|
||||||
@ -128,19 +147,19 @@ public:
|
|||||||
|
|
||||||
if (random) {
|
if (random) {
|
||||||
do {
|
do {
|
||||||
const u64 v = bit_storages[depth][offset];
|
const u64 v = m_bit_storages[depth][offset];
|
||||||
if (v == 0) {
|
if (v == 0) {
|
||||||
// If depth is bigger than zero, then a previous level indicated a block was
|
// If depth is bigger than zero, then a previous level indicated a block was
|
||||||
// free.
|
// free.
|
||||||
ASSERT(depth == 0);
|
ASSERT(depth == 0);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
offset = offset * Common::BitSize<u64>() + rng.SelectRandomBit(v);
|
offset = offset * Common::BitSize<u64>() + m_rng.SelectRandomBit(v);
|
||||||
++depth;
|
++depth;
|
||||||
} while (depth < static_cast<s32>(used_depths));
|
} while (depth < static_cast<s32>(m_used_depths));
|
||||||
} else {
|
} else {
|
||||||
do {
|
do {
|
||||||
const u64 v = bit_storages[depth][offset];
|
const u64 v = m_bit_storages[depth][offset];
|
||||||
if (v == 0) {
|
if (v == 0) {
|
||||||
// If depth is bigger than zero, then a previous level indicated a block was
|
// If depth is bigger than zero, then a previous level indicated a block was
|
||||||
// free.
|
// free.
|
||||||
@ -149,28 +168,69 @@ public:
|
|||||||
}
|
}
|
||||||
offset = offset * Common::BitSize<u64>() + std::countr_zero(v);
|
offset = offset * Common::BitSize<u64>() + std::countr_zero(v);
|
||||||
++depth;
|
++depth;
|
||||||
} while (depth < static_cast<s32>(used_depths));
|
} while (depth < static_cast<s32>(m_used_depths));
|
||||||
}
|
}
|
||||||
|
|
||||||
return static_cast<s64>(offset);
|
return static_cast<s64>(offset);
|
||||||
}
|
}
|
||||||
|
|
||||||
void SetBit(std::size_t offset) {
|
s64 FindFreeRange(size_t count) {
|
||||||
|
// Check that it is possible to find a range.
|
||||||
|
const u64* const storage_start = m_bit_storages[m_used_depths - 1];
|
||||||
|
const u64* const storage_end = m_end_storages[m_used_depths - 1];
|
||||||
|
|
||||||
|
// If we don't have a storage to iterate (or want more blocks than fit in a single storage),
|
||||||
|
// we can't find a free range.
|
||||||
|
if (!(storage_start < storage_end && count <= Common::BitSize<u64>())) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Walk the storages to select a random free range.
|
||||||
|
const size_t options_per_storage = std::max<size_t>(Common::BitSize<u64>() / count, 1);
|
||||||
|
const size_t num_entries = std::max<size_t>(storage_end - storage_start, 1);
|
||||||
|
|
||||||
|
const u64 free_mask = (static_cast<u64>(1) << count) - 1;
|
||||||
|
|
||||||
|
size_t num_valid_options = 0;
|
||||||
|
s64 chosen_offset = -1;
|
||||||
|
for (size_t storage_index = 0; storage_index < num_entries; ++storage_index) {
|
||||||
|
u64 storage = storage_start[storage_index];
|
||||||
|
for (size_t option = 0; option < options_per_storage; ++option) {
|
||||||
|
if ((storage & free_mask) == free_mask) {
|
||||||
|
// We've found a new valid option.
|
||||||
|
++num_valid_options;
|
||||||
|
|
||||||
|
// Select the Kth valid option with probability 1/K. This leads to an overall
|
||||||
|
// uniform distribution.
|
||||||
|
if (num_valid_options == 1 || m_rng.GenerateRandom(num_valid_options) == 0) {
|
||||||
|
// This is our first option, so select it.
|
||||||
|
chosen_offset = storage_index * Common::BitSize<u64>() + option * count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
storage >>= count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return the random offset we chose.*/
|
||||||
|
return chosen_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetBit(size_t offset) {
|
||||||
this->SetBit(this->GetHighestDepthIndex(), offset);
|
this->SetBit(this->GetHighestDepthIndex(), offset);
|
||||||
num_bits++;
|
m_num_bits++;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ClearBit(std::size_t offset) {
|
void ClearBit(size_t offset) {
|
||||||
this->ClearBit(this->GetHighestDepthIndex(), offset);
|
this->ClearBit(this->GetHighestDepthIndex(), offset);
|
||||||
num_bits--;
|
m_num_bits--;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ClearRange(std::size_t offset, std::size_t count) {
|
bool ClearRange(size_t offset, size_t count) {
|
||||||
s32 depth = this->GetHighestDepthIndex();
|
s32 depth = this->GetHighestDepthIndex();
|
||||||
u64* bits = bit_storages[depth];
|
u64* bits = m_bit_storages[depth];
|
||||||
std::size_t bit_ind = offset / Common::BitSize<u64>();
|
size_t bit_ind = offset / Common::BitSize<u64>();
|
||||||
if (count < Common::BitSize<u64>()) {
|
if (count < Common::BitSize<u64>()) [[likely]] {
|
||||||
const std::size_t shift = offset % Common::BitSize<u64>();
|
const size_t shift = offset % Common::BitSize<u64>();
|
||||||
ASSERT(shift + count <= Common::BitSize<u64>());
|
ASSERT(shift + count <= Common::BitSize<u64>());
|
||||||
// Check that all the bits are set.
|
// Check that all the bits are set.
|
||||||
const u64 mask = ((u64(1) << count) - 1) << shift;
|
const u64 mask = ((u64(1) << count) - 1) << shift;
|
||||||
@ -189,8 +249,8 @@ public:
|
|||||||
ASSERT(offset % Common::BitSize<u64>() == 0);
|
ASSERT(offset % Common::BitSize<u64>() == 0);
|
||||||
ASSERT(count % Common::BitSize<u64>() == 0);
|
ASSERT(count % Common::BitSize<u64>() == 0);
|
||||||
// Check that all the bits are set.
|
// Check that all the bits are set.
|
||||||
std::size_t remaining = count;
|
size_t remaining = count;
|
||||||
std::size_t i = 0;
|
size_t i = 0;
|
||||||
do {
|
do {
|
||||||
if (bits[bit_ind + i++] != ~u64(0)) {
|
if (bits[bit_ind + i++] != ~u64(0)) {
|
||||||
return false;
|
return false;
|
||||||
@ -209,18 +269,18 @@ public:
|
|||||||
} while (remaining > 0);
|
} while (remaining > 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
num_bits -= count;
|
m_num_bits -= count;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void SetBit(s32 depth, std::size_t offset) {
|
void SetBit(s32 depth, size_t offset) {
|
||||||
while (depth >= 0) {
|
while (depth >= 0) {
|
||||||
std::size_t ind = offset / Common::BitSize<u64>();
|
size_t ind = offset / Common::BitSize<u64>();
|
||||||
std::size_t which = offset % Common::BitSize<u64>();
|
size_t which = offset % Common::BitSize<u64>();
|
||||||
const u64 mask = u64(1) << which;
|
const u64 mask = u64(1) << which;
|
||||||
|
|
||||||
u64* bit = std::addressof(bit_storages[depth][ind]);
|
u64* bit = std::addressof(m_bit_storages[depth][ind]);
|
||||||
u64 v = *bit;
|
u64 v = *bit;
|
||||||
ASSERT((v & mask) == 0);
|
ASSERT((v & mask) == 0);
|
||||||
*bit = v | mask;
|
*bit = v | mask;
|
||||||
@ -232,13 +292,13 @@ private:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ClearBit(s32 depth, std::size_t offset) {
|
void ClearBit(s32 depth, size_t offset) {
|
||||||
while (depth >= 0) {
|
while (depth >= 0) {
|
||||||
std::size_t ind = offset / Common::BitSize<u64>();
|
size_t ind = offset / Common::BitSize<u64>();
|
||||||
std::size_t which = offset % Common::BitSize<u64>();
|
size_t which = offset % Common::BitSize<u64>();
|
||||||
const u64 mask = u64(1) << which;
|
const u64 mask = u64(1) << which;
|
||||||
|
|
||||||
u64* bit = std::addressof(bit_storages[depth][ind]);
|
u64* bit = std::addressof(m_bit_storages[depth][ind]);
|
||||||
u64 v = *bit;
|
u64 v = *bit;
|
||||||
ASSERT((v & mask) != 0);
|
ASSERT((v & mask) != 0);
|
||||||
v &= ~mask;
|
v &= ~mask;
|
||||||
@ -252,7 +312,7 @@ private:
|
|||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static constexpr s32 GetRequiredDepth(std::size_t region_size) {
|
static constexpr s32 GetRequiredDepth(size_t region_size) {
|
||||||
s32 depth = 0;
|
s32 depth = 0;
|
||||||
while (true) {
|
while (true) {
|
||||||
region_size /= Common::BitSize<u64>();
|
region_size /= Common::BitSize<u64>();
|
||||||
@ -264,8 +324,8 @@ private:
|
|||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
static constexpr std::size_t CalculateManagementOverheadSize(std::size_t region_size) {
|
static constexpr size_t CalculateManagementOverheadSize(size_t region_size) {
|
||||||
std::size_t overhead_bits = 0;
|
size_t overhead_bits = 0;
|
||||||
for (s32 depth = GetRequiredDepth(region_size) - 1; depth >= 0; depth--) {
|
for (s32 depth = GetRequiredDepth(region_size) - 1; depth >= 0; depth--) {
|
||||||
region_size =
|
region_size =
|
||||||
Common::AlignUp(region_size, Common::BitSize<u64>()) / Common::BitSize<u64>();
|
Common::AlignUp(region_size, Common::BitSize<u64>()) / Common::BitSize<u64>();
|
||||||
@ -273,6 +333,13 @@ public:
|
|||||||
}
|
}
|
||||||
return overhead_bits * sizeof(u64);
|
return overhead_bits * sizeof(u64);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::array<u64*, MaxDepth> m_bit_storages{};
|
||||||
|
std::array<u64*, MaxDepth> m_end_storages{};
|
||||||
|
RandomBitGenerator m_rng;
|
||||||
|
size_t m_num_bits{};
|
||||||
|
size_t m_used_depths{};
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Kernel
|
} // namespace Kernel
|
||||||
|
@ -11,6 +11,16 @@
|
|||||||
|
|
||||||
namespace Kernel {
|
namespace Kernel {
|
||||||
|
|
||||||
|
class KernelCore;
|
||||||
|
|
||||||
|
class KPageBufferSlabHeap : protected impl::KSlabHeapImpl {
|
||||||
|
public:
|
||||||
|
static constexpr size_t BufferSize = PageSize;
|
||||||
|
|
||||||
|
public:
|
||||||
|
void Initialize(Core::System& system);
|
||||||
|
};
|
||||||
|
|
||||||
class KPageBuffer final : public KSlabAllocated<KPageBuffer> {
|
class KPageBuffer final : public KSlabAllocated<KPageBuffer> {
|
||||||
public:
|
public:
|
||||||
explicit KPageBuffer(KernelCore&) {}
|
explicit KPageBuffer(KernelCore&) {}
|
||||||
@ -21,8 +31,6 @@ public:
|
|||||||
private:
|
private:
|
||||||
[[maybe_unused]] alignas(PageSize) std::array<u8, PageSize> m_buffer{};
|
[[maybe_unused]] alignas(PageSize) std::array<u8, PageSize> m_buffer{};
|
||||||
};
|
};
|
||||||
|
static_assert(sizeof(KPageBuffer) == KPageBufferSlabHeap::BufferSize);
|
||||||
static_assert(sizeof(KPageBuffer) == PageSize);
|
|
||||||
static_assert(alignof(KPageBuffer) == PageSize);
|
|
||||||
|
|
||||||
} // namespace Kernel
|
} // namespace Kernel
|
||||||
|
@ -5,6 +5,7 @@
|
|||||||
|
|
||||||
#include <list>
|
#include <list>
|
||||||
|
|
||||||
|
#include "common/alignment.h"
|
||||||
#include "common/assert.h"
|
#include "common/assert.h"
|
||||||
#include "common/common_types.h"
|
#include "common/common_types.h"
|
||||||
#include "core/hle/kernel/memory_types.h"
|
#include "core/hle/kernel/memory_types.h"
|
||||||
@ -12,6 +13,89 @@
|
|||||||
|
|
||||||
namespace Kernel {
|
namespace Kernel {
|
||||||
|
|
||||||
|
class KPageGroup;
|
||||||
|
|
||||||
|
class KBlockInfo {
|
||||||
|
private:
|
||||||
|
friend class KPageGroup;
|
||||||
|
|
||||||
|
public:
|
||||||
|
constexpr KBlockInfo() = default;
|
||||||
|
|
||||||
|
constexpr void Initialize(PAddr addr, size_t np) {
|
||||||
|
ASSERT(Common::IsAligned(addr, PageSize));
|
||||||
|
ASSERT(static_cast<u32>(np) == np);
|
||||||
|
|
||||||
|
m_page_index = static_cast<u32>(addr) / PageSize;
|
||||||
|
m_num_pages = static_cast<u32>(np);
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr PAddr GetAddress() const {
|
||||||
|
return m_page_index * PageSize;
|
||||||
|
}
|
||||||
|
constexpr size_t GetNumPages() const {
|
||||||
|
return m_num_pages;
|
||||||
|
}
|
||||||
|
constexpr size_t GetSize() const {
|
||||||
|
return this->GetNumPages() * PageSize;
|
||||||
|
}
|
||||||
|
constexpr PAddr GetEndAddress() const {
|
||||||
|
return (m_page_index + m_num_pages) * PageSize;
|
||||||
|
}
|
||||||
|
constexpr PAddr GetLastAddress() const {
|
||||||
|
return this->GetEndAddress() - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr KBlockInfo* GetNext() const {
|
||||||
|
return m_next;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr bool IsEquivalentTo(const KBlockInfo& rhs) const {
|
||||||
|
return m_page_index == rhs.m_page_index && m_num_pages == rhs.m_num_pages;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr bool operator==(const KBlockInfo& rhs) const {
|
||||||
|
return this->IsEquivalentTo(rhs);
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr bool operator!=(const KBlockInfo& rhs) const {
|
||||||
|
return !(*this == rhs);
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr bool IsStrictlyBefore(PAddr addr) const {
|
||||||
|
const PAddr end = this->GetEndAddress();
|
||||||
|
|
||||||
|
if (m_page_index != 0 && end == 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return end < addr;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr bool operator<(PAddr addr) const {
|
||||||
|
return this->IsStrictlyBefore(addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr bool TryConcatenate(PAddr addr, size_t np) {
|
||||||
|
if (addr != 0 && addr == this->GetEndAddress()) {
|
||||||
|
m_num_pages += static_cast<u32>(np);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
constexpr void SetNext(KBlockInfo* next) {
|
||||||
|
m_next = next;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
KBlockInfo* m_next{};
|
||||||
|
u32 m_page_index{};
|
||||||
|
u32 m_num_pages{};
|
||||||
|
};
|
||||||
|
static_assert(sizeof(KBlockInfo) <= 0x10);
|
||||||
|
|
||||||
class KPageGroup final {
|
class KPageGroup final {
|
||||||
public:
|
public:
|
||||||
class Node final {
|
class Node final {
|
||||||
@ -92,6 +176,8 @@ public:
|
|||||||
return nodes.empty();
|
return nodes.empty();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Finalize() {}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::list<Node> nodes;
|
std::list<Node> nodes;
|
||||||
};
|
};
|
||||||
|
@ -44,11 +44,11 @@ size_t KPageHeap::GetNumFreePages() const {
|
|||||||
return num_free;
|
return num_free;
|
||||||
}
|
}
|
||||||
|
|
||||||
PAddr KPageHeap::AllocateBlock(s32 index, bool random) {
|
PAddr KPageHeap::AllocateByLinearSearch(s32 index) {
|
||||||
const size_t needed_size = m_blocks[index].GetSize();
|
const size_t needed_size = m_blocks[index].GetSize();
|
||||||
|
|
||||||
for (s32 i = index; i < static_cast<s32>(m_num_blocks); i++) {
|
for (s32 i = index; i < static_cast<s32>(m_num_blocks); i++) {
|
||||||
if (const PAddr addr = m_blocks[i].PopBlock(random); addr != 0) {
|
if (const PAddr addr = m_blocks[i].PopBlock(false); addr != 0) {
|
||||||
if (const size_t allocated_size = m_blocks[i].GetSize(); allocated_size > needed_size) {
|
if (const size_t allocated_size = m_blocks[i].GetSize(); allocated_size > needed_size) {
|
||||||
this->Free(addr + needed_size, (allocated_size - needed_size) / PageSize);
|
this->Free(addr + needed_size, (allocated_size - needed_size) / PageSize);
|
||||||
}
|
}
|
||||||
@ -59,6 +59,88 @@ PAddr KPageHeap::AllocateBlock(s32 index, bool random) {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PAddr KPageHeap::AllocateByRandom(s32 index, size_t num_pages, size_t align_pages) {
|
||||||
|
// Get the size and required alignment.
|
||||||
|
const size_t needed_size = num_pages * PageSize;
|
||||||
|
const size_t align_size = align_pages * PageSize;
|
||||||
|
|
||||||
|
// Determine meta-alignment of our desired alignment size.
|
||||||
|
const size_t align_shift = std::countr_zero(align_size);
|
||||||
|
|
||||||
|
// Decide on a block to allocate from.
|
||||||
|
constexpr size_t MinimumPossibleAlignmentsForRandomAllocation = 4;
|
||||||
|
{
|
||||||
|
// By default, we'll want to look at all blocks larger than our current one.
|
||||||
|
s32 max_blocks = static_cast<s32>(m_num_blocks);
|
||||||
|
|
||||||
|
// Determine the maximum block we should try to allocate from.
|
||||||
|
size_t possible_alignments = 0;
|
||||||
|
for (s32 i = index; i < max_blocks; ++i) {
|
||||||
|
// Add the possible alignments from blocks at the current size.
|
||||||
|
possible_alignments += (1 + ((m_blocks[i].GetSize() - needed_size) >> align_shift)) *
|
||||||
|
m_blocks[i].GetNumFreeBlocks();
|
||||||
|
|
||||||
|
// If there are enough possible alignments, we don't need to look at larger blocks.
|
||||||
|
if (possible_alignments >= MinimumPossibleAlignmentsForRandomAllocation) {
|
||||||
|
max_blocks = i + 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we have any possible alignments which require a larger block, we need to pick one.
|
||||||
|
if (possible_alignments > 0 && index + 1 < max_blocks) {
|
||||||
|
// Select a random alignment from the possibilities.
|
||||||
|
const size_t rnd = m_rng.GenerateRandom(possible_alignments);
|
||||||
|
|
||||||
|
// Determine which block corresponds to the random alignment we chose.
|
||||||
|
possible_alignments = 0;
|
||||||
|
for (s32 i = index; i < max_blocks; ++i) {
|
||||||
|
// Add the possible alignments from blocks at the current size.
|
||||||
|
possible_alignments +=
|
||||||
|
(1 + ((m_blocks[i].GetSize() - needed_size) >> align_shift)) *
|
||||||
|
m_blocks[i].GetNumFreeBlocks();
|
||||||
|
|
||||||
|
// If the current block gets us to our random choice, use the current block.
|
||||||
|
if (rnd < possible_alignments) {
|
||||||
|
index = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pop a block from the index we selected.
|
||||||
|
if (PAddr addr = m_blocks[index].PopBlock(true); addr != 0) {
|
||||||
|
// Determine how much size we have left over.
|
||||||
|
if (const size_t leftover_size = m_blocks[index].GetSize() - needed_size;
|
||||||
|
leftover_size > 0) {
|
||||||
|
// Determine how many valid alignments we can have.
|
||||||
|
const size_t possible_alignments = 1 + (leftover_size >> align_shift);
|
||||||
|
|
||||||
|
// Select a random valid alignment.
|
||||||
|
const size_t random_offset = m_rng.GenerateRandom(possible_alignments) << align_shift;
|
||||||
|
|
||||||
|
// Free memory before the random offset.
|
||||||
|
if (random_offset != 0) {
|
||||||
|
this->Free(addr, random_offset / PageSize);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Advance our block by the random offset.
|
||||||
|
addr += random_offset;
|
||||||
|
|
||||||
|
// Free memory after our allocated block.
|
||||||
|
if (random_offset != leftover_size) {
|
||||||
|
this->Free(addr + needed_size, (leftover_size - random_offset) / PageSize);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return the block we allocated.
|
||||||
|
return addr;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
void KPageHeap::FreeBlock(PAddr block, s32 index) {
|
void KPageHeap::FreeBlock(PAddr block, s32 index) {
|
||||||
do {
|
do {
|
||||||
block = m_blocks[index++].PushBlock(block);
|
block = m_blocks[index++].PushBlock(block);
|
||||||
|
@ -14,13 +14,9 @@
|
|||||||
|
|
||||||
namespace Kernel {
|
namespace Kernel {
|
||||||
|
|
||||||
class KPageHeap final {
|
class KPageHeap {
|
||||||
public:
|
public:
|
||||||
YUZU_NON_COPYABLE(KPageHeap);
|
|
||||||
YUZU_NON_MOVEABLE(KPageHeap);
|
|
||||||
|
|
||||||
KPageHeap() = default;
|
KPageHeap() = default;
|
||||||
~KPageHeap() = default;
|
|
||||||
|
|
||||||
constexpr PAddr GetAddress() const {
|
constexpr PAddr GetAddress() const {
|
||||||
return m_heap_address;
|
return m_heap_address;
|
||||||
@ -57,7 +53,20 @@ public:
|
|||||||
m_initial_used_size = m_heap_size - free_size - reserved_size;
|
m_initial_used_size = m_heap_size - free_size - reserved_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
PAddr AllocateBlock(s32 index, bool random);
|
PAddr AllocateBlock(s32 index, bool random) {
|
||||||
|
if (random) {
|
||||||
|
const size_t block_pages = m_blocks[index].GetNumPages();
|
||||||
|
return this->AllocateByRandom(index, block_pages, block_pages);
|
||||||
|
} else {
|
||||||
|
return this->AllocateByLinearSearch(index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
PAddr AllocateAligned(s32 index, size_t num_pages, size_t align_pages) {
|
||||||
|
// TODO: linear search support?
|
||||||
|
return this->AllocateByRandom(index, num_pages, align_pages);
|
||||||
|
}
|
||||||
|
|
||||||
void Free(PAddr addr, size_t num_pages);
|
void Free(PAddr addr, size_t num_pages);
|
||||||
|
|
||||||
static size_t CalculateManagementOverheadSize(size_t region_size) {
|
static size_t CalculateManagementOverheadSize(size_t region_size) {
|
||||||
@ -68,7 +77,7 @@ public:
|
|||||||
static constexpr s32 GetAlignedBlockIndex(size_t num_pages, size_t align_pages) {
|
static constexpr s32 GetAlignedBlockIndex(size_t num_pages, size_t align_pages) {
|
||||||
const size_t target_pages = std::max(num_pages, align_pages);
|
const size_t target_pages = std::max(num_pages, align_pages);
|
||||||
for (size_t i = 0; i < NumMemoryBlockPageShifts; i++) {
|
for (size_t i = 0; i < NumMemoryBlockPageShifts; i++) {
|
||||||
if (target_pages <= (size_t(1) << MemoryBlockPageShifts[i]) / PageSize) {
|
if (target_pages <= (static_cast<size_t>(1) << MemoryBlockPageShifts[i]) / PageSize) {
|
||||||
return static_cast<s32>(i);
|
return static_cast<s32>(i);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -77,7 +86,7 @@ public:
|
|||||||
|
|
||||||
static constexpr s32 GetBlockIndex(size_t num_pages) {
|
static constexpr s32 GetBlockIndex(size_t num_pages) {
|
||||||
for (s32 i = static_cast<s32>(NumMemoryBlockPageShifts) - 1; i >= 0; i--) {
|
for (s32 i = static_cast<s32>(NumMemoryBlockPageShifts) - 1; i >= 0; i--) {
|
||||||
if (num_pages >= (size_t(1) << MemoryBlockPageShifts[i]) / PageSize) {
|
if (num_pages >= (static_cast<size_t>(1) << MemoryBlockPageShifts[i]) / PageSize) {
|
||||||
return i;
|
return i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -85,7 +94,7 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
static constexpr size_t GetBlockSize(size_t index) {
|
static constexpr size_t GetBlockSize(size_t index) {
|
||||||
return size_t(1) << MemoryBlockPageShifts[index];
|
return static_cast<size_t>(1) << MemoryBlockPageShifts[index];
|
||||||
}
|
}
|
||||||
|
|
||||||
static constexpr size_t GetBlockNumPages(size_t index) {
|
static constexpr size_t GetBlockNumPages(size_t index) {
|
||||||
@ -93,13 +102,9 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
class Block final {
|
class Block {
|
||||||
public:
|
public:
|
||||||
YUZU_NON_COPYABLE(Block);
|
|
||||||
YUZU_NON_MOVEABLE(Block);
|
|
||||||
|
|
||||||
Block() = default;
|
Block() = default;
|
||||||
~Block() = default;
|
|
||||||
|
|
||||||
constexpr size_t GetShift() const {
|
constexpr size_t GetShift() const {
|
||||||
return m_block_shift;
|
return m_block_shift;
|
||||||
@ -201,6 +206,9 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
PAddr AllocateByLinearSearch(s32 index);
|
||||||
|
PAddr AllocateByRandom(s32 index, size_t num_pages, size_t align_pages);
|
||||||
|
|
||||||
static size_t CalculateManagementOverheadSize(size_t region_size, const size_t* block_shifts,
|
static size_t CalculateManagementOverheadSize(size_t region_size, const size_t* block_shifts,
|
||||||
size_t num_block_shifts);
|
size_t num_block_shifts);
|
||||||
|
|
||||||
@ -209,7 +217,8 @@ private:
|
|||||||
size_t m_heap_size{};
|
size_t m_heap_size{};
|
||||||
size_t m_initial_used_size{};
|
size_t m_initial_used_size{};
|
||||||
size_t m_num_blocks{};
|
size_t m_num_blocks{};
|
||||||
std::array<Block, NumMemoryBlockPageShifts> m_blocks{};
|
std::array<Block, NumMemoryBlockPageShifts> m_blocks;
|
||||||
|
KPageBitmap::RandomBitGenerator m_rng;
|
||||||
std::vector<u64> m_management_data;
|
std::vector<u64> m_management_data;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -16,6 +16,7 @@
|
|||||||
#include "core/hle/kernel/k_memory_layout.h"
|
#include "core/hle/kernel/k_memory_layout.h"
|
||||||
#include "core/hle/kernel/k_memory_manager.h"
|
#include "core/hle/kernel/k_memory_manager.h"
|
||||||
#include "core/hle/result.h"
|
#include "core/hle/result.h"
|
||||||
|
#include "core/memory.h"
|
||||||
|
|
||||||
namespace Core {
|
namespace Core {
|
||||||
class System;
|
class System;
|
||||||
@ -23,7 +24,10 @@ class System;
|
|||||||
|
|
||||||
namespace Kernel {
|
namespace Kernel {
|
||||||
|
|
||||||
|
class KBlockInfoManager;
|
||||||
class KMemoryBlockManager;
|
class KMemoryBlockManager;
|
||||||
|
class KResourceLimit;
|
||||||
|
class KSystemResource;
|
||||||
|
|
||||||
class KPageTable final {
|
class KPageTable final {
|
||||||
public:
|
public:
|
||||||
@ -36,9 +40,9 @@ public:
|
|||||||
~KPageTable();
|
~KPageTable();
|
||||||
|
|
||||||
Result InitializeForProcess(FileSys::ProgramAddressSpaceType as_type, bool enable_aslr,
|
Result InitializeForProcess(FileSys::ProgramAddressSpaceType as_type, bool enable_aslr,
|
||||||
VAddr code_addr, size_t code_size,
|
bool enable_das_merge, bool from_back, KMemoryManager::Pool pool,
|
||||||
KMemoryBlockSlabManager* mem_block_slab_manager,
|
VAddr code_addr, size_t code_size, KSystemResource* system_resource,
|
||||||
KMemoryManager::Pool pool);
|
KResourceLimit* resource_limit);
|
||||||
|
|
||||||
void Finalize();
|
void Finalize();
|
||||||
|
|
||||||
@ -74,12 +78,20 @@ public:
|
|||||||
KMemoryState state, KMemoryPermission perm,
|
KMemoryState state, KMemoryPermission perm,
|
||||||
PAddr map_addr = 0);
|
PAddr map_addr = 0);
|
||||||
|
|
||||||
Result LockForMapDeviceAddressSpace(VAddr address, size_t size, KMemoryPermission perm,
|
Result LockForMapDeviceAddressSpace(bool* out_is_io, VAddr address, size_t size,
|
||||||
bool is_aligned);
|
KMemoryPermission perm, bool is_aligned, bool check_heap);
|
||||||
Result LockForUnmapDeviceAddressSpace(VAddr address, size_t size);
|
Result LockForUnmapDeviceAddressSpace(VAddr address, size_t size, bool check_heap);
|
||||||
|
|
||||||
Result UnlockForDeviceAddressSpace(VAddr addr, size_t size);
|
Result UnlockForDeviceAddressSpace(VAddr addr, size_t size);
|
||||||
|
|
||||||
|
Result LockForIpcUserBuffer(PAddr* out, VAddr address, size_t size);
|
||||||
|
Result UnlockForIpcUserBuffer(VAddr address, size_t size);
|
||||||
|
|
||||||
|
Result SetupForIpc(VAddr* out_dst_addr, size_t size, VAddr src_addr, KPageTable& src_page_table,
|
||||||
|
KMemoryPermission test_perm, KMemoryState dst_state, bool send);
|
||||||
|
Result CleanupForIpcServer(VAddr address, size_t size, KMemoryState dst_state);
|
||||||
|
Result CleanupForIpcClient(VAddr address, size_t size, KMemoryState dst_state);
|
||||||
|
|
||||||
Result LockForCodeMemory(KPageGroup* out, VAddr addr, size_t size);
|
Result LockForCodeMemory(KPageGroup* out, VAddr addr, size_t size);
|
||||||
Result UnlockForCodeMemory(VAddr addr, size_t size, const KPageGroup& pg);
|
Result UnlockForCodeMemory(VAddr addr, size_t size, const KPageGroup& pg);
|
||||||
Result MakeAndOpenPageGroup(KPageGroup* out, VAddr address, size_t num_pages,
|
Result MakeAndOpenPageGroup(KPageGroup* out, VAddr address, size_t num_pages,
|
||||||
@ -97,13 +109,54 @@ public:
|
|||||||
|
|
||||||
bool CanContain(VAddr addr, size_t size, KMemoryState state) const;
|
bool CanContain(VAddr addr, size_t size, KMemoryState state) const;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
struct PageLinkedList {
|
||||||
|
private:
|
||||||
|
struct Node {
|
||||||
|
Node* m_next;
|
||||||
|
std::array<u8, PageSize - sizeof(Node*)> m_buffer;
|
||||||
|
};
|
||||||
|
|
||||||
|
public:
|
||||||
|
constexpr PageLinkedList() = default;
|
||||||
|
|
||||||
|
void Push(Node* n) {
|
||||||
|
ASSERT(Common::IsAligned(reinterpret_cast<uintptr_t>(n), PageSize));
|
||||||
|
n->m_next = m_root;
|
||||||
|
m_root = n;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Push(Core::Memory::Memory& memory, VAddr addr) {
|
||||||
|
this->Push(memory.GetPointer<Node>(addr));
|
||||||
|
}
|
||||||
|
|
||||||
|
Node* Peek() const {
|
||||||
|
return m_root;
|
||||||
|
}
|
||||||
|
|
||||||
|
Node* Pop() {
|
||||||
|
Node* const r = m_root;
|
||||||
|
|
||||||
|
m_root = r->m_next;
|
||||||
|
r->m_next = nullptr;
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
Node* m_root{};
|
||||||
|
};
|
||||||
|
static_assert(std::is_trivially_destructible<PageLinkedList>::value);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
enum class OperationType : u32 {
|
enum class OperationType : u32 {
|
||||||
Map,
|
Map = 0,
|
||||||
MapGroup,
|
MapFirst = 1,
|
||||||
Unmap,
|
MapGroup = 2,
|
||||||
ChangePermissions,
|
Unmap = 3,
|
||||||
ChangePermissionsAndRefresh,
|
ChangePermissions = 4,
|
||||||
|
ChangePermissionsAndRefresh = 5,
|
||||||
|
Separate = 6,
|
||||||
};
|
};
|
||||||
|
|
||||||
static constexpr KMemoryAttribute DefaultMemoryIgnoreAttr =
|
static constexpr KMemoryAttribute DefaultMemoryIgnoreAttr =
|
||||||
@ -123,6 +176,7 @@ private:
|
|||||||
OperationType operation);
|
OperationType operation);
|
||||||
Result Operate(VAddr addr, size_t num_pages, KMemoryPermission perm, OperationType operation,
|
Result Operate(VAddr addr, size_t num_pages, KMemoryPermission perm, OperationType operation,
|
||||||
PAddr map_addr = 0);
|
PAddr map_addr = 0);
|
||||||
|
void FinalizeUpdate(PageLinkedList* page_list);
|
||||||
VAddr GetRegionAddress(KMemoryState state) const;
|
VAddr GetRegionAddress(KMemoryState state) const;
|
||||||
size_t GetRegionSize(KMemoryState state) const;
|
size_t GetRegionSize(KMemoryState state) const;
|
||||||
|
|
||||||
@ -199,6 +253,18 @@ private:
|
|||||||
return *out != 0;
|
return *out != 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Result SetupForIpcClient(PageLinkedList* page_list, size_t* out_blocks_needed, VAddr address,
|
||||||
|
size_t size, KMemoryPermission test_perm, KMemoryState dst_state);
|
||||||
|
Result SetupForIpcServer(VAddr* out_addr, size_t size, VAddr src_addr,
|
||||||
|
KMemoryPermission test_perm, KMemoryState dst_state,
|
||||||
|
KPageTable& src_page_table, bool send);
|
||||||
|
void CleanupForIpcClientOnServerSetupFailure(PageLinkedList* page_list, VAddr address,
|
||||||
|
size_t size, KMemoryPermission prot_perm);
|
||||||
|
|
||||||
|
// HACK: These will be removed once we automatically manage page reference counts.
|
||||||
|
void HACK_OpenPages(PAddr phys_addr, size_t num_pages);
|
||||||
|
void HACK_ClosePages(VAddr virt_addr, size_t num_pages);
|
||||||
|
|
||||||
mutable KLightLock m_general_lock;
|
mutable KLightLock m_general_lock;
|
||||||
mutable KLightLock m_map_physical_memory_lock;
|
mutable KLightLock m_map_physical_memory_lock;
|
||||||
|
|
||||||
@ -316,6 +382,31 @@ public:
|
|||||||
addr + size - 1 <= m_address_space_end - 1;
|
addr + size - 1 <= m_address_space_end - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
static VAddr GetLinearMappedVirtualAddress(const KMemoryLayout& layout, PAddr addr) {
|
||||||
|
return layout.GetLinearVirtualAddress(addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static PAddr GetLinearMappedPhysicalAddress(const KMemoryLayout& layout, VAddr addr) {
|
||||||
|
return layout.GetLinearPhysicalAddress(addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static VAddr GetHeapVirtualAddress(const KMemoryLayout& layout, PAddr addr) {
|
||||||
|
return GetLinearMappedVirtualAddress(layout, addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static PAddr GetHeapPhysicalAddress(const KMemoryLayout& layout, VAddr addr) {
|
||||||
|
return GetLinearMappedPhysicalAddress(layout, addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static VAddr GetPageTableVirtualAddress(const KMemoryLayout& layout, PAddr addr) {
|
||||||
|
return GetLinearMappedVirtualAddress(layout, addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static PAddr GetPageTablePhysicalAddress(const KMemoryLayout& layout, VAddr addr) {
|
||||||
|
return GetLinearMappedPhysicalAddress(layout, addr);
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
constexpr bool IsKernel() const {
|
constexpr bool IsKernel() const {
|
||||||
return m_is_kernel;
|
return m_is_kernel;
|
||||||
@ -330,6 +421,24 @@ private:
|
|||||||
(addr + num_pages * PageSize - 1 <= m_address_space_end - 1);
|
(addr + num_pages * PageSize - 1 <= m_address_space_end - 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
class KScopedPageTableUpdater {
|
||||||
|
private:
|
||||||
|
KPageTable* m_pt{};
|
||||||
|
PageLinkedList m_ll;
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit KScopedPageTableUpdater(KPageTable* pt) : m_pt(pt) {}
|
||||||
|
explicit KScopedPageTableUpdater(KPageTable& pt) : KScopedPageTableUpdater(&pt) {}
|
||||||
|
~KScopedPageTableUpdater() {
|
||||||
|
m_pt->FinalizeUpdate(this->GetPageList());
|
||||||
|
}
|
||||||
|
|
||||||
|
PageLinkedList* GetPageList() {
|
||||||
|
return &m_ll;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
VAddr m_address_space_start{};
|
VAddr m_address_space_start{};
|
||||||
VAddr m_address_space_end{};
|
VAddr m_address_space_end{};
|
||||||
@ -347,20 +456,27 @@ private:
|
|||||||
VAddr m_alias_code_region_start{};
|
VAddr m_alias_code_region_start{};
|
||||||
VAddr m_alias_code_region_end{};
|
VAddr m_alias_code_region_end{};
|
||||||
|
|
||||||
size_t m_mapped_physical_memory_size{};
|
|
||||||
size_t m_max_heap_size{};
|
size_t m_max_heap_size{};
|
||||||
size_t m_max_physical_memory_size{};
|
size_t m_mapped_physical_memory_size{};
|
||||||
|
size_t m_mapped_unsafe_physical_memory{};
|
||||||
|
size_t m_mapped_insecure_memory{};
|
||||||
|
size_t m_mapped_ipc_server_memory{};
|
||||||
size_t m_address_space_width{};
|
size_t m_address_space_width{};
|
||||||
|
|
||||||
KMemoryBlockManager m_memory_block_manager;
|
KMemoryBlockManager m_memory_block_manager;
|
||||||
|
u32 m_allocate_option{};
|
||||||
|
|
||||||
bool m_is_kernel{};
|
bool m_is_kernel{};
|
||||||
bool m_enable_aslr{};
|
bool m_enable_aslr{};
|
||||||
bool m_enable_device_address_space_merge{};
|
bool m_enable_device_address_space_merge{};
|
||||||
|
|
||||||
KMemoryBlockSlabManager* m_memory_block_slab_manager{};
|
KMemoryBlockSlabManager* m_memory_block_slab_manager{};
|
||||||
|
KBlockInfoManager* m_block_info_manager{};
|
||||||
|
KResourceLimit* m_resource_limit{};
|
||||||
|
|
||||||
u32 m_heap_fill_value{};
|
u32 m_heap_fill_value{};
|
||||||
|
u32 m_ipc_fill_value{};
|
||||||
|
u32 m_stack_fill_value{};
|
||||||
const KMemoryRegion* m_cached_physical_heap_region{};
|
const KMemoryRegion* m_cached_physical_heap_region{};
|
||||||
|
|
||||||
KMemoryManager::Pool m_memory_pool{KMemoryManager::Pool::Application};
|
KMemoryManager::Pool m_memory_pool{KMemoryManager::Pool::Application};
|
||||||
|
55
src/core/hle/kernel/k_page_table_manager.h
Normal file
55
src/core/hle/kernel/k_page_table_manager.h
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <atomic>
|
||||||
|
|
||||||
|
#include "common/common_types.h"
|
||||||
|
#include "core/hle/kernel/k_dynamic_resource_manager.h"
|
||||||
|
#include "core/hle/kernel/k_page_table_slab_heap.h"
|
||||||
|
|
||||||
|
namespace Kernel {
|
||||||
|
|
||||||
|
class KPageTableManager : public KDynamicResourceManager<impl::PageTablePage, true> {
|
||||||
|
public:
|
||||||
|
using RefCount = KPageTableSlabHeap::RefCount;
|
||||||
|
static constexpr size_t PageTableSize = KPageTableSlabHeap::PageTableSize;
|
||||||
|
|
||||||
|
public:
|
||||||
|
KPageTableManager() = default;
|
||||||
|
|
||||||
|
void Initialize(KDynamicPageManager* page_allocator, KPageTableSlabHeap* pt_heap) {
|
||||||
|
m_pt_heap = pt_heap;
|
||||||
|
|
||||||
|
static_assert(std::derived_from<KPageTableSlabHeap, DynamicSlabType>);
|
||||||
|
BaseHeap::Initialize(page_allocator, pt_heap);
|
||||||
|
}
|
||||||
|
|
||||||
|
VAddr Allocate() {
|
||||||
|
return VAddr(BaseHeap::Allocate());
|
||||||
|
}
|
||||||
|
|
||||||
|
RefCount GetRefCount(VAddr addr) const {
|
||||||
|
return m_pt_heap->GetRefCount(addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Open(VAddr addr, int count) {
|
||||||
|
return m_pt_heap->Open(addr, count);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Close(VAddr addr, int count) {
|
||||||
|
return m_pt_heap->Close(addr, count);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsInPageTableHeap(VAddr addr) const {
|
||||||
|
return m_pt_heap->IsInRange(addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
using BaseHeap = KDynamicResourceManager<impl::PageTablePage, true>;
|
||||||
|
|
||||||
|
KPageTableSlabHeap* m_pt_heap{};
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Kernel
|
93
src/core/hle/kernel/k_page_table_slab_heap.h
Normal file
93
src/core/hle/kernel/k_page_table_slab_heap.h
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include "common/common_types.h"
|
||||||
|
#include "core/hle/kernel/k_dynamic_slab_heap.h"
|
||||||
|
#include "core/hle/kernel/slab_helpers.h"
|
||||||
|
|
||||||
|
namespace Kernel {
|
||||||
|
|
||||||
|
namespace impl {
|
||||||
|
|
||||||
|
class PageTablePage {
|
||||||
|
public:
|
||||||
|
// Do not initialize anything.
|
||||||
|
PageTablePage() = default;
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::array<u8, PageSize> m_buffer{};
|
||||||
|
};
|
||||||
|
static_assert(sizeof(PageTablePage) == PageSize);
|
||||||
|
|
||||||
|
} // namespace impl
|
||||||
|
|
||||||
|
class KPageTableSlabHeap : public KDynamicSlabHeap<impl::PageTablePage, true> {
|
||||||
|
public:
|
||||||
|
using RefCount = u16;
|
||||||
|
static constexpr size_t PageTableSize = sizeof(impl::PageTablePage);
|
||||||
|
static_assert(PageTableSize == PageSize);
|
||||||
|
|
||||||
|
public:
|
||||||
|
KPageTableSlabHeap() = default;
|
||||||
|
|
||||||
|
static constexpr size_t CalculateReferenceCountSize(size_t size) {
|
||||||
|
return (size / PageSize) * sizeof(RefCount);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Initialize(KDynamicPageManager* page_allocator, size_t object_count, RefCount* rc) {
|
||||||
|
BaseHeap::Initialize(page_allocator, object_count);
|
||||||
|
this->Initialize(rc);
|
||||||
|
}
|
||||||
|
|
||||||
|
RefCount GetRefCount(VAddr addr) {
|
||||||
|
ASSERT(this->IsInRange(addr));
|
||||||
|
return *this->GetRefCountPointer(addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Open(VAddr addr, int count) {
|
||||||
|
ASSERT(this->IsInRange(addr));
|
||||||
|
|
||||||
|
*this->GetRefCountPointer(addr) += static_cast<RefCount>(count);
|
||||||
|
|
||||||
|
ASSERT(this->GetRefCount(addr) > 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Close(VAddr addr, int count) {
|
||||||
|
ASSERT(this->IsInRange(addr));
|
||||||
|
ASSERT(this->GetRefCount(addr) >= count);
|
||||||
|
|
||||||
|
*this->GetRefCountPointer(addr) -= static_cast<RefCount>(count);
|
||||||
|
return this->GetRefCount(addr) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsInPageTableHeap(VAddr addr) const {
|
||||||
|
return this->IsInRange(addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
void Initialize([[maybe_unused]] RefCount* rc) {
|
||||||
|
// TODO(bunnei): Use rc once we support kernel virtual memory allocations.
|
||||||
|
const auto count = this->GetSize() / PageSize;
|
||||||
|
m_ref_counts.resize(count);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < count; i++) {
|
||||||
|
m_ref_counts[i] = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
RefCount* GetRefCountPointer(VAddr addr) {
|
||||||
|
return m_ref_counts.data() + ((addr - this->GetAddress()) / PageSize);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
using BaseHeap = KDynamicSlabHeap<impl::PageTablePage, true>;
|
||||||
|
|
||||||
|
std::vector<RefCount> m_ref_counts;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Kernel
|
@ -358,8 +358,8 @@ Result KProcess::LoadFromMetadata(const FileSys::ProgramMetadata& metadata, std:
|
|||||||
}
|
}
|
||||||
// Initialize proces address space
|
// Initialize proces address space
|
||||||
if (const Result result{page_table.InitializeForProcess(
|
if (const Result result{page_table.InitializeForProcess(
|
||||||
metadata.GetAddressSpaceType(), false, 0x8000000, code_size,
|
metadata.GetAddressSpaceType(), false, false, false, KMemoryManager::Pool::Application,
|
||||||
&kernel.GetApplicationMemoryBlockManager(), KMemoryManager::Pool::Application)};
|
0x8000000, code_size, &kernel.GetSystemSystemResource(), resource_limit)};
|
||||||
result.IsError()) {
|
result.IsError()) {
|
||||||
R_RETURN(result);
|
R_RETURN(result);
|
||||||
}
|
}
|
||||||
|
26
src/core/hle/kernel/k_system_resource.cpp
Normal file
26
src/core/hle/kernel/k_system_resource.cpp
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include "core/hle/kernel/k_system_resource.h"
|
||||||
|
|
||||||
|
namespace Kernel {
|
||||||
|
|
||||||
|
Result KSecureSystemResource::Initialize([[maybe_unused]] size_t size,
|
||||||
|
[[maybe_unused]] KResourceLimit* resource_limit,
|
||||||
|
[[maybe_unused]] KMemoryManager::Pool pool) {
|
||||||
|
// Unimplemented
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
|
||||||
|
void KSecureSystemResource::Finalize() {
|
||||||
|
// Unimplemented
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t KSecureSystemResource::CalculateRequiredSecureMemorySize(
|
||||||
|
[[maybe_unused]] size_t size, [[maybe_unused]] KMemoryManager::Pool pool) {
|
||||||
|
// Unimplemented
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Kernel
|
137
src/core/hle/kernel/k_system_resource.h
Normal file
137
src/core/hle/kernel/k_system_resource.h
Normal file
@ -0,0 +1,137 @@
|
|||||||
|
// SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "common/assert.h"
|
||||||
|
#include "common/common_types.h"
|
||||||
|
#include "core/hle/kernel/k_auto_object.h"
|
||||||
|
#include "core/hle/kernel/k_dynamic_resource_manager.h"
|
||||||
|
#include "core/hle/kernel/k_memory_manager.h"
|
||||||
|
#include "core/hle/kernel/k_page_table_manager.h"
|
||||||
|
#include "core/hle/kernel/k_resource_limit.h"
|
||||||
|
#include "core/hle/kernel/slab_helpers.h"
|
||||||
|
|
||||||
|
namespace Kernel {
|
||||||
|
|
||||||
|
// NOTE: Nintendo's implementation does not have the "is_secure_resource" field, and instead uses
|
||||||
|
// virtual IsSecureResource().
|
||||||
|
|
||||||
|
class KSystemResource : public KAutoObject {
|
||||||
|
KERNEL_AUTOOBJECT_TRAITS(KSystemResource, KAutoObject);
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit KSystemResource(KernelCore& kernel_) : KAutoObject(kernel_) {}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void SetSecureResource() {
|
||||||
|
m_is_secure_resource = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
virtual void Destroy() override {
|
||||||
|
UNREACHABLE_MSG("KSystemResource::Destroy() was called");
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsSecureResource() const {
|
||||||
|
return m_is_secure_resource;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetManagers(KMemoryBlockSlabManager& mb, KBlockInfoManager& bi, KPageTableManager& pt) {
|
||||||
|
ASSERT(m_p_memory_block_slab_manager == nullptr);
|
||||||
|
ASSERT(m_p_block_info_manager == nullptr);
|
||||||
|
ASSERT(m_p_page_table_manager == nullptr);
|
||||||
|
|
||||||
|
m_p_memory_block_slab_manager = std::addressof(mb);
|
||||||
|
m_p_block_info_manager = std::addressof(bi);
|
||||||
|
m_p_page_table_manager = std::addressof(pt);
|
||||||
|
}
|
||||||
|
|
||||||
|
const KMemoryBlockSlabManager& GetMemoryBlockSlabManager() const {
|
||||||
|
return *m_p_memory_block_slab_manager;
|
||||||
|
}
|
||||||
|
const KBlockInfoManager& GetBlockInfoManager() const {
|
||||||
|
return *m_p_block_info_manager;
|
||||||
|
}
|
||||||
|
const KPageTableManager& GetPageTableManager() const {
|
||||||
|
return *m_p_page_table_manager;
|
||||||
|
}
|
||||||
|
|
||||||
|
KMemoryBlockSlabManager& GetMemoryBlockSlabManager() {
|
||||||
|
return *m_p_memory_block_slab_manager;
|
||||||
|
}
|
||||||
|
KBlockInfoManager& GetBlockInfoManager() {
|
||||||
|
return *m_p_block_info_manager;
|
||||||
|
}
|
||||||
|
KPageTableManager& GetPageTableManager() {
|
||||||
|
return *m_p_page_table_manager;
|
||||||
|
}
|
||||||
|
|
||||||
|
KMemoryBlockSlabManager* GetMemoryBlockSlabManagerPointer() {
|
||||||
|
return m_p_memory_block_slab_manager;
|
||||||
|
}
|
||||||
|
KBlockInfoManager* GetBlockInfoManagerPointer() {
|
||||||
|
return m_p_block_info_manager;
|
||||||
|
}
|
||||||
|
KPageTableManager* GetPageTableManagerPointer() {
|
||||||
|
return m_p_page_table_manager;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
KMemoryBlockSlabManager* m_p_memory_block_slab_manager{};
|
||||||
|
KBlockInfoManager* m_p_block_info_manager{};
|
||||||
|
KPageTableManager* m_p_page_table_manager{};
|
||||||
|
bool m_is_secure_resource{false};
|
||||||
|
};
|
||||||
|
|
||||||
|
class KSecureSystemResource final
|
||||||
|
: public KAutoObjectWithSlabHeap<KSecureSystemResource, KSystemResource> {
|
||||||
|
public:
|
||||||
|
explicit KSecureSystemResource(KernelCore& kernel_)
|
||||||
|
: KAutoObjectWithSlabHeap<KSecureSystemResource, KSystemResource>(kernel_) {
|
||||||
|
// Mark ourselves as being a secure resource.
|
||||||
|
this->SetSecureResource();
|
||||||
|
}
|
||||||
|
|
||||||
|
Result Initialize(size_t size, KResourceLimit* resource_limit, KMemoryManager::Pool pool);
|
||||||
|
void Finalize();
|
||||||
|
|
||||||
|
bool IsInitialized() const {
|
||||||
|
return m_is_initialized;
|
||||||
|
}
|
||||||
|
static void PostDestroy([[maybe_unused]] uintptr_t arg) {}
|
||||||
|
|
||||||
|
size_t CalculateRequiredSecureMemorySize() const {
|
||||||
|
return CalculateRequiredSecureMemorySize(m_resource_size, m_resource_pool);
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t GetSize() const {
|
||||||
|
return m_resource_size;
|
||||||
|
}
|
||||||
|
size_t GetUsedSize() const {
|
||||||
|
return m_dynamic_page_manager.GetUsed() * PageSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
const KDynamicPageManager& GetDynamicPageManager() const {
|
||||||
|
return m_dynamic_page_manager;
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
static size_t CalculateRequiredSecureMemorySize(size_t size, KMemoryManager::Pool pool);
|
||||||
|
|
||||||
|
private:
|
||||||
|
bool m_is_initialized{};
|
||||||
|
KMemoryManager::Pool m_resource_pool{};
|
||||||
|
KDynamicPageManager m_dynamic_page_manager;
|
||||||
|
KMemoryBlockSlabManager m_memory_block_slab_manager;
|
||||||
|
KBlockInfoManager m_block_info_manager;
|
||||||
|
KPageTableManager m_page_table_manager;
|
||||||
|
KMemoryBlockSlabHeap m_memory_block_heap;
|
||||||
|
KBlockInfoSlabHeap m_block_info_heap;
|
||||||
|
KPageTableSlabHeap m_page_table_heap;
|
||||||
|
KResourceLimit* m_resource_limit{};
|
||||||
|
VAddr m_resource_address{};
|
||||||
|
size_t m_resource_size{};
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Kernel
|
@ -28,10 +28,12 @@
|
|||||||
#include "core/hle/kernel/k_handle_table.h"
|
#include "core/hle/kernel/k_handle_table.h"
|
||||||
#include "core/hle/kernel/k_memory_layout.h"
|
#include "core/hle/kernel/k_memory_layout.h"
|
||||||
#include "core/hle/kernel/k_memory_manager.h"
|
#include "core/hle/kernel/k_memory_manager.h"
|
||||||
|
#include "core/hle/kernel/k_page_buffer.h"
|
||||||
#include "core/hle/kernel/k_process.h"
|
#include "core/hle/kernel/k_process.h"
|
||||||
#include "core/hle/kernel/k_resource_limit.h"
|
#include "core/hle/kernel/k_resource_limit.h"
|
||||||
#include "core/hle/kernel/k_scheduler.h"
|
#include "core/hle/kernel/k_scheduler.h"
|
||||||
#include "core/hle/kernel/k_shared_memory.h"
|
#include "core/hle/kernel/k_shared_memory.h"
|
||||||
|
#include "core/hle/kernel/k_system_resource.h"
|
||||||
#include "core/hle/kernel/k_thread.h"
|
#include "core/hle/kernel/k_thread.h"
|
||||||
#include "core/hle/kernel/k_worker_task_manager.h"
|
#include "core/hle/kernel/k_worker_task_manager.h"
|
||||||
#include "core/hle/kernel/kernel.h"
|
#include "core/hle/kernel/kernel.h"
|
||||||
@ -47,6 +49,11 @@ MICROPROFILE_DEFINE(Kernel_SVC, "Kernel", "SVC", MP_RGB(70, 200, 70));
|
|||||||
namespace Kernel {
|
namespace Kernel {
|
||||||
|
|
||||||
struct KernelCore::Impl {
|
struct KernelCore::Impl {
|
||||||
|
static constexpr size_t ApplicationMemoryBlockSlabHeapSize = 20000;
|
||||||
|
static constexpr size_t SystemMemoryBlockSlabHeapSize = 10000;
|
||||||
|
static constexpr size_t BlockInfoSlabHeapSize = 4000;
|
||||||
|
static constexpr size_t ReservedDynamicPageCount = 64;
|
||||||
|
|
||||||
explicit Impl(Core::System& system_, KernelCore& kernel_)
|
explicit Impl(Core::System& system_, KernelCore& kernel_)
|
||||||
: time_manager{system_}, service_threads_manager{1, "ServiceThreadsManager"},
|
: time_manager{system_}, service_threads_manager{1, "ServiceThreadsManager"},
|
||||||
service_thread_barrier{2}, system{system_} {}
|
service_thread_barrier{2}, system{system_} {}
|
||||||
@ -71,7 +78,6 @@ struct KernelCore::Impl {
|
|||||||
// Initialize kernel memory and resources.
|
// Initialize kernel memory and resources.
|
||||||
InitializeSystemResourceLimit(kernel, system.CoreTiming());
|
InitializeSystemResourceLimit(kernel, system.CoreTiming());
|
||||||
InitializeMemoryLayout();
|
InitializeMemoryLayout();
|
||||||
Init::InitializeKPageBufferSlabHeap(system);
|
|
||||||
InitializeShutdownThreads();
|
InitializeShutdownThreads();
|
||||||
InitializePhysicalCores();
|
InitializePhysicalCores();
|
||||||
InitializePreemption(kernel);
|
InitializePreemption(kernel);
|
||||||
@ -81,7 +87,8 @@ struct KernelCore::Impl {
|
|||||||
const auto& pt_heap_region = memory_layout->GetPageTableHeapRegion();
|
const auto& pt_heap_region = memory_layout->GetPageTableHeapRegion();
|
||||||
ASSERT(pt_heap_region.GetEndAddress() != 0);
|
ASSERT(pt_heap_region.GetEndAddress() != 0);
|
||||||
|
|
||||||
InitializeResourceManagers(pt_heap_region.GetAddress(), pt_heap_region.GetSize());
|
InitializeResourceManagers(kernel, pt_heap_region.GetAddress(),
|
||||||
|
pt_heap_region.GetSize());
|
||||||
}
|
}
|
||||||
|
|
||||||
RegisterHostThread();
|
RegisterHostThread();
|
||||||
@ -253,16 +260,82 @@ struct KernelCore::Impl {
|
|||||||
system.CoreTiming().ScheduleLoopingEvent(time_interval, time_interval, preemption_event);
|
system.CoreTiming().ScheduleLoopingEvent(time_interval, time_interval, preemption_event);
|
||||||
}
|
}
|
||||||
|
|
||||||
void InitializeResourceManagers(VAddr address, size_t size) {
|
void InitializeResourceManagers(KernelCore& kernel, VAddr address, size_t size) {
|
||||||
dynamic_page_manager = std::make_unique<KDynamicPageManager>();
|
// Ensure that the buffer is suitable for our use.
|
||||||
memory_block_heap = std::make_unique<KMemoryBlockSlabHeap>();
|
ASSERT(Common::IsAligned(address, PageSize));
|
||||||
app_memory_block_manager = std::make_unique<KMemoryBlockSlabManager>();
|
ASSERT(Common::IsAligned(size, PageSize));
|
||||||
|
|
||||||
dynamic_page_manager->Initialize(address, size);
|
// Ensure that we have space for our reference counts.
|
||||||
static constexpr size_t ApplicationMemoryBlockSlabHeapSize = 20000;
|
const size_t rc_size =
|
||||||
memory_block_heap->Initialize(dynamic_page_manager.get(),
|
Common::AlignUp(KPageTableSlabHeap::CalculateReferenceCountSize(size), PageSize);
|
||||||
|
ASSERT(rc_size < size);
|
||||||
|
size -= rc_size;
|
||||||
|
|
||||||
|
// Initialize the resource managers' shared page manager.
|
||||||
|
resource_manager_page_manager = std::make_unique<KDynamicPageManager>();
|
||||||
|
resource_manager_page_manager->Initialize(
|
||||||
|
address, size, std::max<size_t>(PageSize, KPageBufferSlabHeap::BufferSize));
|
||||||
|
|
||||||
|
// Initialize the KPageBuffer slab heap.
|
||||||
|
page_buffer_slab_heap.Initialize(system);
|
||||||
|
|
||||||
|
// Initialize the fixed-size slabheaps.
|
||||||
|
app_memory_block_heap = std::make_unique<KMemoryBlockSlabHeap>();
|
||||||
|
sys_memory_block_heap = std::make_unique<KMemoryBlockSlabHeap>();
|
||||||
|
block_info_heap = std::make_unique<KBlockInfoSlabHeap>();
|
||||||
|
app_memory_block_heap->Initialize(resource_manager_page_manager.get(),
|
||||||
ApplicationMemoryBlockSlabHeapSize);
|
ApplicationMemoryBlockSlabHeapSize);
|
||||||
app_memory_block_manager->Initialize(nullptr, memory_block_heap.get());
|
sys_memory_block_heap->Initialize(resource_manager_page_manager.get(),
|
||||||
|
SystemMemoryBlockSlabHeapSize);
|
||||||
|
block_info_heap->Initialize(resource_manager_page_manager.get(), BlockInfoSlabHeapSize);
|
||||||
|
|
||||||
|
// Reserve all but a fixed number of remaining pages for the page table heap.
|
||||||
|
const size_t num_pt_pages = resource_manager_page_manager->GetCount() -
|
||||||
|
resource_manager_page_manager->GetUsed() -
|
||||||
|
ReservedDynamicPageCount;
|
||||||
|
page_table_heap = std::make_unique<KPageTableSlabHeap>();
|
||||||
|
|
||||||
|
// TODO(bunnei): Pass in address once we support kernel virtual memory allocations.
|
||||||
|
page_table_heap->Initialize(
|
||||||
|
resource_manager_page_manager.get(), num_pt_pages,
|
||||||
|
/*GetPointer<KPageTableManager::RefCount>(address + size)*/ nullptr);
|
||||||
|
|
||||||
|
// Setup the slab managers.
|
||||||
|
KDynamicPageManager* const app_dynamic_page_manager = nullptr;
|
||||||
|
KDynamicPageManager* const sys_dynamic_page_manager =
|
||||||
|
/*KTargetSystem::IsDynamicResourceLimitsEnabled()*/ true
|
||||||
|
? resource_manager_page_manager.get()
|
||||||
|
: nullptr;
|
||||||
|
app_memory_block_manager = std::make_unique<KMemoryBlockSlabManager>();
|
||||||
|
sys_memory_block_manager = std::make_unique<KMemoryBlockSlabManager>();
|
||||||
|
app_block_info_manager = std::make_unique<KBlockInfoManager>();
|
||||||
|
sys_block_info_manager = std::make_unique<KBlockInfoManager>();
|
||||||
|
app_page_table_manager = std::make_unique<KPageTableManager>();
|
||||||
|
sys_page_table_manager = std::make_unique<KPageTableManager>();
|
||||||
|
|
||||||
|
app_memory_block_manager->Initialize(app_dynamic_page_manager, app_memory_block_heap.get());
|
||||||
|
sys_memory_block_manager->Initialize(sys_dynamic_page_manager, sys_memory_block_heap.get());
|
||||||
|
|
||||||
|
app_block_info_manager->Initialize(app_dynamic_page_manager, block_info_heap.get());
|
||||||
|
sys_block_info_manager->Initialize(sys_dynamic_page_manager, block_info_heap.get());
|
||||||
|
|
||||||
|
app_page_table_manager->Initialize(app_dynamic_page_manager, page_table_heap.get());
|
||||||
|
sys_page_table_manager->Initialize(sys_dynamic_page_manager, page_table_heap.get());
|
||||||
|
|
||||||
|
// Check that we have the correct number of dynamic pages available.
|
||||||
|
ASSERT(resource_manager_page_manager->GetCount() -
|
||||||
|
resource_manager_page_manager->GetUsed() ==
|
||||||
|
ReservedDynamicPageCount);
|
||||||
|
|
||||||
|
// Create the system page table managers.
|
||||||
|
app_system_resource = std::make_unique<KSystemResource>(kernel);
|
||||||
|
sys_system_resource = std::make_unique<KSystemResource>(kernel);
|
||||||
|
|
||||||
|
// Set the managers for the system resources.
|
||||||
|
app_system_resource->SetManagers(*app_memory_block_manager, *app_block_info_manager,
|
||||||
|
*app_page_table_manager);
|
||||||
|
sys_system_resource->SetManagers(*sys_memory_block_manager, *sys_block_info_manager,
|
||||||
|
*sys_page_table_manager);
|
||||||
}
|
}
|
||||||
|
|
||||||
void InitializeShutdownThreads() {
|
void InitializeShutdownThreads() {
|
||||||
@ -446,6 +519,9 @@ struct KernelCore::Impl {
|
|||||||
ASSERT(memory_layout->GetVirtualMemoryRegionTree().Insert(
|
ASSERT(memory_layout->GetVirtualMemoryRegionTree().Insert(
|
||||||
misc_region_start, misc_region_size, KMemoryRegionType_KernelMisc));
|
misc_region_start, misc_region_size, KMemoryRegionType_KernelMisc));
|
||||||
|
|
||||||
|
// Determine if we'll use extra thread resources.
|
||||||
|
const bool use_extra_resources = KSystemControl::Init::ShouldIncreaseThreadResourceLimit();
|
||||||
|
|
||||||
// Setup the stack region.
|
// Setup the stack region.
|
||||||
constexpr size_t StackRegionSize = 14_MiB;
|
constexpr size_t StackRegionSize = 14_MiB;
|
||||||
constexpr size_t StackRegionAlign = KernelAslrAlignment;
|
constexpr size_t StackRegionAlign = KernelAslrAlignment;
|
||||||
@ -456,7 +532,8 @@ struct KernelCore::Impl {
|
|||||||
stack_region_start, StackRegionSize, KMemoryRegionType_KernelStack));
|
stack_region_start, StackRegionSize, KMemoryRegionType_KernelStack));
|
||||||
|
|
||||||
// Determine the size of the resource region.
|
// Determine the size of the resource region.
|
||||||
const size_t resource_region_size = memory_layout->GetResourceRegionSizeForInit();
|
const size_t resource_region_size =
|
||||||
|
memory_layout->GetResourceRegionSizeForInit(use_extra_resources);
|
||||||
|
|
||||||
// Determine the size of the slab region.
|
// Determine the size of the slab region.
|
||||||
const size_t slab_region_size =
|
const size_t slab_region_size =
|
||||||
@ -751,6 +828,8 @@ struct KernelCore::Impl {
|
|||||||
Init::KSlabResourceCounts slab_resource_counts{};
|
Init::KSlabResourceCounts slab_resource_counts{};
|
||||||
KResourceLimit* system_resource_limit{};
|
KResourceLimit* system_resource_limit{};
|
||||||
|
|
||||||
|
KPageBufferSlabHeap page_buffer_slab_heap;
|
||||||
|
|
||||||
std::shared_ptr<Core::Timing::EventType> preemption_event;
|
std::shared_ptr<Core::Timing::EventType> preemption_event;
|
||||||
|
|
||||||
// This is the kernel's handle table or supervisor handle table which
|
// This is the kernel's handle table or supervisor handle table which
|
||||||
@ -776,10 +855,20 @@ struct KernelCore::Impl {
|
|||||||
// Kernel memory management
|
// Kernel memory management
|
||||||
std::unique_ptr<KMemoryManager> memory_manager;
|
std::unique_ptr<KMemoryManager> memory_manager;
|
||||||
|
|
||||||
// Dynamic slab managers
|
// Resource managers
|
||||||
std::unique_ptr<KDynamicPageManager> dynamic_page_manager;
|
std::unique_ptr<KDynamicPageManager> resource_manager_page_manager;
|
||||||
std::unique_ptr<KMemoryBlockSlabHeap> memory_block_heap;
|
std::unique_ptr<KPageTableSlabHeap> page_table_heap;
|
||||||
|
std::unique_ptr<KMemoryBlockSlabHeap> app_memory_block_heap;
|
||||||
|
std::unique_ptr<KMemoryBlockSlabHeap> sys_memory_block_heap;
|
||||||
|
std::unique_ptr<KBlockInfoSlabHeap> block_info_heap;
|
||||||
|
std::unique_ptr<KPageTableManager> app_page_table_manager;
|
||||||
|
std::unique_ptr<KPageTableManager> sys_page_table_manager;
|
||||||
std::unique_ptr<KMemoryBlockSlabManager> app_memory_block_manager;
|
std::unique_ptr<KMemoryBlockSlabManager> app_memory_block_manager;
|
||||||
|
std::unique_ptr<KMemoryBlockSlabManager> sys_memory_block_manager;
|
||||||
|
std::unique_ptr<KBlockInfoManager> app_block_info_manager;
|
||||||
|
std::unique_ptr<KBlockInfoManager> sys_block_info_manager;
|
||||||
|
std::unique_ptr<KSystemResource> app_system_resource;
|
||||||
|
std::unique_ptr<KSystemResource> sys_system_resource;
|
||||||
|
|
||||||
// Shared memory for services
|
// Shared memory for services
|
||||||
Kernel::KSharedMemory* hid_shared_mem{};
|
Kernel::KSharedMemory* hid_shared_mem{};
|
||||||
@ -1057,12 +1146,12 @@ const KMemoryManager& KernelCore::MemoryManager() const {
|
|||||||
return *impl->memory_manager;
|
return *impl->memory_manager;
|
||||||
}
|
}
|
||||||
|
|
||||||
KMemoryBlockSlabManager& KernelCore::GetApplicationMemoryBlockManager() {
|
KSystemResource& KernelCore::GetSystemSystemResource() {
|
||||||
return *impl->app_memory_block_manager;
|
return *impl->sys_system_resource;
|
||||||
}
|
}
|
||||||
|
|
||||||
const KMemoryBlockSlabManager& KernelCore::GetApplicationMemoryBlockManager() const {
|
const KSystemResource& KernelCore::GetSystemSystemResource() const {
|
||||||
return *impl->app_memory_block_manager;
|
return *impl->sys_system_resource;
|
||||||
}
|
}
|
||||||
|
|
||||||
Kernel::KSharedMemory& KernelCore::GetHidSharedMem() {
|
Kernel::KSharedMemory& KernelCore::GetHidSharedMem() {
|
||||||
|
@ -34,13 +34,16 @@ class KClientPort;
|
|||||||
class GlobalSchedulerContext;
|
class GlobalSchedulerContext;
|
||||||
class KAutoObjectWithListContainer;
|
class KAutoObjectWithListContainer;
|
||||||
class KClientSession;
|
class KClientSession;
|
||||||
|
class KDebug;
|
||||||
|
class KDynamicPageManager;
|
||||||
class KEvent;
|
class KEvent;
|
||||||
|
class KEventInfo;
|
||||||
class KHandleTable;
|
class KHandleTable;
|
||||||
class KLinkedListNode;
|
class KLinkedListNode;
|
||||||
class KMemoryBlockSlabManager;
|
|
||||||
class KMemoryLayout;
|
class KMemoryLayout;
|
||||||
class KMemoryManager;
|
class KMemoryManager;
|
||||||
class KPageBuffer;
|
class KPageBuffer;
|
||||||
|
class KPageBufferSlabHeap;
|
||||||
class KPort;
|
class KPort;
|
||||||
class KProcess;
|
class KProcess;
|
||||||
class KResourceLimit;
|
class KResourceLimit;
|
||||||
@ -51,6 +54,7 @@ class KSession;
|
|||||||
class KSessionRequest;
|
class KSessionRequest;
|
||||||
class KSharedMemory;
|
class KSharedMemory;
|
||||||
class KSharedMemoryInfo;
|
class KSharedMemoryInfo;
|
||||||
|
class KSecureSystemResource;
|
||||||
class KThread;
|
class KThread;
|
||||||
class KThreadLocalPage;
|
class KThreadLocalPage;
|
||||||
class KTransferMemory;
|
class KTransferMemory;
|
||||||
@ -244,11 +248,11 @@ public:
|
|||||||
/// Gets the virtual memory manager for the kernel.
|
/// Gets the virtual memory manager for the kernel.
|
||||||
const KMemoryManager& MemoryManager() const;
|
const KMemoryManager& MemoryManager() const;
|
||||||
|
|
||||||
/// Gets the application memory block manager for the kernel.
|
/// Gets the system resource manager.
|
||||||
KMemoryBlockSlabManager& GetApplicationMemoryBlockManager();
|
KSystemResource& GetSystemSystemResource();
|
||||||
|
|
||||||
/// Gets the application memory block manager for the kernel.
|
/// Gets the system resource manager.
|
||||||
const KMemoryBlockSlabManager& GetApplicationMemoryBlockManager() const;
|
const KSystemResource& GetSystemSystemResource() const;
|
||||||
|
|
||||||
/// Gets the shared memory object for HID services.
|
/// Gets the shared memory object for HID services.
|
||||||
Kernel::KSharedMemory& GetHidSharedMem();
|
Kernel::KSharedMemory& GetHidSharedMem();
|
||||||
@ -364,6 +368,12 @@ public:
|
|||||||
return slab_heap_container->thread_local_page;
|
return slab_heap_container->thread_local_page;
|
||||||
} else if constexpr (std::is_same_v<T, KSessionRequest>) {
|
} else if constexpr (std::is_same_v<T, KSessionRequest>) {
|
||||||
return slab_heap_container->session_request;
|
return slab_heap_container->session_request;
|
||||||
|
} else if constexpr (std::is_same_v<T, KSecureSystemResource>) {
|
||||||
|
return slab_heap_container->secure_system_resource;
|
||||||
|
} else if constexpr (std::is_same_v<T, KEventInfo>) {
|
||||||
|
return slab_heap_container->event_info;
|
||||||
|
} else if constexpr (std::is_same_v<T, KDebug>) {
|
||||||
|
return slab_heap_container->debug;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -427,6 +437,9 @@ private:
|
|||||||
KSlabHeap<KPageBuffer> page_buffer;
|
KSlabHeap<KPageBuffer> page_buffer;
|
||||||
KSlabHeap<KThreadLocalPage> thread_local_page;
|
KSlabHeap<KThreadLocalPage> thread_local_page;
|
||||||
KSlabHeap<KSessionRequest> session_request;
|
KSlabHeap<KSessionRequest> session_request;
|
||||||
|
KSlabHeap<KSecureSystemResource> secure_system_resource;
|
||||||
|
KSlabHeap<KEventInfo> event_info;
|
||||||
|
KSlabHeap<KDebug> debug;
|
||||||
};
|
};
|
||||||
|
|
||||||
std::unique_ptr<SlabHeapContainer> slab_heap_container;
|
std::unique_ptr<SlabHeapContainer> slab_heap_container;
|
||||||
|
@ -52,6 +52,84 @@ public:
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template <typename Derived, typename Base>
|
||||||
|
class KAutoObjectWithSlabHeap : public Base {
|
||||||
|
static_assert(std::is_base_of<KAutoObject, Base>::value);
|
||||||
|
|
||||||
|
private:
|
||||||
|
static Derived* Allocate(KernelCore& kernel) {
|
||||||
|
return kernel.SlabHeap<Derived>().Allocate(kernel);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void Free(KernelCore& kernel, Derived* obj) {
|
||||||
|
kernel.SlabHeap<Derived>().Free(obj);
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit KAutoObjectWithSlabHeap(KernelCore& kernel_) : Base(kernel_), kernel(kernel_) {}
|
||||||
|
virtual ~KAutoObjectWithSlabHeap() = default;
|
||||||
|
|
||||||
|
virtual void Destroy() override {
|
||||||
|
const bool is_initialized = this->IsInitialized();
|
||||||
|
uintptr_t arg = 0;
|
||||||
|
if (is_initialized) {
|
||||||
|
arg = this->GetPostDestroyArgument();
|
||||||
|
this->Finalize();
|
||||||
|
}
|
||||||
|
Free(kernel, static_cast<Derived*>(this));
|
||||||
|
if (is_initialized) {
|
||||||
|
Derived::PostDestroy(arg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual bool IsInitialized() const {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
virtual uintptr_t GetPostDestroyArgument() const {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t GetSlabIndex() const {
|
||||||
|
return SlabHeap<Derived>(kernel).GetObjectIndex(static_cast<const Derived*>(this));
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
static void InitializeSlabHeap(KernelCore& kernel, void* memory, size_t memory_size) {
|
||||||
|
kernel.SlabHeap<Derived>().Initialize(memory, memory_size);
|
||||||
|
}
|
||||||
|
|
||||||
|
static Derived* Create(KernelCore& kernel) {
|
||||||
|
Derived* obj = Allocate(kernel);
|
||||||
|
if (obj != nullptr) {
|
||||||
|
KAutoObject::Create(obj);
|
||||||
|
}
|
||||||
|
return obj;
|
||||||
|
}
|
||||||
|
|
||||||
|
static size_t GetObjectSize(KernelCore& kernel) {
|
||||||
|
return kernel.SlabHeap<Derived>().GetObjectSize();
|
||||||
|
}
|
||||||
|
|
||||||
|
static size_t GetSlabHeapSize(KernelCore& kernel) {
|
||||||
|
return kernel.SlabHeap<Derived>().GetSlabHeapSize();
|
||||||
|
}
|
||||||
|
|
||||||
|
static size_t GetPeakIndex(KernelCore& kernel) {
|
||||||
|
return kernel.SlabHeap<Derived>().GetPeakIndex();
|
||||||
|
}
|
||||||
|
|
||||||
|
static uintptr_t GetSlabHeapAddress(KernelCore& kernel) {
|
||||||
|
return kernel.SlabHeap<Derived>().GetSlabHeapAddress();
|
||||||
|
}
|
||||||
|
|
||||||
|
static size_t GetNumRemaining(KernelCore& kernel) {
|
||||||
|
return kernel.SlabHeap<Derived>().GetNumRemaining();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
KernelCore& kernel;
|
||||||
|
};
|
||||||
|
|
||||||
template <typename Derived, typename Base>
|
template <typename Derived, typename Base>
|
||||||
class KAutoObjectWithSlabHeapAndContainer : public Base {
|
class KAutoObjectWithSlabHeapAndContainer : public Base {
|
||||||
static_assert(std::is_base_of<KAutoObjectWithList, Base>::value);
|
static_assert(std::is_base_of<KAutoObjectWithList, Base>::value);
|
||||||
|
@ -2247,7 +2247,7 @@ static u64 GetSystemTick(Core::System& system) {
|
|||||||
auto& core_timing = system.CoreTiming();
|
auto& core_timing = system.CoreTiming();
|
||||||
|
|
||||||
// Returns the value of cntpct_el0 (https://switchbrew.org/wiki/SVC#svcGetSystemTick)
|
// Returns the value of cntpct_el0 (https://switchbrew.org/wiki/SVC#svcGetSystemTick)
|
||||||
const u64 result{system.CoreTiming().GetClockTicks()};
|
const u64 result{core_timing.GetClockTicks()};
|
||||||
|
|
||||||
if (!system.Kernel().IsMulticore()) {
|
if (!system.Kernel().IsMulticore()) {
|
||||||
core_timing.AddTicks(400U);
|
core_timing.AddTicks(400U);
|
||||||
|
@ -37,6 +37,7 @@ constexpr Result ResultInvalidState{ErrorModule::Kernel, 125};
|
|||||||
constexpr Result ResultReservedUsed{ErrorModule::Kernel, 126};
|
constexpr Result ResultReservedUsed{ErrorModule::Kernel, 126};
|
||||||
constexpr Result ResultPortClosed{ErrorModule::Kernel, 131};
|
constexpr Result ResultPortClosed{ErrorModule::Kernel, 131};
|
||||||
constexpr Result ResultLimitReached{ErrorModule::Kernel, 132};
|
constexpr Result ResultLimitReached{ErrorModule::Kernel, 132};
|
||||||
|
constexpr Result ResultOutOfAddressSpace{ErrorModule::Kernel, 259};
|
||||||
constexpr Result ResultInvalidId{ErrorModule::Kernel, 519};
|
constexpr Result ResultInvalidId{ErrorModule::Kernel, 519};
|
||||||
|
|
||||||
} // namespace Kernel
|
} // namespace Kernel
|
||||||
|
@ -22,8 +22,8 @@ enum class MemoryState : u32 {
|
|||||||
Ipc = 0x0A,
|
Ipc = 0x0A,
|
||||||
Stack = 0x0B,
|
Stack = 0x0B,
|
||||||
ThreadLocal = 0x0C,
|
ThreadLocal = 0x0C,
|
||||||
Transferred = 0x0D,
|
Transfered = 0x0D,
|
||||||
SharedTransferred = 0x0E,
|
SharedTransfered = 0x0E,
|
||||||
SharedCode = 0x0F,
|
SharedCode = 0x0F,
|
||||||
Inaccessible = 0x10,
|
Inaccessible = 0x10,
|
||||||
NonSecureIpc = 0x11,
|
NonSecureIpc = 0x11,
|
||||||
@ -32,6 +32,7 @@ enum class MemoryState : u32 {
|
|||||||
GeneratedCode = 0x14,
|
GeneratedCode = 0x14,
|
||||||
CodeOut = 0x15,
|
CodeOut = 0x15,
|
||||||
Coverage = 0x16,
|
Coverage = 0x16,
|
||||||
|
Insecure = 0x17,
|
||||||
};
|
};
|
||||||
DECLARE_ENUM_FLAG_OPERATORS(MemoryState);
|
DECLARE_ENUM_FLAG_OPERATORS(MemoryState);
|
||||||
|
|
||||||
@ -83,6 +84,13 @@ enum class YieldType : s64 {
|
|||||||
ToAnyThread = -2,
|
ToAnyThread = -2,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum class ThreadExitReason : u32 {
|
||||||
|
ExitThread = 0,
|
||||||
|
TerminateThread = 1,
|
||||||
|
ExitProcess = 2,
|
||||||
|
TerminateProcess = 3,
|
||||||
|
};
|
||||||
|
|
||||||
enum class ThreadActivity : u32 {
|
enum class ThreadActivity : u32 {
|
||||||
Runnable = 0,
|
Runnable = 0,
|
||||||
Paused = 1,
|
Paused = 1,
|
||||||
@ -108,6 +116,34 @@ enum class ProcessState : u32 {
|
|||||||
DebugBreak = 7,
|
DebugBreak = 7,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum class ProcessExitReason : u32 {
|
||||||
|
ExitProcess = 0,
|
||||||
|
TerminateProcess = 1,
|
||||||
|
Exception = 2,
|
||||||
|
};
|
||||||
|
|
||||||
constexpr inline size_t ThreadLocalRegionSize = 0x200;
|
constexpr inline size_t ThreadLocalRegionSize = 0x200;
|
||||||
|
|
||||||
|
// Debug types.
|
||||||
|
enum class DebugEvent : u32 {
|
||||||
|
CreateProcess = 0,
|
||||||
|
CreateThread = 1,
|
||||||
|
ExitProcess = 2,
|
||||||
|
ExitThread = 3,
|
||||||
|
Exception = 4,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class DebugException : u32 {
|
||||||
|
UndefinedInstruction = 0,
|
||||||
|
InstructionAbort = 1,
|
||||||
|
DataAbort = 2,
|
||||||
|
AlignmentFault = 3,
|
||||||
|
DebuggerAttached = 4,
|
||||||
|
BreakPoint = 5,
|
||||||
|
UserBreak = 6,
|
||||||
|
DebuggerBreak = 7,
|
||||||
|
UndefinedSystemCall = 8,
|
||||||
|
MemorySystemError = 9,
|
||||||
|
};
|
||||||
|
|
||||||
} // namespace Kernel::Svc
|
} // namespace Kernel::Svc
|
||||||
|
@ -423,16 +423,17 @@ constexpr void UpdateCurrentResultReference<const Result>(Result result_referenc
|
|||||||
} // namespace ResultImpl
|
} // namespace ResultImpl
|
||||||
|
|
||||||
#define DECLARE_CURRENT_RESULT_REFERENCE_AND_STORAGE(COUNTER_VALUE) \
|
#define DECLARE_CURRENT_RESULT_REFERENCE_AND_STORAGE(COUNTER_VALUE) \
|
||||||
[[maybe_unused]] constexpr bool HasPrevRef_##COUNTER_VALUE = \
|
[[maybe_unused]] constexpr bool CONCAT2(HasPrevRef_, COUNTER_VALUE) = \
|
||||||
std::same_as<decltype(__TmpCurrentResultReference), Result&>; \
|
std::same_as<decltype(__TmpCurrentResultReference), Result&>; \
|
||||||
[[maybe_unused]] auto& PrevRef_##COUNTER_VALUE = __TmpCurrentResultReference; \
|
[[maybe_unused]] Result CONCAT2(PrevRef_, COUNTER_VALUE) = __TmpCurrentResultReference; \
|
||||||
[[maybe_unused]] Result __tmp_result_##COUNTER_VALUE = ResultSuccess; \
|
[[maybe_unused]] Result CONCAT2(__tmp_result_, COUNTER_VALUE) = ResultSuccess; \
|
||||||
Result& __TmpCurrentResultReference = \
|
Result& __TmpCurrentResultReference = CONCAT2(HasPrevRef_, COUNTER_VALUE) \
|
||||||
HasPrevRef_##COUNTER_VALUE ? PrevRef_##COUNTER_VALUE : __tmp_result_##COUNTER_VALUE
|
? CONCAT2(PrevRef_, COUNTER_VALUE) \
|
||||||
|
: CONCAT2(__tmp_result_, COUNTER_VALUE)
|
||||||
|
|
||||||
#define ON_RESULT_RETURN_IMPL(...) \
|
#define ON_RESULT_RETURN_IMPL(...) \
|
||||||
static_assert(std::same_as<decltype(__TmpCurrentResultReference), Result&>); \
|
static_assert(std::same_as<decltype(__TmpCurrentResultReference), Result&>); \
|
||||||
auto RESULT_GUARD_STATE_##__COUNTER__ = \
|
auto CONCAT2(RESULT_GUARD_STATE_, __COUNTER__) = \
|
||||||
ResultImpl::ResultReferenceForScopedResultGuard<__VA_ARGS__>( \
|
ResultImpl::ResultReferenceForScopedResultGuard<__VA_ARGS__>( \
|
||||||
__TmpCurrentResultReference) + \
|
__TmpCurrentResultReference) + \
|
||||||
[&]()
|
[&]()
|
||||||
|
@ -126,10 +126,12 @@ NvResult nvmap::IocAlloc(const std::vector<u8>& input, std::vector<u8>& output)
|
|||||||
LOG_CRITICAL(Service_NVDRV, "Object failed to allocate, handle={:08X}", params.handle);
|
LOG_CRITICAL(Service_NVDRV, "Object failed to allocate, handle={:08X}", params.handle);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
bool is_out_io{};
|
||||||
ASSERT(system.CurrentProcess()
|
ASSERT(system.CurrentProcess()
|
||||||
->PageTable()
|
->PageTable()
|
||||||
.LockForMapDeviceAddressSpace(handle_description->address, handle_description->size,
|
.LockForMapDeviceAddressSpace(&is_out_io, handle_description->address,
|
||||||
Kernel::KMemoryPermission::None, true)
|
handle_description->size,
|
||||||
|
Kernel::KMemoryPermission::None, true, false)
|
||||||
.IsSuccess());
|
.IsSuccess());
|
||||||
std::memcpy(output.data(), ¶ms, sizeof(params));
|
std::memcpy(output.data(), ¶ms, sizeof(params));
|
||||||
return result;
|
return result;
|
||||||
|
Loading…
Reference in New Issue
Block a user