@ -96,7 +96,7 @@ void Process::ParseKernelCaps(const u32* kernel_caps, size_t len) {
int minor = kernel_version & 0xFF ;
int minor = kernel_version & 0xFF ;
int major = ( kernel_version > > 8 ) & 0xFF ;
int major = ( kernel_version > > 8 ) & 0xFF ;
LOG_ DEBUG ( Loader , " ExHeader kernel version: %d.%d " , major , minor ) ;
LOG_ INFO ( Loader , " ExHeader kernel version: %d.%d " , major , minor ) ;
} else {
} else {
LOG_ERROR ( Loader , " Unhandled kernel caps descriptor: 0x%08X " , descriptor ) ;
LOG_ERROR ( Loader , " Unhandled kernel caps descriptor: 0x%08X " , descriptor ) ;
}
}
@ -104,6 +104,8 @@ void Process::ParseKernelCaps(const u32* kernel_caps, size_t len) {
}
}
void Process : : Run ( s32 main_thread_priority , u32 stack_size ) {
void Process : : Run ( s32 main_thread_priority , u32 stack_size ) {
memory_region = GetMemoryRegion ( flags . memory_region ) ;
auto MapSegment = [ & ] ( CodeSet : : Segment & segment , VMAPermission permissions , MemoryState memory_state ) {
auto MapSegment = [ & ] ( CodeSet : : Segment & segment , VMAPermission permissions , MemoryState memory_state ) {
auto vma = vm_manager . MapMemoryBlock ( segment . addr , codeset - > memory ,
auto vma = vm_manager . MapMemoryBlock ( segment . addr , codeset - > memory ,
segment . offset , segment . size , memory_state ) . Unwrap ( ) ;
segment . offset , segment . size , memory_state ) . Unwrap ( ) ;
@ -124,6 +126,15 @@ void Process::Run(s32 main_thread_priority, u32 stack_size) {
Kernel : : SetupMainThread ( codeset - > entrypoint , main_thread_priority ) ;
Kernel : : SetupMainThread ( codeset - > entrypoint , main_thread_priority ) ;
}
}
VAddr Process : : GetLinearHeapBase ( ) const {
return ( kernel_version < 0x22C ? Memory : : LINEAR_HEAP_VADDR : Memory : : NEW_LINEAR_HEAP_SIZE )
+ memory_region - > base ;
}
VAddr Process : : GetLinearHeapLimit ( ) const {
return GetLinearHeapBase ( ) + memory_region - > size ;
}
ResultVal < VAddr > Process : : HeapAllocate ( VAddr target , u32 size , VMAPermission perms ) {
ResultVal < VAddr > Process : : HeapAllocate ( VAddr target , u32 size , VMAPermission perms ) {
if ( target < Memory : : HEAP_VADDR | | target + size > Memory : : HEAP_VADDR_END | | target + size < target ) {
if ( target < Memory : : HEAP_VADDR | | target + size > Memory : : HEAP_VADDR_END | | target + size < target ) {
return ERR_INVALID_ADDRESS ;
return ERR_INVALID_ADDRESS ;
@ -166,19 +177,16 @@ ResultCode Process::HeapFree(VAddr target, u32 size) {
}
}
ResultVal < VAddr > Process : : LinearAllocate ( VAddr target , u32 size , VMAPermission perms ) {
ResultVal < VAddr > Process : : LinearAllocate ( VAddr target , u32 size , VMAPermission perms ) {
if ( linear_heap_memory = = nullptr ) {
auto & linheap_memory = memory_region - > linear_heap_memory ;
// Initialize heap
linear_heap_memory = std : : make_shared < std : : vector < u8 > > ( ) ;
}
VAddr heap_end = Memory: : LINEAR_HEAP_VADDR + ( u32 ) lin ear_ heap_memory- > size ( ) ;
VAddr heap_end = GetLinearHeapBase ( ) + ( u32 ) linheap_memory - > size ( ) ;
// Games and homebrew only ever seem to pass 0 here (which lets the kernel decide the address),
// 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.
// but explicit addresses are also accepted and respected.
if ( target = = 0 ) {
if ( target = = 0 ) {
target = heap_end ;
target = heap_end ;
}
}
if ( target < Memory: : LINEAR_HEAP_VADDR | | target + size > Memory : : LINEAR_HEAP_VADDR_END | |
if ( target < GetLinearHeapBase( ) | | target + size > GetLinearHeapLimit ( ) | |
target > heap_end | | target + size < target ) {
target > heap_end | | target + size < target ) {
return ERR_INVALID_ADDRESS ;
return ERR_INVALID_ADDRESS ;
@ -188,25 +196,29 @@ ResultVal<VAddr> Process::LinearAllocate(VAddr target, u32 size, VMAPermission p
// end. It's possible to free gaps in the middle of the heap and then reallocate them later,
// 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.
// but expansions are only allowed at the end.
if ( target = = heap_end ) {
if ( target = = heap_end ) {
lin ear_ heap_memory- > insert ( lin ear_ heap_memory- > end ( ) , size , 0 ) ;
lin heap_memory- > insert ( lin heap_memory- > end ( ) , size , 0 ) ;
vm_manager . RefreshMemoryBlockMappings ( lin ear_ heap_memory. get ( ) ) ;
vm_manager . RefreshMemoryBlockMappings ( lin heap_memory. get ( ) ) ;
}
}
size_t offset = target - Memory : : LINEAR_HEAP_VADDR ;
// TODO(yuriks): As is, this lets processes map memory allocated by other processes from the
CASCADE_RESULT ( auto vma , vm_manager . MapMemoryBlock ( target , linear_heap_memory , offset , size , MemoryState : : Continuous ) ) ;
// same region. It is unknown if or how the 3DS kernel checks against this.
size_t offset = target - GetLinearHeapBase ( ) ;
CASCADE_RESULT ( auto vma , vm_manager . MapMemoryBlock ( target , linheap_memory , offset , size , MemoryState : : Continuous ) ) ;
vm_manager . Reprotect ( vma , perms ) ;
vm_manager . Reprotect ( vma , perms ) ;
return MakeResult < VAddr > ( target ) ;
return MakeResult < VAddr > ( target ) ;
}
}
ResultCode Process : : LinearFree ( VAddr target , u32 size ) {
ResultCode Process : : LinearFree ( VAddr target , u32 size ) {
if ( linear_heap_memory = = nullptr | | target < Memory : : LINEAR_HEAP_VADDR | |
auto & linheap_memory = memory_region - > linear_heap_memory ;
target + size > Memory : : LINEAR_HEAP_VADDR_END | | target + size < target ) {
if ( target < GetLinearHeapBase ( ) | | target + size > GetLinearHeapLimit ( ) | |
target + size < target ) {
return ERR_INVALID_ADDRESS ;
return ERR_INVALID_ADDRESS ;
}
}
VAddr heap_end = Memory: : LINEAR_HEAP_VADDR + ( u32 ) lin ear_ heap_memory- > size ( ) ;
VAddr heap_end = GetLinearHeapBase( ) + ( u32 ) lin heap_memory- > size ( ) ;
if ( target + size > heap_end ) {
if ( target + size > heap_end ) {
return ERR_INVALID_ADDRESS_STATE ;
return ERR_INVALID_ADDRESS_STATE ;
}
}
@ -221,8 +233,8 @@ ResultCode Process::LinearFree(VAddr target, u32 size) {
ASSERT ( vma ! = vm_manager . vma_map . end ( ) ) ;
ASSERT ( vma ! = vm_manager . vma_map . end ( ) ) ;
ASSERT ( vma - > second . type = = VMAType : : Free ) ;
ASSERT ( vma - > second . type = = VMAType : : Free ) ;
VAddr new_end = vma - > second . base ;
VAddr new_end = vma - > second . base ;
if ( new_end > = Memory: : LINEAR_HEAP_VADDR ) {
if ( new_end > = GetLinearHeapBase( ) ) {
lin ear_ heap_memory- > resize ( new_end - Memory: : LINEAR_HEAP_VADDR ) ;
lin heap_memory- > resize ( new_end - GetLinearHeapBase( ) ) ;
}
}
}
}