| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-03 21:58:19 -04:00
										 |  |  | #include <cstddef>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							|  |  |  | #include "core/file_sys/program_metadata.h"
 | 
					
						
							| 
									
										
										
										
											2020-08-23 14:20:37 -04:00
										 |  |  | #include "core/file_sys/vfs.h"
 | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  | #include "core/loader/loader.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace FileSys { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-19 19:19:05 -04:00
										 |  |  | ProgramMetadata::ProgramMetadata() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ProgramMetadata::~ProgramMetadata() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | Loader::ResultStatus ProgramMetadata::Load(VirtualFile file) { | 
					
						
							| 
									
										
										
										
											2019-04-05 15:52:13 -04:00
										 |  |  |     const std::size_t total_size = file->GetSize(); | 
					
						
							|  |  |  |     if (total_size < sizeof(Header)) { | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |         return Loader::ResultStatus::ErrorBadNPDMHeader; | 
					
						
							| 
									
										
										
										
											2019-04-05 15:52:13 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-05 15:52:13 -04:00
										 |  |  |     if (sizeof(Header) != file->ReadObject(&npdm_header)) { | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |         return Loader::ResultStatus::ErrorBadNPDMHeader; | 
					
						
							| 
									
										
										
										
											2019-04-05 15:52:13 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-05 15:52:13 -04:00
										 |  |  |     if (sizeof(AcidHeader) != file->ReadObject(&acid_header, npdm_header.acid_offset)) { | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |         return Loader::ResultStatus::ErrorBadACIDHeader; | 
					
						
							| 
									
										
										
										
											2019-04-05 15:52:13 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-05 15:52:13 -04:00
										 |  |  |     if (sizeof(AciHeader) != file->ReadObject(&aci_header, npdm_header.aci_offset)) { | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |         return Loader::ResultStatus::ErrorBadACIHeader; | 
					
						
							| 
									
										
										
										
											2019-04-05 15:52:13 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-07 20:24:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-10 16:25:25 -04:00
										 |  |  |     // Load acid_file_access per-component instead of the entire struct, since this struct does not
 | 
					
						
							|  |  |  |     // reflect the layout of the real data.
 | 
					
						
							|  |  |  |     std::size_t current_offset = acid_header.fac_offset; | 
					
						
							|  |  |  |     if (sizeof(FileAccessControl::version) != file->ReadBytes(&acid_file_access.version, | 
					
						
							|  |  |  |                                                               sizeof(FileAccessControl::version), | 
					
						
							|  |  |  |                                                               current_offset)) { | 
					
						
							|  |  |  |         return Loader::ResultStatus::ErrorBadFileAccessControl; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (sizeof(FileAccessControl::permissions) != | 
					
						
							|  |  |  |         file->ReadBytes(&acid_file_access.permissions, sizeof(FileAccessControl::permissions), | 
					
						
							|  |  |  |                         current_offset += sizeof(FileAccessControl::version) + 3)) { | 
					
						
							|  |  |  |         return Loader::ResultStatus::ErrorBadFileAccessControl; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (sizeof(FileAccessControl::unknown) != | 
					
						
							|  |  |  |         file->ReadBytes(&acid_file_access.unknown, sizeof(FileAccessControl::unknown), | 
					
						
							|  |  |  |                         current_offset + sizeof(FileAccessControl::permissions))) { | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |         return Loader::ResultStatus::ErrorBadFileAccessControl; | 
					
						
							| 
									
										
										
										
											2019-04-05 15:52:13 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-10 16:25:25 -04:00
										 |  |  |     // Load aci_file_access per-component instead of the entire struct, same as acid_file_access
 | 
					
						
							|  |  |  |     current_offset = aci_header.fah_offset; | 
					
						
							|  |  |  |     if (sizeof(FileAccessHeader::version) != file->ReadBytes(&aci_file_access.version, | 
					
						
							|  |  |  |                                                              sizeof(FileAccessHeader::version), | 
					
						
							|  |  |  |                                                              current_offset)) { | 
					
						
							|  |  |  |         return Loader::ResultStatus::ErrorBadFileAccessHeader; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (sizeof(FileAccessHeader::permissions) != | 
					
						
							|  |  |  |         file->ReadBytes(&aci_file_access.permissions, sizeof(FileAccessHeader::permissions), | 
					
						
							|  |  |  |                         current_offset += sizeof(FileAccessHeader::version) + 3)) { | 
					
						
							|  |  |  |         return Loader::ResultStatus::ErrorBadFileAccessHeader; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (sizeof(FileAccessHeader::unk_offset) != | 
					
						
							|  |  |  |         file->ReadBytes(&aci_file_access.unk_offset, sizeof(FileAccessHeader::unk_offset), | 
					
						
							|  |  |  |                         current_offset += sizeof(FileAccessHeader::permissions))) { | 
					
						
							|  |  |  |         return Loader::ResultStatus::ErrorBadFileAccessHeader; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (sizeof(FileAccessHeader::unk_size) != | 
					
						
							|  |  |  |         file->ReadBytes(&aci_file_access.unk_size, sizeof(FileAccessHeader::unk_size), | 
					
						
							|  |  |  |                         current_offset += sizeof(FileAccessHeader::unk_offset))) { | 
					
						
							|  |  |  |         return Loader::ResultStatus::ErrorBadFileAccessHeader; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (sizeof(FileAccessHeader::unk_offset_2) != | 
					
						
							|  |  |  |         file->ReadBytes(&aci_file_access.unk_offset_2, sizeof(FileAccessHeader::unk_offset_2), | 
					
						
							|  |  |  |                         current_offset += sizeof(FileAccessHeader::unk_size))) { | 
					
						
							|  |  |  |         return Loader::ResultStatus::ErrorBadFileAccessHeader; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (sizeof(FileAccessHeader::unk_size_2) != | 
					
						
							|  |  |  |         file->ReadBytes(&aci_file_access.unk_size_2, sizeof(FileAccessHeader::unk_size_2), | 
					
						
							|  |  |  |                         current_offset + sizeof(FileAccessHeader::unk_offset_2))) { | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |         return Loader::ResultStatus::ErrorBadFileAccessHeader; | 
					
						
							| 
									
										
										
										
											2019-04-05 15:52:13 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-07 20:24:51 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-19 23:50:20 -05:00
										 |  |  |     aci_kernel_capabilities.resize(aci_header.kac_size / sizeof(u32)); | 
					
						
							|  |  |  |     const u64 read_size = aci_header.kac_size; | 
					
						
							|  |  |  |     const u64 read_offset = npdm_header.aci_offset + aci_header.kac_offset; | 
					
						
							|  |  |  |     if (file->ReadBytes(aci_kernel_capabilities.data(), read_size, read_offset) != read_size) { | 
					
						
							|  |  |  |         return Loader::ResultStatus::ErrorBadKernelCapabilityDescriptors; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  |     return Loader::ResultStatus::Success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-20 18:15:52 -04:00
										 |  |  | /*static*/ ProgramMetadata ProgramMetadata::GetDefault() { | 
					
						
							| 
									
										
										
										
											2021-11-10 19:23:59 -05:00
										 |  |  |     // Allow use of cores 0~3 and thread priorities 1~63.
 | 
					
						
							| 
									
										
										
										
											2023-02-11 13:28:03 -05:00
										 |  |  |     constexpr static u32 default_thread_info_capability = 0x30007F7; | 
					
						
							| 
									
										
										
										
											2021-11-10 19:23:59 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-20 18:15:52 -04:00
										 |  |  |     ProgramMetadata result; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     result.LoadManual( | 
					
						
							|  |  |  |         true /*is_64_bit*/, FileSys::ProgramAddressSpaceType::Is39Bit /*address_space*/, | 
					
						
							|  |  |  |         0x2c /*main_thread_prio*/, 0 /*main_thread_core*/, 0x00100000 /*main_thread_stack_size*/, | 
					
						
							| 
									
										
										
										
											2021-04-23 09:37:35 -04:00
										 |  |  |         0 /*title_id*/, 0xFFFFFFFFFFFFFFFF /*filesystem_permissions*/, | 
					
						
							| 
									
										
										
										
											2021-11-10 19:23:59 -05:00
										 |  |  |         0x1FE00000 /*system_resource_size*/, {default_thread_info_capability} /*capabilities*/); | 
					
						
							| 
									
										
										
										
											2020-04-20 18:15:52 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-05 00:20:26 -04:00
										 |  |  | void ProgramMetadata::LoadManual(bool is_64_bit, ProgramAddressSpaceType address_space, | 
					
						
							| 
									
										
										
										
											2019-11-12 05:20:26 -05:00
										 |  |  |                                  s32 main_thread_prio, u32 main_thread_core, | 
					
						
							| 
									
										
										
										
											2019-06-05 00:20:26 -04:00
										 |  |  |                                  u32 main_thread_stack_size, u64 title_id, | 
					
						
							| 
									
										
										
										
											2021-04-23 09:37:35 -04:00
										 |  |  |                                  u64 filesystem_permissions, u32 system_resource_size, | 
					
						
							| 
									
										
										
										
											2019-06-05 00:20:26 -04:00
										 |  |  |                                  KernelCapabilityDescriptors capabilities) { | 
					
						
							|  |  |  |     npdm_header.has_64_bit_instructions.Assign(is_64_bit); | 
					
						
							|  |  |  |     npdm_header.address_space_type.Assign(address_space); | 
					
						
							| 
									
										
										
										
											2019-11-12 05:20:26 -05:00
										 |  |  |     npdm_header.main_thread_priority = static_cast<u8>(main_thread_prio); | 
					
						
							|  |  |  |     npdm_header.main_thread_cpu = static_cast<u8>(main_thread_core); | 
					
						
							| 
									
										
										
										
											2019-06-05 00:20:26 -04:00
										 |  |  |     npdm_header.main_stack_size = main_thread_stack_size; | 
					
						
							|  |  |  |     aci_header.title_id = title_id; | 
					
						
							|  |  |  |     aci_file_access.permissions = filesystem_permissions; | 
					
						
							| 
									
										
										
										
											2021-04-23 09:37:35 -04:00
										 |  |  |     npdm_header.system_resource_size = system_resource_size; | 
					
						
							| 
									
										
										
										
											2021-09-29 01:17:30 -04:00
										 |  |  |     aci_kernel_capabilities = std::move(capabilities); | 
					
						
							| 
									
										
										
										
											2019-06-05 00:20:26 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  | bool ProgramMetadata::Is64BitProgram() const { | 
					
						
							| 
									
										
										
										
											2022-10-21 02:34:06 -04:00
										 |  |  |     return npdm_header.has_64_bit_instructions.As<bool>(); | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ProgramAddressSpaceType ProgramMetadata::GetAddressSpaceType() const { | 
					
						
							|  |  |  |     return npdm_header.address_space_type; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u8 ProgramMetadata::GetMainThreadPriority() const { | 
					
						
							|  |  |  |     return npdm_header.main_thread_priority; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u8 ProgramMetadata::GetMainThreadCore() const { | 
					
						
							|  |  |  |     return npdm_header.main_thread_cpu; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u32 ProgramMetadata::GetMainThreadStackSize() const { | 
					
						
							|  |  |  |     return npdm_header.main_stack_size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u64 ProgramMetadata::GetTitleID() const { | 
					
						
							|  |  |  |     return aci_header.title_id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u64 ProgramMetadata::GetFilesystemPermissions() const { | 
					
						
							|  |  |  |     return aci_file_access.permissions; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-07 09:42:54 -07:00
										 |  |  | u32 ProgramMetadata::GetSystemResourceSize() const { | 
					
						
							|  |  |  |     return npdm_header.system_resource_size; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-19 23:50:20 -05:00
										 |  |  | const ProgramMetadata::KernelCapabilityDescriptors& ProgramMetadata::GetKernelCapabilities() const { | 
					
						
							|  |  |  |     return aci_kernel_capabilities; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  | void ProgramMetadata::Print() const { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |     LOG_DEBUG(Service_FS, "Magic:                  {:.4}", npdm_header.magic.data()); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, "Main thread priority:   0x{:02X}", npdm_header.main_thread_priority); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, "Main thread core:       {}", npdm_header.main_thread_cpu); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, "Main thread stack size: 0x{:X} bytes", npdm_header.main_stack_size); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, "Process category:       {}", npdm_header.process_category); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, "Flags:                  0x{:02X}", npdm_header.flags); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, " > 64-bit instructions: {}", | 
					
						
							| 
									
										
										
										
											2018-07-02 10:20:50 -06:00
										 |  |  |               npdm_header.has_64_bit_instructions ? "YES" : "NO"); | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 19:16:43 -05:00
										 |  |  |     const char* address_space = "Unknown"; | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  |     switch (npdm_header.address_space_type) { | 
					
						
							| 
									
										
										
										
											2018-09-22 20:09:32 -04:00
										 |  |  |     case ProgramAddressSpaceType::Is36Bit: | 
					
						
							| 
									
										
										
										
											2018-12-27 19:16:43 -05:00
										 |  |  |         address_space = "64-bit (36-bit address space)"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2018-09-22 20:09:32 -04:00
										 |  |  |     case ProgramAddressSpaceType::Is39Bit: | 
					
						
							| 
									
										
										
										
											2018-12-27 19:16:43 -05:00
										 |  |  |         address_space = "64-bit (39-bit address space)"; | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case ProgramAddressSpaceType::Is32Bit: | 
					
						
							|  |  |  |         address_space = "32-bit"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2018-12-27 19:16:43 -05:00
										 |  |  |     case ProgramAddressSpaceType::Is32BitNoMap: | 
					
						
							|  |  |  |         address_space = "32-bit (no map region)"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |     LOG_DEBUG(Service_FS, " > Address space:       {}\n", address_space); | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Begin ACID printing (potential perms, signed)
 | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |     LOG_DEBUG(Service_FS, "Magic:                  {:.4}", acid_header.magic.data()); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, "Flags:                  0x{:02X}", acid_header.flags); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, " > Is Retail:           {}", acid_header.is_retail ? "YES" : "NO"); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, "Title ID Min:           0x{:016X}", acid_header.title_id_min); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, "Title ID Max:           0x{:016X}", acid_header.title_id_max); | 
					
						
							| 
									
										
										
										
											2021-06-10 16:25:25 -04:00
										 |  |  |     LOG_DEBUG(Service_FS, "Filesystem Access:      0x{:016X}\n", acid_file_access.permissions); | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Begin ACI0 printing (actual perms, unsigned)
 | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |     LOG_DEBUG(Service_FS, "Magic:                  {:.4}", aci_header.magic.data()); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, "Title ID:               0x{:016X}", aci_header.title_id); | 
					
						
							|  |  |  |     LOG_DEBUG(Service_FS, "Filesystem Access:      0x{:016X}\n", aci_file_access.permissions); | 
					
						
							| 
									
										
										
										
											2018-02-25 03:34:19 -07:00
										 |  |  | } | 
					
						
							|  |  |  | } // namespace FileSys
 |