Browse Source
Merge pull request #2232 from lioncash/transfer-memory
Merge pull request #2232 from lioncash/transfer-memory
core/hle/kernel: Split transfer memory handling out into its own classnce_cpp
committed by
GitHub
6 changed files with 282 additions and 6 deletions
-
2src/core/CMakeLists.txt
-
1src/core/hle/kernel/object.cpp
-
1src/core/hle/kernel/object.h
-
120src/core/hle/kernel/svc.cpp
-
73src/core/hle/kernel/transfer_memory.cpp
-
91src/core/hle/kernel/transfer_memory.h
@ -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 |
|||
Write
Preview
Loading…
Cancel
Save
Reference in new issue