| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-21 00:21:23 +09:00
										 |  |  | #include "core/loader/3dsx.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | #include "core/file_sys/archive_romfs.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | #include "core/hle/kernel/process.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-12 15:25:15 -05:00
										 |  |  | #include "core/hle/kernel/resource_limit.h"
 | 
					
						
							| 
									
										
										
										
											2016-05-17 23:06:33 +01:00
										 |  |  | #include "core/hle/service/fs/archive.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-12 22:38:29 -03:00
										 |  |  | #include "core/memory.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Loader { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 17:11:09 -03:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |  * File layout: | 
					
						
							|  |  |  |  * - File header | 
					
						
							|  |  |  |  * - Code, rodata and data relocation table headers | 
					
						
							|  |  |  |  * - Code segment | 
					
						
							|  |  |  |  * - Rodata segment | 
					
						
							|  |  |  |  * - Loadable (non-BSS) part of the data segment | 
					
						
							|  |  |  |  * - Code relocation table | 
					
						
							|  |  |  |  * - Rodata relocation table | 
					
						
							|  |  |  |  * - Data relocation table | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Memory layout before relocations are applied: | 
					
						
							|  |  |  |  * [0..codeSegSize)             -> code segment | 
					
						
							|  |  |  |  * [codeSegSize..rodataSegSize) -> rodata segment | 
					
						
							|  |  |  |  * [rodataSegSize..dataSegSize) -> data segment | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Memory layout after relocations are applied: well, however the loader sets it up :) | 
					
						
							|  |  |  |  * The entrypoint is always the start of the code segment. | 
					
						
							|  |  |  |  * The BSS section must be cleared manually by the application. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-07-09 17:11:09 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | enum THREEDSX_Error { ERROR_NONE = 0, ERROR_READ = 1, ERROR_FILE = 2, ERROR_ALLOC = 3 }; | 
					
						
							| 
									
										
										
										
											2015-07-09 17:11:09 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | static const u32 RELOCBUFSIZE = 512; | 
					
						
							| 
									
										
										
										
											2015-07-09 17:11:09 -03:00
										 |  |  | static const unsigned int NUM_SEGMENTS = 3; | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | // File header
 | 
					
						
							|  |  |  | #pragma pack(1)
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | struct THREEDSX_Header { | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |     u32 magic; | 
					
						
							|  |  |  |     u16 header_size, reloc_hdr_size; | 
					
						
							|  |  |  |     u32 format_ver; | 
					
						
							|  |  |  |     u32 flags; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Sizes of the code, rodata and data segments +
 | 
					
						
							|  |  |  |     // size of the BSS section (uninitialized latter half of the data segment)
 | 
					
						
							|  |  |  |     u32 code_seg_size, rodata_seg_size, data_seg_size, bss_size; | 
					
						
							| 
									
										
										
										
											2015-09-21 01:30:06 -04:00
										 |  |  |     // offset and size of smdh
 | 
					
						
							|  |  |  |     u32 smdh_offset, smdh_size; | 
					
						
							|  |  |  |     // offset to filesystem
 | 
					
						
							|  |  |  |     u32 fs_offset; | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Relocation header: all fields (even extra unknown fields) are guaranteed to be relocation counts.
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | struct THREEDSX_RelocHdr { | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |     // # of absolute relocations (that is, fix address to post-relocation memory layout)
 | 
					
						
							| 
									
										
										
										
											2015-01-05 20:09:35 +00:00
										 |  |  |     u32 cross_segment_absolute; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     // # of cross-segment relative relocations (that is, 32bit signed offsets that need to be
 | 
					
						
							|  |  |  |     // patched)
 | 
					
						
							| 
									
										
										
										
											2015-01-05 20:09:35 +00:00
										 |  |  |     u32 cross_segment_relative; | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |     // more?
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Relocations are written in this order:
 | 
					
						
							|  |  |  |     // - Absolute relocations
 | 
					
						
							|  |  |  |     // - Relative relocations
 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Relocation entry: from the current pointer, skip X words and patch Y words
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | struct THREEDSX_Reloc { | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |     u16 skip, patch; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | #pragma pack()
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | struct THREEloadinfo { | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |     u8* seg_ptrs[3]; // code, rodata & data
 | 
					
						
							|  |  |  |     u32 seg_addrs[3]; | 
					
						
							|  |  |  |     u32 seg_sizes[3]; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | static u32 TranslateAddr(u32 addr, const THREEloadinfo* loadinfo, u32* offsets) { | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |     if (addr < offsets[0]) | 
					
						
							|  |  |  |         return loadinfo->seg_addrs[0] + addr; | 
					
						
							|  |  |  |     if (addr < offsets[1]) | 
					
						
							|  |  |  |         return loadinfo->seg_addrs[1] + addr - offsets[0]; | 
					
						
							|  |  |  |     return loadinfo->seg_addrs[2] + addr - offsets[1]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | using Kernel::SharedPtr; | 
					
						
							|  |  |  | using Kernel::CodeSet; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | static THREEDSX_Error Load3DSXFile(FileUtil::IOFile& file, u32 base_addr, | 
					
						
							|  |  |  |                                    SharedPtr<CodeSet>* out_codeset) { | 
					
						
							| 
									
										
										
										
											2015-01-06 19:56:26 +00:00
										 |  |  |     if (!file.IsOpen()) | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |         return ERROR_FILE; | 
					
						
							| 
									
										
										
										
											2015-01-06 19:56:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-06 22:47:43 +00:00
										 |  |  |     // Reset read pointer in case this file has been read before.
 | 
					
						
							|  |  |  |     file.Seek(0, SEEK_SET); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |     THREEDSX_Header hdr; | 
					
						
							|  |  |  |     if (file.ReadBytes(&hdr, sizeof(hdr)) != sizeof(hdr)) | 
					
						
							|  |  |  |         return ERROR_READ; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     THREEloadinfo loadinfo; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     // loadinfo segments must be a multiple of 0x1000
 | 
					
						
							|  |  |  |     loadinfo.seg_sizes[0] = (hdr.code_seg_size + 0xFFF) & ~0xFFF; | 
					
						
							|  |  |  |     loadinfo.seg_sizes[1] = (hdr.rodata_seg_size + 0xFFF) & ~0xFFF; | 
					
						
							|  |  |  |     loadinfo.seg_sizes[2] = (hdr.data_seg_size + 0xFFF) & ~0xFFF; | 
					
						
							|  |  |  |     u32 offsets[2] = {loadinfo.seg_sizes[0], loadinfo.seg_sizes[0] + loadinfo.seg_sizes[1]}; | 
					
						
							| 
									
										
										
										
											2015-07-09 17:11:09 -03:00
										 |  |  |     u32 n_reloc_tables = hdr.reloc_hdr_size / sizeof(u32); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     std::vector<u8> program_image(loadinfo.seg_sizes[0] + loadinfo.seg_sizes[1] + | 
					
						
							|  |  |  |                                   loadinfo.seg_sizes[2]); | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     loadinfo.seg_addrs[0] = base_addr; | 
					
						
							|  |  |  |     loadinfo.seg_addrs[1] = loadinfo.seg_addrs[0] + loadinfo.seg_sizes[0]; | 
					
						
							|  |  |  |     loadinfo.seg_addrs[2] = loadinfo.seg_addrs[1] + loadinfo.seg_sizes[1]; | 
					
						
							| 
									
										
										
										
											2015-07-09 17:11:09 -03:00
										 |  |  |     loadinfo.seg_ptrs[0] = program_image.data(); | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |     loadinfo.seg_ptrs[1] = loadinfo.seg_ptrs[0] + loadinfo.seg_sizes[0]; | 
					
						
							|  |  |  |     loadinfo.seg_ptrs[2] = loadinfo.seg_ptrs[1] + loadinfo.seg_sizes[1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Skip header for future compatibility
 | 
					
						
							|  |  |  |     file.Seek(hdr.header_size, SEEK_SET); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Read the relocation headers
 | 
					
						
							| 
									
										
										
										
											2015-07-09 17:11:09 -03:00
										 |  |  |     std::vector<u32> relocs(n_reloc_tables * NUM_SEGMENTS); | 
					
						
							|  |  |  |     for (unsigned int current_segment = 0; current_segment < NUM_SEGMENTS; ++current_segment) { | 
					
						
							|  |  |  |         size_t size = n_reloc_tables * sizeof(u32); | 
					
						
							| 
									
										
										
										
											2015-01-07 01:21:50 +00:00
										 |  |  |         if (file.ReadBytes(&relocs[current_segment * n_reloc_tables], size) != size) | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |             return ERROR_READ; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Read the segments
 | 
					
						
							|  |  |  |     if (file.ReadBytes(loadinfo.seg_ptrs[0], hdr.code_seg_size) != hdr.code_seg_size) | 
					
						
							|  |  |  |         return ERROR_READ; | 
					
						
							|  |  |  |     if (file.ReadBytes(loadinfo.seg_ptrs[1], hdr.rodata_seg_size) != hdr.rodata_seg_size) | 
					
						
							|  |  |  |         return ERROR_READ; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     if (file.ReadBytes(loadinfo.seg_ptrs[2], hdr.data_seg_size - hdr.bss_size) != | 
					
						
							|  |  |  |         hdr.data_seg_size - hdr.bss_size) | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |         return ERROR_READ; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // BSS clear
 | 
					
						
							|  |  |  |     memset((char*)loadinfo.seg_ptrs[2] + hdr.data_seg_size - hdr.bss_size, 0, hdr.bss_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Relocate the segments
 | 
					
						
							| 
									
										
										
										
											2015-07-09 17:11:09 -03:00
										 |  |  |     for (unsigned int current_segment = 0; current_segment < NUM_SEGMENTS; ++current_segment) { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         for (unsigned current_segment_reloc_table = 0; current_segment_reloc_table < n_reloc_tables; | 
					
						
							|  |  |  |              current_segment_reloc_table++) { | 
					
						
							| 
									
										
										
										
											2015-01-07 01:21:50 +00:00
										 |  |  |             u32 n_relocs = relocs[current_segment * n_reloc_tables + current_segment_reloc_table]; | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |             if (current_segment_reloc_table >= 2) { | 
					
						
							|  |  |  |                 // We are not using this table - ignore it because we don't know what it dose
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                 file.Seek(n_relocs * sizeof(THREEDSX_Reloc), SEEK_CUR); | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2015-07-09 17:11:09 -03:00
										 |  |  |             THREEDSX_Reloc reloc_table[RELOCBUFSIZE]; | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             u32* pos = (u32*)loadinfo.seg_ptrs[current_segment]; | 
					
						
							| 
									
										
										
										
											2015-01-07 01:21:50 +00:00
										 |  |  |             const u32* end_pos = pos + (loadinfo.seg_sizes[current_segment] / 4); | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             while (n_relocs) { | 
					
						
							|  |  |  |                 u32 remaining = std::min(RELOCBUFSIZE, n_relocs); | 
					
						
							|  |  |  |                 n_relocs -= remaining; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                 if (file.ReadBytes(reloc_table, remaining * sizeof(THREEDSX_Reloc)) != | 
					
						
							|  |  |  |                     remaining * sizeof(THREEDSX_Reloc)) | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |                     return ERROR_READ; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                 for (unsigned current_inprogress = 0; | 
					
						
							|  |  |  |                      current_inprogress < remaining && pos < end_pos; current_inprogress++) { | 
					
						
							| 
									
										
										
										
											2015-01-07 01:21:50 +00:00
										 |  |  |                     const auto& table = reloc_table[current_inprogress]; | 
					
						
							| 
									
										
										
										
											2015-10-09 22:13:54 +01:00
										 |  |  |                     LOG_TRACE(Loader, "(t=%d,skip=%u,patch=%u)", current_segment_reloc_table, | 
					
						
							| 
									
										
										
										
											2016-04-25 16:10:03 -04:00
										 |  |  |                               static_cast<u32>(table.skip), static_cast<u32>(table.patch)); | 
					
						
							| 
									
										
										
										
											2015-01-07 01:21:50 +00:00
										 |  |  |                     pos += table.skip; | 
					
						
							|  |  |  |                     s32 num_patches = table.patch; | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |                     while (0 < num_patches && pos < end_pos) { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                         u32 in_addr = | 
					
						
							|  |  |  |                             static_cast<u32>(reinterpret_cast<u8*>(pos) - program_image.data()); | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |                         u32 addr = TranslateAddr(*pos, &loadinfo, offsets); | 
					
						
							| 
									
										
										
										
											2015-10-09 22:13:54 +01:00
										 |  |  |                         LOG_TRACE(Loader, "Patching %08X <-- rel(%08X,%d) (%08X)", | 
					
						
							| 
									
										
										
										
											2015-01-07 01:21:50 +00:00
										 |  |  |                                   base_addr + in_addr, addr, current_segment_reloc_table, *pos); | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |                         switch (current_segment_reloc_table) { | 
					
						
							| 
									
										
										
										
											2015-01-07 01:21:50 +00:00
										 |  |  |                         case 0: | 
					
						
							|  |  |  |                             *pos = (addr); | 
					
						
							|  |  |  |                             break; | 
					
						
							|  |  |  |                         case 1: | 
					
						
							| 
									
										
										
										
											2015-06-29 10:21:40 -07:00
										 |  |  |                             *pos = static_cast<u32>(addr - in_addr); | 
					
						
							| 
									
										
										
										
											2015-01-07 01:21:50 +00:00
										 |  |  |                             break; | 
					
						
							|  |  |  |                         default: | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                             break; // this should never happen
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |                         } | 
					
						
							|  |  |  |                         pos++; | 
					
						
							|  |  |  |                         num_patches--; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |     // Create the CodeSet
 | 
					
						
							|  |  |  |     SharedPtr<CodeSet> code_set = CodeSet::Create("", 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     code_set->code.offset = loadinfo.seg_ptrs[0] - program_image.data(); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     code_set->code.addr = loadinfo.seg_addrs[0]; | 
					
						
							|  |  |  |     code_set->code.size = loadinfo.seg_sizes[0]; | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     code_set->rodata.offset = loadinfo.seg_ptrs[1] - program_image.data(); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     code_set->rodata.addr = loadinfo.seg_addrs[1]; | 
					
						
							|  |  |  |     code_set->rodata.size = loadinfo.seg_sizes[1]; | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     code_set->data.offset = loadinfo.seg_ptrs[2] - program_image.data(); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     code_set->data.addr = loadinfo.seg_addrs[2]; | 
					
						
							|  |  |  |     code_set->data.size = loadinfo.seg_sizes[2]; | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     code_set->entrypoint = code_set->code.addr; | 
					
						
							|  |  |  |     code_set->memory = std::make_shared<std::vector<u8>>(std::move(program_image)); | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 02:25:40 -03:00
										 |  |  |     LOG_DEBUG(Loader, "code size:   0x%X", loadinfo.seg_sizes[0]); | 
					
						
							|  |  |  |     LOG_DEBUG(Loader, "rodata size: 0x%X", loadinfo.seg_sizes[1]); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     LOG_DEBUG(Loader, "data size:   0x%X (including 0x%X of bss)", loadinfo.seg_sizes[2], | 
					
						
							|  |  |  |               hdr.bss_size); | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |     *out_codeset = code_set; | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |     return ERROR_NONE; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  | FileType AppLoader_THREEDSX::IdentifyType(FileUtil::IOFile& file) { | 
					
						
							|  |  |  |     u32 magic; | 
					
						
							|  |  |  |     file.Seek(0, SEEK_SET); | 
					
						
							|  |  |  |     if (1 != file.ReadArray<u32>(&magic, 1)) | 
					
						
							|  |  |  |         return FileType::Error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (MakeMagic('3', 'D', 'S', 'X') == magic) | 
					
						
							|  |  |  |         return FileType::THREEDSX; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return FileType::Error; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-05 20:09:35 +00:00
										 |  |  | ResultStatus AppLoader_THREEDSX::Load() { | 
					
						
							| 
									
										
										
										
											2015-01-06 19:49:25 +00:00
										 |  |  |     if (is_loaded) | 
					
						
							|  |  |  |         return ResultStatus::ErrorAlreadyLoaded; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-11 19:16:33 -03:00
										 |  |  |     if (!file.IsOpen()) | 
					
						
							| 
									
										
										
										
											2015-01-05 20:09:35 +00:00
										 |  |  |         return ResultStatus::Error; | 
					
						
							| 
									
										
										
										
											2015-01-06 21:30:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |     SharedPtr<CodeSet> codeset; | 
					
						
							| 
									
										
										
										
											2015-07-11 19:16:33 -03:00
										 |  |  |     if (Load3DSXFile(file, Memory::PROCESS_IMAGE_VADDR, &codeset) != ERROR_NONE) | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |         return ResultStatus::Error; | 
					
						
							|  |  |  |     codeset->name = filename; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Kernel::g_current_process = Kernel::Process::Create(std::move(codeset)); | 
					
						
							| 
									
										
										
										
											2015-05-08 16:51:48 -03:00
										 |  |  |     Kernel::g_current_process->svc_access_mask.set(); | 
					
						
							| 
									
										
										
										
											2015-05-08 18:12:25 -03:00
										 |  |  |     Kernel::g_current_process->address_mappings = default_address_mappings; | 
					
						
							| 
									
										
										
										
											2015-05-25 20:34:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-12 15:25:15 -05:00
										 |  |  |     // Attach the default resource limit (APPLICATION) to the process
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     Kernel::g_current_process->resource_limit = | 
					
						
							|  |  |  |         Kernel::ResourceLimit::GetForCategory(Kernel::ResourceLimitCategory::APPLICATION); | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |     Kernel::g_current_process->Run(48, Kernel::DEFAULT_STACK_SIZE); | 
					
						
							| 
									
										
										
										
											2015-01-06 19:49:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     Service::FS::RegisterArchiveType(std::make_unique<FileSys::ArchiveFactory_RomFS>(*this), | 
					
						
							|  |  |  |                                      Service::FS::ArchiveIdCode::RomFS); | 
					
						
							| 
									
										
										
										
											2016-05-17 23:06:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-06 19:49:25 +00:00
										 |  |  |     is_loaded = true; | 
					
						
							| 
									
										
										
										
											2015-01-05 20:09:35 +00:00
										 |  |  |     return ResultStatus::Success; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | ResultStatus AppLoader_THREEDSX::ReadRomFS(std::shared_ptr<FileUtil::IOFile>& romfs_file, | 
					
						
							|  |  |  |                                            u64& offset, u64& size) { | 
					
						
							| 
									
										
										
										
											2015-09-21 01:30:06 -04:00
										 |  |  |     if (!file.IsOpen()) | 
					
						
							|  |  |  |         return ResultStatus::Error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Reset read pointer in case this file has been read before.
 | 
					
						
							|  |  |  |     file.Seek(0, SEEK_SET); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     THREEDSX_Header hdr; | 
					
						
							|  |  |  |     if (file.ReadBytes(&hdr, sizeof(THREEDSX_Header)) != sizeof(THREEDSX_Header)) | 
					
						
							|  |  |  |         return ResultStatus::Error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (hdr.header_size != sizeof(THREEDSX_Header)) | 
					
						
							|  |  |  |         return ResultStatus::Error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Check if the 3DSX has a RomFS...
 | 
					
						
							|  |  |  |     if (hdr.fs_offset != 0) { | 
					
						
							|  |  |  |         u32 romfs_offset = hdr.fs_offset; | 
					
						
							| 
									
										
										
										
											2016-04-25 16:10:03 -04:00
										 |  |  |         u32 romfs_size = static_cast<u32>(file.GetSize()) - hdr.fs_offset; | 
					
						
							| 
									
										
										
										
											2015-09-21 01:30:06 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         LOG_DEBUG(Loader, "RomFS offset:           0x%08X", romfs_offset); | 
					
						
							|  |  |  |         LOG_DEBUG(Loader, "RomFS size:             0x%08X", romfs_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // We reopen the file, to allow its position to be independent from file's
 | 
					
						
							|  |  |  |         romfs_file = std::make_shared<FileUtil::IOFile>(filepath, "rb"); | 
					
						
							|  |  |  |         if (!romfs_file->IsOpen()) | 
					
						
							|  |  |  |             return ResultStatus::Error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         offset = romfs_offset; | 
					
						
							|  |  |  |         size = romfs_size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return ResultStatus::Success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     LOG_DEBUG(Loader, "3DSX has no RomFS"); | 
					
						
							|  |  |  |     return ResultStatus::ErrorNotUsed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-14 00:04:05 +03:00
										 |  |  | ResultStatus AppLoader_THREEDSX::ReadIcon(std::vector<u8>& buffer) { | 
					
						
							|  |  |  |     if (!file.IsOpen()) | 
					
						
							|  |  |  |         return ResultStatus::Error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Reset read pointer in case this file has been read before.
 | 
					
						
							|  |  |  |     file.Seek(0, SEEK_SET); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     THREEDSX_Header hdr; | 
					
						
							|  |  |  |     if (file.ReadBytes(&hdr, sizeof(THREEDSX_Header)) != sizeof(THREEDSX_Header)) | 
					
						
							|  |  |  |         return ResultStatus::Error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (hdr.header_size != sizeof(THREEDSX_Header)) | 
					
						
							|  |  |  |         return ResultStatus::Error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Check if the 3DSX has a SMDH...
 | 
					
						
							|  |  |  |     if (hdr.smdh_offset != 0) { | 
					
						
							|  |  |  |         file.Seek(hdr.smdh_offset, SEEK_SET); | 
					
						
							|  |  |  |         buffer.resize(hdr.smdh_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (file.ReadBytes(&buffer[0], hdr.smdh_size) != hdr.smdh_size) | 
					
						
							|  |  |  |             return ResultStatus::Error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return ResultStatus::Success; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return ResultStatus::ErrorNotUsed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | } // namespace Loader
 |