| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:12:14 -04:00
										 |  |  | #include <algorithm>
 | 
					
						
							| 
									
										
										
										
											2018-09-03 21:58:19 -04:00
										 |  |  | #include <cstring>
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  | #include <optional>
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:12:14 -04:00
										 |  |  | #include <utility>
 | 
					
						
							| 
									
										
										
										
											2018-09-03 21:58:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | #include "common/polyfill_ranges.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | #include "core/crypto/aes_util.h"
 | 
					
						
							|  |  |  | #include "core/crypto/ctr_encryption_layer.h"
 | 
					
						
							| 
									
										
										
										
											2020-08-23 14:20:37 -04:00
										 |  |  | #include "core/crypto/key_manager.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | #include "core/file_sys/content_archive.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-03 21:58:19 -04:00
										 |  |  | #include "core/file_sys/partition_filesystem.h"
 | 
					
						
							| 
									
										
										
										
											2024-01-16 06:23:01 +01:00
										 |  |  | #include "core/file_sys/vfs/vfs_offset.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | #include "core/loader/loader.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  | #include "core/file_sys/fssystem/fssystem_compression_configuration.h"
 | 
					
						
							|  |  |  | #include "core/file_sys/fssystem/fssystem_crypto_configuration.h"
 | 
					
						
							|  |  |  | #include "core/file_sys/fssystem/fssystem_nca_file_system_driver.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  | namespace FileSys { | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-28 22:22:00 -04:00
										 |  |  | static u8 MasterKeyIdForKeyGeneration(u8 key_generation) { | 
					
						
							|  |  |  |     return std::max<u8>(key_generation, 1) - 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  | NCA::NCA(VirtualFile file_, const NCA* base_nca) | 
					
						
							|  |  |  |     : file(std::move(file_)), keys{Core::Crypto::KeyManager::Instance()} { | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  |     if (file == nullptr) { | 
					
						
							|  |  |  |         status = Loader::ResultStatus::ErrorNullFile; | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |     reader = std::make_shared<NcaReader>(); | 
					
						
							|  |  |  |     if (Result rc = | 
					
						
							| 
									
										
										
										
											2023-08-12 15:18:55 -04:00
										 |  |  |             reader->Initialize(file, GetCryptoConfiguration(), GetNcaCompressionConfiguration()); | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |         R_FAILED(rc)) { | 
					
						
							|  |  |  |         if (rc != ResultInvalidNcaSignature) { | 
					
						
							|  |  |  |             LOG_ERROR(Loader, "File reader errored out during header read: {:#x}", | 
					
						
							|  |  |  |                       rc.GetInnerValue()); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  |         status = Loader::ResultStatus::ErrorBadNCAHeader; | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-28 22:22:00 -04:00
										 |  |  |     // Ensure we have the proper key area keys to continue.
 | 
					
						
							|  |  |  |     const u8 master_key_id = MasterKeyIdForKeyGeneration(reader->GetKeyGeneration()); | 
					
						
							|  |  |  |     if (!keys.HasKey(Core::Crypto::S128KeyType::KeyArea, master_key_id, reader->GetKeyIndex())) { | 
					
						
							|  |  |  |         status = Loader::ResultStatus::ErrorMissingKeyAreaKey; | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |     RightsId rights_id{}; | 
					
						
							|  |  |  |     reader->GetRightsId(rights_id.data(), rights_id.size()); | 
					
						
							|  |  |  |     if (rights_id != RightsId{}) { | 
					
						
							|  |  |  |         // External decryption key required; provide it here.
 | 
					
						
							|  |  |  |         u128 rights_id_u128; | 
					
						
							|  |  |  |         std::memcpy(rights_id_u128.data(), rights_id.data(), sizeof(rights_id)); | 
					
						
							| 
									
										
										
										
											2023-07-12 23:17:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |         auto titlekey = | 
					
						
							|  |  |  |             keys.GetKey(Core::Crypto::S128KeyType::Titlekey, rights_id_u128[1], rights_id_u128[0]); | 
					
						
							|  |  |  |         if (titlekey == Core::Crypto::Key128{}) { | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorMissingTitlekey; | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2023-07-12 23:17:18 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-28 22:22:00 -04:00
										 |  |  |         if (!keys.HasKey(Core::Crypto::S128KeyType::Titlekek, master_key_id)) { | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |             status = Loader::ResultStatus::ErrorMissingTitlekek; | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-28 22:22:00 -04:00
										 |  |  |         auto titlekek = keys.GetKey(Core::Crypto::S128KeyType::Titlekek, master_key_id); | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |         Core::Crypto::AESCipher<Core::Crypto::Key128> cipher(titlekek, Core::Crypto::Mode::ECB); | 
					
						
							|  |  |  |         cipher.Transcode(titlekey.data(), titlekey.size(), titlekey.data(), | 
					
						
							|  |  |  |                          Core::Crypto::Op::Decrypt); | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |         reader->SetExternalDecryptionKey(titlekey.data(), titlekey.size()); | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |     const s32 fs_count = reader->GetFsCount(); | 
					
						
							|  |  |  |     NcaFileSystemDriver fs(base_nca ? base_nca->reader : nullptr, reader); | 
					
						
							|  |  |  |     std::vector<VirtualFile> filesystems(fs_count); | 
					
						
							|  |  |  |     for (s32 i = 0; i < fs_count; i++) { | 
					
						
							|  |  |  |         NcaFsHeaderReader header_reader; | 
					
						
							|  |  |  |         const Result rc = fs.OpenStorage(&filesystems[i], &header_reader, i); | 
					
						
							|  |  |  |         if (R_FAILED(rc)) { | 
					
						
							|  |  |  |             LOG_ERROR(Loader, "File reader errored out during read of section {}: {:#x}", i, | 
					
						
							|  |  |  |                       rc.GetInnerValue()); | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorBadNCAHeader; | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |         if (header_reader.GetFsType() == NcaFsHeader::FsType::RomFs) { | 
					
						
							|  |  |  |             files.push_back(filesystems[i]); | 
					
						
							|  |  |  |             romfs = files.back(); | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |         if (header_reader.GetFsType() == NcaFsHeader::FsType::PartitionFs) { | 
					
						
							|  |  |  |             auto npfs = std::make_shared<PartitionFilesystem>(filesystems[i]); | 
					
						
							|  |  |  |             if (npfs->GetStatus() == Loader::ResultStatus::Success) { | 
					
						
							|  |  |  |                 dirs.push_back(npfs); | 
					
						
							|  |  |  |                 if (IsDirectoryExeFS(npfs)) { | 
					
						
							|  |  |  |                     exefs = dirs.back(); | 
					
						
							|  |  |  |                 } else if (IsDirectoryLogoPartition(npfs)) { | 
					
						
							|  |  |  |                     logo = dirs.back(); | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     continue; | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |         if (header_reader.GetEncryptionType() == NcaFsHeader::EncryptionType::AesCtrEx) { | 
					
						
							|  |  |  |             is_update = true; | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-12-07 22:00:34 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-29 20:47:33 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |     if (is_update && base_nca == nullptr) { | 
					
						
							|  |  |  |         status = Loader::ResultStatus::ErrorMissingBKTRBaseRomFS; | 
					
						
							| 
									
										
										
										
											2020-12-07 22:00:34 -05:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |         status = Loader::ResultStatus::Success; | 
					
						
							| 
									
										
										
										
											2020-12-07 22:00:34 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  | NCA::~NCA() = default; | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | Loader::ResultStatus NCA::GetStatus() const { | 
					
						
							|  |  |  |     return status; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-10 01:31:58 -05:00
										 |  |  | std::vector<VirtualFile> NCA::GetFiles() const { | 
					
						
							|  |  |  |     if (status != Loader::ResultStatus::Success) { | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-12-10 01:31:58 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  |     return files; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-10 01:31:58 -05:00
										 |  |  | std::vector<VirtualDir> NCA::GetSubdirectories() const { | 
					
						
							|  |  |  |     if (status != Loader::ResultStatus::Success) { | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-12-10 01:31:58 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  |     return dirs; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string NCA::GetName() const { | 
					
						
							|  |  |  |     return file->GetName(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-10 01:31:58 -05:00
										 |  |  | VirtualDir NCA::GetParentDirectory() const { | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  |     return file->GetContainingDirectory(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NCAContentType NCA::GetType() const { | 
					
						
							| 
									
										
										
										
											2023-08-12 15:18:55 -04:00
										 |  |  |     return static_cast<NCAContentType>(reader->GetContentType()); | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u64 NCA::GetTitleId() const { | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |     if (is_update) { | 
					
						
							|  |  |  |         return reader->GetProgramId() | 0x800; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-08-12 15:18:55 -04:00
										 |  |  |     return reader->GetProgramId(); | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  | RightsId NCA::GetRightsId() const { | 
					
						
							|  |  |  |     RightsId result; | 
					
						
							|  |  |  |     reader->GetRightsId(result.data(), result.size()); | 
					
						
							|  |  |  |     return result; | 
					
						
							| 
									
										
										
										
											2019-04-10 10:23:13 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u32 NCA::GetSDKVersion() const { | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |     return reader->GetSdkAddonVersion(); | 
					
						
							| 
									
										
										
										
											2019-04-10 10:23:13 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-09 11:36:40 -05:00
										 |  |  | u8 NCA::GetKeyGeneration() const { | 
					
						
							|  |  |  |     return reader->GetKeyGeneration(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-25 19:01:46 -04:00
										 |  |  | bool NCA::IsUpdate() const { | 
					
						
							|  |  |  |     return is_update; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | VirtualFile NCA::GetRomFS() const { | 
					
						
							|  |  |  |     return romfs; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | VirtualDir NCA::GetExeFS() const { | 
					
						
							|  |  |  |     return exefs; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | VirtualFile NCA::GetBaseFile() const { | 
					
						
							|  |  |  |     return file; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-15 15:56:32 -05:00
										 |  |  | VirtualDir NCA::GetLogoPartition() const { | 
					
						
							|  |  |  |     return logo; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | } // namespace FileSys
 |