| 
									
										
										
										
											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.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #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"
 | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | #include "core/hle/service/fs/archive.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | #include "core/loader/elf.h"
 | 
					
						
							|  |  |  | #include "core/loader/ncch.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-12 22:38:29 -03:00
										 |  |  | #include "core/memory.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "3dsx.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01: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)
 | 
					
						
							|  |  |  | struct THREEDSX_Header | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Relocation header: all fields (even extra unknown fields) are guaranteed to be relocation counts.
 | 
					
						
							|  |  |  | struct THREEDSX_RelocHdr | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // # of absolute relocations (that is, fix address to post-relocation memory layout)
 | 
					
						
							| 
									
										
										
										
											2015-01-05 20:09:35 +00:00
										 |  |  |     u32 cross_segment_absolute; | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01: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
 | 
					
						
							|  |  |  | struct THREEDSX_Reloc | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     u16 skip, patch; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | #pragma pack()
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct THREEloadinfo | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     u8* seg_ptrs[3]; // code, rodata & data
 | 
					
						
							|  |  |  |     u32 seg_addrs[3]; | 
					
						
							|  |  |  |     u32 seg_sizes[3]; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 01:21:50 +00: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-01-06 19:56:26 +00:00
										 |  |  | static THREEDSX_Error Load3DSXFile(FileUtil::IOFile& file, u32 base_addr) | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  |     //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); | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  |     if (file.ReadBytes(loadinfo.seg_ptrs[2], hdr.data_seg_size - hdr.bss_size) != hdr.data_seg_size - hdr.bss_size) | 
					
						
							|  |  |  |         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) { | 
					
						
							| 
									
										
										
										
											2015-01-07 01:21:50 +00:00
										 |  |  |         for (unsigned current_segment_reloc_table = 0; current_segment_reloc_table < n_reloc_tables; current_segment_reloc_table++) { | 
					
						
							|  |  |  |             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
 | 
					
						
							|  |  |  |                 file.Seek(n_relocs*sizeof(THREEDSX_Reloc), SEEK_CUR); | 
					
						
							|  |  |  |                 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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 01:21:50 +00: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; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 01:21:50 +00:00
										 |  |  |                 for (unsigned current_inprogress = 0; current_inprogress < remaining && pos < end_pos; current_inprogress++) { | 
					
						
							|  |  |  |                     const auto& table = reloc_table[current_inprogress]; | 
					
						
							|  |  |  |                     LOG_TRACE(Loader, "(t=%d,skip=%u,patch=%u)\n", current_segment_reloc_table, | 
					
						
							|  |  |  |                               (u32)table.skip, (u32)table.patch); | 
					
						
							|  |  |  |                     pos += table.skip; | 
					
						
							|  |  |  |                     s32 num_patches = table.patch; | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |                     while (0 < num_patches && pos < end_pos) { | 
					
						
							| 
									
										
										
										
											2015-07-09 17:11:09 -03:00
										 |  |  |                         u32 in_addr = (u8*)pos - program_image.data(); | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |                         u32 addr = TranslateAddr(*pos, &loadinfo, offsets); | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |                         LOG_TRACE(Loader, "Patching %08X <-- rel(%08X,%d) (%08X)\n", | 
					
						
							| 
									
										
										
										
											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: | 
					
						
							|  |  |  |                             *pos = (addr - in_addr); | 
					
						
							|  |  |  |                             break; | 
					
						
							|  |  |  |                         default: | 
					
						
							|  |  |  |                             break; //this should never happen
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |                         } | 
					
						
							|  |  |  |                         pos++; | 
					
						
							|  |  |  |                         num_patches--; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Write the data
 | 
					
						
							| 
									
										
										
										
											2015-07-09 17:11:09 -03:00
										 |  |  |     memcpy(Memory::GetPointer(base_addr), program_image.data(), program_image.size()); | 
					
						
							| 
									
										
										
										
											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]); | 
					
						
							|  |  |  |     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
										 |  |  | 
 | 
					
						
							|  |  |  |     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-01-06 21:30:40 +00:00
										 |  |  |     if (!file->IsOpen()) | 
					
						
							| 
									
										
										
										
											2015-01-05 20:09:35 +00:00
										 |  |  |         return ResultStatus::Error; | 
					
						
							| 
									
										
										
										
											2015-01-06 21:30:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  |     Kernel::g_current_process = Kernel::Process::Create(filename, 0); | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							|  |  |  |     Kernel::g_current_process->resource_limit = Kernel::ResourceLimit::GetForCategory(Kernel::ResourceLimitCategory::APPLICATION); | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 17:11:09 -03:00
										 |  |  |     if (Load3DSXFile(*file, Memory::PROCESS_IMAGE_VADDR) != ERROR_NONE) | 
					
						
							|  |  |  |         return ResultStatus::Error; | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-09 00:39:56 -03:00
										 |  |  |     Kernel::g_current_process->Run(Memory::PROCESS_IMAGE_VADDR, 48, Kernel::DEFAULT_STACK_SIZE); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							|  |  |  | } // namespace Loader
 |