/* * Copyright (c) 2018-2020 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 . */ #include #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); 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; return ResultSuccess(); } 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), size, [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; return ResultSuccess(); }, [handle](uintptr_t map_address, size_t map_size) -> void { return IoRegionImpl::UnmapIoRegion(handle, reinterpret_cast(map_address), map_size); } )); /* Return the address we mapped at. */ *out = reinterpret_cast(mapped_address); return ResultSuccess(); } void IoRegionImpl::UnmapIoRegion(NativeHandle handle, void *address, size_t size) { R_ABORT_UNLESS(svc::UnmapIoRegion(handle, reinterpret_cast(address), size)); } }