mirror of
				https://github.com/Atmosphere-NX/Atmosphere.git
				synced 2025-10-31 19:25:46 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			178 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			178 lines
		
	
	
		
			7.2 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/>.
 | |
|  */
 | |
| #pragma once
 | |
| #include <stratosphere.hpp>
 | |
| 
 | |
| namespace ams::ro::impl {
 | |
| 
 | |
|     constexpr inline auto RetrySearchCount = 512;
 | |
| 
 | |
|     Result SearchFreeRegion(uintptr_t *out, size_t mapping_size);
 | |
| 
 | |
|     class ProcessRegionInfo {
 | |
|         NON_COPYABLE(ProcessRegionInfo);
 | |
|         NON_MOVEABLE(ProcessRegionInfo);
 | |
|         private:
 | |
|             static constexpr size_t StackGuardSize = 4 * os::MemoryPageSize;
 | |
|         private:
 | |
|             u64 m_heap_start;
 | |
|             u64 m_heap_size;
 | |
|             u64 m_alias_start;
 | |
|             u64 m_alias_size;
 | |
|             u64 m_aslr_start;
 | |
|             u64 m_aslr_size;
 | |
|         public:
 | |
|             ProcessRegionInfo(os::NativeHandle process) {
 | |
|                 R_ABORT_UNLESS(svc::GetInfo(std::addressof(m_heap_start),  svc::InfoType_HeapRegionAddress,  process, 0));
 | |
|                 R_ABORT_UNLESS(svc::GetInfo(std::addressof(m_heap_size),   svc::InfoType_HeapRegionSize,     process, 0));
 | |
|                 R_ABORT_UNLESS(svc::GetInfo(std::addressof(m_alias_start), svc::InfoType_AliasRegionAddress, process, 0));
 | |
|                 R_ABORT_UNLESS(svc::GetInfo(std::addressof(m_alias_size),  svc::InfoType_AliasRegionSize,    process, 0));
 | |
|                 R_ABORT_UNLESS(svc::GetInfo(std::addressof(m_aslr_start),  svc::InfoType_AslrRegionAddress,  process, 0));
 | |
|                 R_ABORT_UNLESS(svc::GetInfo(std::addressof(m_aslr_size),   svc::InfoType_AslrRegionSize,     process, 0));
 | |
|             }
 | |
| 
 | |
|             u64 GetAslrRegion(u64 mapping_size) const {
 | |
|                 /* If we can, look for a region. */
 | |
|                 if (mapping_size <= m_aslr_size) {
 | |
|                     for (auto i = 0; i < RetrySearchCount; ++i) {
 | |
|                         /* Get a random address. */
 | |
|                         const u64 address = m_aslr_start + os::GenerateRandomU64((m_aslr_size - mapping_size) / os::MemoryPageSize) * os::MemoryPageSize;
 | |
| 
 | |
|                         /* Check that it's not contained within heap. */
 | |
|                         if (m_heap_size != 0 && !(address + mapping_size - 1 < m_heap_start || m_heap_start + m_heap_size - 1 < address)) {
 | |
|                             continue;
 | |
|                         }
 | |
| 
 | |
|                         /* Check that it's not contained within alias. */
 | |
|                         if (m_alias_size != 0 && !(address + mapping_size - 1 < m_alias_start || m_alias_start + m_alias_size - 1 < address)) {
 | |
|                             continue;
 | |
|                         }
 | |
| 
 | |
|                         /* Return the address. */
 | |
|                         return address;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 /* We failed to find a region. */
 | |
|                 return 0;
 | |
|             }
 | |
| 
 | |
|             bool CanEmplaceGuardSpaces(os::NativeHandle process, u64 address, u64 size) {
 | |
|                 /* NOTE: Nintendo does not check the results of these svc calls. */
 | |
|                 svc::MemoryInfo mem_info;
 | |
|                 svc::PageInfo page_info;
 | |
| 
 | |
|                 /* Check for guard availability before the region. */
 | |
|                 R_ABORT_UNLESS(svc::QueryProcessMemory(std::addressof(mem_info), std::addressof(page_info), process, address - 1));
 | |
|                 if (!(mem_info.state == svc::MemoryState_Free && mem_info.base_address <= address - StackGuardSize)) {
 | |
|                     return false;
 | |
|                 }
 | |
| 
 | |
|                 /* Check for guard availability after the region. */
 | |
|                 R_ABORT_UNLESS(svc::QueryProcessMemory(std::addressof(mem_info), std::addressof(page_info), process, address + size));
 | |
|                 if (!(mem_info.state == svc::MemoryState_Free && address + size + StackGuardSize <= mem_info.base_address + mem_info.size)) {
 | |
|                     return false;
 | |
|                 }
 | |
| 
 | |
|                 return true;
 | |
|             }
 | |
|     };
 | |
| 
 | |
|     class AutoCloseMap {
 | |
|         private:
 | |
|             Result m_result;
 | |
|             uintptr_t m_map_address;
 | |
|             os::NativeHandle m_handle;
 | |
|             u64 m_address;
 | |
|             u64 m_size;
 | |
|         public:
 | |
|             AutoCloseMap(uintptr_t map, os::NativeHandle handle, u64 addr, u64 size) : m_map_address(map), m_handle(handle), m_address(addr), m_size(size) {
 | |
|                 m_result = svc::MapProcessMemory(m_map_address, m_handle, m_address, m_size);
 | |
|             }
 | |
| 
 | |
|             ~AutoCloseMap() {
 | |
|                 if (m_handle != os::InvalidNativeHandle && R_SUCCEEDED(m_result)) {
 | |
|                     R_ABORT_UNLESS(svc::UnmapProcessMemory(m_map_address, m_handle, m_address, m_size));
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             Result GetResult() const {
 | |
|                 return m_result;
 | |
|             }
 | |
| 
 | |
|             bool IsSuccess() const {
 | |
|                 return R_SUCCEEDED(m_result);
 | |
|             }
 | |
| 
 | |
|             void Cancel() {
 | |
|                 m_handle = os::InvalidNativeHandle;
 | |
|             }
 | |
|     };
 | |
| 
 | |
|     class MappedCodeMemory {
 | |
|         NON_COPYABLE(MappedCodeMemory);
 | |
|         private:
 | |
|             os::NativeHandle m_handle;
 | |
|             Result m_result;
 | |
|             u64 m_dst_address;
 | |
|             u64 m_src_address;
 | |
|             u64 m_size;
 | |
|         public:
 | |
|             constexpr MappedCodeMemory() : m_handle(os::InvalidNativeHandle), m_result(ro::ResultInternalError()), m_dst_address(0), m_src_address(0), m_size(0) {
 | |
|                 /* ... */
 | |
|             }
 | |
| 
 | |
|             MappedCodeMemory(os::NativeHandle handle, u64 dst, u64 src, u64 size) : m_handle(handle), m_dst_address(dst), m_src_address(src), m_size(size) {
 | |
|                 m_result = svc::MapProcessCodeMemory(m_handle, m_dst_address, m_src_address, m_size);
 | |
|             }
 | |
| 
 | |
|             ~MappedCodeMemory() {
 | |
|                 if (m_handle != os::InvalidNativeHandle && R_SUCCEEDED(m_result) && m_size > 0) {
 | |
|                     R_ABORT_UNLESS(svc::UnmapProcessCodeMemory(m_handle, m_dst_address, m_src_address, m_size));
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             MappedCodeMemory(MappedCodeMemory &&rhs) : m_handle(rhs.m_handle), m_result(rhs.m_result), m_dst_address(rhs.m_dst_address), m_src_address(rhs.m_src_address), m_size(rhs.m_size) {
 | |
|                 rhs.m_handle = os::InvalidNativeHandle;
 | |
|             }
 | |
| 
 | |
|             MappedCodeMemory &operator=(MappedCodeMemory &&rhs) {
 | |
|                 m_handle      = rhs.m_handle;
 | |
|                 m_result      = rhs.m_result;
 | |
|                 m_dst_address = rhs.m_dst_address;
 | |
|                 m_src_address = rhs.m_src_address;
 | |
|                 m_size        = rhs.m_size;
 | |
| 
 | |
|                 rhs.m_handle  = os::InvalidNativeHandle;
 | |
| 
 | |
|                 return *this;
 | |
|             }
 | |
| 
 | |
|             Result GetResult() const {
 | |
|                 return m_result;
 | |
|             }
 | |
| 
 | |
|             bool IsSuccess() const {
 | |
|                 return R_SUCCEEDED(m_result);
 | |
|             }
 | |
| 
 | |
|             void Cancel() {
 | |
|                 m_handle = os::InvalidNativeHandle;
 | |
|             }
 | |
|     };
 | |
| 
 | |
| }
 |