mirror of
https://github.com/Atmosphere-NX/Atmosphere-libs.git
synced 2025-06-21 19:12:42 +02:00
91 lines
3.8 KiB
C++
91 lines
3.8 KiB
C++
/*
|
|
* Copyright (c) Atmosphère-NX
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms and conditions of the GNU General Public License,
|
|
* version 2, as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
* more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#include <stratosphere.hpp>
|
|
#include "os_io_region_impl.hpp"
|
|
#include "os_aslr_space_manager_types.hpp"
|
|
#include "os_aslr_space_manager.hpp"
|
|
|
|
namespace ams::os::impl {
|
|
|
|
namespace {
|
|
|
|
constexpr svc::MemoryPermission ConvertToSvcMemoryPermission(os::MemoryPermission perm) {
|
|
switch (perm) {
|
|
case os::MemoryPermission_None: return svc::MemoryPermission_None;
|
|
case os::MemoryPermission_ReadOnly: return svc::MemoryPermission_Read;
|
|
case os::MemoryPermission_WriteOnly: return svc::MemoryPermission_Write;
|
|
case os::MemoryPermission_ReadWrite: return svc::MemoryPermission_ReadWrite;
|
|
AMS_UNREACHABLE_DEFAULT_CASE();
|
|
}
|
|
}
|
|
|
|
constexpr svc::MemoryMapping ConvertToSvcMemoryMapping(os::MemoryMapping mapping) {
|
|
static_assert(std::same_as<svc::MemoryMapping, os::MemoryMapping>);
|
|
return mapping;
|
|
}
|
|
|
|
}
|
|
|
|
Result IoRegionImpl::CreateIoRegion(NativeHandle *out, NativeHandle io_pool_handle, uintptr_t address, size_t size, MemoryMapping mapping, MemoryPermission permission) {
|
|
/* Convert mapping/permission. */
|
|
const auto svc_mapping = ConvertToSvcMemoryMapping(mapping);
|
|
const auto svc_perm = ConvertToSvcMemoryPermission(permission);
|
|
|
|
/* Create the io region. */
|
|
/* TODO: Result conversion/abort on unexpected result? */
|
|
svc::Handle handle;
|
|
R_TRY(svc::CreateIoRegion(std::addressof(handle), io_pool_handle, address, size, svc_mapping, svc_perm));
|
|
|
|
*out = handle;
|
|
R_SUCCEED();
|
|
}
|
|
|
|
Result IoRegionImpl::MapIoRegion(void **out, NativeHandle handle, size_t size, MemoryPermission perm) {
|
|
/* Convert permission. */
|
|
const auto svc_perm = ConvertToSvcMemoryPermission(perm);
|
|
|
|
/* Map at a random address. */
|
|
uintptr_t mapped_address;
|
|
R_TRY(impl::GetAslrSpaceManager().MapAtRandomAddress(std::addressof(mapped_address),
|
|
[handle, svc_perm](uintptr_t map_address, size_t map_size) -> Result {
|
|
R_TRY_CATCH(svc::MapIoRegion(handle, map_address, map_size, svc_perm)) {
|
|
/* TODO: What's the correct result for these? */
|
|
// R_CONVERT(svc::ResultInvalidHandle, os::ResultInvalidHandle())
|
|
// R_CONVERT(svc::ResultInvalidSize, os::Result???())
|
|
// R_CONVERT(svc::ResultInvalidState, os::Result???())
|
|
R_CONVERT(svc::ResultInvalidCurrentMemory, os::ResultInvalidCurrentMemoryState())
|
|
} R_END_TRY_CATCH_WITH_ABORT_UNLESS;
|
|
|
|
R_SUCCEED();
|
|
},
|
|
[handle](uintptr_t map_address, size_t map_size) -> void {
|
|
return IoRegionImpl::UnmapIoRegion(handle, reinterpret_cast<void *>(map_address), map_size);
|
|
},
|
|
size,
|
|
0
|
|
));
|
|
|
|
/* Return the address we mapped at. */
|
|
*out = reinterpret_cast<void *>(mapped_address);
|
|
R_SUCCEED();
|
|
}
|
|
|
|
void IoRegionImpl::UnmapIoRegion(NativeHandle handle, void *address, size_t size) {
|
|
R_ABORT_UNLESS(svc::UnmapIoRegion(handle, reinterpret_cast<uintptr_t>(address), size));
|
|
}
|
|
|
|
}
|