| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | // Copyright 2018 yuzu emulator team
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <array>
 | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  | #include <functional>
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | #include <memory>
 | 
					
						
							|  |  |  | #include <string>
 | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | #include <boost/container/flat_map.hpp>
 | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2018-11-01 20:23:38 -04:00
										 |  |  | #include "core/crypto/key_manager.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | #include "core/file_sys/vfs.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace FileSys { | 
					
						
							|  |  |  | class CNMT; | 
					
						
							| 
									
										
										
										
											2018-09-03 21:58:19 -04:00
										 |  |  | class NCA; | 
					
						
							| 
									
										
										
										
											2018-09-04 14:44:40 -04:00
										 |  |  | class NSP; | 
					
						
							| 
									
										
										
										
											2018-09-03 21:58:19 -04:00
										 |  |  | class XCI; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum class ContentRecordType : u8; | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | enum class NCAContentType : u8; | 
					
						
							| 
									
										
										
										
											2018-09-03 21:58:19 -04:00
										 |  |  | enum class TitleType : u8; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct ContentRecord; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | using NcaID = std::array<u8, 0x10>; | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | using ContentProviderParsingFunction = std::function<VirtualFile(const VirtualFile&, const NcaID&)>; | 
					
						
							| 
									
										
										
										
											2018-09-19 22:09:23 -04:00
										 |  |  | using VfsCopyFunction = std::function<bool(const VirtualFile&, const VirtualFile&, size_t)>; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  | enum class InstallResult { | 
					
						
							|  |  |  |     Success, | 
					
						
							|  |  |  |     ErrorAlreadyExists, | 
					
						
							|  |  |  |     ErrorCopyFailed, | 
					
						
							|  |  |  |     ErrorMetaFailed, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | struct ContentProviderEntry { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     u64 title_id; | 
					
						
							|  |  |  |     ContentRecordType type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::string DebugInfo() const; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-26 10:53:31 -04:00
										 |  |  | constexpr u64 GetUpdateTitleID(u64 base_title_id) { | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  |     return base_title_id | 0x800; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | ContentRecordType GetCRTypeFromNCAType(NCAContentType type); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | // boost flat_map requires operator< for O(log(n)) lookups.
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | bool operator<(const ContentProviderEntry& lhs, const ContentProviderEntry& rhs); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-17 14:04:18 -04:00
										 |  |  | // std unique requires operator== to identify duplicates.
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | bool operator==(const ContentProviderEntry& lhs, const ContentProviderEntry& 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; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2018-10-17 14:04:18 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * 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) | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2018-10-17 14:04:18 -04:00
										 |  |  |  * (This impl also supports substituting the nca dir for an nca file, as that's more convenient | 
					
						
							|  |  |  |  * when 4GB splitting can be ignored.) | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | class RegisteredCache : public ContentProvider { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 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.
 | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  |     explicit RegisteredCache(VirtualDir dir, | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |                              ContentProviderParsingFunction parsing_function = | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  |                                  [](const VirtualFile& file, const NcaID& id) { return file; }); | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     ~RegisteredCache() override; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     void Refresh() override; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     bool HasEntry(u64 title_id, ContentRecordType type) const override; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     std::optional<u32> GetEntryVersion(u64 title_id) const override; | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     VirtualFile GetEntryUnparsed(u64 title_id, ContentRecordType type) const override; | 
					
						
							| 
									
										
										
										
											2018-08-16 17:07:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     VirtualFile GetEntryRaw(u64 title_id, ContentRecordType type) const override; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     std::unique_ptr<NCA> GetEntry(u64 title_id, ContentRecordType type) const override; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |     // If a parameter is not std::nullopt, it will be filtered for from all entries.
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     std::vector<ContentProviderEntry> ListEntriesFilter( | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |         std::optional<TitleType> title_type = {}, std::optional<ContentRecordType> record_type = {}, | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |         std::optional<u64> title_id = {}) const override; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-25 11:50:04 -04:00
										 |  |  |     // 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.
 | 
					
						
							| 
									
										
										
										
											2018-11-27 13:52:13 -05:00
										 |  |  |     InstallResult InstallEntry(const XCI& xci, bool overwrite_if_exists = false, | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |                                const VfsCopyFunction& copy = &VfsRawCopy); | 
					
						
							| 
									
										
										
										
											2018-11-27 13:52:13 -05:00
										 |  |  |     InstallResult InstallEntry(const NSP& nsp, bool overwrite_if_exists = false, | 
					
						
							| 
									
										
										
										
											2018-08-25 11:50:04 -04:00
										 |  |  |                                const VfsCopyFunction& copy = &VfsRawCopy); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // 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.
 | 
					
						
							| 
									
										
										
										
											2018-11-27 13:52:13 -05:00
										 |  |  |     InstallResult InstallEntry(const NCA& nca, TitleType type, bool overwrite_if_exists = false, | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |                                const VfsCopyFunction& copy = &VfsRawCopy); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 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; | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  |     std::vector<NcaID> AccumulateFiles() const; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     void ProcessFiles(const std::vector<NcaID>& ids); | 
					
						
							|  |  |  |     void AccumulateYuzuMeta(); | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |     std::optional<NcaID> GetNcaIDFromMetadata(u64 title_id, ContentRecordType type) const; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     VirtualFile GetFileAtID(NcaID id) const; | 
					
						
							|  |  |  |     VirtualFile OpenFileOrDirectoryConcat(const VirtualDir& dir, std::string_view path) const; | 
					
						
							| 
									
										
										
										
											2018-11-27 13:52:13 -05:00
										 |  |  |     InstallResult RawInstallNCA(const NCA& nca, const VfsCopyFunction& copy, | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |                                 bool overwrite_if_exists, std::optional<NcaID> override_id = {}); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     bool RawInstallYuzuMeta(const CNMT& cnmt); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     VirtualDir dir; | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     ContentProviderParsingFunction parser; | 
					
						
							| 
									
										
										
										
											2018-11-01 20:23:38 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     // maps tid -> NcaID of meta
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     std::map<u64, NcaID> meta_id; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     // maps tid -> meta
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     std::map<u64, CNMT> meta; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  |     // maps tid -> meta for CNMT in yuzu_meta
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     std::map<u64, CNMT> yuzu_meta; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | enum class ContentProviderUnionSlot { | 
					
						
							|  |  |  |     SysNAND,        ///< System NAND
 | 
					
						
							|  |  |  |     UserNAND,       ///< User NAND
 | 
					
						
							|  |  |  |     SDMC,           ///< SD Card
 | 
					
						
							|  |  |  |     FrontendManual, ///< Frontend-defined game list or similar
 | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  | // Combines multiple ContentProvider(s) (i.e. SysNAND, UserNAND, SDMC) into one interface.
 | 
					
						
							|  |  |  | class ContentProviderUnion : public ContentProvider { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     ~ContentProviderUnion() override; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void SetSlot(ContentProviderUnionSlot slot, ContentProvider* provider); | 
					
						
							|  |  |  |     void ClearSlot(ContentProviderUnionSlot slot); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::vector<std::pair<ContentProviderUnionSlot, ContentProviderEntry>> ListEntriesFilterOrigin( | 
					
						
							|  |  |  |         std::optional<ContentProviderUnionSlot> origin = {}, | 
					
						
							| 
									
										
										
										
											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-08-25 19:00:36 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 19:10:17 -04:00
										 |  |  |     std::optional<ContentProviderUnionSlot> GetSlotForEntry(u64 title_id, | 
					
						
							|  |  |  |                                                             ContentRecordType type) const; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2018-12-27 23:58:18 -05:00
										 |  |  |     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; | 
					
						
							| 
									
										
										
										
											2018-08-25 19:00:36 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:52:27 -04:00
										 |  |  | } // namespace FileSys
 |