Merge pull request #2232 from lioncash/transfer-memory
core/hle/kernel: Split transfer memory handling out into its own class
This commit is contained in:
commit
3f74518e19
|
@ -146,6 +146,8 @@ add_library(core STATIC
|
||||||
hle/kernel/svc_wrap.h
|
hle/kernel/svc_wrap.h
|
||||||
hle/kernel/thread.cpp
|
hle/kernel/thread.cpp
|
||||||
hle/kernel/thread.h
|
hle/kernel/thread.h
|
||||||
|
hle/kernel/transfer_memory.cpp
|
||||||
|
hle/kernel/transfer_memory.h
|
||||||
hle/kernel/vm_manager.cpp
|
hle/kernel/vm_manager.cpp
|
||||||
hle/kernel/vm_manager.h
|
hle/kernel/vm_manager.h
|
||||||
hle/kernel/wait_object.cpp
|
hle/kernel/wait_object.cpp
|
||||||
|
|
|
@ -23,6 +23,7 @@ bool Object::IsWaitable() const {
|
||||||
case HandleType::Unknown:
|
case HandleType::Unknown:
|
||||||
case HandleType::WritableEvent:
|
case HandleType::WritableEvent:
|
||||||
case HandleType::SharedMemory:
|
case HandleType::SharedMemory:
|
||||||
|
case HandleType::TransferMemory:
|
||||||
case HandleType::AddressArbiter:
|
case HandleType::AddressArbiter:
|
||||||
case HandleType::ResourceLimit:
|
case HandleType::ResourceLimit:
|
||||||
case HandleType::ClientPort:
|
case HandleType::ClientPort:
|
||||||
|
|
|
@ -22,6 +22,7 @@ enum class HandleType : u32 {
|
||||||
WritableEvent,
|
WritableEvent,
|
||||||
ReadableEvent,
|
ReadableEvent,
|
||||||
SharedMemory,
|
SharedMemory,
|
||||||
|
TransferMemory,
|
||||||
Thread,
|
Thread,
|
||||||
Process,
|
Process,
|
||||||
AddressArbiter,
|
AddressArbiter,
|
||||||
|
|
|
@ -32,6 +32,7 @@
|
||||||
#include "core/hle/kernel/svc.h"
|
#include "core/hle/kernel/svc.h"
|
||||||
#include "core/hle/kernel/svc_wrap.h"
|
#include "core/hle/kernel/svc_wrap.h"
|
||||||
#include "core/hle/kernel/thread.h"
|
#include "core/hle/kernel/thread.h"
|
||||||
|
#include "core/hle/kernel/transfer_memory.h"
|
||||||
#include "core/hle/kernel/writable_event.h"
|
#include "core/hle/kernel/writable_event.h"
|
||||||
#include "core/hle/lock.h"
|
#include "core/hle/lock.h"
|
||||||
#include "core/hle/result.h"
|
#include "core/hle/result.h"
|
||||||
|
@ -1586,14 +1587,121 @@ static ResultCode CreateTransferMemory(Handle* handle, VAddr addr, u64 size, u32
|
||||||
}
|
}
|
||||||
|
|
||||||
auto& kernel = Core::System::GetInstance().Kernel();
|
auto& kernel = Core::System::GetInstance().Kernel();
|
||||||
auto process = kernel.CurrentProcess();
|
auto transfer_mem_handle = TransferMemory::Create(kernel, addr, size, perms);
|
||||||
auto& handle_table = process->GetHandleTable();
|
|
||||||
const auto shared_mem_handle = SharedMemory::Create(kernel, process, size, perms, perms, addr);
|
|
||||||
|
|
||||||
CASCADE_RESULT(*handle, handle_table.Create(shared_mem_handle));
|
auto& handle_table = kernel.CurrentProcess()->GetHandleTable();
|
||||||
|
const auto result = handle_table.Create(std::move(transfer_mem_handle));
|
||||||
|
if (result.Failed()) {
|
||||||
|
return result.Code();
|
||||||
|
}
|
||||||
|
|
||||||
|
*handle = *result;
|
||||||
return RESULT_SUCCESS;
|
return RESULT_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static ResultCode MapTransferMemory(Handle handle, VAddr address, u64 size, u32 permission_raw) {
|
||||||
|
LOG_DEBUG(Kernel_SVC,
|
||||||
|
"called. handle=0x{:08X}, address=0x{:016X}, size=0x{:016X}, permissions=0x{:08X}",
|
||||||
|
handle, address, size, permission_raw);
|
||||||
|
|
||||||
|
if (!Common::Is4KBAligned(address)) {
|
||||||
|
LOG_ERROR(Kernel_SVC, "Transfer memory addresses must be 4KB aligned (size=0x{:016X}).",
|
||||||
|
address);
|
||||||
|
return ERR_INVALID_ADDRESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (size == 0 || !Common::Is4KBAligned(size)) {
|
||||||
|
LOG_ERROR(Kernel_SVC,
|
||||||
|
"Transfer memory sizes must be 4KB aligned and not be zero (size=0x{:016X}).",
|
||||||
|
size);
|
||||||
|
return ERR_INVALID_SIZE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!IsValidAddressRange(address, size)) {
|
||||||
|
LOG_ERROR(Kernel_SVC,
|
||||||
|
"Given address and size overflows the 64-bit range (address=0x{:016X}, "
|
||||||
|
"size=0x{:016X}).",
|
||||||
|
address, size);
|
||||||
|
return ERR_INVALID_ADDRESS_STATE;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto permissions = static_cast<MemoryPermission>(permission_raw);
|
||||||
|
if (permissions != MemoryPermission::None && permissions != MemoryPermission::Read &&
|
||||||
|
permissions != MemoryPermission::ReadWrite) {
|
||||||
|
LOG_ERROR(Kernel_SVC, "Invalid transfer memory permissions given (permissions=0x{:08X}).",
|
||||||
|
permission_raw);
|
||||||
|
return ERR_INVALID_STATE;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto& kernel = Core::System::GetInstance().Kernel();
|
||||||
|
const auto* const current_process = kernel.CurrentProcess();
|
||||||
|
const auto& handle_table = current_process->GetHandleTable();
|
||||||
|
|
||||||
|
auto transfer_memory = handle_table.Get<TransferMemory>(handle);
|
||||||
|
if (!transfer_memory) {
|
||||||
|
LOG_ERROR(Kernel_SVC, "Nonexistent transfer memory handle given (handle=0x{:08X}).",
|
||||||
|
handle);
|
||||||
|
return ERR_INVALID_HANDLE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!current_process->VMManager().IsWithinASLRRegion(address, size)) {
|
||||||
|
LOG_ERROR(Kernel_SVC,
|
||||||
|
"Given address and size don't fully fit within the ASLR region "
|
||||||
|
"(address=0x{:016X}, size=0x{:016X}).",
|
||||||
|
address, size);
|
||||||
|
return ERR_INVALID_MEMORY_RANGE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return transfer_memory->MapMemory(address, size, permissions);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ResultCode UnmapTransferMemory(Handle handle, VAddr address, u64 size) {
|
||||||
|
LOG_DEBUG(Kernel_SVC, "called. handle=0x{:08X}, address=0x{:016X}, size=0x{:016X}", handle,
|
||||||
|
address, size);
|
||||||
|
|
||||||
|
if (!Common::Is4KBAligned(address)) {
|
||||||
|
LOG_ERROR(Kernel_SVC, "Transfer memory addresses must be 4KB aligned (size=0x{:016X}).",
|
||||||
|
address);
|
||||||
|
return ERR_INVALID_ADDRESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (size == 0 || !Common::Is4KBAligned(size)) {
|
||||||
|
LOG_ERROR(Kernel_SVC,
|
||||||
|
"Transfer memory sizes must be 4KB aligned and not be zero (size=0x{:016X}).",
|
||||||
|
size);
|
||||||
|
return ERR_INVALID_SIZE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!IsValidAddressRange(address, size)) {
|
||||||
|
LOG_ERROR(Kernel_SVC,
|
||||||
|
"Given address and size overflows the 64-bit range (address=0x{:016X}, "
|
||||||
|
"size=0x{:016X}).",
|
||||||
|
address, size);
|
||||||
|
return ERR_INVALID_ADDRESS_STATE;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto& kernel = Core::System::GetInstance().Kernel();
|
||||||
|
const auto* const current_process = kernel.CurrentProcess();
|
||||||
|
const auto& handle_table = current_process->GetHandleTable();
|
||||||
|
|
||||||
|
auto transfer_memory = handle_table.Get<TransferMemory>(handle);
|
||||||
|
if (!transfer_memory) {
|
||||||
|
LOG_ERROR(Kernel_SVC, "Nonexistent transfer memory handle given (handle=0x{:08X}).",
|
||||||
|
handle);
|
||||||
|
return ERR_INVALID_HANDLE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!current_process->VMManager().IsWithinASLRRegion(address, size)) {
|
||||||
|
LOG_ERROR(Kernel_SVC,
|
||||||
|
"Given address and size don't fully fit within the ASLR region "
|
||||||
|
"(address=0x{:016X}, size=0x{:016X}).",
|
||||||
|
address, size);
|
||||||
|
return ERR_INVALID_MEMORY_RANGE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return transfer_memory->UnmapMemory(address, size);
|
||||||
|
}
|
||||||
|
|
||||||
static ResultCode GetThreadCoreMask(Handle thread_handle, u32* core, u64* mask) {
|
static ResultCode GetThreadCoreMask(Handle thread_handle, u32* core, u64* mask) {
|
||||||
LOG_TRACE(Kernel_SVC, "called, handle=0x{:08X}", thread_handle);
|
LOG_TRACE(Kernel_SVC, "called, handle=0x{:08X}", thread_handle);
|
||||||
|
|
||||||
|
@ -1969,8 +2077,8 @@ static const FunctionDef SVC_Table[] = {
|
||||||
{0x4E, nullptr, "ReadWriteRegister"},
|
{0x4E, nullptr, "ReadWriteRegister"},
|
||||||
{0x4F, nullptr, "SetProcessActivity"},
|
{0x4F, nullptr, "SetProcessActivity"},
|
||||||
{0x50, SvcWrap<CreateSharedMemory>, "CreateSharedMemory"},
|
{0x50, SvcWrap<CreateSharedMemory>, "CreateSharedMemory"},
|
||||||
{0x51, nullptr, "MapTransferMemory"},
|
{0x51, SvcWrap<MapTransferMemory>, "MapTransferMemory"},
|
||||||
{0x52, nullptr, "UnmapTransferMemory"},
|
{0x52, SvcWrap<UnmapTransferMemory>, "UnmapTransferMemory"},
|
||||||
{0x53, nullptr, "CreateInterruptEvent"},
|
{0x53, nullptr, "CreateInterruptEvent"},
|
||||||
{0x54, nullptr, "QueryPhysicalAddress"},
|
{0x54, nullptr, "QueryPhysicalAddress"},
|
||||||
{0x55, nullptr, "QueryIoMapping"},
|
{0x55, nullptr, "QueryIoMapping"},
|
||||||
|
|
|
@ -0,0 +1,73 @@
|
||||||
|
// Copyright 2019 yuzu emulator team
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include "core/hle/kernel/errors.h"
|
||||||
|
#include "core/hle/kernel/kernel.h"
|
||||||
|
#include "core/hle/kernel/process.h"
|
||||||
|
#include "core/hle/kernel/shared_memory.h"
|
||||||
|
#include "core/hle/kernel/transfer_memory.h"
|
||||||
|
#include "core/hle/result.h"
|
||||||
|
|
||||||
|
namespace Kernel {
|
||||||
|
|
||||||
|
TransferMemory::TransferMemory(KernelCore& kernel) : Object{kernel} {}
|
||||||
|
TransferMemory::~TransferMemory() = default;
|
||||||
|
|
||||||
|
SharedPtr<TransferMemory> TransferMemory::Create(KernelCore& kernel, VAddr base_address,
|
||||||
|
size_t size, MemoryPermission permissions) {
|
||||||
|
SharedPtr<TransferMemory> transfer_memory{new TransferMemory(kernel)};
|
||||||
|
|
||||||
|
transfer_memory->base_address = base_address;
|
||||||
|
transfer_memory->memory_size = size;
|
||||||
|
transfer_memory->owner_permissions = permissions;
|
||||||
|
transfer_memory->owner_process = kernel.CurrentProcess();
|
||||||
|
|
||||||
|
return transfer_memory;
|
||||||
|
}
|
||||||
|
|
||||||
|
ResultCode TransferMemory::MapMemory(VAddr address, size_t size, MemoryPermission permissions) {
|
||||||
|
if (memory_size != size) {
|
||||||
|
return ERR_INVALID_SIZE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (owner_permissions != permissions) {
|
||||||
|
return ERR_INVALID_STATE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (is_mapped) {
|
||||||
|
return ERR_INVALID_STATE;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto map_state = owner_permissions == MemoryPermission::None
|
||||||
|
? MemoryState::TransferMemoryIsolated
|
||||||
|
: MemoryState::TransferMemory;
|
||||||
|
auto& vm_manager = owner_process->VMManager();
|
||||||
|
const auto map_result = vm_manager.MapMemoryBlock(
|
||||||
|
address, std::make_shared<std::vector<u8>>(size), 0, size, map_state);
|
||||||
|
|
||||||
|
if (map_result.Failed()) {
|
||||||
|
return map_result.Code();
|
||||||
|
}
|
||||||
|
|
||||||
|
is_mapped = true;
|
||||||
|
return RESULT_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
ResultCode TransferMemory::UnmapMemory(VAddr address, size_t size) {
|
||||||
|
if (memory_size != size) {
|
||||||
|
return ERR_INVALID_SIZE;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto& vm_manager = owner_process->VMManager();
|
||||||
|
const auto result = vm_manager.UnmapRange(address, size);
|
||||||
|
|
||||||
|
if (result.IsError()) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
is_mapped = false;
|
||||||
|
return RESULT_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Kernel
|
|
@ -0,0 +1,91 @@
|
||||||
|
// Copyright 2019 yuzu emulator team
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "core/hle/kernel/object.h"
|
||||||
|
|
||||||
|
union ResultCode;
|
||||||
|
|
||||||
|
namespace Kernel {
|
||||||
|
|
||||||
|
class KernelCore;
|
||||||
|
class Process;
|
||||||
|
|
||||||
|
enum class MemoryPermission : u32;
|
||||||
|
|
||||||
|
/// Defines the interface for transfer memory objects.
|
||||||
|
///
|
||||||
|
/// Transfer memory is typically used for the purpose of
|
||||||
|
/// transferring memory between separate process instances,
|
||||||
|
/// thus the name.
|
||||||
|
///
|
||||||
|
class TransferMemory final : public Object {
|
||||||
|
public:
|
||||||
|
static constexpr HandleType HANDLE_TYPE = HandleType::TransferMemory;
|
||||||
|
|
||||||
|
static SharedPtr<TransferMemory> Create(KernelCore& kernel, VAddr base_address, size_t size,
|
||||||
|
MemoryPermission permissions);
|
||||||
|
|
||||||
|
TransferMemory(const TransferMemory&) = delete;
|
||||||
|
TransferMemory& operator=(const TransferMemory&) = delete;
|
||||||
|
|
||||||
|
TransferMemory(TransferMemory&&) = delete;
|
||||||
|
TransferMemory& operator=(TransferMemory&&) = delete;
|
||||||
|
|
||||||
|
std::string GetTypeName() const override {
|
||||||
|
return "TransferMemory";
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string GetName() const override {
|
||||||
|
return GetTypeName();
|
||||||
|
}
|
||||||
|
|
||||||
|
HandleType GetHandleType() const override {
|
||||||
|
return HANDLE_TYPE;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Attempts to map transfer memory with the given range and memory permissions.
|
||||||
|
///
|
||||||
|
/// @param address The base address to being mapping memory at.
|
||||||
|
/// @param size The size of the memory to map, in bytes.
|
||||||
|
/// @param permissions The memory permissions to check against when mapping memory.
|
||||||
|
///
|
||||||
|
/// @pre The given address, size, and memory permissions must all match
|
||||||
|
/// the same values that were given when creating the transfer memory
|
||||||
|
/// instance.
|
||||||
|
///
|
||||||
|
ResultCode MapMemory(VAddr address, size_t size, MemoryPermission permissions);
|
||||||
|
|
||||||
|
/// Unmaps the transfer memory with the given range
|
||||||
|
///
|
||||||
|
/// @param address The base address to begin unmapping memory at.
|
||||||
|
/// @param size The size of the memory to unmap, in bytes.
|
||||||
|
///
|
||||||
|
/// @pre The given address and size must be the same as the ones used
|
||||||
|
/// to create the transfer memory instance.
|
||||||
|
///
|
||||||
|
ResultCode UnmapMemory(VAddr address, size_t size);
|
||||||
|
|
||||||
|
private:
|
||||||
|
explicit TransferMemory(KernelCore& kernel);
|
||||||
|
~TransferMemory() override;
|
||||||
|
|
||||||
|
/// The base address for the memory managed by this instance.
|
||||||
|
VAddr base_address = 0;
|
||||||
|
|
||||||
|
/// Size of the memory, in bytes, that this instance manages.
|
||||||
|
size_t memory_size = 0;
|
||||||
|
|
||||||
|
/// The memory permissions that are applied to this instance.
|
||||||
|
MemoryPermission owner_permissions{};
|
||||||
|
|
||||||
|
/// The process that this transfer memory instance was created under.
|
||||||
|
Process* owner_process = nullptr;
|
||||||
|
|
||||||
|
/// Whether or not this transfer memory instance has mapped memory.
|
||||||
|
bool is_mapped = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Kernel
|
Reference in New Issue