| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | // Copyright 2015 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-05 13:29:55 +01:00
										 |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-08 16:51:48 -03:00
										 |  |  | #include "common/common_funcs.h"
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2015-07-29 12:08:00 -03:00
										 |  |  | #include "core/hle/kernel/memory.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-20 23:52:38 -07:00
										 |  |  | #include "core/hle/kernel/process.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-12 15:25:15 -05:00
										 |  |  | #include "core/hle/kernel/resource_limit.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | #include "core/hle/kernel/thread.h"
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | #include "core/hle/kernel/vm_manager.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-12 22:38:29 -03:00
										 |  |  | #include "core/memory.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Kernel { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | SharedPtr<CodeSet> CodeSet::Create(std::string name, u64 program_id) { | 
					
						
							|  |  |  |     SharedPtr<CodeSet> codeset(new CodeSet); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     codeset->name = std::move(name); | 
					
						
							|  |  |  |     codeset->program_id = program_id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return codeset; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  | CodeSet::CodeSet() {} | 
					
						
							|  |  |  | CodeSet::~CodeSet() {} | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-11 09:15:10 -05:00
										 |  |  | u32 Process::next_process_id; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | SharedPtr<Process> Process::Create(SharedPtr<CodeSet> code_set) { | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  |     SharedPtr<Process> process(new Process); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |     process->codeset = std::move(code_set); | 
					
						
							| 
									
										
										
										
											2015-05-08 17:53:19 -03:00
										 |  |  |     process->flags.raw = 0; | 
					
						
							| 
									
										
										
										
											2016-02-11 17:41:15 +00:00
										 |  |  |     process->flags.memory_region.Assign(MemoryRegion::APPLICATION); | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |     Memory::InitLegacyAddressSpace(process->vm_manager); | 
					
						
							| 
									
										
										
										
											2015-05-08 17:53:19 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  |     return process; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 16:51:48 -03:00
										 |  |  | void Process::ParseKernelCaps(const u32* kernel_caps, size_t len) { | 
					
						
							| 
									
										
										
										
											2015-05-14 12:59:12 -04:00
										 |  |  |     for (size_t i = 0; i < len; ++i) { | 
					
						
							| 
									
										
										
										
											2015-05-08 16:51:48 -03:00
										 |  |  |         u32 descriptor = kernel_caps[i]; | 
					
						
							|  |  |  |         u32 type = descriptor >> 20; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (descriptor == 0xFFFFFFFF) { | 
					
						
							|  |  |  |             // Unused descriptor entry
 | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } else if ((type & 0xF00) == 0xE00) { // 0x0FFF
 | 
					
						
							|  |  |  |             // Allowed interrupts list
 | 
					
						
							|  |  |  |             LOG_WARNING(Loader, "ExHeader allowed interrupts list ignored"); | 
					
						
							|  |  |  |         } else if ((type & 0xF80) == 0xF00) { // 0x07FF
 | 
					
						
							|  |  |  |             // Allowed syscalls mask
 | 
					
						
							|  |  |  |             unsigned int index = ((descriptor >> 24) & 7) * 24; | 
					
						
							|  |  |  |             u32 bits = descriptor & 0xFFFFFF; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             while (bits && index < svc_access_mask.size()) { | 
					
						
							|  |  |  |                 svc_access_mask.set(index, bits & 1); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                 ++index; | 
					
						
							|  |  |  |                 bits >>= 1; | 
					
						
							| 
									
										
										
										
											2015-05-08 16:51:48 -03:00
										 |  |  |             } | 
					
						
							|  |  |  |         } else if ((type & 0xFF0) == 0xFE0) { // 0x00FF
 | 
					
						
							|  |  |  |             // Handle table size
 | 
					
						
							|  |  |  |             handle_table_size = descriptor & 0x3FF; | 
					
						
							|  |  |  |         } else if ((type & 0xFF8) == 0xFF0) { // 0x007F
 | 
					
						
							|  |  |  |             // Misc. flags
 | 
					
						
							| 
									
										
										
										
											2015-05-08 17:53:19 -03:00
										 |  |  |             flags.raw = descriptor & 0xFFFF; | 
					
						
							| 
									
										
										
										
											2015-05-08 16:51:48 -03:00
										 |  |  |         } else if ((type & 0xFFE) == 0xFF8) { // 0x001F
 | 
					
						
							|  |  |  |             // Mapped memory range
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |             if (i + 1 >= len || ((kernel_caps[i + 1] >> 20) & 0xFFE) != 0xFF8) { | 
					
						
							| 
									
										
										
										
											2015-05-08 16:51:48 -03:00
										 |  |  |                 LOG_WARNING(Loader, "Incomplete exheader memory range descriptor ignored."); | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |             u32 end_desc = kernel_caps[i + 1]; | 
					
						
							| 
									
										
										
										
											2015-05-08 16:51:48 -03:00
										 |  |  |             ++i; // Skip over the second descriptor on the next iteration
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 18:12:25 -03:00
										 |  |  |             AddressMapping mapping; | 
					
						
							| 
									
										
										
										
											2015-05-08 16:51:48 -03:00
										 |  |  |             mapping.address = descriptor << 12; | 
					
						
							|  |  |  |             mapping.size = (end_desc << 12) - mapping.address; | 
					
						
							| 
									
										
										
										
											2015-05-14 12:59:12 -04:00
										 |  |  |             mapping.writable = (descriptor & (1 << 20)) != 0; | 
					
						
							|  |  |  |             mapping.unk_flag = (end_desc & (1 << 20)) != 0; | 
					
						
							| 
									
										
										
										
											2015-05-08 16:51:48 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 18:12:25 -03:00
										 |  |  |             address_mappings.push_back(mapping); | 
					
						
							| 
									
										
										
										
											2015-05-08 16:51:48 -03:00
										 |  |  |         } else if ((type & 0xFFF) == 0xFFE) { // 0x000F
 | 
					
						
							|  |  |  |             // Mapped memory page
 | 
					
						
							| 
									
										
										
										
											2015-05-08 18:12:25 -03:00
										 |  |  |             AddressMapping mapping; | 
					
						
							| 
									
										
										
										
											2015-05-08 16:51:48 -03:00
										 |  |  |             mapping.address = descriptor << 12; | 
					
						
							|  |  |  |             mapping.size = Memory::PAGE_SIZE; | 
					
						
							|  |  |  |             mapping.writable = true; // TODO: Not sure if correct
 | 
					
						
							|  |  |  |             mapping.unk_flag = false; | 
					
						
							|  |  |  |         } else if ((type & 0xFE0) == 0xFC0) { // 0x01FF
 | 
					
						
							|  |  |  |             // Kernel version
 | 
					
						
							| 
									
										
										
										
											2015-07-19 15:18:57 -03:00
										 |  |  |             kernel_version = descriptor & 0xFFFF; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             int minor = kernel_version & 0xFF; | 
					
						
							|  |  |  |             int major = (kernel_version >> 8) & 0xFF; | 
					
						
							| 
									
										
										
										
											2015-08-05 21:26:52 -03:00
										 |  |  |             LOG_INFO(Loader, "ExHeader kernel version: %d.%d", major, minor); | 
					
						
							| 
									
										
										
										
											2015-05-08 16:51:48 -03:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             LOG_ERROR(Loader, "Unhandled kernel caps descriptor: 0x%08X", descriptor); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | void Process::Run(s32 main_thread_priority, u32 stack_size) { | 
					
						
							| 
									
										
										
										
											2015-08-05 21:26:52 -03:00
										 |  |  |     memory_region = GetMemoryRegion(flags.memory_region); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     auto MapSegment = [&](CodeSet::Segment& segment, VMAPermission permissions, | 
					
						
							|  |  |  |                           MemoryState memory_state) { | 
					
						
							|  |  |  |         auto vma = vm_manager | 
					
						
							|  |  |  |                        .MapMemoryBlock(segment.addr, codeset->memory, segment.offset, segment.size, | 
					
						
							|  |  |  |                                        memory_state) | 
					
						
							|  |  |  |                        .Unwrap(); | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |         vm_manager.Reprotect(vma, permissions); | 
					
						
							| 
									
										
										
										
											2015-08-05 21:39:53 -03:00
										 |  |  |         misc_memory_used += segment.size; | 
					
						
							| 
									
										
										
										
											2015-11-26 19:00:16 -08:00
										 |  |  |         memory_region->used += segment.size; | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |     // Map CodeSet segments
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     MapSegment(codeset->code, VMAPermission::ReadExecute, MemoryState::Code); | 
					
						
							|  |  |  |     MapSegment(codeset->rodata, VMAPermission::Read, MemoryState::Code); | 
					
						
							|  |  |  |     MapSegment(codeset->data, VMAPermission::ReadWrite, MemoryState::Private); | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |     // Allocate and map stack
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     vm_manager | 
					
						
							|  |  |  |         .MapMemoryBlock(Memory::HEAP_VADDR_END - stack_size, | 
					
						
							|  |  |  |                         std::make_shared<std::vector<u8>>(stack_size, 0), 0, stack_size, | 
					
						
							|  |  |  |                         MemoryState::Locked) | 
					
						
							|  |  |  |         .Unwrap(); | 
					
						
							| 
									
										
										
										
											2015-08-05 21:39:53 -03:00
										 |  |  |     misc_memory_used += stack_size; | 
					
						
							| 
									
										
										
										
											2015-11-26 19:00:16 -08:00
										 |  |  |     memory_region->used += stack_size; | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     vm_manager.LogLayout(Log::Level::Debug); | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |     Kernel::SetupMainThread(codeset->entrypoint, main_thread_priority); | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-05 22:09:59 -08:00
										 |  |  | VAddr Process::GetLinearHeapAreaAddress() const { | 
					
						
							|  |  |  |     return kernel_version < 0x22C ? Memory::LINEAR_HEAP_VADDR : Memory::NEW_LINEAR_HEAP_VADDR; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-08-05 21:26:52 -03:00
										 |  |  | VAddr Process::GetLinearHeapBase() const { | 
					
						
							| 
									
										
										
										
											2016-03-05 22:09:59 -08:00
										 |  |  |     return GetLinearHeapAreaAddress() + memory_region->base; | 
					
						
							| 
									
										
										
										
											2015-08-05 21:26:52 -03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | VAddr Process::GetLinearHeapLimit() const { | 
					
						
							|  |  |  |     return GetLinearHeapBase() + memory_region->size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  | ResultVal<VAddr> Process::HeapAllocate(VAddr target, u32 size, VMAPermission perms) { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     if (target < Memory::HEAP_VADDR || target + size > Memory::HEAP_VADDR_END || | 
					
						
							|  |  |  |         target + size < target) { | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |         return ERR_INVALID_ADDRESS; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (heap_memory == nullptr) { | 
					
						
							|  |  |  |         // Initialize heap
 | 
					
						
							|  |  |  |         heap_memory = std::make_shared<std::vector<u8>>(); | 
					
						
							|  |  |  |         heap_start = heap_end = target; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If necessary, expand backing vector to cover new heap extents.
 | 
					
						
							|  |  |  |     if (target < heap_start) { | 
					
						
							|  |  |  |         heap_memory->insert(begin(*heap_memory), heap_start - target, 0); | 
					
						
							|  |  |  |         heap_start = target; | 
					
						
							|  |  |  |         vm_manager.RefreshMemoryBlockMappings(heap_memory.get()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (target + size > heap_end) { | 
					
						
							|  |  |  |         heap_memory->insert(end(*heap_memory), (target + size) - heap_end, 0); | 
					
						
							|  |  |  |         heap_end = target + size; | 
					
						
							|  |  |  |         vm_manager.RefreshMemoryBlockMappings(heap_memory.get()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ASSERT(heap_end - heap_start == heap_memory->size()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     CASCADE_RESULT(auto vma, vm_manager.MapMemoryBlock(target, heap_memory, target - heap_start, | 
					
						
							|  |  |  |                                                        size, MemoryState::Private)); | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |     vm_manager.Reprotect(vma, perms); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-05 21:39:53 -03:00
										 |  |  |     heap_used += size; | 
					
						
							| 
									
										
										
										
											2015-11-26 19:00:16 -08:00
										 |  |  |     memory_region->used += size; | 
					
						
							| 
									
										
										
										
											2015-08-05 21:39:53 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |     return MakeResult<VAddr>(heap_end - size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode Process::HeapFree(VAddr target, u32 size) { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     if (target < Memory::HEAP_VADDR || target + size > Memory::HEAP_VADDR_END || | 
					
						
							|  |  |  |         target + size < target) { | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |         return ERR_INVALID_ADDRESS; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-26 06:38:26 -03:00
										 |  |  |     if (size == 0) { | 
					
						
							|  |  |  |         return RESULT_SUCCESS; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |     ResultCode result = vm_manager.UnmapRange(target, size); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     if (result.IsError()) | 
					
						
							|  |  |  |         return result; | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-05 21:39:53 -03:00
										 |  |  |     heap_used -= size; | 
					
						
							| 
									
										
										
										
											2015-11-26 19:00:16 -08:00
										 |  |  |     memory_region->used -= size; | 
					
						
							| 
									
										
										
										
											2015-08-05 21:39:53 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultVal<VAddr> Process::LinearAllocate(VAddr target, u32 size, VMAPermission perms) { | 
					
						
							| 
									
										
										
										
											2015-08-05 21:26:52 -03:00
										 |  |  |     auto& linheap_memory = memory_region->linear_heap_memory; | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-05 21:26:52 -03:00
										 |  |  |     VAddr heap_end = GetLinearHeapBase() + (u32)linheap_memory->size(); | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |     // Games and homebrew only ever seem to pass 0 here (which lets the kernel decide the address),
 | 
					
						
							|  |  |  |     // but explicit addresses are also accepted and respected.
 | 
					
						
							|  |  |  |     if (target == 0) { | 
					
						
							|  |  |  |         target = heap_end; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     if (target < GetLinearHeapBase() || target + size > GetLinearHeapLimit() || target > heap_end || | 
					
						
							|  |  |  |         target + size < target) { | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return ERR_INVALID_ADDRESS; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-17 13:36:15 -05:00
										 |  |  |     // Expansion of the linear heap is only allowed if you do an allocation immediately at its
 | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |     // end. It's possible to free gaps in the middle of the heap and then reallocate them later,
 | 
					
						
							|  |  |  |     // but expansions are only allowed at the end.
 | 
					
						
							|  |  |  |     if (target == heap_end) { | 
					
						
							| 
									
										
										
										
											2015-08-05 21:26:52 -03:00
										 |  |  |         linheap_memory->insert(linheap_memory->end(), size, 0); | 
					
						
							|  |  |  |         vm_manager.RefreshMemoryBlockMappings(linheap_memory.get()); | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-05 21:26:52 -03:00
										 |  |  |     // TODO(yuriks): As is, this lets processes map memory allocated by other processes from the
 | 
					
						
							|  |  |  |     // same region. It is unknown if or how the 3DS kernel checks against this.
 | 
					
						
							|  |  |  |     size_t offset = target - GetLinearHeapBase(); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     CASCADE_RESULT(auto vma, vm_manager.MapMemoryBlock(target, linheap_memory, offset, size, | 
					
						
							|  |  |  |                                                        MemoryState::Continuous)); | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |     vm_manager.Reprotect(vma, perms); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-05 21:39:53 -03:00
										 |  |  |     linear_heap_used += size; | 
					
						
							| 
									
										
										
										
											2015-11-26 19:00:16 -08:00
										 |  |  |     memory_region->used += size; | 
					
						
							| 
									
										
										
										
											2015-08-05 21:39:53 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |     return MakeResult<VAddr>(target); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ResultCode Process::LinearFree(VAddr target, u32 size) { | 
					
						
							| 
									
										
										
										
											2015-08-05 21:26:52 -03:00
										 |  |  |     auto& linheap_memory = memory_region->linear_heap_memory; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (target < GetLinearHeapBase() || target + size > GetLinearHeapLimit() || | 
					
						
							|  |  |  |         target + size < target) { | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return ERR_INVALID_ADDRESS; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-26 06:38:26 -03:00
										 |  |  |     if (size == 0) { | 
					
						
							|  |  |  |         return RESULT_SUCCESS; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-05 21:26:52 -03:00
										 |  |  |     VAddr heap_end = GetLinearHeapBase() + (u32)linheap_memory->size(); | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |     if (target + size > heap_end) { | 
					
						
							|  |  |  |         return ERR_INVALID_ADDRESS_STATE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ResultCode result = vm_manager.UnmapRange(target, size); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     if (result.IsError()) | 
					
						
							|  |  |  |         return result; | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-05 21:39:53 -03:00
										 |  |  |     linear_heap_used -= size; | 
					
						
							| 
									
										
										
										
											2015-11-26 19:00:16 -08:00
										 |  |  |     memory_region->used -= size; | 
					
						
							| 
									
										
										
										
											2015-08-05 21:39:53 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |     if (target + size == heap_end) { | 
					
						
							|  |  |  |         // End of linear heap has been freed, so check what's the last allocated block in it and
 | 
					
						
							|  |  |  |         // reduce the size.
 | 
					
						
							|  |  |  |         auto vma = vm_manager.FindVMA(target); | 
					
						
							|  |  |  |         ASSERT(vma != vm_manager.vma_map.end()); | 
					
						
							|  |  |  |         ASSERT(vma->second.type == VMAType::Free); | 
					
						
							|  |  |  |         VAddr new_end = vma->second.base; | 
					
						
							| 
									
										
										
										
											2015-08-05 21:26:52 -03:00
										 |  |  |         if (new_end >= GetLinearHeapBase()) { | 
					
						
							|  |  |  |             linheap_memory->resize(new_end - GetLinearHeapBase()); | 
					
						
							| 
									
										
										
										
											2015-07-17 23:19:16 -03:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return RESULT_SUCCESS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  | Kernel::Process::Process() {} | 
					
						
							|  |  |  | Kernel::Process::~Process() {} | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | SharedPtr<Process> g_current_process; | 
					
						
							|  |  |  | } |