forked from eden-emu/eden
		
	file_sys: Create ContentProvider interface and default implementations
This commit is contained in:
		
							parent
							
								
									595511876e
								
							
						
					
					
						commit
						a6c7ae6fe8
					
				
					 2 changed files with 277 additions and 150 deletions
				
			
		|  | @ -23,19 +23,19 @@ namespace FileSys { | ||||||
| // The size of blocks to use when vfs raw copying into nand.
 | // The size of blocks to use when vfs raw copying into nand.
 | ||||||
| constexpr size_t VFS_RC_LARGE_COPY_BLOCK = 0x400000; | constexpr size_t VFS_RC_LARGE_COPY_BLOCK = 0x400000; | ||||||
| 
 | 
 | ||||||
| std::string RegisteredCacheEntry::DebugInfo() const { | std::string ContentProviderEntry::DebugInfo() const { | ||||||
|     return fmt::format("title_id={:016X}, content_type={:02X}", title_id, static_cast<u8>(type)); |     return fmt::format("title_id={:016X}, content_type={:02X}", title_id, static_cast<u8>(type)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool operator<(const RegisteredCacheEntry& lhs, const RegisteredCacheEntry& rhs) { | bool operator<(const ContentProviderEntry& lhs, const ContentProviderEntry& rhs) { | ||||||
|     return (lhs.title_id < rhs.title_id) || (lhs.title_id == rhs.title_id && lhs.type < rhs.type); |     return (lhs.title_id < rhs.title_id) || (lhs.title_id == rhs.title_id && lhs.type < rhs.type); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool operator==(const RegisteredCacheEntry& lhs, const RegisteredCacheEntry& rhs) { | bool operator==(const ContentProviderEntry& lhs, const ContentProviderEntry& rhs) { | ||||||
|     return std::tie(lhs.title_id, lhs.type) == std::tie(rhs.title_id, rhs.type); |     return std::tie(lhs.title_id, lhs.type) == std::tie(rhs.title_id, rhs.type); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool operator!=(const RegisteredCacheEntry& lhs, const RegisteredCacheEntry& rhs) { | bool operator!=(const ContentProviderEntry& lhs, const ContentProviderEntry& rhs) { | ||||||
|     return !operator==(lhs, rhs); |     return !operator==(lhs, rhs); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -84,7 +84,7 @@ static std::string GetCNMTName(TitleType type, u64 title_id) { | ||||||
|     return fmt::format("{}_{:016x}.cnmt", TITLE_TYPE_NAMES[index], title_id); |     return fmt::format("{}_{:016x}.cnmt", TITLE_TYPE_NAMES[index], title_id); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static ContentRecordType GetCRTypeFromNCAType(NCAContentType type) { | ContentRecordType GetCRTypeFromNCAType(NCAContentType type) { | ||||||
|     switch (type) { |     switch (type) { | ||||||
|     case NCAContentType::Program: |     case NCAContentType::Program: | ||||||
|         // TODO(DarkLordZach): Differentiate between Program and Patch
 |         // TODO(DarkLordZach): Differentiate between Program and Patch
 | ||||||
|  | @ -104,6 +104,28 @@ static ContentRecordType GetCRTypeFromNCAType(NCAContentType type) { | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | 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); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| VirtualFile RegisteredCache::OpenFileOrDirectoryConcat(const VirtualDir& dir, | VirtualFile RegisteredCache::OpenFileOrDirectoryConcat(const VirtualDir& dir, | ||||||
|                                                        std::string_view path) const { |                                                        std::string_view path) const { | ||||||
|     const auto file = dir->GetFileRelative(path); |     const auto file = dir->GetFileRelative(path); | ||||||
|  | @ -161,8 +183,8 @@ VirtualFile RegisteredCache::GetFileAtID(NcaID id) const { | ||||||
|     return file; |     return file; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static std::optional<NcaID> CheckMapForContentRecord( | static std::optional<NcaID> CheckMapForContentRecord(const std::map<u64, CNMT>& map, u64 title_id, | ||||||
|     const boost::container::flat_map<u64, CNMT>& map, u64 title_id, ContentRecordType type) { |                                                      ContentRecordType type) { | ||||||
|     if (map.find(title_id) == map.end()) |     if (map.find(title_id) == map.end()) | ||||||
|         return {}; |         return {}; | ||||||
| 
 | 
 | ||||||
|  | @ -268,7 +290,7 @@ void RegisteredCache::Refresh() { | ||||||
|     AccumulateYuzuMeta(); |     AccumulateYuzuMeta(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| RegisteredCache::RegisteredCache(VirtualDir dir_, RegisteredCacheParsingFunction parsing_function) | RegisteredCache::RegisteredCache(VirtualDir dir_, ContentProviderParsingFunction parsing_function) | ||||||
|     : dir(std::move(dir_)), parser(std::move(parsing_function)) { |     : dir(std::move(dir_)), parser(std::move(parsing_function)) { | ||||||
|     Refresh(); |     Refresh(); | ||||||
| } | } | ||||||
|  | @ -279,19 +301,11 @@ bool RegisteredCache::HasEntry(u64 title_id, ContentRecordType type) const { | ||||||
|     return GetEntryRaw(title_id, type) != nullptr; |     return GetEntryRaw(title_id, type) != nullptr; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool RegisteredCache::HasEntry(RegisteredCacheEntry entry) const { |  | ||||||
|     return GetEntryRaw(entry) != nullptr; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| VirtualFile RegisteredCache::GetEntryUnparsed(u64 title_id, ContentRecordType type) const { | VirtualFile RegisteredCache::GetEntryUnparsed(u64 title_id, ContentRecordType type) const { | ||||||
|     const auto id = GetNcaIDFromMetadata(title_id, type); |     const auto id = GetNcaIDFromMetadata(title_id, type); | ||||||
|     return id ? GetFileAtID(*id) : nullptr; |     return id ? GetFileAtID(*id) : nullptr; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| VirtualFile RegisteredCache::GetEntryUnparsed(RegisteredCacheEntry entry) const { |  | ||||||
|     return GetEntryUnparsed(entry.title_id, entry.type); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| std::optional<u32> RegisteredCache::GetEntryVersion(u64 title_id) const { | std::optional<u32> RegisteredCache::GetEntryVersion(u64 title_id) const { | ||||||
|     const auto meta_iter = meta.find(title_id); |     const auto meta_iter = meta.find(title_id); | ||||||
|     if (meta_iter != meta.end()) |     if (meta_iter != meta.end()) | ||||||
|  | @ -309,10 +323,6 @@ VirtualFile RegisteredCache::GetEntryRaw(u64 title_id, ContentRecordType type) c | ||||||
|     return id ? parser(GetFileAtID(*id), *id) : nullptr; |     return id ? parser(GetFileAtID(*id), *id) : nullptr; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| VirtualFile RegisteredCache::GetEntryRaw(RegisteredCacheEntry entry) const { |  | ||||||
|     return GetEntryRaw(entry.title_id, entry.type); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| std::unique_ptr<NCA> RegisteredCache::GetEntry(u64 title_id, ContentRecordType type) const { | std::unique_ptr<NCA> RegisteredCache::GetEntry(u64 title_id, ContentRecordType type) const { | ||||||
|     const auto raw = GetEntryRaw(title_id, type); |     const auto raw = GetEntryRaw(title_id, type); | ||||||
|     if (raw == nullptr) |     if (raw == nullptr) | ||||||
|  | @ -320,10 +330,6 @@ std::unique_ptr<NCA> RegisteredCache::GetEntry(u64 title_id, ContentRecordType t | ||||||
|     return std::make_unique<NCA>(raw, nullptr, 0, keys); |     return std::make_unique<NCA>(raw, nullptr, 0, keys); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::unique_ptr<NCA> RegisteredCache::GetEntry(RegisteredCacheEntry entry) const { |  | ||||||
|     return GetEntry(entry.title_id, entry.type); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| template <typename T> | template <typename T> | ||||||
| void RegisteredCache::IterateAllMetadata( | void RegisteredCache::IterateAllMetadata( | ||||||
|     std::vector<T>& out, std::function<T(const CNMT&, const ContentRecord&)> proc, |     std::vector<T>& out, std::function<T(const CNMT&, const ContentRecord&)> proc, | ||||||
|  | @ -348,25 +354,14 @@ void RegisteredCache::IterateAllMetadata( | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::vector<RegisteredCacheEntry> RegisteredCache::ListEntries() const { | std::vector<ContentProviderEntry> RegisteredCache::ListEntriesFilter( | ||||||
|     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( |  | ||||||
|     std::optional<TitleType> title_type, std::optional<ContentRecordType> record_type, |     std::optional<TitleType> title_type, std::optional<ContentRecordType> record_type, | ||||||
|     std::optional<u64> title_id) const { |     std::optional<u64> title_id) const { | ||||||
|     std::vector<RegisteredCacheEntry> out; |     std::vector<ContentProviderEntry> out; | ||||||
|     IterateAllMetadata<RegisteredCacheEntry>( |     IterateAllMetadata<ContentProviderEntry>( | ||||||
|         out, |         out, | ||||||
|         [](const CNMT& c, const ContentRecord& r) { |         [](const CNMT& c, const ContentRecord& r) { | ||||||
|             return RegisteredCacheEntry{c.GetTitleID(), r.type}; |             return ContentProviderEntry{c.GetTitleID(), r.type}; | ||||||
|         }, |         }, | ||||||
|         [&title_type, &record_type, &title_id](const CNMT& c, const ContentRecord& r) { |         [&title_type, &record_type, &title_id](const CNMT& c, const ContentRecord& r) { | ||||||
|             if (title_type && *title_type != c.GetType()) |             if (title_type && *title_type != c.GetType()) | ||||||
|  | @ -521,37 +516,56 @@ bool RegisteredCache::RawInstallYuzuMeta(const CNMT& cnmt) { | ||||||
|                         }) != yuzu_meta.end(); |                         }) != yuzu_meta.end(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| RegisteredCacheUnion::RegisteredCacheUnion(std::vector<RegisteredCache*> caches) | ContentProviderUnion::~ContentProviderUnion() = default; | ||||||
|     : caches(std::move(caches)) {} |  | ||||||
| 
 | 
 | ||||||
| void RegisteredCacheUnion::Refresh() { | void ContentProviderUnion::SetSlot(ContentProviderUnionSlot slot, ContentProvider* provider) { | ||||||
|     for (const auto& c : caches) |     providers[slot] = provider; | ||||||
|         c->Refresh(); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool RegisteredCacheUnion::HasEntry(u64 title_id, ContentRecordType type) const { | void ContentProviderUnion::ClearSlot(ContentProviderUnionSlot slot) { | ||||||
|     return std::any_of(caches.begin(), caches.end(), [title_id, type](const auto& cache) { |     providers[slot] = nullptr; | ||||||
|         return cache->HasEntry(title_id, type); |  | ||||||
|     }); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool RegisteredCacheUnion::HasEntry(RegisteredCacheEntry entry) const { | void ContentProviderUnion::Refresh() { | ||||||
|     return HasEntry(entry.title_id, entry.type); |     for (auto& provider : providers) { | ||||||
|  |         if (provider.second == nullptr) | ||||||
|  |             continue; | ||||||
|  | 
 | ||||||
|  |         provider.second->Refresh(); | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::optional<u32> RegisteredCacheUnion::GetEntryVersion(u64 title_id) const { | bool ContentProviderUnion::HasEntry(u64 title_id, ContentRecordType type) const { | ||||||
|     for (const auto& c : caches) { |     for (const auto& provider : providers) { | ||||||
|         const auto res = c->GetEntryVersion(title_id); |         if (provider.second == nullptr) | ||||||
|         if (res) |             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) | ||||||
|             return res; |             return res; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return {}; |     return std::nullopt; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| VirtualFile RegisteredCacheUnion::GetEntryUnparsed(u64 title_id, ContentRecordType type) const { | VirtualFile ContentProviderUnion::GetEntryUnparsed(u64 title_id, ContentRecordType type) const { | ||||||
|     for (const auto& c : caches) { |     for (const auto& provider : providers) { | ||||||
|         const auto res = c->GetEntryUnparsed(title_id, type); |         if (provider.second == nullptr) | ||||||
|  |             continue; | ||||||
|  | 
 | ||||||
|  |         const auto res = provider.second->GetEntryUnparsed(title_id, type); | ||||||
|         if (res != nullptr) |         if (res != nullptr) | ||||||
|             return res; |             return res; | ||||||
|     } |     } | ||||||
|  | @ -559,13 +573,12 @@ VirtualFile RegisteredCacheUnion::GetEntryUnparsed(u64 title_id, ContentRecordTy | ||||||
|     return nullptr; |     return nullptr; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| VirtualFile RegisteredCacheUnion::GetEntryUnparsed(RegisteredCacheEntry entry) const { | VirtualFile ContentProviderUnion::GetEntryRaw(u64 title_id, ContentRecordType type) const { | ||||||
|     return GetEntryUnparsed(entry.title_id, entry.type); |     for (const auto& provider : providers) { | ||||||
| } |         if (provider.second == nullptr) | ||||||
|  |             continue; | ||||||
| 
 | 
 | ||||||
| VirtualFile RegisteredCacheUnion::GetEntryRaw(u64 title_id, ContentRecordType type) const { |         const auto res = provider.second->GetEntryRaw(title_id, type); | ||||||
|     for (const auto& c : caches) { |  | ||||||
|         const auto res = c->GetEntryRaw(title_id, type); |  | ||||||
|         if (res != nullptr) |         if (res != nullptr) | ||||||
|             return res; |             return res; | ||||||
|     } |     } | ||||||
|  | @ -573,30 +586,30 @@ VirtualFile RegisteredCacheUnion::GetEntryRaw(u64 title_id, ContentRecordType ty | ||||||
|     return nullptr; |     return nullptr; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| VirtualFile RegisteredCacheUnion::GetEntryRaw(RegisteredCacheEntry entry) const { | std::unique_ptr<NCA> ContentProviderUnion::GetEntry(u64 title_id, ContentRecordType type) const { | ||||||
|     return GetEntryRaw(entry.title_id, entry.type); |     for (const auto& provider : providers) { | ||||||
| } |         if (provider.second == nullptr) | ||||||
|  |             continue; | ||||||
|  | 
 | ||||||
|  |         auto res = provider.second->GetEntry(title_id, type); | ||||||
|  |         if (res != nullptr) | ||||||
|  |             return res; | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
| std::unique_ptr<NCA> RegisteredCacheUnion::GetEntry(u64 title_id, ContentRecordType type) const { |  | ||||||
|     const auto raw = GetEntryRaw(title_id, type); |  | ||||||
|     if (raw == nullptr) |  | ||||||
|     return nullptr; |     return nullptr; | ||||||
|     return std::make_unique<NCA>(raw); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::unique_ptr<NCA> RegisteredCacheUnion::GetEntry(RegisteredCacheEntry entry) const { | std::vector<ContentProviderEntry> ContentProviderUnion::ListEntriesFilter( | ||||||
|     return GetEntry(entry.title_id, entry.type); |     std::optional<TitleType> title_type, std::optional<ContentRecordType> record_type, | ||||||
| } |     std::optional<u64> title_id) const { | ||||||
|  |     std::vector<ContentProviderEntry> out; | ||||||
| 
 | 
 | ||||||
| std::vector<RegisteredCacheEntry> RegisteredCacheUnion::ListEntries() const { |     for (const auto& provider : providers) { | ||||||
|     std::vector<RegisteredCacheEntry> out; |         if (provider.second == nullptr) | ||||||
|     for (const auto& c : caches) { |             continue; | ||||||
|         c->IterateAllMetadata<RegisteredCacheEntry>( | 
 | ||||||
|             out, |         const auto vec = provider.second->ListEntriesFilter(title_type, record_type, title_id); | ||||||
|             [](const CNMT& c, const ContentRecord& r) { |         std::copy(vec.begin(), vec.end(), std::back_inserter(out)); | ||||||
|                 return RegisteredCacheEntry{c.GetTitleID(), r.type}; |  | ||||||
|             }, |  | ||||||
|             [](const CNMT& c, const ContentRecord& r) { return true; }); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     std::sort(out.begin(), out.end()); |     std::sort(out.begin(), out.end()); | ||||||
|  | @ -604,25 +617,87 @@ std::vector<RegisteredCacheEntry> RegisteredCacheUnion::ListEntries() const { | ||||||
|     return out; |     return out; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::vector<RegisteredCacheEntry> RegisteredCacheUnion::ListEntriesFilter( | 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); | ||||||
|  |                        }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     std::sort(out.begin(), out.end()); | ||||||
|  |     out.erase(std::unique(out.begin(), out.end()), out.end()); | ||||||
|  |     return out; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 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) { | ||||||
|  |             const auto [title_type, content_type, e_title_id] = entry.first; | ||||||
|  |             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; | ||||||
|  |     return std::make_unique<NCA>(res, nullptr, 0, keys); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | std::vector<ContentProviderEntry> ManualContentProvider::ListEntriesFilter( | ||||||
|     std::optional<TitleType> title_type, std::optional<ContentRecordType> record_type, |     std::optional<TitleType> title_type, std::optional<ContentRecordType> record_type, | ||||||
|     std::optional<u64> title_id) const { |     std::optional<u64> title_id) const { | ||||||
|     std::vector<RegisteredCacheEntry> out; |     std::vector<ContentProviderEntry> out; | ||||||
|     for (const auto& c : caches) { | 
 | ||||||
|         c->IterateAllMetadata<RegisteredCacheEntry>( |     for (const auto& entry : entries) { | ||||||
|             out, |         const auto [e_title_type, e_content_type, e_title_id] = entry.first; | ||||||
|             [](const CNMT& c, const ContentRecord& r) { |         if ((title_type == std::nullopt || e_title_type == *title_type) && | ||||||
|                 return RegisteredCacheEntry{c.GetTitleID(), r.type}; |             (record_type == std::nullopt || e_content_type == *record_type) && | ||||||
|             }, |             (title_id == std::nullopt || e_title_id == *title_id)) { | ||||||
|             [&title_type, &record_type, &title_id](const CNMT& c, const ContentRecord& r) { |             out.emplace_back(ContentProviderEntry{e_title_id, e_content_type}); | ||||||
|                 if (title_type && *title_type != c.GetType()) |         } | ||||||
|                     return false; |  | ||||||
|                 if (record_type && *record_type != r.type) |  | ||||||
|                     return false; |  | ||||||
|                 if (title_id && *title_id != c.GetTitleID()) |  | ||||||
|                     return false; |  | ||||||
|                 return true; |  | ||||||
|             }); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     std::sort(out.begin(), out.end()); |     std::sort(out.begin(), out.end()); | ||||||
|  |  | ||||||
|  | @ -21,12 +21,13 @@ class NSP; | ||||||
| class XCI; | class XCI; | ||||||
| 
 | 
 | ||||||
| enum class ContentRecordType : u8; | enum class ContentRecordType : u8; | ||||||
|  | enum class NCAContentType : u8; | ||||||
| enum class TitleType : u8; | enum class TitleType : u8; | ||||||
| 
 | 
 | ||||||
| struct ContentRecord; | struct ContentRecord; | ||||||
| 
 | 
 | ||||||
| using NcaID = std::array<u8, 0x10>; | using NcaID = std::array<u8, 0x10>; | ||||||
| using RegisteredCacheParsingFunction = std::function<VirtualFile(const VirtualFile&, const NcaID&)>; | using ContentProviderParsingFunction = std::function<VirtualFile(const VirtualFile&, const NcaID&)>; | ||||||
| using VfsCopyFunction = std::function<bool(const VirtualFile&, const VirtualFile&, size_t)>; | using VfsCopyFunction = std::function<bool(const VirtualFile&, const VirtualFile&, size_t)>; | ||||||
| 
 | 
 | ||||||
| enum class InstallResult { | enum class InstallResult { | ||||||
|  | @ -36,7 +37,7 @@ enum class InstallResult { | ||||||
|     ErrorMetaFailed, |     ErrorMetaFailed, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| struct RegisteredCacheEntry { | struct ContentProviderEntry { | ||||||
|     u64 title_id; |     u64 title_id; | ||||||
|     ContentRecordType type; |     ContentRecordType type; | ||||||
| 
 | 
 | ||||||
|  | @ -47,12 +48,46 @@ constexpr u64 GetUpdateTitleID(u64 base_title_id) { | ||||||
|     return base_title_id | 0x800; |     return base_title_id | 0x800; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | ContentRecordType GetCRTypeFromNCAType(NCAContentType type); | ||||||
|  | 
 | ||||||
| // boost flat_map requires operator< for O(log(n)) lookups.
 | // boost flat_map requires operator< for O(log(n)) lookups.
 | ||||||
| bool operator<(const RegisteredCacheEntry& lhs, const RegisteredCacheEntry& rhs); | bool operator<(const ContentProviderEntry& lhs, const ContentProviderEntry& rhs); | ||||||
| 
 | 
 | ||||||
| // std unique requires operator== to identify duplicates.
 | // std unique requires operator== to identify duplicates.
 | ||||||
| bool operator==(const RegisteredCacheEntry& lhs, const RegisteredCacheEntry& rhs); | bool operator==(const ContentProviderEntry& lhs, const ContentProviderEntry& rhs); | ||||||
| bool operator!=(const RegisteredCacheEntry& lhs, const RegisteredCacheEntry& rhs); | bool operator!=(const ContentProviderEntry& lhs, const ContentProviderEntry& rhs); | ||||||
|  | 
 | ||||||
|  | class ContentProvider { | ||||||
|  | public: | ||||||
|  |     virtual ~ContentProvider(); | ||||||
|  | 
 | ||||||
|  |     virtual void Refresh() = 0; | ||||||
|  | 
 | ||||||
|  |     virtual bool HasEntry(u64 title_id, ContentRecordType type) const = 0; | ||||||
|  |     virtual bool HasEntry(ContentProviderEntry entry) const; | ||||||
|  | 
 | ||||||
|  |     virtual std::optional<u32> GetEntryVersion(u64 title_id) const = 0; | ||||||
|  | 
 | ||||||
|  |     virtual VirtualFile GetEntryUnparsed(u64 title_id, ContentRecordType type) const = 0; | ||||||
|  |     virtual VirtualFile GetEntryUnparsed(ContentProviderEntry entry) const; | ||||||
|  | 
 | ||||||
|  |     virtual VirtualFile GetEntryRaw(u64 title_id, ContentRecordType type) const = 0; | ||||||
|  |     virtual VirtualFile GetEntryRaw(ContentProviderEntry entry) const; | ||||||
|  | 
 | ||||||
|  |     virtual std::unique_ptr<NCA> GetEntry(u64 title_id, ContentRecordType type) const = 0; | ||||||
|  |     virtual std::unique_ptr<NCA> GetEntry(ContentProviderEntry entry) const; | ||||||
|  | 
 | ||||||
|  |     virtual std::vector<ContentProviderEntry> ListEntries() const; | ||||||
|  | 
 | ||||||
|  |     // If a parameter is not std::nullopt, it will be filtered for from all entries.
 | ||||||
|  |     virtual std::vector<ContentProviderEntry> ListEntriesFilter( | ||||||
|  |         std::optional<TitleType> title_type = {}, std::optional<ContentRecordType> record_type = {}, | ||||||
|  |         std::optional<u64> title_id = {}) const = 0; | ||||||
|  | 
 | ||||||
|  | protected: | ||||||
|  |     // A single instance of KeyManager to be used by GetEntry()
 | ||||||
|  |     Core::Crypto::KeyManager keys; | ||||||
|  | }; | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * A class that catalogues NCAs in the registered directory structure. |  * A class that catalogues NCAs in the registered directory structure. | ||||||
|  | @ -67,39 +102,32 @@ bool operator!=(const RegisteredCacheEntry& lhs, const RegisteredCacheEntry& rhs | ||||||
|  * (This impl also supports substituting the nca dir for an nca file, as that's more convenient |  * (This impl also supports substituting the nca dir for an nca file, as that's more convenient | ||||||
|  * when 4GB splitting can be ignored.) |  * when 4GB splitting can be ignored.) | ||||||
|  */ |  */ | ||||||
| class RegisteredCache { | class RegisteredCache : public ContentProvider { | ||||||
|     friend class RegisteredCacheUnion; |  | ||||||
| 
 |  | ||||||
| public: | public: | ||||||
|     // Parsing function defines the conversion from raw file to NCA. If there are other steps
 |     // 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
 |     // besides creating the NCA from the file (e.g. NAX0 on SD Card), that should go in a custom
 | ||||||
|     // parsing function.
 |     // parsing function.
 | ||||||
|     explicit RegisteredCache(VirtualDir dir, |     explicit RegisteredCache(VirtualDir dir, | ||||||
|                              RegisteredCacheParsingFunction parsing_function = |                              ContentProviderParsingFunction parsing_function = | ||||||
|                                  [](const VirtualFile& file, const NcaID& id) { return file; }); |                                  [](const VirtualFile& file, const NcaID& id) { return file; }); | ||||||
|     ~RegisteredCache(); |     ~RegisteredCache() override; | ||||||
| 
 | 
 | ||||||
|     void Refresh(); |     void Refresh() override; | ||||||
| 
 | 
 | ||||||
|     bool HasEntry(u64 title_id, ContentRecordType type) const; |     bool HasEntry(u64 title_id, ContentRecordType type) const override; | ||||||
|     bool HasEntry(RegisteredCacheEntry entry) const; |  | ||||||
| 
 | 
 | ||||||
|     std::optional<u32> GetEntryVersion(u64 title_id) const; |     std::optional<u32> GetEntryVersion(u64 title_id) const override; | ||||||
| 
 | 
 | ||||||
|     VirtualFile GetEntryUnparsed(u64 title_id, ContentRecordType type) const; |     VirtualFile GetEntryUnparsed(u64 title_id, ContentRecordType type) const override; | ||||||
|     VirtualFile GetEntryUnparsed(RegisteredCacheEntry entry) const; |  | ||||||
| 
 | 
 | ||||||
|     VirtualFile GetEntryRaw(u64 title_id, ContentRecordType type) const; |     VirtualFile GetEntryRaw(u64 title_id, ContentRecordType type) const override; | ||||||
|     VirtualFile GetEntryRaw(RegisteredCacheEntry entry) const; |  | ||||||
| 
 | 
 | ||||||
|     std::unique_ptr<NCA> GetEntry(u64 title_id, ContentRecordType type) const; |     std::unique_ptr<NCA> GetEntry(u64 title_id, ContentRecordType type) const override; | ||||||
|     std::unique_ptr<NCA> GetEntry(RegisteredCacheEntry entry) const; |  | ||||||
| 
 | 
 | ||||||
|     std::vector<RegisteredCacheEntry> ListEntries() const; |  | ||||||
|     // If a parameter is not std::nullopt, it will be filtered for from all entries.
 |     // If a parameter is not std::nullopt, it will be filtered for from all entries.
 | ||||||
|     std::vector<RegisteredCacheEntry> ListEntriesFilter( |     std::vector<ContentProviderEntry> ListEntriesFilter( | ||||||
|         std::optional<TitleType> title_type = {}, std::optional<ContentRecordType> record_type = {}, |         std::optional<TitleType> title_type = {}, std::optional<ContentRecordType> record_type = {}, | ||||||
|         std::optional<u64> title_id = {}) const; |         std::optional<u64> title_id = {}) const override; | ||||||
| 
 | 
 | ||||||
|     // Raw copies all the ncas from the xci/nsp to the csache. Does some quick checks to make sure
 |     // Raw copies all the ncas from the xci/nsp to the csache. Does some quick checks to make sure
 | ||||||
|     // there is a meta NCA and all of them are accessible.
 |     // there is a meta NCA and all of them are accessible.
 | ||||||
|  | @ -131,46 +159,70 @@ private: | ||||||
|     bool RawInstallYuzuMeta(const CNMT& cnmt); |     bool RawInstallYuzuMeta(const CNMT& cnmt); | ||||||
| 
 | 
 | ||||||
|     VirtualDir dir; |     VirtualDir dir; | ||||||
|     RegisteredCacheParsingFunction parser; |     ContentProviderParsingFunction parser; | ||||||
|     Core::Crypto::KeyManager keys; |  | ||||||
| 
 | 
 | ||||||
|     // maps tid -> NcaID of meta
 |     // maps tid -> NcaID of meta
 | ||||||
|     boost::container::flat_map<u64, NcaID> meta_id; |     std::map<u64, NcaID> meta_id; | ||||||
|     // maps tid -> meta
 |     // maps tid -> meta
 | ||||||
|     boost::container::flat_map<u64, CNMT> meta; |     std::map<u64, CNMT> meta; | ||||||
|     // maps tid -> meta for CNMT in yuzu_meta
 |     // maps tid -> meta for CNMT in yuzu_meta
 | ||||||
|     boost::container::flat_map<u64, CNMT> yuzu_meta; |     std::map<u64, CNMT> yuzu_meta; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| // Combines multiple RegisteredCaches (i.e. SysNAND, UserNAND, SDMC) into one interface.
 | enum class ContentProviderUnionSlot { | ||||||
| class RegisteredCacheUnion { |     SysNAND,        ///< System NAND
 | ||||||
|  |     UserNAND,       ///< User NAND
 | ||||||
|  |     SDMC,           ///< SD Card
 | ||||||
|  |     FrontendManual, ///< Frontend-defined game list or similar
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | // Combines multiple ContentProvider(s) (i.e. SysNAND, UserNAND, SDMC) into one interface.
 | ||||||
|  | class ContentProviderUnion : public ContentProvider { | ||||||
| public: | public: | ||||||
|     explicit RegisteredCacheUnion(std::vector<RegisteredCache*> caches); |     ~ContentProviderUnion() override; | ||||||
| 
 | 
 | ||||||
|     void Refresh(); |     void SetSlot(ContentProviderUnionSlot slot, ContentProvider* provider); | ||||||
|  |     void ClearSlot(ContentProviderUnionSlot slot); | ||||||
| 
 | 
 | ||||||
|     bool HasEntry(u64 title_id, ContentRecordType type) const; |     void Refresh() override; | ||||||
|     bool HasEntry(RegisteredCacheEntry entry) const; |     bool HasEntry(u64 title_id, ContentRecordType type) const override; | ||||||
|  |     std::optional<u32> GetEntryVersion(u64 title_id) const override; | ||||||
|  |     VirtualFile GetEntryUnparsed(u64 title_id, ContentRecordType type) const override; | ||||||
|  |     VirtualFile GetEntryRaw(u64 title_id, ContentRecordType type) const override; | ||||||
|  |     std::unique_ptr<NCA> GetEntry(u64 title_id, ContentRecordType type) const override; | ||||||
|  |     std::vector<ContentProviderEntry> ListEntriesFilter( | ||||||
|  |         std::optional<TitleType> title_type, std::optional<ContentRecordType> record_type, | ||||||
|  |         std::optional<u64> title_id) const override; | ||||||
| 
 | 
 | ||||||
|     std::optional<u32> GetEntryVersion(u64 title_id) const; |     std::vector<std::pair<ContentProviderUnionSlot, ContentProviderEntry>> ListEntriesFilterOrigin( | ||||||
| 
 |         std::optional<ContentProviderUnionSlot> origin = {}, | ||||||
|     VirtualFile GetEntryUnparsed(u64 title_id, ContentRecordType type) const; |  | ||||||
|     VirtualFile GetEntryUnparsed(RegisteredCacheEntry entry) const; |  | ||||||
| 
 |  | ||||||
|     VirtualFile GetEntryRaw(u64 title_id, ContentRecordType type) const; |  | ||||||
|     VirtualFile GetEntryRaw(RegisteredCacheEntry entry) const; |  | ||||||
| 
 |  | ||||||
|     std::unique_ptr<NCA> GetEntry(u64 title_id, ContentRecordType type) const; |  | ||||||
|     std::unique_ptr<NCA> GetEntry(RegisteredCacheEntry entry) const; |  | ||||||
| 
 |  | ||||||
|     std::vector<RegisteredCacheEntry> ListEntries() const; |  | ||||||
|     // If a parameter is not std::nullopt, it will be filtered for from all entries.
 |  | ||||||
|     std::vector<RegisteredCacheEntry> ListEntriesFilter( |  | ||||||
|         std::optional<TitleType> title_type = {}, std::optional<ContentRecordType> record_type = {}, |         std::optional<TitleType> title_type = {}, std::optional<ContentRecordType> record_type = {}, | ||||||
|         std::optional<u64> title_id = {}) const; |         std::optional<u64> title_id = {}) const; | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|     std::vector<RegisteredCache*> caches; |     std::map<ContentProviderUnionSlot, ContentProvider*> providers; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | class ManualContentProvider : public ContentProvider { | ||||||
|  | public: | ||||||
|  |     ~ManualContentProvider() override; | ||||||
|  | 
 | ||||||
|  |     void AddEntry(TitleType title_type, ContentRecordType content_type, u64 title_id, | ||||||
|  |                   VirtualFile file); | ||||||
|  |     void ClearAllEntries(); | ||||||
|  | 
 | ||||||
|  |     void Refresh() override; | ||||||
|  |     bool HasEntry(u64 title_id, ContentRecordType type) const override; | ||||||
|  |     std::optional<u32> GetEntryVersion(u64 title_id) const override; | ||||||
|  |     VirtualFile GetEntryUnparsed(u64 title_id, ContentRecordType type) const override; | ||||||
|  |     VirtualFile GetEntryRaw(u64 title_id, ContentRecordType type) const override; | ||||||
|  |     std::unique_ptr<NCA> GetEntry(u64 title_id, ContentRecordType type) const override; | ||||||
|  |     std::vector<ContentProviderEntry> ListEntriesFilter( | ||||||
|  |         std::optional<TitleType> title_type, std::optional<ContentRecordType> record_type, | ||||||
|  |         std::optional<u64> title_id) const override; | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  |     std::map<std::tuple<TitleType, ContentRecordType, u64>, VirtualFile> entries; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| } // namespace FileSys
 | } // namespace FileSys
 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Zach Hilman
						Zach Hilman