| 
									
										
										
										
											2014-04-08 19:15:46 -04:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-04-08 19:15:46 -04:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							| 
									
										
										
										
											2013-09-19 23:21:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-21 13:40:28 +01:00
										 |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2014-12-20 03:21:23 -02:00
										 |  |  | #include <string>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-20 03:21:23 -02:00
										 |  |  | #include "common/make_unique.h"
 | 
					
						
							| 
									
										
										
										
											2015-06-20 22:24:01 +01:00
										 |  |  | #include "common/string_util.h"
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | #include "core/file_sys/archive_romfs.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | #include "core/hle/kernel/process.h"
 | 
					
						
							|  |  |  | #include "core/hle/service/fs/archive.h"
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | #include "core/loader/3dsx.h"
 | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | #include "core/loader/elf.h"
 | 
					
						
							| 
									
										
										
										
											2014-06-16 22:57:09 -04:00
										 |  |  | #include "core/loader/ncch.h"
 | 
					
						
							| 
									
										
										
										
											2014-04-21 23:09:10 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-19 23:21:22 -04:00
										 |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Loader { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-08 18:12:25 -03:00
										 |  |  | const std::initializer_list<Kernel::AddressMapping> default_address_mappings = { | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  |     { 0x1FF50000,   0x8000, true  }, // part of DSP RAM
 | 
					
						
							|  |  |  |     { 0x1FF70000,   0x8000, true  }, // part of DSP RAM
 | 
					
						
							|  |  |  |     { 0x1F000000, 0x600000, false }, // entire VRAM
 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-19 23:21:22 -04:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Identifies the type of a bootable file | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  |  * @param file open file | 
					
						
							|  |  |  |  * @return FileType of file | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static FileType IdentifyFile(FileUtil::IOFile& file) { | 
					
						
							|  |  |  |     FileType type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define CHECK_TYPE(loader) \
 | 
					
						
							|  |  |  |     type = AppLoader_##loader::IdentifyType(file); \ | 
					
						
							|  |  |  |     if (FileType::Error != type) \ | 
					
						
							|  |  |  |         return type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CHECK_TYPE(THREEDSX) | 
					
						
							|  |  |  |     CHECK_TYPE(ELF) | 
					
						
							|  |  |  |     CHECK_TYPE(NCCH) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #undef CHECK_TYPE
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return FileType::Unknown; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Guess the type of a bootable file from its extension | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  |  * @param extension String extension of bootable file | 
					
						
							| 
									
										
										
										
											2013-09-19 23:21:22 -04:00
										 |  |  |  * @return FileType of file | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | static FileType GuessFromExtension(const std::string& extension_) { | 
					
						
							|  |  |  |     std::string extension = Common::ToLower(extension_); | 
					
						
							| 
									
										
										
										
											2014-09-07 00:49:52 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  |     if (extension == ".elf") | 
					
						
							| 
									
										
										
										
											2014-09-07 11:50:43 -07:00
										 |  |  |         return FileType::ELF; | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  |     else if (extension == ".axf") | 
					
						
							| 
									
										
										
										
											2014-09-07 11:50:43 -07:00
										 |  |  |         return FileType::ELF; | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  |     else if (extension == ".cxi") | 
					
						
							| 
									
										
										
										
											2014-09-07 11:50:43 -07:00
										 |  |  |         return FileType::CXI; | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  |     else if (extension == ".cci") | 
					
						
							| 
									
										
										
										
											2014-09-07 11:50:43 -07:00
										 |  |  |         return FileType::CCI; | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  |     else if (extension == ".3ds") | 
					
						
							| 
									
										
										
										
											2015-01-05 04:45:09 +01:00
										 |  |  |         return FileType::CCI; | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  |     else if (extension == ".3dsx") | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |         return FileType::THREEDSX; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     return FileType::Unknown; | 
					
						
							| 
									
										
										
										
											2013-09-19 23:21:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 00:33:00 +00:00
										 |  |  | static const char* GetFileTypeString(FileType type) { | 
					
						
							|  |  |  |     switch (type) { | 
					
						
							|  |  |  |     case FileType::CCI: | 
					
						
							|  |  |  |         return "NCSD"; | 
					
						
							|  |  |  |     case FileType::CXI: | 
					
						
							|  |  |  |         return "NCCH"; | 
					
						
							|  |  |  |     case FileType::ELF: | 
					
						
							|  |  |  |         return "ELF"; | 
					
						
							|  |  |  |     case FileType::THREEDSX: | 
					
						
							|  |  |  |         return "3DSX"; | 
					
						
							|  |  |  |     case FileType::Error: | 
					
						
							|  |  |  |     case FileType::Unknown: | 
					
						
							| 
									
										
										
										
											2015-02-01 15:31:21 -05:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2015-01-07 00:33:00 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-02-01 15:31:21 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return "unknown"; | 
					
						
							| 
									
										
										
										
											2015-01-07 00:33:00 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-03-31 22:23:55 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 00:33:00 +00:00
										 |  |  | ResultStatus LoadFile(const std::string& filename) { | 
					
						
							| 
									
										
										
										
											2015-07-11 19:16:33 -03:00
										 |  |  |     FileUtil::IOFile file(filename, "rb"); | 
					
						
							|  |  |  |     if (!file.IsOpen()) { | 
					
						
							| 
									
										
										
										
											2015-01-07 00:33:00 +00:00
										 |  |  |         LOG_ERROR(Loader, "Failed to load file %s", filename.c_str()); | 
					
						
							| 
									
										
										
										
											2015-01-06 21:30:40 +00:00
										 |  |  |         return ResultStatus::Error; | 
					
						
							| 
									
										
										
										
											2015-01-07 00:33:00 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-01-06 21:30:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  |     std::string filename_filename, filename_extension; | 
					
						
							|  |  |  |     Common::SplitPath(filename, nullptr, &filename_filename, &filename_extension); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-11 19:16:33 -03:00
										 |  |  |     FileType type = IdentifyFile(file); | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  |     FileType filename_type = GuessFromExtension(filename_extension); | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (type != filename_type) { | 
					
						
							|  |  |  |         LOG_WARNING(Loader, "File %s has a different type than its extension.", filename.c_str()); | 
					
						
							|  |  |  |         if (FileType::Unknown == type) | 
					
						
							|  |  |  |             type = filename_type; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 00:33:00 +00:00
										 |  |  |     LOG_INFO(Loader, "Loading file %s as %s...", filename.c_str(), GetFileTypeString(type)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  |     switch (type) { | 
					
						
							| 
									
										
										
										
											2014-03-31 22:23:55 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  |     //3DSX file format...
 | 
					
						
							|  |  |  |     case FileType::THREEDSX: | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  |         return AppLoader_THREEDSX(std::move(file), filename_filename).Load(); | 
					
						
							| 
									
										
										
										
											2014-12-07 21:47:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     // Standard ELF file format...
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |     case FileType::ELF: | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  |         return AppLoader_ELF(std::move(file), filename_filename).Load(); | 
					
						
							| 
									
										
										
										
											2014-04-30 23:50:14 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     // NCCH/NCSD container formats...
 | 
					
						
							|  |  |  |     case FileType::CXI: | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  |     case FileType::CCI: | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2015-07-10 00:55:23 +03:00
										 |  |  |         AppLoader_NCCH app_loader(std::move(file), filename); | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Load application and RomFS
 | 
					
						
							|  |  |  |         if (ResultStatus::Success == app_loader.Load()) { | 
					
						
							| 
									
										
										
										
											2015-02-06 11:53:14 -02:00
										 |  |  |             Service::FS::RegisterArchiveType(Common::make_unique<FileSys::ArchiveFactory_RomFS>(app_loader), Service::FS::ArchiveIdCode::RomFS); | 
					
						
							| 
									
										
										
										
											2014-06-27 16:18:56 -04:00
										 |  |  |             return ResultStatus::Success; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-06-16 22:57:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     // Error occurred durring IdentifyFile...
 | 
					
						
							|  |  |  |     case FileType::Error: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // IdentifyFile could know identify file type...
 | 
					
						
							|  |  |  |     case FileType::Unknown: | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2015-01-30 11:33:20 -05:00
										 |  |  |         LOG_CRITICAL(Loader, "File %s is of unknown type.", filename.c_str()); | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |         return ResultStatus::ErrorInvalidFormat; | 
					
						
							| 
									
										
										
										
											2014-03-31 22:23:55 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     return ResultStatus::Error; | 
					
						
							| 
									
										
										
										
											2013-09-19 23:21:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-16 23:18:10 -04:00
										 |  |  | } // namespace Loader
 |