file_sys: Add RegisteredCache
Manages NAND NCA get and install.
This commit is contained in:
		
							parent
							
								
									7fdfa63ce3
								
							
						
					
					
						commit
						2d3617c723
					
				
					 2 changed files with 543 additions and 0 deletions
				
			
		
							
								
								
									
										435
									
								
								src/core/file_sys/registered_cache.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										435
									
								
								src/core/file_sys/registered_cache.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,435 @@ | ||||||
|  | // Copyright 2018 yuzu emulator team
 | ||||||
|  | // Licensed under GPLv2 or any later version
 | ||||||
|  | // Refer to the license.txt file included.
 | ||||||
|  | 
 | ||||||
|  | #include <regex> | ||||||
|  | #include <mbedtls/sha256.h> | ||||||
|  | #include "common/assert.h" | ||||||
|  | #include "common/hex_util.h" | ||||||
|  | #include "common/logging/log.h" | ||||||
|  | #include "core/crypto/encryption_layer.h" | ||||||
|  | #include "core/file_sys/card_image.h" | ||||||
|  | #include "core/file_sys/nca_metadata.h" | ||||||
|  | #include "core/file_sys/registered_cache.h" | ||||||
|  | #include "core/file_sys/vfs_concat.h" | ||||||
|  | 
 | ||||||
|  | namespace FileSys { | ||||||
|  | std::string RegisteredCacheEntry::DebugInfo() const { | ||||||
|  |     return fmt::format("title_id={:016X}, content_type={:02X}", title_id, static_cast<u8>(type)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool operator<(const RegisteredCacheEntry& lhs, const RegisteredCacheEntry& rhs) { | ||||||
|  |     return (lhs.title_id < rhs.title_id) || (lhs.title_id == rhs.title_id && lhs.type < rhs.type); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static bool FollowsTwoDigitDirFormat(std::string_view name) { | ||||||
|  |     const static std::regex two_digit_regex( | ||||||
|  |         "000000[0123456789abcdefABCDEF][0123456789abcdefABCDEF]"); | ||||||
|  |     return std::regex_match(name.begin(), name.end(), two_digit_regex); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static bool FollowsNcaIdFormat(std::string_view name) { | ||||||
|  |     const static std::regex nca_id_regex("[0123456789abcdefABCDEF]+.nca"); | ||||||
|  |     return name.size() == 36 && std::regex_match(name.begin(), name.end(), nca_id_regex); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static std::string GetRelativePathFromNcaID(const std::array<u8, 16>& nca_id, bool second_hex_upper, | ||||||
|  |                                             bool within_two_digit) { | ||||||
|  |     if (!within_two_digit) | ||||||
|  |         return fmt::format("/{}.nca", HexArrayToString(nca_id, second_hex_upper)); | ||||||
|  | 
 | ||||||
|  |     Core::Crypto::SHA256Hash hash{}; | ||||||
|  |     mbedtls_sha256(nca_id.data(), nca_id.size(), hash.data(), 0); | ||||||
|  |     return fmt::format("/000000{:02X}/{}.nca", hash[0], HexArrayToString(nca_id, second_hex_upper)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static std::string GetCNMTName(TitleType type, u64 title_id) { | ||||||
|  |     constexpr std::array<const char*, 9> TITLE_TYPE_NAMES{ | ||||||
|  |         "SystemProgram", | ||||||
|  |         "SystemData", | ||||||
|  |         "SystemUpdate", | ||||||
|  |         "BootImagePackage", | ||||||
|  |         "BootImagePackageSafe", | ||||||
|  |         "Application", | ||||||
|  |         "Patch", | ||||||
|  |         "AddOnContent", | ||||||
|  |         "" ///< Currently unknown 'DeltaTitle'
 | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     size_t index = static_cast<size_t>(type); | ||||||
|  |     if (index >= 0x80) | ||||||
|  |         index -= 0x80; | ||||||
|  |     return fmt::format("{}_{:016x}.cnmt", TITLE_TYPE_NAMES[index], title_id); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static ContentRecordType GetCRTypeFromNCAType(NCAContentType type) { | ||||||
|  |     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: | ||||||
|  |         return ContentRecordType::Data; | ||||||
|  |     case NCAContentType::Manual: | ||||||
|  |         // TODO(DarkLordZach): Peek at NCA contents to differentiate Manual and Legal.
 | ||||||
|  |         return ContentRecordType::Manual; | ||||||
|  |     default: | ||||||
|  |         UNREACHABLE(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | VirtualFile RegisteredCache::OpenFileOrDirectoryConcat(const VirtualDir& dir, | ||||||
|  |                                                        std::string_view path) const { | ||||||
|  |     if (dir->GetFileRelative(path) != nullptr) | ||||||
|  |         return dir->GetFileRelative(path); | ||||||
|  |     if (dir->GetDirectoryRelative(path) != nullptr) { | ||||||
|  |         const auto nca_dir = dir->GetDirectoryRelative(path); | ||||||
|  |         VirtualFile file = nullptr; | ||||||
|  | 
 | ||||||
|  |         const auto files = nca_dir->GetFiles(); | ||||||
|  |         if (files.size() == 1 && files[0]->GetName() == "00") | ||||||
|  |             file = files[0]; | ||||||
|  |         else { | ||||||
|  |             std::vector<VirtualFile> concat; | ||||||
|  |             for (u8 i = 0; i < 0x10; ++i) { | ||||||
|  |                 auto next = nca_dir->GetFile(fmt::format("{:02X}", i)); | ||||||
|  |                 if (next != nullptr) | ||||||
|  |                     concat.push_back(std::move(next)); | ||||||
|  |                 else { | ||||||
|  |                     next = nca_dir->GetFile(fmt::format("{:02x}", i)); | ||||||
|  |                     if (next != nullptr) | ||||||
|  |                         concat.push_back(std::move(next)); | ||||||
|  |                     else | ||||||
|  |                         break; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (concat.empty()) | ||||||
|  |                 return nullptr; | ||||||
|  | 
 | ||||||
|  |             file = FileSys::ConcatenateFiles(concat); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return file; | ||||||
|  |     } | ||||||
|  |     return nullptr; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | VirtualFile RegisteredCache::GetFileAtID(NcaID id) const { | ||||||
|  |     VirtualFile file; | ||||||
|  |     for (u8 i = 0; i < 4; ++i) { | ||||||
|  |         file = OpenFileOrDirectoryConcat( | ||||||
|  |             dir, GetRelativePathFromNcaID(id, (i & 0b10) == 0, (i & 0b01) == 0)); | ||||||
|  |         if (file != nullptr) | ||||||
|  |             return file; | ||||||
|  |     } | ||||||
|  |     return file; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | boost::optional<NcaID> RegisteredCache::GetNcaIDFromMetadata(u64 title_id, | ||||||
|  |                                                              ContentRecordType type) const { | ||||||
|  |     if (type == ContentRecordType::Meta && meta_id.find(title_id) != meta_id.end()) | ||||||
|  |         return meta_id.at(title_id); | ||||||
|  |     if (meta.find(title_id) == meta.end()) | ||||||
|  |         return boost::none; | ||||||
|  | 
 | ||||||
|  |     const auto& cnmt = meta.at(title_id); | ||||||
|  | 
 | ||||||
|  |     const auto iter = std::find_if(cnmt.GetContentRecords().begin(), cnmt.GetContentRecords().end(), | ||||||
|  |                                    [type](const ContentRecord& rec) { return rec.type == type; }); | ||||||
|  |     if (iter == cnmt.GetContentRecords().end()) | ||||||
|  |         return boost::none; | ||||||
|  | 
 | ||||||
|  |     return boost::make_optional(iter->nca_id); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void RegisteredCache::AccumulateFiles(std::vector<NcaID>& ids) const { | ||||||
|  |     for (const auto& d2_dir : dir->GetSubdirectories()) { | ||||||
|  |         if (FollowsNcaIdFormat(d2_dir->GetName())) { | ||||||
|  |             ids.push_back(HexStringToArray<0x10, true>(d2_dir->GetName().substr(0, 0x20))); | ||||||
|  |             continue; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (!FollowsTwoDigitDirFormat(d2_dir->GetName())) | ||||||
|  |             continue; | ||||||
|  | 
 | ||||||
|  |         for (const auto& nca_dir : d2_dir->GetSubdirectories()) { | ||||||
|  |             if (!FollowsNcaIdFormat(nca_dir->GetName())) | ||||||
|  |                 continue; | ||||||
|  | 
 | ||||||
|  |             ids.push_back(HexStringToArray<0x10, true>(nca_dir->GetName().substr(0, 0x20))); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         for (const auto& nca_file : d2_dir->GetFiles()) { | ||||||
|  |             if (!FollowsNcaIdFormat(nca_file->GetName())) | ||||||
|  |                 continue; | ||||||
|  | 
 | ||||||
|  |             ids.push_back(HexStringToArray<0x10, true>(nca_file->GetName().substr(0, 0x20))); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     for (const auto& d2_file : dir->GetFiles()) { | ||||||
|  |         if (FollowsNcaIdFormat(d2_file->GetName())) | ||||||
|  |             ids.push_back(HexStringToArray<0x10, true>(d2_file->GetName().substr(0, 0x20))); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void RegisteredCache::ProcessFiles(const std::vector<NcaID>& ids) { | ||||||
|  |     for (const auto& id : ids) { | ||||||
|  |         const auto file = GetFileAtID(id); | ||||||
|  | 
 | ||||||
|  |         if (file == nullptr) | ||||||
|  |             continue; | ||||||
|  |         const auto nca = std::make_shared<NCA>(parser(file, id)); | ||||||
|  |         if (nca->GetStatus() != Loader::ResultStatus::Success || | ||||||
|  |             nca->GetType() != NCAContentType::Meta) | ||||||
|  |             continue; | ||||||
|  | 
 | ||||||
|  |         const auto section0 = nca->GetSubdirectories()[0]; | ||||||
|  | 
 | ||||||
|  |         for (const auto& file : section0->GetFiles()) { | ||||||
|  |             if (file->GetExtension() != "cnmt") | ||||||
|  |                 continue; | ||||||
|  | 
 | ||||||
|  |             meta.insert_or_assign(nca->GetTitleId(), CNMT(file)); | ||||||
|  |             meta_id.insert_or_assign(nca->GetTitleId(), id); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void RegisteredCache::AccumulateYuzuMeta() { | ||||||
|  |     const auto dir = this->dir->GetSubdirectory("yuzu_meta"); | ||||||
|  |     if (dir == nullptr) | ||||||
|  |         return; | ||||||
|  | 
 | ||||||
|  |     for (const auto& file : dir->GetFiles()) { | ||||||
|  |         if (file->GetExtension() != "cnmt") | ||||||
|  |             continue; | ||||||
|  | 
 | ||||||
|  |         CNMT cnmt(file); | ||||||
|  |         yuzu_meta.insert_or_assign(cnmt.GetTitleID(), std::move(cnmt)); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void RegisteredCache::Refresh() { | ||||||
|  |     if (dir == nullptr) | ||||||
|  |         return; | ||||||
|  |     std::vector<NcaID> ids; | ||||||
|  |     AccumulateFiles(ids); | ||||||
|  |     ProcessFiles(ids); | ||||||
|  |     AccumulateYuzuMeta(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | RegisteredCache::RegisteredCache(VirtualDir dir_, RegisteredCacheParsingFunction parsing_function) | ||||||
|  |     : dir(std::move(dir_)), parser(std::move(parsing_function)) { | ||||||
|  |     Refresh(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool RegisteredCache::HasEntry(u64 title_id, ContentRecordType type) const { | ||||||
|  |     return GetEntryRaw(title_id, type) != nullptr; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool RegisteredCache::HasEntry(RegisteredCacheEntry entry) const { | ||||||
|  |     return GetEntryRaw(entry) != nullptr; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | VirtualFile RegisteredCache::GetEntryRaw(u64 title_id, ContentRecordType type) const { | ||||||
|  |     const auto id = GetNcaIDFromMetadata(title_id, type); | ||||||
|  |     if (id == boost::none) | ||||||
|  |         return nullptr; | ||||||
|  | 
 | ||||||
|  |     return parser(GetFileAtID(id.get()), id.get()); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | VirtualFile RegisteredCache::GetEntryRaw(RegisteredCacheEntry entry) const { | ||||||
|  |     return GetEntryRaw(entry.title_id, entry.type); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | std::shared_ptr<NCA> RegisteredCache::GetEntry(u64 title_id, ContentRecordType type) const { | ||||||
|  |     const auto raw = GetEntryRaw(title_id, type); | ||||||
|  |     if (raw == nullptr) | ||||||
|  |         return nullptr; | ||||||
|  |     return std::make_shared<NCA>(raw); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | std::shared_ptr<NCA> RegisteredCache::GetEntry(RegisteredCacheEntry entry) const { | ||||||
|  |     return GetEntry(entry.title_id, entry.type); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 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)); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | std::vector<RegisteredCacheEntry> RegisteredCache::ListEntries() const { | ||||||
|  |     std::vector<RegisteredCacheEntry> out; | ||||||
|  |     IterateAllMetadata<RegisteredCacheEntry>( | ||||||
|  |         out, | ||||||
|  |         [](const CNMT& c, const ContentRecord& r) { | ||||||
|  |             return RegisteredCacheEntry{c.GetTitleID(), r.type}; | ||||||
|  |         }, | ||||||
|  |         [](const CNMT& c, const ContentRecord& r) { return true; }); | ||||||
|  |     return out; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | std::vector<RegisteredCacheEntry> RegisteredCache::ListEntriesFilter( | ||||||
|  |     boost::optional<TitleType> title_type, boost::optional<ContentRecordType> record_type, | ||||||
|  |     boost::optional<u64> title_id) const { | ||||||
|  |     std::vector<RegisteredCacheEntry> out; | ||||||
|  |     IterateAllMetadata<RegisteredCacheEntry>( | ||||||
|  |         out, | ||||||
|  |         [](const CNMT& c, const ContentRecord& r) { | ||||||
|  |             return RegisteredCacheEntry{c.GetTitleID(), r.type}; | ||||||
|  |         }, | ||||||
|  |         [&title_type, &record_type, &title_id](const CNMT& c, const ContentRecord& r) { | ||||||
|  |             if (title_type != boost::none && title_type.get() != c.GetType()) | ||||||
|  |                 return false; | ||||||
|  |             if (record_type != boost::none && record_type.get() != r.type) | ||||||
|  |                 return false; | ||||||
|  |             if (title_id != boost::none && title_id.get() != c.GetTitleID()) | ||||||
|  |                 return false; | ||||||
|  |             return true; | ||||||
|  |         }); | ||||||
|  |     return out; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static std::shared_ptr<NCA> GetNCAFromXCIForID(std::shared_ptr<XCI> xci, const NcaID& id) { | ||||||
|  |     const auto filename = fmt::format("{}.nca", HexArrayToString(id, false)); | ||||||
|  |     const auto iter = | ||||||
|  |         std::find_if(xci->GetNCAs().begin(), xci->GetNCAs().end(), | ||||||
|  |                      [&filename](std::shared_ptr<NCA> nca) { return nca->GetName() == filename; }); | ||||||
|  |     return iter == xci->GetNCAs().end() ? nullptr : *iter; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool RegisteredCache::InstallEntry(std::shared_ptr<XCI> xci) { | ||||||
|  |     const auto& ncas = xci->GetNCAs(); | ||||||
|  |     const auto& meta_iter = std::find_if(ncas.begin(), ncas.end(), [](std::shared_ptr<NCA> nca) { | ||||||
|  |         return nca->GetType() == NCAContentType::Meta; | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     if (meta_iter == ncas.end()) { | ||||||
|  |         LOG_ERROR(Loader, "The XCI you are attempting to install does not have a metadata NCA and " | ||||||
|  |                           "is therefore malformed. Double check your encryption keys."); | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Install Metadata File
 | ||||||
|  |     const auto meta_id_raw = (*meta_iter)->GetName().substr(0, 32); | ||||||
|  |     const auto meta_id = HexStringToArray<16>(meta_id_raw); | ||||||
|  |     if (!RawInstallNCA(*meta_iter, meta_id)) | ||||||
|  |         return false; | ||||||
|  | 
 | ||||||
|  |     // Install all the other NCAs
 | ||||||
|  |     const auto section0 = (*meta_iter)->GetSubdirectories()[0]; | ||||||
|  |     const auto cnmt_file = section0->GetFiles()[0]; | ||||||
|  |     const CNMT cnmt(cnmt_file); | ||||||
|  |     for (const auto& record : cnmt.GetContentRecords()) { | ||||||
|  |         const auto nca = GetNCAFromXCIForID(xci, record.nca_id); | ||||||
|  |         if (nca == nullptr || !RawInstallNCA(nca, record.nca_id)) | ||||||
|  |             return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     Refresh(); | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool RegisteredCache::InstallEntry(std::shared_ptr<NCA> nca, TitleType type) { | ||||||
|  |     CNMTHeader header{ | ||||||
|  |         nca->GetTitleId(), ///< Title ID
 | ||||||
|  |         0,                 ///< Ignore/Default title version
 | ||||||
|  |         type,              ///< Type
 | ||||||
|  |         {},                ///< Padding
 | ||||||
|  |         0x10,              ///< Default table offset
 | ||||||
|  |         1,                 ///< 1 Content Entry
 | ||||||
|  |         0,                 ///< No Meta Entries
 | ||||||
|  |         {},                ///< Padding
 | ||||||
|  |     }; | ||||||
|  |     OptionalHeader opt_header{0, 0}; | ||||||
|  |     ContentRecord c_rec{{}, {}, {}, GetCRTypeFromNCAType(nca->GetType()), {}}; | ||||||
|  |     const auto& data = nca->GetBaseFile()->ReadBytes(0x100000); | ||||||
|  |     mbedtls_sha256(data.data(), data.size(), c_rec.hash.data(), 0); | ||||||
|  |     memcpy(&c_rec.nca_id, &c_rec.hash, 16); | ||||||
|  |     const CNMT new_cnmt(header, opt_header, {c_rec}, {}); | ||||||
|  |     return RawInstallYuzuMeta(new_cnmt) && RawInstallNCA(nca, c_rec.nca_id); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool RegisteredCache::RawInstallNCA(std::shared_ptr<NCA> nca, boost::optional<NcaID> override_id) { | ||||||
|  |     const auto in = nca->GetBaseFile(); | ||||||
|  |     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{}; | ||||||
|  |     if (override_id == boost::none) { | ||||||
|  |         const auto& data = in->ReadBytes(0x100000); | ||||||
|  |         mbedtls_sha256(data.data(), data.size(), hash.data(), 0); | ||||||
|  |         memcpy(id.data(), hash.data(), 16); | ||||||
|  |     } else { | ||||||
|  |         id = override_id.get(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     std::string path = GetRelativePathFromNcaID(id, false, true); | ||||||
|  | 
 | ||||||
|  |     if (GetFileAtID(id) != nullptr) { | ||||||
|  |         LOG_WARNING(Loader, "OW Attempt"); | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     auto out = dir->CreateFileRelative(path); | ||||||
|  |     if (out == nullptr) | ||||||
|  |         return false; | ||||||
|  |     return VfsRawCopy(in, out); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool RegisteredCache::RawInstallYuzuMeta(const CNMT& cnmt) { | ||||||
|  |     const auto dir = this->dir->CreateDirectoryRelative("yuzu_meta"); | ||||||
|  |     const auto filename = GetCNMTName(cnmt.GetType(), cnmt.GetTitleID()); | ||||||
|  |     if (dir->GetFile(filename) == nullptr) { | ||||||
|  |         auto out = dir->CreateFile(filename); | ||||||
|  |         const auto buffer = cnmt.Serialize(); | ||||||
|  |         out->Resize(buffer.size()); | ||||||
|  |         out->WriteBytes(buffer); | ||||||
|  |     } else { | ||||||
|  |         auto out = dir->GetFile(filename); | ||||||
|  |         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(), | ||||||
|  |                         [&cnmt](const std::pair<const u64, CNMT>& kv) { | ||||||
|  |                             return kv.second.GetType() == cnmt.GetType() && | ||||||
|  |                                    kv.second.GetTitleID() == cnmt.GetTitleID(); | ||||||
|  |                         }) != yuzu_meta.end(); | ||||||
|  | } | ||||||
|  | } // namespace FileSys
 | ||||||
							
								
								
									
										108
									
								
								src/core/file_sys/registered_cache.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										108
									
								
								src/core/file_sys/registered_cache.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,108 @@ | ||||||
|  | // Copyright 2018 yuzu emulator team
 | ||||||
|  | // Licensed under GPLv2 or any later version
 | ||||||
|  | // Refer to the license.txt file included.
 | ||||||
|  | 
 | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include <array> | ||||||
|  | #include <map> | ||||||
|  | #include <memory> | ||||||
|  | #include <string> | ||||||
|  | #include <boost/container/flat_map.hpp> | ||||||
|  | #include "common/common_funcs.h" | ||||||
|  | #include "content_archive.h" | ||||||
|  | #include "core/file_sys/vfs.h" | ||||||
|  | #include "nca_metadata.h" | ||||||
|  | 
 | ||||||
|  | namespace FileSys { | ||||||
|  | class XCI; | ||||||
|  | class CNMT; | ||||||
|  | 
 | ||||||
|  | using NcaID = std::array<u8, 0x10>; | ||||||
|  | using RegisteredCacheParsingFunction = std::function<VirtualFile(const VirtualFile&, const NcaID&)>; | ||||||
|  | 
 | ||||||
|  | struct RegisteredCacheEntry { | ||||||
|  |     u64 title_id; | ||||||
|  |     ContentRecordType type; | ||||||
|  | 
 | ||||||
|  |     std::string DebugInfo() const; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // boost flat_map requires operator< for O(log(n)) lookups.
 | ||||||
|  | bool operator<(const RegisteredCacheEntry& lhs, const RegisteredCacheEntry& rhs); | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * A class that catalogues NCAs in the registered directory structure. | ||||||
|  |  * Nintendo's registered format follows this structure: | ||||||
|  |  * | ||||||
|  |  * Root | ||||||
|  |  *   | 000000XX <- XX is the ____ two digits of the NcaID | ||||||
|  |  *       | <hash>.nca <- hash is the NcaID (first half of SHA256 over entire file) (folder) | ||||||
|  |  *         | 00 | ||||||
|  |  *         | 01 <- Actual content split along 4GB boundaries. (optional) | ||||||
|  |  * | ||||||
|  |  * (This impl also supports substituting the nca dir for an nca file, as that's more convenient when | ||||||
|  |  * 4GB splitting can be ignored.) | ||||||
|  |  */ | ||||||
|  | class RegisteredCache { | ||||||
|  | public: | ||||||
|  |     // Parsing function defines the conversion from raw file to NCA. If there are other steps
 | ||||||
|  |     // besides creating the NCA from the file (e.g. NAX0 on SD Card), that should go in a custom
 | ||||||
|  |     // parsing function.
 | ||||||
|  |     RegisteredCache(VirtualDir dir, | ||||||
|  |                     RegisteredCacheParsingFunction parsing_function = | ||||||
|  |                         [](const VirtualFile& file, const NcaID& id) { return file; }); | ||||||
|  | 
 | ||||||
|  |     void Refresh(); | ||||||
|  | 
 | ||||||
|  |     bool HasEntry(u64 title_id, ContentRecordType type) const; | ||||||
|  |     bool HasEntry(RegisteredCacheEntry entry) const; | ||||||
|  | 
 | ||||||
|  |     VirtualFile GetEntryRaw(u64 title_id, ContentRecordType type) const; | ||||||
|  |     VirtualFile GetEntryRaw(RegisteredCacheEntry entry) const; | ||||||
|  | 
 | ||||||
|  |     std::shared_ptr<NCA> GetEntry(u64 title_id, ContentRecordType type) const; | ||||||
|  |     std::shared_ptr<NCA> GetEntry(RegisteredCacheEntry entry) const; | ||||||
|  | 
 | ||||||
|  |     std::vector<RegisteredCacheEntry> ListEntries() const; | ||||||
|  |     // If a parameter is not boost::none, it will be filtered for from all entries.
 | ||||||
|  |     std::vector<RegisteredCacheEntry> ListEntriesFilter( | ||||||
|  |         boost::optional<TitleType> title_type = boost::none, | ||||||
|  |         boost::optional<ContentRecordType> record_type = boost::none, | ||||||
|  |         boost::optional<u64> title_id = boost::none) const; | ||||||
|  | 
 | ||||||
|  |     // Raw copies all the ncas from the xci to the csache. Does some quick checks to make sure there
 | ||||||
|  |     // is a meta NCA and all of them are accessible.
 | ||||||
|  |     bool InstallEntry(std::shared_ptr<XCI> xci); | ||||||
|  | 
 | ||||||
|  |     // Due to the fact that we must use Meta-type NCAs to determine the existance of files, this
 | ||||||
|  |     // poses quite a challenge. Instead of creating a new meta NCA for this file, yuzu will create a
 | ||||||
|  |     // dir inside the NAND called 'yuzu_meta' and store the raw CNMT there.
 | ||||||
|  |     // TODO(DarkLordZach): Author real meta-type NCAs and install those.
 | ||||||
|  |     bool InstallEntry(std::shared_ptr<NCA> nca, TitleType type); | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  |     template <typename T> | ||||||
|  |     void IterateAllMetadata(std::vector<T>& out, | ||||||
|  |                             std::function<T(const CNMT&, const ContentRecord&)> proc, | ||||||
|  |                             std::function<bool(const CNMT&, const ContentRecord&)> filter) const; | ||||||
|  |     void AccumulateFiles(std::vector<NcaID>& ids) const; | ||||||
|  |     void ProcessFiles(const std::vector<NcaID>& ids); | ||||||
|  |     void AccumulateYuzuMeta(); | ||||||
|  |     boost::optional<NcaID> GetNcaIDFromMetadata(u64 title_id, ContentRecordType type) const; | ||||||
|  |     VirtualFile GetFileAtID(NcaID id) const; | ||||||
|  |     VirtualFile OpenFileOrDirectoryConcat(const VirtualDir& dir, std::string_view path) const; | ||||||
|  |     bool RawInstallNCA(std::shared_ptr<NCA> nca, boost::optional<NcaID> override_id = boost::none); | ||||||
|  |     bool RawInstallYuzuMeta(const CNMT& cnmt); | ||||||
|  | 
 | ||||||
|  |     VirtualDir dir; | ||||||
|  |     RegisteredCacheParsingFunction parser; | ||||||
|  |     // maps tid -> NcaID of meta
 | ||||||
|  |     boost::container::flat_map<u64, NcaID> meta_id; | ||||||
|  |     // maps tid -> meta
 | ||||||
|  |     boost::container::flat_map<u64, CNMT> meta; | ||||||
|  |     // maps tid -> meta for CNMT in yuzu_meta
 | ||||||
|  |     boost::container::flat_map<u64, CNMT> yuzu_meta; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | } // namespace FileSys
 | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Zach Hilman
						Zach Hilman