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