Browse Source
kernel: memory: Add MemoryBlock class, for managing memory blocks and their state.
pull/15/merge
kernel: memory: Add MemoryBlock class, for managing memory blocks and their state.
pull/15/merge
2 changed files with 316 additions and 0 deletions
@ -0,0 +1,315 @@ |
|||
// Copyright 2020 yuzu Emulator Project |
|||
// Licensed under GPLv2 or any later version |
|||
// Refer to the license.txt file included. |
|||
|
|||
#pragma once |
|||
|
|||
#include "common/alignment.h" |
|||
#include "common/assert.h" |
|||
#include "common/common_types.h" |
|||
#include "core/hle/kernel/memory/memory_types.h" |
|||
#include "core/hle/kernel/svc_types.h" |
|||
|
|||
namespace Kernel::Memory { |
|||
|
|||
enum class MemoryState : u32 { |
|||
None = 0, |
|||
Mask = 0xFFFFFFFF, // TODO(bunnei): This should probable be 0xFF |
|||
All = ~None, |
|||
|
|||
FlagCanReprotect = (1 << 8), |
|||
FlagCanDebug = (1 << 9), |
|||
FlagCanUseIpc = (1 << 10), |
|||
FlagCanUseNonDeviceIpc = (1 << 11), |
|||
FlagCanUseNonSecureIpc = (1 << 12), |
|||
FlagMapped = (1 << 13), |
|||
FlagCode = (1 << 14), |
|||
FlagCanAlias = (1 << 15), |
|||
FlagCanCodeAlias = (1 << 16), |
|||
FlagCanTransfer = (1 << 17), |
|||
FlagCanQueryPhysical = (1 << 18), |
|||
FlagCanDeviceMap = (1 << 19), |
|||
FlagCanAlignedDeviceMap = (1 << 20), |
|||
FlagCanIpcUserBuffer = (1 << 21), |
|||
FlagReferenceCounted = (1 << 22), |
|||
FlagCanMapProcess = (1 << 23), |
|||
FlagCanChangeAttribute = (1 << 24), |
|||
FlagCanCodeMemory = (1 << 25), |
|||
|
|||
FlagsData = FlagCanReprotect | FlagCanUseIpc | FlagCanUseNonDeviceIpc | FlagCanUseNonSecureIpc | |
|||
FlagMapped | FlagCanAlias | FlagCanTransfer | FlagCanQueryPhysical | |
|||
FlagCanDeviceMap | FlagCanAlignedDeviceMap | FlagCanIpcUserBuffer | |
|||
FlagReferenceCounted | FlagCanChangeAttribute, |
|||
|
|||
FlagsCode = FlagCanDebug | FlagCanUseIpc | FlagCanUseNonDeviceIpc | FlagCanUseNonSecureIpc | |
|||
FlagMapped | FlagCode | FlagCanQueryPhysical | FlagCanDeviceMap | |
|||
FlagCanAlignedDeviceMap | FlagReferenceCounted, |
|||
|
|||
FlagsMisc = FlagMapped | FlagReferenceCounted | FlagCanQueryPhysical | FlagCanDeviceMap, |
|||
|
|||
Free = static_cast<u32>(Svc::MemoryState::Free), |
|||
Io = static_cast<u32>(Svc::MemoryState::Io) | FlagMapped, |
|||
Static = static_cast<u32>(Svc::MemoryState::Static) | FlagMapped | FlagCanQueryPhysical, |
|||
Code = static_cast<u32>(Svc::MemoryState::Code) | FlagsCode | FlagCanMapProcess, |
|||
CodeData = static_cast<u32>(Svc::MemoryState::CodeData) | FlagsData | FlagCanMapProcess | |
|||
FlagCanCodeMemory, |
|||
Shared = static_cast<u32>(Svc::MemoryState::Shared) | FlagMapped | FlagReferenceCounted, |
|||
Normal = static_cast<u32>(Svc::MemoryState::Normal) | FlagsData | FlagCanCodeMemory, |
|||
|
|||
AliasCode = static_cast<u32>(Svc::MemoryState::AliasCode) | FlagsCode | FlagCanMapProcess | |
|||
FlagCanCodeAlias, |
|||
AliasCodeData = static_cast<u32>(Svc::MemoryState::AliasCodeData) | FlagsData | |
|||
FlagCanMapProcess | FlagCanCodeAlias | FlagCanCodeMemory, |
|||
|
|||
Ipc = static_cast<u32>(Svc::MemoryState::Ipc) | FlagsMisc | FlagCanAlignedDeviceMap | |
|||
FlagCanUseIpc | FlagCanUseNonSecureIpc | FlagCanUseNonDeviceIpc, |
|||
|
|||
Stack = static_cast<u32>(Svc::MemoryState::Stack) | FlagsMisc | FlagCanAlignedDeviceMap | |
|||
FlagCanUseIpc | FlagCanUseNonSecureIpc | FlagCanUseNonDeviceIpc, |
|||
|
|||
ThreadLocal = |
|||
static_cast<u32>(Svc::MemoryState::ThreadLocal) | FlagMapped | FlagReferenceCounted, |
|||
|
|||
Transfered = static_cast<u32>(Svc::MemoryState::Transfered) | FlagsMisc | |
|||
FlagCanAlignedDeviceMap | FlagCanChangeAttribute | FlagCanUseIpc | |
|||
FlagCanUseNonSecureIpc | FlagCanUseNonDeviceIpc, |
|||
|
|||
SharedTransfered = static_cast<u32>(Svc::MemoryState::SharedTransfered) | FlagsMisc | |
|||
FlagCanAlignedDeviceMap | FlagCanUseNonSecureIpc | FlagCanUseNonDeviceIpc, |
|||
|
|||
SharedCode = static_cast<u32>(Svc::MemoryState::SharedCode) | FlagMapped | |
|||
FlagReferenceCounted | FlagCanUseNonSecureIpc | FlagCanUseNonDeviceIpc, |
|||
|
|||
Inaccessible = static_cast<u32>(Svc::MemoryState::Inaccessible), |
|||
|
|||
NonSecureIpc = static_cast<u32>(Svc::MemoryState::NonSecureIpc) | FlagsMisc | |
|||
FlagCanAlignedDeviceMap | FlagCanUseNonSecureIpc | FlagCanUseNonDeviceIpc, |
|||
|
|||
NonDeviceIpc = |
|||
static_cast<u32>(Svc::MemoryState::NonDeviceIpc) | FlagsMisc | FlagCanUseNonDeviceIpc, |
|||
|
|||
Kernel = static_cast<u32>(Svc::MemoryState::Kernel) | FlagMapped, |
|||
|
|||
GeneratedCode = static_cast<u32>(Svc::MemoryState::GeneratedCode) | FlagMapped | |
|||
FlagReferenceCounted | FlagCanDebug, |
|||
CodeOut = static_cast<u32>(Svc::MemoryState::CodeOut) | FlagMapped | FlagReferenceCounted, |
|||
}; |
|||
DECLARE_ENUM_FLAG_OPERATORS(MemoryState); |
|||
|
|||
static_assert(static_cast<u32>(MemoryState::Free) == 0x00000000); |
|||
static_assert(static_cast<u32>(MemoryState::Io) == 0x00002001); |
|||
static_assert(static_cast<u32>(MemoryState::Static) == 0x00042002); |
|||
static_assert(static_cast<u32>(MemoryState::Code) == 0x00DC7E03); |
|||
static_assert(static_cast<u32>(MemoryState::CodeData) == 0x03FEBD04); |
|||
static_assert(static_cast<u32>(MemoryState::Normal) == 0x037EBD05); |
|||
static_assert(static_cast<u32>(MemoryState::Shared) == 0x00402006); |
|||
static_assert(static_cast<u32>(MemoryState::AliasCode) == 0x00DD7E08); |
|||
static_assert(static_cast<u32>(MemoryState::AliasCodeData) == 0x03FFBD09); |
|||
static_assert(static_cast<u32>(MemoryState::Ipc) == 0x005C3C0A); |
|||
static_assert(static_cast<u32>(MemoryState::Stack) == 0x005C3C0B); |
|||
static_assert(static_cast<u32>(MemoryState::ThreadLocal) == 0x0040200C); |
|||
static_assert(static_cast<u32>(MemoryState::Transfered) == 0x015C3C0D); |
|||
static_assert(static_cast<u32>(MemoryState::SharedTransfered) == 0x005C380E); |
|||
static_assert(static_cast<u32>(MemoryState::SharedCode) == 0x0040380F); |
|||
static_assert(static_cast<u32>(MemoryState::Inaccessible) == 0x00000010); |
|||
static_assert(static_cast<u32>(MemoryState::NonSecureIpc) == 0x005C3811); |
|||
static_assert(static_cast<u32>(MemoryState::NonDeviceIpc) == 0x004C2812); |
|||
static_assert(static_cast<u32>(MemoryState::Kernel) == 0x00002013); |
|||
static_assert(static_cast<u32>(MemoryState::GeneratedCode) == 0x00402214); |
|||
static_assert(static_cast<u32>(MemoryState::CodeOut) == 0x00402015); |
|||
|
|||
enum class MemoryPermission : u8 { |
|||
None = 0, |
|||
Mask = 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), |
|||
}; |
|||
DECLARE_ENUM_FLAG_OPERATORS(MemoryPermission); |
|||
|
|||
enum class MemoryAttribute : u8 { |
|||
None = 0x00, |
|||
Mask = 0x7F, |
|||
All = Mask, |
|||
DontCareMask = 0x80, |
|||
|
|||
Locked = static_cast<u8>(Svc::MemoryAttribute::Locked), |
|||
IpcLocked = static_cast<u8>(Svc::MemoryAttribute::IpcLocked), |
|||
DeviceShared = static_cast<u8>(Svc::MemoryAttribute::DeviceShared), |
|||
Uncached = static_cast<u8>(Svc::MemoryAttribute::Uncached), |
|||
|
|||
IpcAndDeviceMapped = IpcLocked | DeviceShared, |
|||
LockedAndIpcLocked = Locked | IpcLocked, |
|||
DeviceSharedAndUncached = DeviceShared | Uncached |
|||
}; |
|||
DECLARE_ENUM_FLAG_OPERATORS(MemoryAttribute); |
|||
|
|||
static_assert((static_cast<u8>(MemoryAttribute::Mask) & |
|||
static_cast<u8>(MemoryAttribute::DontCareMask)) == 0); |
|||
|
|||
struct MemoryInfo { |
|||
VAddr addr{}; |
|||
std::size_t size{}; |
|||
MemoryState state{}; |
|||
MemoryPermission perm{}; |
|||
MemoryAttribute attribute{}; |
|||
MemoryPermission original_perm{}; |
|||
u16 ipc_lock_count{}; |
|||
u16 device_use_count{}; |
|||
|
|||
constexpr Svc::MemoryInfo GetSvcMemoryInfo() const { |
|||
return { |
|||
addr, |
|||
size, |
|||
static_cast<Svc::MemoryState>(state & MemoryState::Mask), |
|||
static_cast<Svc::MemoryAttribute>(attribute & MemoryAttribute::Mask), |
|||
static_cast<Svc::MemoryPermission>(perm & MemoryPermission::UserMask), |
|||
ipc_lock_count, |
|||
device_use_count, |
|||
}; |
|||
} |
|||
|
|||
constexpr VAddr GetAddress() const { |
|||
return addr; |
|||
} |
|||
constexpr std::size_t GetSize() const { |
|||
return size; |
|||
} |
|||
constexpr std::size_t GetNumPages() const { |
|||
return GetSize() / PageSize; |
|||
} |
|||
constexpr VAddr GetEndAddress() const { |
|||
return GetAddress() + GetSize(); |
|||
} |
|||
constexpr VAddr GetLastAddress() const { |
|||
return GetEndAddress() - 1; |
|||
} |
|||
}; |
|||
|
|||
class MemoryBlock final { |
|||
friend class MemoryBlockManager; |
|||
|
|||
private: |
|||
VAddr addr{}; |
|||
std::size_t num_pages{}; |
|||
MemoryState state{MemoryState::None}; |
|||
u16 ipc_lock_count{}; |
|||
u16 device_use_count{}; |
|||
MemoryPermission perm{MemoryPermission::None}; |
|||
MemoryPermission original_perm{MemoryPermission::None}; |
|||
MemoryAttribute attribute{MemoryAttribute::None}; |
|||
|
|||
public: |
|||
static constexpr int Compare(const MemoryBlock& lhs, const MemoryBlock& rhs) { |
|||
if (lhs.GetAddress() < rhs.GetAddress()) { |
|||
return -1; |
|||
} else if (lhs.GetAddress() <= rhs.GetLastAddress()) { |
|||
return 0; |
|||
} else { |
|||
return 1; |
|||
} |
|||
} |
|||
|
|||
public: |
|||
constexpr MemoryBlock() = default; |
|||
constexpr MemoryBlock(VAddr addr, std::size_t num_pages, MemoryState state, |
|||
MemoryPermission perm, MemoryAttribute attribute) |
|||
: addr{addr}, num_pages(num_pages), state{state}, perm{perm}, attribute{attribute} {} |
|||
|
|||
constexpr VAddr GetAddress() const { |
|||
return addr; |
|||
} |
|||
|
|||
constexpr std::size_t GetNumPages() const { |
|||
return num_pages; |
|||
} |
|||
|
|||
constexpr std::size_t GetSize() const { |
|||
return GetNumPages() * PageSize; |
|||
} |
|||
|
|||
constexpr VAddr GetEndAddress() const { |
|||
return GetAddress() + GetSize(); |
|||
} |
|||
|
|||
constexpr VAddr GetLastAddress() const { |
|||
return GetEndAddress() - 1; |
|||
} |
|||
|
|||
constexpr MemoryInfo GetMemoryInfo() const { |
|||
return { |
|||
GetAddress(), GetSize(), state, perm, |
|||
attribute, original_perm, ipc_lock_count, device_use_count, |
|||
}; |
|||
} |
|||
|
|||
private: |
|||
constexpr bool HasProperties(MemoryState s, MemoryPermission p, MemoryAttribute a) const { |
|||
constexpr MemoryAttribute AttributeIgnoreMask{MemoryAttribute::DontCareMask | |
|||
MemoryAttribute::IpcLocked | |
|||
MemoryAttribute::DeviceShared}; |
|||
return state == s && perm == p && |
|||
(attribute | AttributeIgnoreMask) == (a | AttributeIgnoreMask); |
|||
} |
|||
|
|||
constexpr bool HasSameProperties(const MemoryBlock& rhs) const { |
|||
return state == rhs.state && perm == rhs.perm && original_perm == rhs.original_perm && |
|||
attribute == rhs.attribute && ipc_lock_count == rhs.ipc_lock_count && |
|||
device_use_count == rhs.device_use_count; |
|||
} |
|||
|
|||
constexpr bool Contains(VAddr start) const { |
|||
return GetAddress() <= start && start <= GetEndAddress(); |
|||
} |
|||
|
|||
constexpr void Add(std::size_t count) { |
|||
ASSERT(count > 0); |
|||
ASSERT(GetAddress() + count * PageSize - 1 < GetEndAddress() + count * PageSize - 1); |
|||
|
|||
num_pages += count; |
|||
} |
|||
|
|||
constexpr void Update(MemoryState new_state, MemoryPermission new_perm, |
|||
MemoryAttribute new_attribute) { |
|||
ASSERT(original_perm == MemoryPermission::None); |
|||
ASSERT((attribute & MemoryAttribute::IpcLocked) == MemoryAttribute::None); |
|||
|
|||
state = new_state; |
|||
perm = new_perm; |
|||
|
|||
// TODO(bunnei): Is this right? |
|||
attribute = static_cast<MemoryAttribute>( |
|||
new_attribute /*| (attribute & (MemoryAttribute::IpcLocked | MemoryAttribute::DeviceShared))*/); |
|||
} |
|||
|
|||
constexpr MemoryBlock Split(VAddr split_addr) { |
|||
ASSERT(GetAddress() < split_addr); |
|||
ASSERT(Contains(split_addr)); |
|||
ASSERT(Common::IsAligned(split_addr, PageSize)); |
|||
|
|||
MemoryBlock block; |
|||
block.addr = addr; |
|||
block.num_pages = (split_addr - GetAddress()) / PageSize; |
|||
block.state = state; |
|||
block.ipc_lock_count = ipc_lock_count; |
|||
block.device_use_count = device_use_count; |
|||
block.perm = perm; |
|||
block.original_perm = original_perm; |
|||
block.attribute = attribute; |
|||
|
|||
addr = split_addr; |
|||
num_pages -= block.num_pages; |
|||
|
|||
return block; |
|||
} |
|||
}; |
|||
static_assert(std::is_trivially_destructible<MemoryBlock>::value); |
|||
|
|||
} // namespace Kernel::Memory |
|||
Write
Preview
Loading…
Cancel
Save
Reference in new issue