mirror of
https://github.com/yuzu-emu/yuzu.git
synced 2024-07-04 23:31:19 +01:00
hle: kernel: Move KMemoryRegion to its own module and update.
This commit is contained in:
parent
2e85ee250d
commit
778e0f8ec1
4 changed files with 322 additions and 31 deletions
|
@ -172,6 +172,7 @@ add_library(core STATIC
|
||||||
hle/kernel/k_memory_layout.h
|
hle/kernel/k_memory_layout.h
|
||||||
hle/kernel/k_memory_manager.cpp
|
hle/kernel/k_memory_manager.cpp
|
||||||
hle/kernel/k_memory_manager.h
|
hle/kernel/k_memory_manager.h
|
||||||
|
hle/kernel/k_memory_region.h
|
||||||
hle/kernel/k_page_bitmap.h
|
hle/kernel/k_page_bitmap.h
|
||||||
hle/kernel/k_page_heap.cpp
|
hle/kernel/k_page_heap.cpp
|
||||||
hle/kernel/k_page_heap.h
|
hle/kernel/k_page_heap.h
|
||||||
|
|
|
@ -6,6 +6,7 @@
|
||||||
|
|
||||||
#include "common/common_types.h"
|
#include "common/common_types.h"
|
||||||
#include "core/device_memory.h"
|
#include "core/device_memory.h"
|
||||||
|
#include "core/hle/kernel/k_memory_region.h"
|
||||||
|
|
||||||
namespace Kernel {
|
namespace Kernel {
|
||||||
|
|
||||||
|
@ -27,27 +28,6 @@ constexpr bool IsKernelAddress(VAddr address) {
|
||||||
return KernelVirtualAddressSpaceBase <= address && address < KernelVirtualAddressSpaceEnd;
|
return KernelVirtualAddressSpaceBase <= address && address < KernelVirtualAddressSpaceEnd;
|
||||||
}
|
}
|
||||||
|
|
||||||
class KMemoryRegion final {
|
|
||||||
friend class KMemoryLayout;
|
|
||||||
|
|
||||||
public:
|
|
||||||
constexpr PAddr StartAddress() const {
|
|
||||||
return start_address;
|
|
||||||
}
|
|
||||||
|
|
||||||
constexpr PAddr EndAddress() const {
|
|
||||||
return end_address;
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
|
||||||
constexpr KMemoryRegion() = default;
|
|
||||||
constexpr KMemoryRegion(PAddr start_address, PAddr end_address)
|
|
||||||
: start_address{start_address}, end_address{end_address} {}
|
|
||||||
|
|
||||||
const PAddr start_address{};
|
|
||||||
const PAddr end_address{};
|
|
||||||
};
|
|
||||||
|
|
||||||
class KMemoryLayout final {
|
class KMemoryLayout final {
|
||||||
public:
|
public:
|
||||||
constexpr const KMemoryRegion& Application() const {
|
constexpr const KMemoryRegion& Application() const {
|
||||||
|
|
310
src/core/hle/kernel/k_memory_region.h
Normal file
310
src/core/hle/kernel/k_memory_region.h
Normal file
|
@ -0,0 +1,310 @@
|
||||||
|
// Copyright 2021 yuzu Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "common/assert.h"
|
||||||
|
#include "common/common_types.h"
|
||||||
|
#include "common/intrusive_red_black_tree.h"
|
||||||
|
|
||||||
|
namespace Kernel {
|
||||||
|
|
||||||
|
class KMemoryRegion final : public Common::IntrusiveRedBlackTreeBaseNode<KMemoryRegion>,
|
||||||
|
NonCopyable {
|
||||||
|
friend class KMemoryLayout;
|
||||||
|
friend class KMemoryRegionTree;
|
||||||
|
|
||||||
|
public:
|
||||||
|
static constexpr int Compare(const KMemoryRegion& lhs, const KMemoryRegion& rhs) {
|
||||||
|
if (lhs.GetAddress() < rhs.GetAddress()) {
|
||||||
|
return -1;
|
||||||
|
} else if (lhs.GetAddress() <= rhs.GetLastAddress()) {
|
||||||
|
return 0;
|
||||||
|
} else {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr u64 GetAddress() const {
|
||||||
|
return address;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr u64 GetPairAddress() const {
|
||||||
|
return pair_address;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr u64 GetLastAddress() const {
|
||||||
|
return last_address;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr u64 GetEndAddress() const {
|
||||||
|
return GetLastAddress() + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr std::size_t GetSize() const {
|
||||||
|
return GetEndAddress() - GetAddress();
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr u32 GetAttributes() const {
|
||||||
|
return attributes;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr u32 GetType() const {
|
||||||
|
return type_id;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr void SetType(u32 type) {
|
||||||
|
ASSERT(this->CanDerive(type));
|
||||||
|
type_id = type;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr bool Contains(uintptr_t address) const {
|
||||||
|
ASSERT(this->GetEndAddress() != 0);
|
||||||
|
return this->GetAddress() <= address && address <= this->GetLastAddress();
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr bool IsDerivedFrom(u32 type) const {
|
||||||
|
return (this->GetType() | type) == this->GetType();
|
||||||
|
}
|
||||||
|
|
||||||
|
// constexpr bool HasTypeAttribute(KMemoryRegionAttr attr) const {
|
||||||
|
// return (this->GetType() | attr) == this->GetType();
|
||||||
|
//}
|
||||||
|
|
||||||
|
constexpr bool CanDerive(u32 type) const {
|
||||||
|
return (this->GetType() | type) == type;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr void SetPairAddress(u64 a) {
|
||||||
|
pair_address = a;
|
||||||
|
}
|
||||||
|
|
||||||
|
// constexpr void SetTypeAttribute(KMemoryRegionAttr attr) {
|
||||||
|
// type_id |= attr;
|
||||||
|
//}
|
||||||
|
|
||||||
|
private:
|
||||||
|
constexpr KMemoryRegion() = default;
|
||||||
|
constexpr KMemoryRegion(u64 address_, u64 last_address_)
|
||||||
|
: address{address_}, last_address{last_address_} {}
|
||||||
|
constexpr KMemoryRegion(u64 address_, u64 last_address_, u64 pair_address_, u32 attributes_,
|
||||||
|
u32 type_id_)
|
||||||
|
: address(address_), last_address(last_address_), pair_address(pair_address_),
|
||||||
|
attributes(attributes_), type_id(type_id_) {}
|
||||||
|
constexpr KMemoryRegion(u64 address_, u64 last_address_, u32 attributes_, u32 type_id_)
|
||||||
|
: KMemoryRegion(address_, last_address_, std::numeric_limits<uintptr_t>::max(), attributes_,
|
||||||
|
type_id_) {}
|
||||||
|
|
||||||
|
const u64 address{};
|
||||||
|
const u64 last_address{};
|
||||||
|
u64 pair_address{};
|
||||||
|
u32 attributes{};
|
||||||
|
u32 type_id{};
|
||||||
|
};
|
||||||
|
|
||||||
|
class KMemoryRegionTree final : NonCopyable {
|
||||||
|
public:
|
||||||
|
struct DerivedRegionExtents {
|
||||||
|
const KMemoryRegion* first_region{};
|
||||||
|
const KMemoryRegion* last_region{};
|
||||||
|
|
||||||
|
constexpr DerivedRegionExtents() = default;
|
||||||
|
|
||||||
|
constexpr u64 GetAddress() const {
|
||||||
|
return this->first_region->GetAddress();
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr u64 GetLastAddress() const {
|
||||||
|
return this->last_region->GetLastAddress();
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr u64 GetEndAddress() const {
|
||||||
|
return this->GetLastAddress() + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr size_t GetSize() const {
|
||||||
|
return this->GetEndAddress() - this->GetAddress();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
private:
|
||||||
|
using TreeType =
|
||||||
|
Common::IntrusiveRedBlackTreeBaseTraits<KMemoryRegion>::TreeType<KMemoryRegion>;
|
||||||
|
|
||||||
|
public:
|
||||||
|
using value_type = TreeType::value_type;
|
||||||
|
using size_type = TreeType::size_type;
|
||||||
|
using difference_type = TreeType::difference_type;
|
||||||
|
using pointer = TreeType::pointer;
|
||||||
|
using const_pointer = TreeType::const_pointer;
|
||||||
|
using reference = TreeType::reference;
|
||||||
|
using const_reference = TreeType::const_reference;
|
||||||
|
using iterator = TreeType::iterator;
|
||||||
|
using const_iterator = TreeType::const_iterator;
|
||||||
|
|
||||||
|
private:
|
||||||
|
TreeType m_tree{};
|
||||||
|
|
||||||
|
public:
|
||||||
|
constexpr KMemoryRegionTree() = default;
|
||||||
|
|
||||||
|
public:
|
||||||
|
KMemoryRegion* FindModifiable(u64 address) {
|
||||||
|
if (auto it = this->find(KMemoryRegion(address, address, 0, 0)); it != this->end()) {
|
||||||
|
return std::addressof(*it);
|
||||||
|
} else {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const KMemoryRegion* Find(u64 address) const {
|
||||||
|
if (auto it = this->find(KMemoryRegion(address, address, 0, 0)); it != this->cend()) {
|
||||||
|
return std::addressof(*it);
|
||||||
|
} else {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const KMemoryRegion* FindByType(u32 type_id) const {
|
||||||
|
for (auto it = this->cbegin(); it != this->cend(); ++it) {
|
||||||
|
if (it->GetType() == type_id) {
|
||||||
|
return std::addressof(*it);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
const KMemoryRegion* FindByTypeAndAttribute(u32 type_id, u32 attr) const {
|
||||||
|
for (auto it = this->cbegin(); it != this->cend(); ++it) {
|
||||||
|
if (it->GetType() == type_id && it->GetAttributes() == attr) {
|
||||||
|
return std::addressof(*it);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
const KMemoryRegion* FindFirstDerived(u32 type_id) const {
|
||||||
|
for (auto it = this->cbegin(); it != this->cend(); it++) {
|
||||||
|
if (it->IsDerivedFrom(type_id)) {
|
||||||
|
return std::addressof(*it);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
const KMemoryRegion* FindLastDerived(u32 type_id) const {
|
||||||
|
const KMemoryRegion* region = nullptr;
|
||||||
|
for (auto it = this->begin(); it != this->end(); it++) {
|
||||||
|
if (it->IsDerivedFrom(type_id)) {
|
||||||
|
region = std::addressof(*it);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return region;
|
||||||
|
}
|
||||||
|
|
||||||
|
DerivedRegionExtents GetDerivedRegionExtents(u32 type_id) const {
|
||||||
|
DerivedRegionExtents extents;
|
||||||
|
|
||||||
|
ASSERT(extents.first_region == nullptr);
|
||||||
|
ASSERT(extents.last_region == nullptr);
|
||||||
|
|
||||||
|
for (auto it = this->cbegin(); it != this->cend(); it++) {
|
||||||
|
if (it->IsDerivedFrom(type_id)) {
|
||||||
|
if (extents.first_region == nullptr) {
|
||||||
|
extents.first_region = std::addressof(*it);
|
||||||
|
}
|
||||||
|
extents.last_region = std::addressof(*it);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ASSERT(extents.first_region != nullptr);
|
||||||
|
ASSERT(extents.last_region != nullptr);
|
||||||
|
|
||||||
|
return extents;
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
void InsertDirectly(u64 address, u64 last_address, u32 attr = 0, u32 type_id = 0);
|
||||||
|
bool Insert(u64 address, size_t size, u32 type_id, u32 new_attr = 0, u32 old_attr = 0);
|
||||||
|
|
||||||
|
VAddr GetRandomAlignedRegion(size_t size, size_t alignment, u32 type_id);
|
||||||
|
|
||||||
|
VAddr GetRandomAlignedRegionWithGuard(size_t size, size_t alignment, u32 type_id,
|
||||||
|
size_t guard_size) {
|
||||||
|
return this->GetRandomAlignedRegion(size + 2 * guard_size, alignment, type_id) + guard_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
public:
|
||||||
|
// Iterator accessors.
|
||||||
|
iterator begin() {
|
||||||
|
return m_tree.begin();
|
||||||
|
}
|
||||||
|
|
||||||
|
const_iterator begin() const {
|
||||||
|
return m_tree.begin();
|
||||||
|
}
|
||||||
|
|
||||||
|
iterator end() {
|
||||||
|
return m_tree.end();
|
||||||
|
}
|
||||||
|
|
||||||
|
const_iterator end() const {
|
||||||
|
return m_tree.end();
|
||||||
|
}
|
||||||
|
|
||||||
|
const_iterator cbegin() const {
|
||||||
|
return this->begin();
|
||||||
|
}
|
||||||
|
|
||||||
|
const_iterator cend() const {
|
||||||
|
return this->end();
|
||||||
|
}
|
||||||
|
|
||||||
|
iterator iterator_to(reference ref) {
|
||||||
|
return m_tree.iterator_to(ref);
|
||||||
|
}
|
||||||
|
|
||||||
|
const_iterator iterator_to(const_reference ref) const {
|
||||||
|
return m_tree.iterator_to(ref);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Content management.
|
||||||
|
bool empty() const {
|
||||||
|
return m_tree.empty();
|
||||||
|
}
|
||||||
|
|
||||||
|
reference back() {
|
||||||
|
return m_tree.back();
|
||||||
|
}
|
||||||
|
|
||||||
|
const_reference back() const {
|
||||||
|
return m_tree.back();
|
||||||
|
}
|
||||||
|
|
||||||
|
reference front() {
|
||||||
|
return m_tree.front();
|
||||||
|
}
|
||||||
|
|
||||||
|
const_reference front() const {
|
||||||
|
return m_tree.front();
|
||||||
|
}
|
||||||
|
|
||||||
|
iterator insert(reference ref) {
|
||||||
|
return m_tree.insert(ref);
|
||||||
|
}
|
||||||
|
|
||||||
|
iterator erase(iterator it) {
|
||||||
|
return m_tree.erase(it);
|
||||||
|
}
|
||||||
|
|
||||||
|
iterator find(const_reference ref) const {
|
||||||
|
return m_tree.find(ref);
|
||||||
|
}
|
||||||
|
|
||||||
|
iterator nfind(const_reference ref) const {
|
||||||
|
return m_tree.nfind(ref);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Kernel
|
|
@ -275,22 +275,22 @@ struct KernelCore::Impl {
|
||||||
constexpr std::size_t font_size{0x1100000};
|
constexpr std::size_t font_size{0x1100000};
|
||||||
constexpr std::size_t irs_size{0x8000};
|
constexpr std::size_t irs_size{0x8000};
|
||||||
constexpr std::size_t time_size{0x1000};
|
constexpr std::size_t time_size{0x1000};
|
||||||
constexpr PAddr hid_addr{layout.System().StartAddress()};
|
constexpr PAddr hid_addr{layout.System().GetAddress()};
|
||||||
constexpr PAddr font_pa{layout.System().StartAddress() + hid_size};
|
constexpr PAddr font_pa{layout.System().GetAddress() + hid_size};
|
||||||
constexpr PAddr irs_addr{layout.System().StartAddress() + hid_size + font_size};
|
constexpr PAddr irs_addr{layout.System().GetAddress() + hid_size + font_size};
|
||||||
constexpr PAddr time_addr{layout.System().StartAddress() + hid_size + font_size + irs_size};
|
constexpr PAddr time_addr{layout.System().GetAddress() + hid_size + font_size + irs_size};
|
||||||
|
|
||||||
// Initialize memory manager
|
// Initialize memory manager
|
||||||
memory_manager = std::make_unique<KMemoryManager>();
|
memory_manager = std::make_unique<KMemoryManager>();
|
||||||
memory_manager->InitializeManager(KMemoryManager::Pool::Application,
|
memory_manager->InitializeManager(KMemoryManager::Pool::Application,
|
||||||
layout.Application().StartAddress(),
|
layout.Application().GetAddress(),
|
||||||
layout.Application().EndAddress());
|
layout.Application().GetLastAddress());
|
||||||
memory_manager->InitializeManager(KMemoryManager::Pool::Applet,
|
memory_manager->InitializeManager(KMemoryManager::Pool::Applet,
|
||||||
layout.Applet().StartAddress(),
|
layout.Applet().GetAddress(),
|
||||||
layout.Applet().EndAddress());
|
layout.Applet().GetLastAddress());
|
||||||
memory_manager->InitializeManager(KMemoryManager::Pool::System,
|
memory_manager->InitializeManager(KMemoryManager::Pool::System,
|
||||||
layout.System().StartAddress(),
|
layout.System().GetAddress(),
|
||||||
layout.System().EndAddress());
|
layout.System().GetLastAddress());
|
||||||
|
|
||||||
hid_shared_mem = Kernel::KSharedMemory::Create(
|
hid_shared_mem = Kernel::KSharedMemory::Create(
|
||||||
system.Kernel(), system.DeviceMemory(), nullptr, {hid_addr, hid_size / PageSize},
|
system.Kernel(), system.DeviceMemory(), nullptr, {hid_addr, hid_size / PageSize},
|
||||||
|
|
Loading…
Reference in a new issue