| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-17 14:04:18 -04:00
										 |  |  | #include <algorithm>
 | 
					
						
							| 
									
										
										
										
											2019-04-22 17:56:56 -04:00
										 |  |  | #include <random>
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | #include <regex>
 | 
					
						
							|  |  |  | #include <mbedtls/sha256.h>
 | 
					
						
							|  |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  | #include "common/fs/path_util.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | #include "common/hex_util.h"
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2023-09-11 23:50:36 -04:00
										 |  |  | #include "common/scope_exit.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-03 21:58:19 -04:00
										 |  |  | #include "core/crypto/key_manager.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | #include "core/file_sys/card_image.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-16 00:24:06 -04:00
										 |  |  | #include "core/file_sys/common_funcs.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-03 21:58:19 -04:00
										 |  |  | #include "core/file_sys/content_archive.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | #include "core/file_sys/nca_metadata.h"
 | 
					
						
							|  |  |  | #include "core/file_sys/registered_cache.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-04 14:44:40 -04:00
										 |  |  | #include "core/file_sys/submission_package.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | #include "core/file_sys/vfs_concat.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-03 21:58:19 -04:00
										 |  |  | #include "core/loader/loader.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace FileSys { | 
					
						
							| 
									
										
										
										
											2018-09-23 21:50:16 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | // The size of blocks to use when vfs raw copying into nand.
 | 
					
						
							|  |  |  | constexpr size_t VFS_RC_LARGE_COPY_BLOCK = 0x400000; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | std::string ContentProviderEntry::DebugInfo() const { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     return fmt::format("title_id={:016X}, content_type={:02X}", title_id, static_cast<u8>(type)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | bool operator<(const ContentProviderEntry& lhs, const ContentProviderEntry& rhs) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     return (lhs.title_id < rhs.title_id) || (lhs.title_id == rhs.title_id && lhs.type < rhs.type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | bool operator==(const ContentProviderEntry& lhs, const ContentProviderEntry& rhs) { | 
					
						
							| 
									
										
										
										
											2018-10-17 14:04:18 -04:00
										 |  |  |     return std::tie(lhs.title_id, lhs.type) == std::tie(rhs.title_id, rhs.type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | bool operator!=(const ContentProviderEntry& lhs, const ContentProviderEntry& rhs) { | 
					
						
							| 
									
										
										
										
											2018-10-17 18:27:23 -04:00
										 |  |  |     return !operator==(lhs, rhs); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | static bool FollowsTwoDigitDirFormat(std::string_view name) { | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |     static const std::regex two_digit_regex("000000[0-9A-F]{2}", std::regex_constants::ECMAScript | | 
					
						
							|  |  |  |                                                                      std::regex_constants::icase); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     return std::regex_match(name.begin(), name.end(), two_digit_regex); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool FollowsNcaIdFormat(std::string_view name) { | 
					
						
							| 
									
										
										
										
											2018-08-12 15:55:44 -04:00
										 |  |  |     static const std::regex nca_id_regex("[0-9A-F]{32}\\.nca", std::regex_constants::ECMAScript | | 
					
						
							|  |  |  |                                                                    std::regex_constants::icase); | 
					
						
							| 
									
										
										
										
											2019-04-10 12:24:44 -04:00
										 |  |  |     static const std::regex nca_id_cnmt_regex( | 
					
						
							|  |  |  |         "[0-9A-F]{32}\\.cnmt.nca", std::regex_constants::ECMAScript | std::regex_constants::icase); | 
					
						
							|  |  |  |     return (name.size() == 36 && std::regex_match(name.begin(), name.end(), nca_id_regex)) || | 
					
						
							|  |  |  |            (name.size() == 41 && std::regex_match(name.begin(), name.end(), nca_id_cnmt_regex)); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static std::string GetRelativePathFromNcaID(const std::array<u8, 16>& nca_id, bool second_hex_upper, | 
					
						
							| 
									
										
										
										
											2019-04-10 12:24:44 -04:00
										 |  |  |                                             bool within_two_digit, bool cnmt_suffix) { | 
					
						
							| 
									
										
										
										
											2021-06-23 09:59:56 -04:00
										 |  |  |     if (!within_two_digit) { | 
					
						
							|  |  |  |         const auto format_str = fmt::runtime(cnmt_suffix ? "{}.cnmt.nca" : "/{}.nca"); | 
					
						
							|  |  |  |         return fmt::format(format_str, Common::HexToString(nca_id, second_hex_upper)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Core::Crypto::SHA256Hash hash{}; | 
					
						
							| 
									
										
										
										
											2019-11-12 08:37:58 -05:00
										 |  |  |     mbedtls_sha256_ret(nca_id.data(), nca_id.size(), hash.data(), 0); | 
					
						
							| 
									
										
										
										
											2021-06-23 09:59:56 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const auto format_str = | 
					
						
							|  |  |  |         fmt::runtime(cnmt_suffix ? "/000000{:02X}/{}.cnmt.nca" : "/000000{:02X}/{}.nca"); | 
					
						
							|  |  |  |     return fmt::format(format_str, hash[0], Common::HexToString(nca_id, second_hex_upper)); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static std::string GetCNMTName(TitleType type, u64 title_id) { | 
					
						
							| 
									
										
										
										
											2023-02-14 11:13:47 -05:00
										 |  |  |     static constexpr std::array<const char*, 9> TITLE_TYPE_NAMES{ | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         "SystemProgram", | 
					
						
							|  |  |  |         "SystemData", | 
					
						
							|  |  |  |         "SystemUpdate", | 
					
						
							|  |  |  |         "BootImagePackage", | 
					
						
							|  |  |  |         "BootImagePackageSafe", | 
					
						
							|  |  |  |         "Application", | 
					
						
							|  |  |  |         "Patch", | 
					
						
							|  |  |  |         "AddOnContent", | 
					
						
							|  |  |  |         "" ///< Currently unknown 'DeltaTitle'
 | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |     auto index = static_cast<std::size_t>(type); | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |     // If the index is after the jump in TitleType, subtract it out.
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |     if (index >= static_cast<std::size_t>(TitleType::Application)) { | 
					
						
							|  |  |  |         index -= static_cast<std::size_t>(TitleType::Application) - | 
					
						
							|  |  |  |                  static_cast<std::size_t>(TitleType::FirmwarePackageB); | 
					
						
							| 
									
										
										
										
											2018-08-12 15:55:44 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     return fmt::format("{}_{:016x}.cnmt", TITLE_TYPE_NAMES[index], title_id); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | ContentRecordType GetCRTypeFromNCAType(NCAContentType type) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     switch (type) { | 
					
						
							|  |  |  |     case NCAContentType::Program: | 
					
						
							|  |  |  |         // TODO(DarkLordZach): Differentiate between Program and Patch
 | 
					
						
							|  |  |  |         return ContentRecordType::Program; | 
					
						
							|  |  |  |     case NCAContentType::Meta: | 
					
						
							|  |  |  |         return ContentRecordType::Meta; | 
					
						
							|  |  |  |     case NCAContentType::Control: | 
					
						
							|  |  |  |         return ContentRecordType::Control; | 
					
						
							|  |  |  |     case NCAContentType::Data: | 
					
						
							| 
									
										
										
										
											2019-03-19 15:14:52 -04:00
										 |  |  |     case NCAContentType::PublicData: | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         return ContentRecordType::Data; | 
					
						
							|  |  |  |     case NCAContentType::Manual: | 
					
						
							|  |  |  |         // TODO(DarkLordZach): Peek at NCA contents to differentiate Manual and Legal.
 | 
					
						
							| 
									
										
										
										
											2019-07-02 00:57:23 +01:00
										 |  |  |         return ContentRecordType::HtmlDocument; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     default: | 
					
						
							| 
									
										
										
										
											2022-06-07 17:02:29 -04:00
										 |  |  |         ASSERT_MSG(false, "Invalid NCAContentType={:02X}", type); | 
					
						
							| 
									
										
										
										
											2021-01-05 04:18:16 -03:00
										 |  |  |         return ContentRecordType{}; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | ContentProvider::~ContentProvider() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool ContentProvider::HasEntry(ContentProviderEntry entry) const { | 
					
						
							|  |  |  |     return HasEntry(entry.title_id, entry.type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | VirtualFile ContentProvider::GetEntryUnparsed(ContentProviderEntry entry) const { | 
					
						
							|  |  |  |     return GetEntryUnparsed(entry.title_id, entry.type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | VirtualFile ContentProvider::GetEntryRaw(ContentProviderEntry entry) const { | 
					
						
							|  |  |  |     return GetEntryRaw(entry.title_id, entry.type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::unique_ptr<NCA> ContentProvider::GetEntry(ContentProviderEntry entry) const { | 
					
						
							|  |  |  |     return GetEntry(entry.title_id, entry.type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::vector<ContentProviderEntry> ContentProvider::ListEntries() const { | 
					
						
							|  |  |  |     return ListEntriesFilter(std::nullopt, std::nullopt, std::nullopt); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 12:21:51 -04:00
										 |  |  | PlaceholderCache::PlaceholderCache(VirtualDir dir_) : dir(std::move(dir_)) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool PlaceholderCache::Create(const NcaID& id, u64 size) const { | 
					
						
							|  |  |  |     const auto path = GetRelativePathFromNcaID(id, false, true, false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (dir->GetFileRelative(path) != nullptr) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Core::Crypto::SHA256Hash hash{}; | 
					
						
							| 
									
										
										
										
											2019-11-12 08:37:58 -05:00
										 |  |  |     mbedtls_sha256_ret(id.data(), id.size(), hash.data(), 0); | 
					
						
							| 
									
										
										
										
											2019-04-10 12:21:51 -04:00
										 |  |  |     const auto dirname = fmt::format("000000{:02X}", hash[0]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto dir2 = GetOrCreateDirectoryRelative(dir, dirname); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (dir2 == nullptr) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-21 18:43:11 -04:00
										 |  |  |     const auto file = dir2->CreateFile(fmt::format("{}.nca", Common::HexToString(id, false))); | 
					
						
							| 
									
										
										
										
											2019-04-10 12:21:51 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (file == nullptr) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return file->Resize(size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool PlaceholderCache::Delete(const NcaID& id) const { | 
					
						
							|  |  |  |     const auto path = GetRelativePathFromNcaID(id, false, true, false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (dir->GetFileRelative(path) == nullptr) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Core::Crypto::SHA256Hash hash{}; | 
					
						
							| 
									
										
										
										
											2019-11-12 08:37:58 -05:00
										 |  |  |     mbedtls_sha256_ret(id.data(), id.size(), hash.data(), 0); | 
					
						
							| 
									
										
										
										
											2019-04-10 12:21:51 -04:00
										 |  |  |     const auto dirname = fmt::format("000000{:02X}", hash[0]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto dir2 = GetOrCreateDirectoryRelative(dir, dirname); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-21 18:43:11 -04:00
										 |  |  |     const auto res = dir2->DeleteFile(fmt::format("{}.nca", Common::HexToString(id, false))); | 
					
						
							| 
									
										
										
										
											2019-04-10 12:21:51 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return res; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool PlaceholderCache::Exists(const NcaID& id) const { | 
					
						
							|  |  |  |     const auto path = GetRelativePathFromNcaID(id, false, true, false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return dir->GetFileRelative(path) != nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool PlaceholderCache::Write(const NcaID& id, u64 offset, const std::vector<u8>& data) const { | 
					
						
							|  |  |  |     const auto path = GetRelativePathFromNcaID(id, false, true, false); | 
					
						
							|  |  |  |     const auto file = dir->GetFileRelative(path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (file == nullptr) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return file->WriteBytes(data, offset) == data.size(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool PlaceholderCache::Register(RegisteredCache* cache, const NcaID& placeholder, | 
					
						
							|  |  |  |                                 const NcaID& install) const { | 
					
						
							|  |  |  |     const auto path = GetRelativePathFromNcaID(placeholder, false, true, false); | 
					
						
							|  |  |  |     const auto file = dir->GetFileRelative(path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (file == nullptr) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto res = cache->RawInstallNCA(NCA{file}, &VfsRawCopy, false, install); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (res != InstallResult::Success) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return Delete(placeholder); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool PlaceholderCache::CleanAll() const { | 
					
						
							|  |  |  |     return dir->GetParentDirectory()->CleanSubdirectoryRecursive(dir->GetName()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::optional<std::array<u8, 0x10>> PlaceholderCache::GetRightsID(const NcaID& id) const { | 
					
						
							|  |  |  |     const auto path = GetRelativePathFromNcaID(id, false, true, false); | 
					
						
							|  |  |  |     const auto file = dir->GetFileRelative(path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (file == nullptr) | 
					
						
							|  |  |  |         return std::nullopt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     NCA nca{file}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (nca.GetStatus() != Loader::ResultStatus::Success && | 
					
						
							|  |  |  |         nca.GetStatus() != Loader::ResultStatus::ErrorMissingBKTRBaseRomFS) { | 
					
						
							|  |  |  |         return std::nullopt; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto rights_id = nca.GetRightsId(); | 
					
						
							|  |  |  |     if (rights_id == NcaID{}) | 
					
						
							|  |  |  |         return std::nullopt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return rights_id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u64 PlaceholderCache::Size(const NcaID& id) const { | 
					
						
							|  |  |  |     const auto path = GetRelativePathFromNcaID(id, false, true, false); | 
					
						
							|  |  |  |     const auto file = dir->GetFileRelative(path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (file == nullptr) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return file->GetSize(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool PlaceholderCache::SetSize(const NcaID& id, u64 new_size) const { | 
					
						
							|  |  |  |     const auto path = GetRelativePathFromNcaID(id, false, true, false); | 
					
						
							|  |  |  |     const auto file = dir->GetFileRelative(path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (file == nullptr) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return file->Resize(new_size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::vector<NcaID> PlaceholderCache::List() const { | 
					
						
							|  |  |  |     std::vector<NcaID> out; | 
					
						
							|  |  |  |     for (const auto& sdir : dir->GetSubdirectories()) { | 
					
						
							|  |  |  |         for (const auto& file : sdir->GetFiles()) { | 
					
						
							|  |  |  |             const auto name = file->GetName(); | 
					
						
							| 
									
										
										
										
											2020-08-23 11:25:43 -04:00
										 |  |  |             if (name.length() == 36 && name.ends_with(".nca")) { | 
					
						
							| 
									
										
										
										
											2019-04-10 12:21:51 -04:00
										 |  |  |                 out.push_back(Common::HexStringToArray<0x10>(name.substr(0, 32))); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return out; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NcaID PlaceholderCache::Generate() { | 
					
						
							|  |  |  |     std::random_device device; | 
					
						
							|  |  |  |     std::mt19937 gen(device()); | 
					
						
							|  |  |  |     std::uniform_int_distribution<u64> distribution(1, std::numeric_limits<u64>::max()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     NcaID out{}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto v1 = distribution(gen); | 
					
						
							|  |  |  |     const auto v2 = distribution(gen); | 
					
						
							|  |  |  |     std::memcpy(out.data(), &v1, sizeof(u64)); | 
					
						
							|  |  |  |     std::memcpy(out.data() + sizeof(u64), &v2, sizeof(u64)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return out; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-02 02:34:40 -04:00
										 |  |  | VirtualFile RegisteredCache::OpenFileOrDirectoryConcat(const VirtualDir& open_dir, | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |                                                        std::string_view path) const { | 
					
						
							| 
									
										
										
										
											2021-05-02 02:34:40 -04:00
										 |  |  |     const auto file = open_dir->GetFileRelative(path); | 
					
						
							| 
									
										
										
										
											2018-12-01 23:50:10 -05:00
										 |  |  |     if (file != nullptr) { | 
					
						
							| 
									
										
										
										
											2018-11-01 20:23:38 -04:00
										 |  |  |         return file; | 
					
						
							| 
									
										
										
										
											2018-12-01 23:50:10 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-11-01 20:23:38 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-02 02:34:40 -04:00
										 |  |  |     const auto nca_dir = open_dir->GetDirectoryRelative(path); | 
					
						
							| 
									
										
										
										
											2018-12-01 23:50:10 -05:00
										 |  |  |     if (nca_dir == nullptr) { | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-01 23:50:10 -05:00
										 |  |  |     const auto files = nca_dir->GetFiles(); | 
					
						
							|  |  |  |     if (files.size() == 1 && files[0]->GetName() == "00") { | 
					
						
							|  |  |  |         return files[0]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::vector<VirtualFile> concat; | 
					
						
							|  |  |  |     // Since the files are a two-digit hex number, max is FF.
 | 
					
						
							|  |  |  |     for (std::size_t i = 0; i < 0x100; ++i) { | 
					
						
							|  |  |  |         auto next = nca_dir->GetFile(fmt::format("{:02X}", i)); | 
					
						
							|  |  |  |         if (next != nullptr) { | 
					
						
							|  |  |  |             concat.push_back(std::move(next)); | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2018-12-01 23:50:10 -05:00
										 |  |  |             next = nca_dir->GetFile(fmt::format("{:02x}", i)); | 
					
						
							|  |  |  |             if (next != nullptr) { | 
					
						
							|  |  |  |                 concat.push_back(std::move(next)); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-12-01 23:50:10 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-01 23:50:10 -05:00
										 |  |  |     if (concat.empty()) { | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-12-01 23:50:10 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-13 13:34:41 -04:00
										 |  |  |     auto name = concat.front()->GetName(); | 
					
						
							|  |  |  |     return ConcatenatedVfsFile::MakeConcatenatedFile(std::move(name), std::move(concat)); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | VirtualFile RegisteredCache::GetFileAtID(NcaID id) const { | 
					
						
							|  |  |  |     VirtualFile file; | 
					
						
							| 
									
										
										
										
											2019-04-10 12:24:44 -04:00
										 |  |  |     // Try all five relevant modes of file storage:
 | 
					
						
							|  |  |  |     // (bit 2 = uppercase/lower, bit 1 = within a two-digit dir, bit 0 = .cnmt suffix)
 | 
					
						
							|  |  |  |     // 000: /000000**/{:032X}.nca
 | 
					
						
							|  |  |  |     // 010: /{:032X}.nca
 | 
					
						
							|  |  |  |     // 100: /000000**/{:032x}.nca
 | 
					
						
							|  |  |  |     // 110: /{:032x}.nca
 | 
					
						
							|  |  |  |     // 111: /{:032x}.cnmt.nca
 | 
					
						
							|  |  |  |     for (u8 i = 0; i < 8; ++i) { | 
					
						
							|  |  |  |         if ((i % 2) == 1 && i != 7) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         const auto path = | 
					
						
							|  |  |  |             GetRelativePathFromNcaID(id, (i & 0b100) == 0, (i & 0b010) == 0, (i & 0b001) == 0b001); | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |         file = OpenFileOrDirectoryConcat(dir, path); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         if (file != nullptr) | 
					
						
							|  |  |  |             return file; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return file; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | static std::optional<NcaID> CheckMapForContentRecord(const std::map<u64, CNMT>& map, u64 title_id, | 
					
						
							|  |  |  |                                                      ContentRecordType type) { | 
					
						
							| 
									
										
										
										
											2020-08-03 07:34:42 -04:00
										 |  |  |     const auto cmnt_iter = map.find(title_id); | 
					
						
							|  |  |  |     if (cmnt_iter == map.cend()) { | 
					
						
							|  |  |  |         return std::nullopt; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-03 07:34:42 -04:00
										 |  |  |     const auto& cnmt = cmnt_iter->second; | 
					
						
							|  |  |  |     const auto& content_records = cnmt.GetContentRecords(); | 
					
						
							|  |  |  |     const auto iter = std::find_if(content_records.cbegin(), content_records.cend(), | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |                                    [type](const ContentRecord& rec) { return rec.type == type; }); | 
					
						
							| 
									
										
										
										
											2020-08-03 07:34:42 -04:00
										 |  |  |     if (iter == content_records.cend()) { | 
					
						
							|  |  |  |         return std::nullopt; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |     return std::make_optional(iter->nca_id); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  | std::optional<NcaID> RegisteredCache::GetNcaIDFromMetadata(u64 title_id, | 
					
						
							|  |  |  |                                                            ContentRecordType type) const { | 
					
						
							| 
									
										
										
										
											2018-08-10 11:10:44 -04:00
										 |  |  |     if (type == ContentRecordType::Meta && meta_id.find(title_id) != meta_id.end()) | 
					
						
							|  |  |  |         return meta_id.at(title_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto res1 = CheckMapForContentRecord(yuzu_meta, title_id, type); | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |     if (res1) | 
					
						
							| 
									
										
										
										
											2018-08-10 11:10:44 -04:00
										 |  |  |         return res1; | 
					
						
							|  |  |  |     return CheckMapForContentRecord(meta, title_id, type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  | std::vector<NcaID> RegisteredCache::AccumulateFiles() const { | 
					
						
							|  |  |  |     std::vector<NcaID> ids; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     for (const auto& d2_dir : dir->GetSubdirectories()) { | 
					
						
							|  |  |  |         if (FollowsNcaIdFormat(d2_dir->GetName())) { | 
					
						
							| 
									
										
										
										
											2018-08-15 23:16:11 -04:00
										 |  |  |             ids.push_back(Common::HexStringToArray<0x10, true>(d2_dir->GetName().substr(0, 0x20))); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!FollowsTwoDigitDirFormat(d2_dir->GetName())) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (const auto& nca_dir : d2_dir->GetSubdirectories()) { | 
					
						
							| 
									
										
										
										
											2022-03-27 17:06:27 -04:00
										 |  |  |             if (nca_dir == nullptr || !FollowsNcaIdFormat(nca_dir->GetName())) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2022-03-27 17:06:27 -04:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-15 23:16:11 -04:00
										 |  |  |             ids.push_back(Common::HexStringToArray<0x10, true>(nca_dir->GetName().substr(0, 0x20))); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (const auto& nca_file : d2_dir->GetFiles()) { | 
					
						
							| 
									
										
										
										
											2022-03-27 17:06:27 -04:00
										 |  |  |             if (nca_file == nullptr || !FollowsNcaIdFormat(nca_file->GetName())) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |                 continue; | 
					
						
							| 
									
										
										
										
											2022-03-27 17:06:27 -04:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-15 23:16:11 -04:00
										 |  |  |             ids.push_back( | 
					
						
							|  |  |  |                 Common::HexStringToArray<0x10, true>(nca_file->GetName().substr(0, 0x20))); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const auto& d2_file : dir->GetFiles()) { | 
					
						
							|  |  |  |         if (FollowsNcaIdFormat(d2_file->GetName())) | 
					
						
							| 
									
										
										
										
											2018-08-15 23:16:11 -04:00
										 |  |  |             ids.push_back(Common::HexStringToArray<0x10, true>(d2_file->GetName().substr(0, 0x20))); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  |     return ids; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void RegisteredCache::ProcessFiles(const std::vector<NcaID>& ids) { | 
					
						
							|  |  |  |     for (const auto& id : ids) { | 
					
						
							|  |  |  |         const auto file = GetFileAtID(id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (file == nullptr) | 
					
						
							|  |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |         const auto nca = std::make_shared<NCA>(parser(file, id)); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         if (nca->GetStatus() != Loader::ResultStatus::Success || | 
					
						
							| 
									
										
										
										
											2023-08-16 12:25:36 -04:00
										 |  |  |             nca->GetType() != NCAContentType::Meta || nca->GetSubdirectories().empty()) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         const auto section0 = nca->GetSubdirectories()[0]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-27 15:55:43 -04:00
										 |  |  |         for (const auto& section0_file : section0->GetFiles()) { | 
					
						
							|  |  |  |             if (section0_file->GetExtension() != "cnmt") | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |                 continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-27 15:55:43 -04:00
										 |  |  |             meta.insert_or_assign(nca->GetTitleId(), CNMT(section0_file)); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |             meta_id.insert_or_assign(nca->GetTitleId(), id); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void RegisteredCache::AccumulateYuzuMeta() { | 
					
						
							| 
									
										
										
										
											2021-05-02 02:34:40 -04:00
										 |  |  |     const auto meta_dir = dir->GetSubdirectory("yuzu_meta"); | 
					
						
							|  |  |  |     if (meta_dir == nullptr) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2021-05-02 02:34:40 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-02 02:34:40 -04:00
										 |  |  |     for (const auto& file : meta_dir->GetFiles()) { | 
					
						
							|  |  |  |         if (file->GetExtension() != "cnmt") { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2021-05-02 02:34:40 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         CNMT cnmt(file); | 
					
						
							|  |  |  |         yuzu_meta.insert_or_assign(cnmt.GetTitleID(), std::move(cnmt)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void RegisteredCache::Refresh() { | 
					
						
							| 
									
										
										
										
											2021-05-02 02:34:40 -04:00
										 |  |  |     if (dir == nullptr) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         return; | 
					
						
							| 
									
										
										
										
											2021-05-02 02:34:40 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  |     const auto ids = AccumulateFiles(); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     ProcessFiles(ids); | 
					
						
							|  |  |  |     AccumulateYuzuMeta(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | RegisteredCache::RegisteredCache(VirtualDir dir_, ContentProviderParsingFunction parsing_function) | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     : dir(std::move(dir_)), parser(std::move(parsing_function)) { | 
					
						
							|  |  |  |     Refresh(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-18 21:16:20 -04:00
										 |  |  | RegisteredCache::~RegisteredCache() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | bool RegisteredCache::HasEntry(u64 title_id, ContentRecordType type) const { | 
					
						
							|  |  |  |     return GetEntryRaw(title_id, type) != nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-16 17:07:37 -04:00
										 |  |  | VirtualFile RegisteredCache::GetEntryUnparsed(u64 title_id, ContentRecordType type) const { | 
					
						
							|  |  |  |     const auto id = GetNcaIDFromMetadata(title_id, type); | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |     return id ? GetFileAtID(*id) : nullptr; | 
					
						
							| 
									
										
										
										
											2018-08-16 17:07:37 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  | std::optional<u32> RegisteredCache::GetEntryVersion(u64 title_id) const { | 
					
						
							| 
									
										
										
										
											2018-08-26 10:53:31 -04:00
										 |  |  |     const auto meta_iter = meta.find(title_id); | 
					
						
							| 
									
										
										
										
											2020-08-03 07:34:42 -04:00
										 |  |  |     if (meta_iter != meta.cend()) { | 
					
						
							| 
									
										
										
										
											2018-08-26 10:53:31 -04:00
										 |  |  |         return meta_iter->second.GetTitleVersion(); | 
					
						
							| 
									
										
										
										
											2020-08-03 07:34:42 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-26 10:53:31 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const auto yuzu_meta_iter = yuzu_meta.find(title_id); | 
					
						
							| 
									
										
										
										
											2020-08-03 07:34:42 -04:00
										 |  |  |     if (yuzu_meta_iter != yuzu_meta.cend()) { | 
					
						
							| 
									
										
										
										
											2018-08-26 10:53:31 -04:00
										 |  |  |         return yuzu_meta_iter->second.GetTitleVersion(); | 
					
						
							| 
									
										
										
										
											2020-08-03 07:34:42 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-26 10:53:31 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-03 07:34:42 -04:00
										 |  |  |     return std::nullopt; | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | VirtualFile RegisteredCache::GetEntryRaw(u64 title_id, ContentRecordType type) const { | 
					
						
							|  |  |  |     const auto id = GetNcaIDFromMetadata(title_id, type); | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |     return id ? parser(GetFileAtID(*id), *id) : nullptr; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-16 09:05:47 -04:00
										 |  |  | std::unique_ptr<NCA> RegisteredCache::GetEntry(u64 title_id, ContentRecordType type) const { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     const auto raw = GetEntryRaw(title_id, type); | 
					
						
							|  |  |  |     if (raw == nullptr) | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |     return std::make_unique<NCA>(raw); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename T> | 
					
						
							|  |  |  | void RegisteredCache::IterateAllMetadata( | 
					
						
							|  |  |  |     std::vector<T>& out, std::function<T(const CNMT&, const ContentRecord&)> proc, | 
					
						
							|  |  |  |     std::function<bool(const CNMT&, const ContentRecord&)> filter) const { | 
					
						
							|  |  |  |     for (const auto& kv : meta) { | 
					
						
							|  |  |  |         const auto& cnmt = kv.second; | 
					
						
							|  |  |  |         if (filter(cnmt, EMPTY_META_CONTENT_RECORD)) | 
					
						
							|  |  |  |             out.push_back(proc(cnmt, EMPTY_META_CONTENT_RECORD)); | 
					
						
							|  |  |  |         for (const auto& rec : cnmt.GetContentRecords()) { | 
					
						
							|  |  |  |             if (GetFileAtID(rec.nca_id) != nullptr && filter(cnmt, rec)) { | 
					
						
							|  |  |  |                 out.push_back(proc(cnmt, rec)); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     for (const auto& kv : yuzu_meta) { | 
					
						
							|  |  |  |         const auto& cnmt = kv.second; | 
					
						
							|  |  |  |         for (const auto& rec : cnmt.GetContentRecords()) { | 
					
						
							|  |  |  |             if (GetFileAtID(rec.nca_id) != nullptr && filter(cnmt, rec)) { | 
					
						
							|  |  |  |                 out.push_back(proc(cnmt, rec)); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | std::vector<ContentProviderEntry> RegisteredCache::ListEntriesFilter( | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |     std::optional<TitleType> title_type, std::optional<ContentRecordType> record_type, | 
					
						
							|  |  |  |     std::optional<u64> title_id) const { | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     std::vector<ContentProviderEntry> out; | 
					
						
							|  |  |  |     IterateAllMetadata<ContentProviderEntry>( | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         out, | 
					
						
							|  |  |  |         [](const CNMT& c, const ContentRecord& r) { | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |             return ContentProviderEntry{c.GetTitleID(), r.type}; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         }, | 
					
						
							|  |  |  |         [&title_type, &record_type, &title_id](const CNMT& c, const ContentRecord& r) { | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |             if (title_type && *title_type != c.GetType()) | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |                 return false; | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |             if (record_type && *record_type != r.type) | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |                 return false; | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |             if (title_id && *title_id != c.GetTitleID()) | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |                 return false; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |     return out; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-27 13:52:13 -05:00
										 |  |  | static std::shared_ptr<NCA> GetNCAFromNSPForID(const NSP& nsp, const NcaID& id) { | 
					
						
							| 
									
										
										
										
											2019-06-12 17:27:06 -04:00
										 |  |  |     auto file = nsp.GetFile(fmt::format("{}.nca", Common::HexToString(id, false))); | 
					
						
							|  |  |  |     if (file == nullptr) { | 
					
						
							| 
									
										
										
										
											2018-08-25 11:50:04 -04:00
										 |  |  |         return nullptr; | 
					
						
							| 
									
										
										
										
											2019-06-12 17:27:06 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     return std::make_shared<NCA>(std::move(file)); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-27 13:52:13 -05:00
										 |  |  | InstallResult RegisteredCache::InstallEntry(const XCI& xci, bool overwrite_if_exists, | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |                                             const VfsCopyFunction& copy) { | 
					
						
							| 
									
										
										
										
											2018-11-27 13:52:13 -05:00
										 |  |  |     return InstallEntry(*xci.GetSecurePartitionNSP(), overwrite_if_exists, copy); | 
					
						
							| 
									
										
										
										
											2018-08-25 11:50:04 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-27 13:52:13 -05:00
										 |  |  | InstallResult RegisteredCache::InstallEntry(const NSP& nsp, bool overwrite_if_exists, | 
					
						
							| 
									
										
										
										
											2018-08-25 11:50:04 -04:00
										 |  |  |                                             const VfsCopyFunction& copy) { | 
					
						
							| 
									
										
										
										
											2018-11-27 13:52:13 -05:00
										 |  |  |     const auto ncas = nsp.GetNCAsCollapsed(); | 
					
						
							|  |  |  |     const auto meta_iter = std::find_if(ncas.begin(), ncas.end(), [](const auto& nca) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         return nca->GetType() == NCAContentType::Meta; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (meta_iter == ncas.end()) { | 
					
						
							| 
									
										
										
										
											2019-07-01 07:31:32 +01:00
										 |  |  |         LOG_ERROR(Loader, "The file you are attempting to install does not have a metadata NCA and " | 
					
						
							|  |  |  |                           "is therefore malformed. Check your encryption keys."); | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |         return InstallResult::ErrorMetaFailed; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto meta_id_raw = (*meta_iter)->GetName().substr(0, 32); | 
					
						
							| 
									
										
										
										
											2021-05-02 02:34:40 -04:00
										 |  |  |     const auto meta_id_data = Common::HexStringToArray<16>(meta_id_raw); | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-16 05:22:51 -04:00
										 |  |  |     if ((*meta_iter)->GetSubdirectories().empty()) { | 
					
						
							|  |  |  |         LOG_ERROR(Loader, | 
					
						
							|  |  |  |                   "The file you are attempting to install does not contain a section0 within the " | 
					
						
							|  |  |  |                   "metadata NCA and is therefore malformed. Verify that the file is valid."); | 
					
						
							|  |  |  |         return InstallResult::ErrorMetaFailed; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |     const auto section0 = (*meta_iter)->GetSubdirectories()[0]; | 
					
						
							| 
									
										
										
										
											2020-07-16 05:22:51 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (section0->GetFiles().empty()) { | 
					
						
							|  |  |  |         LOG_ERROR(Loader, | 
					
						
							|  |  |  |                   "The file you are attempting to install does not contain a CNMT within the " | 
					
						
							|  |  |  |                   "metadata NCA and is therefore malformed. Verify that the file is valid."); | 
					
						
							|  |  |  |         return InstallResult::ErrorMetaFailed; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |     const auto cnmt_file = section0->GetFiles()[0]; | 
					
						
							|  |  |  |     const CNMT cnmt(cnmt_file); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto title_id = cnmt.GetTitleID(); | 
					
						
							| 
									
										
										
										
											2021-05-16 00:24:06 -04:00
										 |  |  |     const auto version = cnmt.GetTitleVersion(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (title_id == GetBaseTitleID(title_id) && version == 0) { | 
					
						
							|  |  |  |         return InstallResult::ErrorBaseInstall; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |     const auto result = RemoveExistingEntry(title_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Install Metadata File
 | 
					
						
							| 
									
										
										
										
											2023-08-25 18:00:15 -04:00
										 |  |  |     const auto meta_result = RawInstallNCA(**meta_iter, copy, overwrite_if_exists, meta_id_data); | 
					
						
							|  |  |  |     if (meta_result != InstallResult::Success) { | 
					
						
							|  |  |  |         return meta_result; | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Install all the other NCAs
 | 
					
						
							|  |  |  |     for (const auto& record : cnmt.GetContentRecords()) { | 
					
						
							| 
									
										
										
										
											2019-07-01 06:37:22 +01:00
										 |  |  |         // Ignore DeltaFragments, they are not useful to us
 | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |         if (record.type == ContentRecordType::DeltaFragment) { | 
					
						
							| 
									
										
										
										
											2019-07-01 06:37:22 +01:00
										 |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-08-25 11:50:04 -04:00
										 |  |  |         const auto nca = GetNCAFromNSPForID(nsp, record.nca_id); | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |         if (nca == nullptr) { | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |             return InstallResult::ErrorCopyFailed; | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-08-25 18:00:15 -04:00
										 |  |  |         if (nca->GetStatus() == Loader::ResultStatus::ErrorMissingBKTRBaseRomFS && | 
					
						
							|  |  |  |             nca->GetTitleId() != title_id) { | 
					
						
							|  |  |  |             // Create fake cnmt for patch to multiprogram application
 | 
					
						
							|  |  |  |             const auto sub_nca_result = | 
					
						
							| 
									
										
										
										
											2023-09-11 23:50:36 -04:00
										 |  |  |                 InstallEntry(*nca, cnmt.GetHeader(), record, overwrite_if_exists, copy); | 
					
						
							| 
									
										
										
										
											2023-08-25 18:00:15 -04:00
										 |  |  |             if (sub_nca_result != InstallResult::Success) { | 
					
						
							|  |  |  |                 return sub_nca_result; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         const auto nca_result = RawInstallNCA(*nca, copy, overwrite_if_exists, record.nca_id); | 
					
						
							|  |  |  |         if (nca_result != InstallResult::Success) { | 
					
						
							|  |  |  |             return nca_result; | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Refresh(); | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |     if (result) { | 
					
						
							| 
									
										
										
										
											2020-07-12 12:28:18 -04:00
										 |  |  |         return InstallResult::OverwriteExisting; | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |     return InstallResult::Success; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-27 13:52:13 -05:00
										 |  |  | InstallResult RegisteredCache::InstallEntry(const NCA& nca, TitleType type, | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |                                             bool overwrite_if_exists, const VfsCopyFunction& copy) { | 
					
						
							| 
									
										
										
										
											2020-08-23 10:26:18 -04:00
										 |  |  |     const CNMTHeader header{ | 
					
						
							|  |  |  |         .title_id = nca.GetTitleId(), | 
					
						
							|  |  |  |         .title_version = 0, | 
					
						
							|  |  |  |         .type = type, | 
					
						
							|  |  |  |         .reserved = {}, | 
					
						
							|  |  |  |         .table_offset = 0x10, | 
					
						
							|  |  |  |         .number_content_entries = 1, | 
					
						
							|  |  |  |         .number_meta_entries = 0, | 
					
						
							|  |  |  |         .attributes = 0, | 
					
						
							|  |  |  |         .reserved2 = {}, | 
					
						
							|  |  |  |         .is_committed = 0, | 
					
						
							|  |  |  |         .required_download_system_version = 0, | 
					
						
							|  |  |  |         .reserved3 = {}, | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2020-08-23 10:26:18 -04:00
										 |  |  |     const OptionalHeader opt_header{0, 0}; | 
					
						
							| 
									
										
										
										
											2018-11-27 13:52:13 -05:00
										 |  |  |     ContentRecord c_rec{{}, {}, {}, GetCRTypeFromNCAType(nca.GetType()), {}}; | 
					
						
							|  |  |  |     const auto& data = nca.GetBaseFile()->ReadBytes(0x100000); | 
					
						
							| 
									
										
										
										
											2019-11-12 08:37:58 -05:00
										 |  |  |     mbedtls_sha256_ret(data.data(), data.size(), c_rec.hash.data(), 0); | 
					
						
							| 
									
										
										
										
											2020-08-23 10:26:18 -04:00
										 |  |  |     std::memcpy(&c_rec.nca_id, &c_rec.hash, 16); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     const CNMT new_cnmt(header, opt_header, {c_rec}, {}); | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |     if (!RawInstallYuzuMeta(new_cnmt)) { | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |         return InstallResult::ErrorMetaFailed; | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |     return RawInstallNCA(nca, copy, overwrite_if_exists, c_rec.nca_id); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-11 23:50:36 -04:00
										 |  |  | InstallResult RegisteredCache::InstallEntry(const NCA& nca, const CNMTHeader& base_header, | 
					
						
							|  |  |  |                                             const ContentRecord& base_record, | 
					
						
							|  |  |  |                                             bool overwrite_if_exists, const VfsCopyFunction& copy) { | 
					
						
							|  |  |  |     const CNMTHeader header{ | 
					
						
							|  |  |  |         .title_id = nca.GetTitleId(), | 
					
						
							|  |  |  |         .title_version = base_header.title_version, | 
					
						
							|  |  |  |         .type = base_header.type, | 
					
						
							|  |  |  |         .reserved = {}, | 
					
						
							|  |  |  |         .table_offset = 0x10, | 
					
						
							|  |  |  |         .number_content_entries = 1, | 
					
						
							|  |  |  |         .number_meta_entries = 0, | 
					
						
							|  |  |  |         .attributes = 0, | 
					
						
							|  |  |  |         .reserved2 = {}, | 
					
						
							|  |  |  |         .is_committed = 0, | 
					
						
							|  |  |  |         .required_download_system_version = 0, | 
					
						
							|  |  |  |         .reserved3 = {}, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     const OptionalHeader opt_header{0, 0}; | 
					
						
							|  |  |  |     const CNMT new_cnmt(header, opt_header, {base_record}, {}); | 
					
						
							|  |  |  |     if (!RawInstallYuzuMeta(new_cnmt)) { | 
					
						
							|  |  |  |         return InstallResult::ErrorMetaFailed; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return RawInstallNCA(nca, copy, overwrite_if_exists, base_record.nca_id); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-20 10:28:44 -04:00
										 |  |  | bool RegisteredCache::RemoveExistingEntry(u64 title_id) const { | 
					
						
							| 
									
										
										
										
											2023-08-25 18:00:15 -04:00
										 |  |  |     bool removed_data = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-20 10:28:44 -04:00
										 |  |  |     const auto delete_nca = [this](const NcaID& id) { | 
					
						
							|  |  |  |         const auto path = GetRelativePathFromNcaID(id, false, true, false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const bool isFile = dir->GetFileRelative(path) != nullptr; | 
					
						
							|  |  |  |         const bool isDir = dir->GetDirectoryRelative(path) != nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (isFile) { | 
					
						
							|  |  |  |             return dir->DeleteFile(path); | 
					
						
							|  |  |  |         } else if (isDir) { | 
					
						
							|  |  |  |             return dir->DeleteSubdirectoryRecursive(path); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If an entry exists in the registered cache, remove it
 | 
					
						
							|  |  |  |     if (HasEntry(title_id, ContentRecordType::Meta)) { | 
					
						
							|  |  |  |         LOG_INFO(Loader, | 
					
						
							|  |  |  |                  "Previously installed entry (v{}) for title_id={:016X} detected! " | 
					
						
							|  |  |  |                  "Attempting to remove...", | 
					
						
							|  |  |  |                  GetEntryVersion(title_id).value_or(0), title_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Get all the ncas associated with the current CNMT and delete them
 | 
					
						
							|  |  |  |         const auto meta_old_id = | 
					
						
							|  |  |  |             GetNcaIDFromMetadata(title_id, ContentRecordType::Meta).value_or(NcaID{}); | 
					
						
							|  |  |  |         const auto program_id = | 
					
						
							|  |  |  |             GetNcaIDFromMetadata(title_id, ContentRecordType::Program).value_or(NcaID{}); | 
					
						
							|  |  |  |         const auto data_id = | 
					
						
							|  |  |  |             GetNcaIDFromMetadata(title_id, ContentRecordType::Data).value_or(NcaID{}); | 
					
						
							|  |  |  |         const auto control_id = | 
					
						
							|  |  |  |             GetNcaIDFromMetadata(title_id, ContentRecordType::Control).value_or(NcaID{}); | 
					
						
							|  |  |  |         const auto html_id = | 
					
						
							|  |  |  |             GetNcaIDFromMetadata(title_id, ContentRecordType::HtmlDocument).value_or(NcaID{}); | 
					
						
							|  |  |  |         const auto legal_id = | 
					
						
							|  |  |  |             GetNcaIDFromMetadata(title_id, ContentRecordType::LegalInformation).value_or(NcaID{}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const auto deleted_meta = delete_nca(meta_old_id); | 
					
						
							|  |  |  |         const auto deleted_program = delete_nca(program_id); | 
					
						
							|  |  |  |         const auto deleted_data = delete_nca(data_id); | 
					
						
							|  |  |  |         const auto deleted_control = delete_nca(control_id); | 
					
						
							|  |  |  |         const auto deleted_html = delete_nca(html_id); | 
					
						
							|  |  |  |         const auto deleted_legal = delete_nca(legal_id); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-25 18:00:15 -04:00
										 |  |  |         removed_data |= (deleted_meta || deleted_program || deleted_data || deleted_control || | 
					
						
							|  |  |  |                          deleted_html || deleted_legal); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If patch entries for any program exist in yuzu meta, remove them
 | 
					
						
							|  |  |  |     for (u8 i = 0; i < 0x10; i++) { | 
					
						
							|  |  |  |         const auto meta_dir = dir->CreateDirectoryRelative("yuzu_meta"); | 
					
						
							|  |  |  |         const auto filename = GetCNMTName(TitleType::Update, title_id + i); | 
					
						
							| 
									
										
										
										
											2023-09-16 18:25:17 -04:00
										 |  |  |         if (meta_dir->GetFile(filename)) { | 
					
						
							|  |  |  |             removed_data |= meta_dir->DeleteFile(filename); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-07-20 10:28:44 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-25 18:00:15 -04:00
										 |  |  |     return removed_data; | 
					
						
							| 
									
										
										
										
											2020-07-20 10:28:44 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-27 13:52:13 -05:00
										 |  |  | InstallResult RegisteredCache::RawInstallNCA(const NCA& nca, const VfsCopyFunction& copy, | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |                                              bool overwrite_if_exists, | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |                                              std::optional<NcaID> override_id) { | 
					
						
							| 
									
										
										
										
											2018-11-27 13:52:13 -05:00
										 |  |  |     const auto in = nca.GetBaseFile(); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     Core::Crypto::SHA256Hash hash{}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Calculate NcaID
 | 
					
						
							|  |  |  |     // NOTE: Because computing the SHA256 of an entire NCA is quite expensive (especially if the
 | 
					
						
							|  |  |  |     // game is massive), we're going to cheat and only hash the first MB of the NCA.
 | 
					
						
							|  |  |  |     // Also, for XCIs the NcaID matters, so if the override id isn't none, use that.
 | 
					
						
							|  |  |  |     NcaID id{}; | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |     if (override_id) { | 
					
						
							|  |  |  |         id = *override_id; | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         const auto& data = in->ReadBytes(0x100000); | 
					
						
							| 
									
										
										
										
											2019-11-12 08:37:58 -05:00
										 |  |  |         mbedtls_sha256_ret(data.data(), data.size(), hash.data(), 0); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         memcpy(id.data(), hash.data(), 16); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 12:24:44 -04:00
										 |  |  |     std::string path = GetRelativePathFromNcaID(id, false, true, false); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |     if (GetFileAtID(id) != nullptr && !overwrite_if_exists) { | 
					
						
							| 
									
										
										
										
											2018-08-10 11:10:44 -04:00
										 |  |  |         LOG_WARNING(Loader, "Attempting to overwrite existing NCA. Skipping..."); | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |         return InstallResult::ErrorAlreadyExists; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (GetFileAtID(id) != nullptr) { | 
					
						
							|  |  |  |         LOG_WARNING(Loader, "Overwriting existing NCA..."); | 
					
						
							|  |  |  |         VirtualDir c_dir; | 
					
						
							|  |  |  |         { c_dir = dir->GetFileRelative(path)->GetContainingDirectory(); } | 
					
						
							| 
									
										
										
										
											2020-08-15 08:33:16 -04:00
										 |  |  |         c_dir->DeleteFile(Common::FS::GetFilename(path)); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto out = dir->CreateFileRelative(path); | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |     if (out == nullptr) { | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |         return InstallResult::ErrorCopyFailed; | 
					
						
							| 
									
										
										
										
											2020-07-05 09:37:50 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-23 21:50:16 -04:00
										 |  |  |     return copy(in, out, VFS_RC_LARGE_COPY_BLOCK) ? InstallResult::Success | 
					
						
							|  |  |  |                                                   : InstallResult::ErrorCopyFailed; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool RegisteredCache::RawInstallYuzuMeta(const CNMT& cnmt) { | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |     // Reasoning behind this method can be found in the comment for InstallEntry, NCA overload.
 | 
					
						
							| 
									
										
										
										
											2021-05-02 02:34:40 -04:00
										 |  |  |     const auto meta_dir = dir->CreateDirectoryRelative("yuzu_meta"); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     const auto filename = GetCNMTName(cnmt.GetType(), cnmt.GetTitleID()); | 
					
						
							| 
									
										
										
										
											2021-05-02 02:34:40 -04:00
										 |  |  |     if (meta_dir->GetFile(filename) == nullptr) { | 
					
						
							|  |  |  |         auto out = meta_dir->CreateFile(filename); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         const auto buffer = cnmt.Serialize(); | 
					
						
							|  |  |  |         out->Resize(buffer.size()); | 
					
						
							|  |  |  |         out->WriteBytes(buffer); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2021-05-02 02:34:40 -04:00
										 |  |  |         auto out = meta_dir->GetFile(filename); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |         CNMT old_cnmt(out); | 
					
						
							|  |  |  |         // Returns true on change
 | 
					
						
							|  |  |  |         if (old_cnmt.UnionRecords(cnmt)) { | 
					
						
							|  |  |  |             out->Resize(0); | 
					
						
							|  |  |  |             const auto buffer = old_cnmt.Serialize(); | 
					
						
							|  |  |  |             out->Resize(buffer.size()); | 
					
						
							|  |  |  |             out->WriteBytes(buffer); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     Refresh(); | 
					
						
							|  |  |  |     return std::find_if(yuzu_meta.begin(), yuzu_meta.end(), | 
					
						
							| 
									
										
										
										
											2018-08-09 21:33:13 -04:00
										 |  |  |                         [&cnmt](const std::pair<u64, CNMT>& kv) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |                             return kv.second.GetType() == cnmt.GetType() && | 
					
						
							|  |  |  |                                    kv.second.GetTitleID() == cnmt.GetTitleID(); | 
					
						
							|  |  |  |                         }) != yuzu_meta.end(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | ContentProviderUnion::~ContentProviderUnion() = default; | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | void ContentProviderUnion::SetSlot(ContentProviderUnionSlot slot, ContentProvider* provider) { | 
					
						
							|  |  |  |     providers[slot] = provider; | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | void ContentProviderUnion::ClearSlot(ContentProviderUnionSlot slot) { | 
					
						
							|  |  |  |     providers[slot] = nullptr; | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | void ContentProviderUnion::Refresh() { | 
					
						
							|  |  |  |     for (auto& provider : providers) { | 
					
						
							|  |  |  |         if (provider.second == nullptr) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         provider.second->Refresh(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | bool ContentProviderUnion::HasEntry(u64 title_id, ContentRecordType type) const { | 
					
						
							|  |  |  |     for (const auto& provider : providers) { | 
					
						
							|  |  |  |         if (provider.second == nullptr) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (provider.second->HasEntry(title_id, type)) | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::optional<u32> ContentProviderUnion::GetEntryVersion(u64 title_id) const { | 
					
						
							|  |  |  |     for (const auto& provider : providers) { | 
					
						
							|  |  |  |         if (provider.second == nullptr) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const auto res = provider.second->GetEntryVersion(title_id); | 
					
						
							|  |  |  |         if (res != std::nullopt) | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  |             return res; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     return std::nullopt; | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | VirtualFile ContentProviderUnion::GetEntryUnparsed(u64 title_id, ContentRecordType type) const { | 
					
						
							|  |  |  |     for (const auto& provider : providers) { | 
					
						
							|  |  |  |         if (provider.second == nullptr) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const auto res = provider.second->GetEntryUnparsed(title_id, type); | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  |         if (res != nullptr) | 
					
						
							|  |  |  |             return res; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | VirtualFile ContentProviderUnion::GetEntryRaw(u64 title_id, ContentRecordType type) const { | 
					
						
							|  |  |  |     for (const auto& provider : providers) { | 
					
						
							|  |  |  |         if (provider.second == nullptr) | 
					
						
							|  |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |         const auto res = provider.second->GetEntryRaw(title_id, type); | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  |         if (res != nullptr) | 
					
						
							|  |  |  |             return res; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | std::unique_ptr<NCA> ContentProviderUnion::GetEntry(u64 title_id, ContentRecordType type) const { | 
					
						
							|  |  |  |     for (const auto& provider : providers) { | 
					
						
							|  |  |  |         if (provider.second == nullptr) | 
					
						
							|  |  |  |             continue; | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |         auto res = provider.second->GetEntry(title_id, type); | 
					
						
							|  |  |  |         if (res != nullptr) | 
					
						
							|  |  |  |             return res; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | std::vector<ContentProviderEntry> ContentProviderUnion::ListEntriesFilter( | 
					
						
							|  |  |  |     std::optional<TitleType> title_type, std::optional<ContentRecordType> record_type, | 
					
						
							|  |  |  |     std::optional<u64> title_id) const { | 
					
						
							|  |  |  |     std::vector<ContentProviderEntry> out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const auto& provider : providers) { | 
					
						
							|  |  |  |         if (provider.second == nullptr) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const auto vec = provider.second->ListEntriesFilter(title_type, record_type, title_id); | 
					
						
							|  |  |  |         std::copy(vec.begin(), vec.end(), std::back_inserter(out)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::sort(out.begin(), out.end()); | 
					
						
							|  |  |  |     out.erase(std::unique(out.begin(), out.end()), out.end()); | 
					
						
							|  |  |  |     return out; | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | std::vector<std::pair<ContentProviderUnionSlot, ContentProviderEntry>> | 
					
						
							|  |  |  | ContentProviderUnion::ListEntriesFilterOrigin(std::optional<ContentProviderUnionSlot> origin, | 
					
						
							|  |  |  |                                               std::optional<TitleType> title_type, | 
					
						
							|  |  |  |                                               std::optional<ContentRecordType> record_type, | 
					
						
							|  |  |  |                                               std::optional<u64> title_id) const { | 
					
						
							|  |  |  |     std::vector<std::pair<ContentProviderUnionSlot, ContentProviderEntry>> out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const auto& provider : providers) { | 
					
						
							|  |  |  |         if (provider.second == nullptr) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (origin.has_value() && *origin != provider.first) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const auto vec = provider.second->ListEntriesFilter(title_type, record_type, title_id); | 
					
						
							|  |  |  |         std::transform(vec.begin(), vec.end(), std::back_inserter(out), | 
					
						
							|  |  |  |                        [&provider](const ContentProviderEntry& entry) { | 
					
						
							|  |  |  |                            return std::make_pair(provider.first, entry); | 
					
						
							|  |  |  |                        }); | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-10-17 14:04:18 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     std::sort(out.begin(), out.end()); | 
					
						
							|  |  |  |     out.erase(std::unique(out.begin(), out.end()), out.end()); | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  |     return out; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:10:17 -04:00
										 |  |  | std::optional<ContentProviderUnionSlot> ContentProviderUnion::GetSlotForEntry( | 
					
						
							|  |  |  |     u64 title_id, ContentRecordType type) const { | 
					
						
							| 
									
										
										
										
											2019-06-25 22:25:10 -04:00
										 |  |  |     const auto iter = | 
					
						
							|  |  |  |         std::find_if(providers.begin(), providers.end(), [title_id, type](const auto& provider) { | 
					
						
							|  |  |  |             return provider.second != nullptr && provider.second->HasEntry(title_id, type); | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2019-06-24 19:10:17 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-25 22:25:10 -04:00
										 |  |  |     if (iter == providers.end()) { | 
					
						
							|  |  |  |         return std::nullopt; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:10:17 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-25 22:25:10 -04:00
										 |  |  |     return iter->first; | 
					
						
							| 
									
										
										
										
											2019-06-24 19:10:17 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | ManualContentProvider::~ManualContentProvider() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ManualContentProvider::AddEntry(TitleType title_type, ContentRecordType content_type, | 
					
						
							|  |  |  |                                      u64 title_id, VirtualFile file) { | 
					
						
							|  |  |  |     entries.insert_or_assign({title_type, content_type, title_id}, file); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ManualContentProvider::ClearAllEntries() { | 
					
						
							|  |  |  |     entries.clear(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ManualContentProvider::Refresh() {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool ManualContentProvider::HasEntry(u64 title_id, ContentRecordType type) const { | 
					
						
							|  |  |  |     return GetEntryRaw(title_id, type) != nullptr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::optional<u32> ManualContentProvider::GetEntryVersion(u64 title_id) const { | 
					
						
							|  |  |  |     return std::nullopt; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | VirtualFile ManualContentProvider::GetEntryUnparsed(u64 title_id, ContentRecordType type) const { | 
					
						
							|  |  |  |     return GetEntryRaw(title_id, type); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | VirtualFile ManualContentProvider::GetEntryRaw(u64 title_id, ContentRecordType type) const { | 
					
						
							|  |  |  |     const auto iter = | 
					
						
							|  |  |  |         std::find_if(entries.begin(), entries.end(), [title_id, type](const auto& entry) { | 
					
						
							| 
									
										
										
										
											2020-04-15 15:59:23 -04:00
										 |  |  |             const auto content_type = std::get<1>(entry.first); | 
					
						
							|  |  |  |             const auto e_title_id = std::get<2>(entry.first); | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |             return content_type == type && e_title_id == title_id; | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |     if (iter == entries.end()) | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  |     return iter->second; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::unique_ptr<NCA> ManualContentProvider::GetEntry(u64 title_id, ContentRecordType type) const { | 
					
						
							|  |  |  |     const auto res = GetEntryRaw(title_id, type); | 
					
						
							|  |  |  |     if (res == nullptr) | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							| 
									
										
										
										
											2023-08-10 21:34:43 -04:00
										 |  |  |     return std::make_unique<NCA>(res); | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::vector<ContentProviderEntry> ManualContentProvider::ListEntriesFilter( | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |     std::optional<TitleType> title_type, std::optional<ContentRecordType> record_type, | 
					
						
							|  |  |  |     std::optional<u64> title_id) const { | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     std::vector<ContentProviderEntry> out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const auto& entry : entries) { | 
					
						
							|  |  |  |         const auto [e_title_type, e_content_type, e_title_id] = entry.first; | 
					
						
							|  |  |  |         if ((title_type == std::nullopt || e_title_type == *title_type) && | 
					
						
							|  |  |  |             (record_type == std::nullopt || e_content_type == *record_type) && | 
					
						
							|  |  |  |             (title_id == std::nullopt || e_title_id == *title_id)) { | 
					
						
							|  |  |  |             out.emplace_back(ContentProviderEntry{e_title_id, e_content_type}); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-10-17 14:04:18 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     std::sort(out.begin(), out.end()); | 
					
						
							|  |  |  |     out.erase(std::unique(out.begin(), out.end()), out.end()); | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  |     return out; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | } // namespace FileSys
 |