| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | // Copyright 2018 yuzu emulator team
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:12:14 -04:00
										 |  |  | #include <algorithm>
 | 
					
						
							| 
									
										
										
										
											2018-09-03 21:58:19 -04:00
										 |  |  | #include <cstring>
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  | #include <optional>
 | 
					
						
							| 
									
										
										
										
											2018-07-18 23:12:14 -04:00
										 |  |  | #include <utility>
 | 
					
						
							| 
									
										
										
										
											2018-09-03 21:58:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | #include "core/crypto/aes_util.h"
 | 
					
						
							|  |  |  | #include "core/crypto/ctr_encryption_layer.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | #include "core/file_sys/content_archive.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-25 19:01:46 -04:00
										 |  |  | #include "core/file_sys/nca_patch.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-03 21:58:19 -04:00
										 |  |  | #include "core/file_sys/partition_filesystem.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  | #include "core/file_sys/romfs.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | #include "core/file_sys/vfs_offset.h"
 | 
					
						
							|  |  |  | #include "core/loader/loader.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace FileSys { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Media offsets in headers are stored divided by 512. Mult. by this to get real offset.
 | 
					
						
							|  |  |  | constexpr u64 MEDIA_OFFSET_MULTIPLIER = 0x200; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | constexpr u64 SECTION_HEADER_SIZE = 0x200; | 
					
						
							|  |  |  | constexpr u64 SECTION_HEADER_OFFSET = 0x400; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | constexpr u32 IVFC_MAX_LEVEL = 6; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum class NCASectionFilesystemType : u8 { | 
					
						
							|  |  |  |     PFS0 = 0x2, | 
					
						
							|  |  |  |     ROMFS = 0x3, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct NCASectionHeaderBlock { | 
					
						
							|  |  |  |     INSERT_PADDING_BYTES(3); | 
					
						
							|  |  |  |     NCASectionFilesystemType filesystem_type; | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |     NCASectionCryptoType crypto_type; | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  |     INSERT_PADDING_BYTES(3); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | static_assert(sizeof(NCASectionHeaderBlock) == 0x8, "NCASectionHeaderBlock has incorrect size."); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | struct NCASectionRaw { | 
					
						
							|  |  |  |     NCASectionHeaderBlock header; | 
					
						
							|  |  |  |     std::array<u8, 0x138> block_data; | 
					
						
							|  |  |  |     std::array<u8, 0x8> section_ctr; | 
					
						
							|  |  |  |     INSERT_PADDING_BYTES(0xB8); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | static_assert(sizeof(NCASectionRaw) == 0x200, "NCASectionRaw has incorrect size."); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | struct PFS0Superblock { | 
					
						
							|  |  |  |     NCASectionHeaderBlock header_block; | 
					
						
							|  |  |  |     std::array<u8, 0x20> hash; | 
					
						
							|  |  |  |     u32_le size; | 
					
						
							|  |  |  |     INSERT_PADDING_BYTES(4); | 
					
						
							|  |  |  |     u64_le hash_table_offset; | 
					
						
							|  |  |  |     u64_le hash_table_size; | 
					
						
							|  |  |  |     u64_le pfs0_header_offset; | 
					
						
							|  |  |  |     u64_le pfs0_size; | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |     INSERT_PADDING_BYTES(0x1B0); | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | static_assert(sizeof(PFS0Superblock) == 0x200, "PFS0Superblock has incorrect size."); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct RomFSSuperblock { | 
					
						
							|  |  |  |     NCASectionHeaderBlock header_block; | 
					
						
							| 
									
										
										
										
											2018-07-27 18:14:03 -04:00
										 |  |  |     IVFCHeader ivfc; | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |     INSERT_PADDING_BYTES(0x118); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | static_assert(sizeof(RomFSSuperblock) == 0x200, "RomFSSuperblock has incorrect size."); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-25 19:01:46 -04:00
										 |  |  | struct BKTRHeader { | 
					
						
							|  |  |  |     u64_le offset; | 
					
						
							|  |  |  |     u64_le size; | 
					
						
							|  |  |  |     u32_le magic; | 
					
						
							|  |  |  |     INSERT_PADDING_BYTES(0x4); | 
					
						
							|  |  |  |     u32_le number_entries; | 
					
						
							|  |  |  |     INSERT_PADDING_BYTES(0x4); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | static_assert(sizeof(BKTRHeader) == 0x20, "BKTRHeader has incorrect size."); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct BKTRSuperblock { | 
					
						
							|  |  |  |     NCASectionHeaderBlock header_block; | 
					
						
							|  |  |  |     IVFCHeader ivfc; | 
					
						
							|  |  |  |     INSERT_PADDING_BYTES(0x18); | 
					
						
							|  |  |  |     BKTRHeader relocation; | 
					
						
							|  |  |  |     BKTRHeader subsection; | 
					
						
							|  |  |  |     INSERT_PADDING_BYTES(0xC0); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | static_assert(sizeof(BKTRSuperblock) == 0x200, "BKTRSuperblock has incorrect size."); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | union NCASectionHeader { | 
					
						
							|  |  |  |     NCASectionRaw raw; | 
					
						
							|  |  |  |     PFS0Superblock pfs0; | 
					
						
							|  |  |  |     RomFSSuperblock romfs; | 
					
						
							| 
									
										
										
										
											2018-08-25 19:01:46 -04:00
										 |  |  |     BKTRSuperblock bktr; | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | static_assert(sizeof(NCASectionHeader) == 0x200, "NCASectionHeader has incorrect size."); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-16 13:13:40 -04:00
										 |  |  | static bool IsValidNCA(const NCAHeader& header) { | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |     // TODO(DarkLordZach): Add NCA2/NCA0 support.
 | 
					
						
							|  |  |  |     return header.magic == Common::MakeMagic('N', 'C', 'A', '3'); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-01 20:22:29 -04:00
										 |  |  | NCA::NCA(VirtualFile file_, VirtualFile bktr_base_romfs_, u64 bktr_base_ivfc_offset, | 
					
						
							|  |  |  |          Core::Crypto::KeyManager keys_) | 
					
						
							|  |  |  |     : file(std::move(file_)), bktr_base_romfs(std::move(bktr_base_romfs_)), keys(std::move(keys_)) { | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  |     if (file == nullptr) { | 
					
						
							|  |  |  |         status = Loader::ResultStatus::ErrorNullFile; | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (sizeof(NCAHeader) != file->ReadObject(&header)) { | 
					
						
							|  |  |  |         LOG_ERROR(Loader, "File reader errored out during header read."); | 
					
						
							|  |  |  |         status = Loader::ResultStatus::ErrorBadNCAHeader; | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!HandlePotentialHeaderDecryption()) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-16 13:08:42 -04:00
										 |  |  |     has_rights_id = std::any_of(header.rights_id.begin(), header.rights_id.end(), | 
					
						
							|  |  |  |                                 [](char c) { return c != '\0'; }); | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const std::vector<NCASectionHeader> sections = ReadSectionHeaders(); | 
					
						
							|  |  |  |     is_update = std::any_of(sections.begin(), sections.end(), [](const NCASectionHeader& header) { | 
					
						
							|  |  |  |         return header.raw.header.crypto_type == NCASectionCryptoType::BKTR; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!ReadSections(sections, bktr_base_ivfc_offset)) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     status = Loader::ResultStatus::Success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NCA::~NCA() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool NCA::CheckSupportedNCA(const NCAHeader& nca_header) { | 
					
						
							|  |  |  |     if (nca_header.magic == Common::MakeMagic('N', 'C', 'A', '2')) { | 
					
						
							|  |  |  |         status = Loader::ResultStatus::ErrorNCA2; | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (nca_header.magic == Common::MakeMagic('N', 'C', 'A', '0')) { | 
					
						
							|  |  |  |         status = Loader::ResultStatus::ErrorNCA0; | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool NCA::HandlePotentialHeaderDecryption() { | 
					
						
							|  |  |  |     if (IsValidNCA(header)) { | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!CheckSupportedNCA(header)) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     NCAHeader dec_header{}; | 
					
						
							|  |  |  |     Core::Crypto::AESCipher<Core::Crypto::Key256> cipher( | 
					
						
							|  |  |  |         keys.GetKey(Core::Crypto::S256KeyType::Header), Core::Crypto::Mode::XTS); | 
					
						
							|  |  |  |     cipher.XTSTranscode(&header, sizeof(NCAHeader), &dec_header, 0, 0x200, | 
					
						
							|  |  |  |                         Core::Crypto::Op::Decrypt); | 
					
						
							|  |  |  |     if (IsValidNCA(dec_header)) { | 
					
						
							|  |  |  |         header = dec_header; | 
					
						
							|  |  |  |         encrypted = true; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         if (!CheckSupportedNCA(dec_header)) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (keys.HasKey(Core::Crypto::S256KeyType::Header)) { | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorIncorrectHeaderKey; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorMissingHeaderKey; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::vector<NCASectionHeader> NCA::ReadSectionHeaders() const { | 
					
						
							|  |  |  |     const std::ptrdiff_t number_sections = | 
					
						
							|  |  |  |         std::count_if(std::begin(header.section_tables), std::end(header.section_tables), | 
					
						
							|  |  |  |                       [](NCASectionTableEntry entry) { return entry.media_offset > 0; }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::vector<NCASectionHeader> sections(number_sections); | 
					
						
							|  |  |  |     const auto length_sections = SECTION_HEADER_SIZE * number_sections; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (encrypted) { | 
					
						
							|  |  |  |         auto raw = file->ReadBytes(length_sections, SECTION_HEADER_OFFSET); | 
					
						
							|  |  |  |         Core::Crypto::AESCipher<Core::Crypto::Key256> cipher( | 
					
						
							|  |  |  |             keys.GetKey(Core::Crypto::S256KeyType::Header), Core::Crypto::Mode::XTS); | 
					
						
							|  |  |  |         cipher.XTSTranscode(raw.data(), length_sections, sections.data(), 2, SECTION_HEADER_SIZE, | 
					
						
							|  |  |  |                             Core::Crypto::Op::Decrypt); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         file->ReadBytes(sections.data(), length_sections, SECTION_HEADER_OFFSET); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return sections; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool NCA::ReadSections(const std::vector<NCASectionHeader>& sections, u64 bktr_base_ivfc_offset) { | 
					
						
							|  |  |  |     for (std::size_t i = 0; i < sections.size(); ++i) { | 
					
						
							|  |  |  |         const auto& section = sections[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (section.raw.header.filesystem_type == NCASectionFilesystemType::ROMFS) { | 
					
						
							|  |  |  |             if (!ReadRomFSSection(section, header.section_tables[i], bktr_base_ivfc_offset)) { | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else if (section.raw.header.filesystem_type == NCASectionFilesystemType::PFS0) { | 
					
						
							|  |  |  |             if (!ReadPFS0Section(section, header.section_tables[i])) { | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool NCA::ReadRomFSSection(const NCASectionHeader& section, const NCASectionTableEntry& entry, | 
					
						
							|  |  |  |                            u64 bktr_base_ivfc_offset) { | 
					
						
							|  |  |  |     const std::size_t base_offset = entry.media_offset * MEDIA_OFFSET_MULTIPLIER; | 
					
						
							|  |  |  |     ivfc_offset = section.romfs.ivfc.levels[IVFC_MAX_LEVEL - 1].offset; | 
					
						
							|  |  |  |     const std::size_t romfs_offset = base_offset + ivfc_offset; | 
					
						
							|  |  |  |     const std::size_t romfs_size = section.romfs.ivfc.levels[IVFC_MAX_LEVEL - 1].size; | 
					
						
							|  |  |  |     auto raw = std::make_shared<OffsetVfsFile>(file, romfs_size, romfs_offset); | 
					
						
							|  |  |  |     auto dec = Decrypt(section, raw, romfs_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (dec == nullptr) { | 
					
						
							|  |  |  |         if (status != Loader::ResultStatus::Success) | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         if (has_rights_id) | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorIncorrectTitlekeyOrTitlekek; | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorIncorrectKeyAreaKey; | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (section.raw.header.crypto_type == NCASectionCryptoType::BKTR) { | 
					
						
							|  |  |  |         if (section.bktr.relocation.magic != Common::MakeMagic('B', 'K', 'T', 'R') || | 
					
						
							|  |  |  |             section.bktr.subsection.magic != Common::MakeMagic('B', 'K', 'T', 'R')) { | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorBadBKTRHeader; | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (section.bktr.relocation.offset + section.bktr.relocation.size != | 
					
						
							|  |  |  |             section.bktr.subsection.offset) { | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorBKTRSubsectionNotAfterRelocation; | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const u64 size = MEDIA_OFFSET_MULTIPLIER * (entry.media_end_offset - entry.media_offset); | 
					
						
							|  |  |  |         if (section.bktr.subsection.offset + section.bktr.subsection.size != size) { | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorBKTRSubsectionNotAtEnd; | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const u64 offset = section.romfs.ivfc.levels[IVFC_MAX_LEVEL - 1].offset; | 
					
						
							|  |  |  |         RelocationBlock relocation_block{}; | 
					
						
							|  |  |  |         if (dec->ReadObject(&relocation_block, section.bktr.relocation.offset - offset) != | 
					
						
							|  |  |  |             sizeof(RelocationBlock)) { | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorBadRelocationBlock; | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         SubsectionBlock subsection_block{}; | 
					
						
							|  |  |  |         if (dec->ReadObject(&subsection_block, section.bktr.subsection.offset - offset) != | 
					
						
							|  |  |  |             sizeof(RelocationBlock)) { | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorBadSubsectionBlock; | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         std::vector<RelocationBucketRaw> relocation_buckets_raw( | 
					
						
							|  |  |  |             (section.bktr.relocation.size - sizeof(RelocationBlock)) / sizeof(RelocationBucketRaw)); | 
					
						
							|  |  |  |         if (dec->ReadBytes(relocation_buckets_raw.data(), | 
					
						
							|  |  |  |                            section.bktr.relocation.size - sizeof(RelocationBlock), | 
					
						
							|  |  |  |                            section.bktr.relocation.offset + sizeof(RelocationBlock) - offset) != | 
					
						
							|  |  |  |             section.bktr.relocation.size - sizeof(RelocationBlock)) { | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorBadRelocationBuckets; | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         std::vector<SubsectionBucketRaw> subsection_buckets_raw( | 
					
						
							|  |  |  |             (section.bktr.subsection.size - sizeof(SubsectionBlock)) / sizeof(SubsectionBucketRaw)); | 
					
						
							|  |  |  |         if (dec->ReadBytes(subsection_buckets_raw.data(), | 
					
						
							|  |  |  |                            section.bktr.subsection.size - sizeof(SubsectionBlock), | 
					
						
							|  |  |  |                            section.bktr.subsection.offset + sizeof(SubsectionBlock) - offset) != | 
					
						
							|  |  |  |             section.bktr.subsection.size - sizeof(SubsectionBlock)) { | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorBadSubsectionBuckets; | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         std::vector<RelocationBucket> relocation_buckets(relocation_buckets_raw.size()); | 
					
						
							|  |  |  |         std::transform(relocation_buckets_raw.begin(), relocation_buckets_raw.end(), | 
					
						
							|  |  |  |                        relocation_buckets.begin(), &ConvertRelocationBucketRaw); | 
					
						
							|  |  |  |         std::vector<SubsectionBucket> subsection_buckets(subsection_buckets_raw.size()); | 
					
						
							|  |  |  |         std::transform(subsection_buckets_raw.begin(), subsection_buckets_raw.end(), | 
					
						
							|  |  |  |                        subsection_buckets.begin(), &ConvertSubsectionBucketRaw); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         u32 ctr_low; | 
					
						
							|  |  |  |         std::memcpy(&ctr_low, section.raw.section_ctr.data(), sizeof(ctr_low)); | 
					
						
							|  |  |  |         subsection_buckets.back().entries.push_back({section.bktr.relocation.offset, {0}, ctr_low}); | 
					
						
							|  |  |  |         subsection_buckets.back().entries.push_back({size, {0}, 0}); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |         std::optional<Core::Crypto::Key128> key = {}; | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  |         if (encrypted) { | 
					
						
							|  |  |  |             if (has_rights_id) { | 
					
						
							|  |  |  |                 status = Loader::ResultStatus::Success; | 
					
						
							|  |  |  |                 key = GetTitlekey(); | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |                 if (!key) { | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  |                     status = Loader::ResultStatus::ErrorMissingTitlekey; | 
					
						
							|  |  |  |                     return false; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 key = GetKeyAreaKey(NCASectionCryptoType::BKTR); | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |                 if (!key) { | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  |                     status = Loader::ResultStatus::ErrorMissingKeyAreaKey; | 
					
						
							|  |  |  |                     return false; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (bktr_base_romfs == nullptr) { | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorMissingBKTRBaseRomFS; | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         auto bktr = std::make_shared<BKTR>( | 
					
						
							|  |  |  |             bktr_base_romfs, std::make_shared<OffsetVfsFile>(file, romfs_size, base_offset), | 
					
						
							|  |  |  |             relocation_block, relocation_buckets, subsection_block, subsection_buckets, encrypted, | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |             encrypted ? *key : Core::Crypto::Key128{}, base_offset, bktr_base_ivfc_offset, | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  |             section.raw.section_ctr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // BKTR applies to entire IVFC, so make an offset version to level 6
 | 
					
						
							|  |  |  |         files.push_back(std::make_shared<OffsetVfsFile>( | 
					
						
							|  |  |  |             bktr, romfs_size, section.romfs.ivfc.levels[IVFC_MAX_LEVEL - 1].offset)); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         files.push_back(std::move(dec)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     romfs = files.back(); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool NCA::ReadPFS0Section(const NCASectionHeader& section, const NCASectionTableEntry& entry) { | 
					
						
							|  |  |  |     const u64 offset = (static_cast<u64>(entry.media_offset) * MEDIA_OFFSET_MULTIPLIER) + | 
					
						
							|  |  |  |                        section.pfs0.pfs0_header_offset; | 
					
						
							|  |  |  |     const u64 size = MEDIA_OFFSET_MULTIPLIER * (entry.media_end_offset - entry.media_offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto dec = Decrypt(section, std::make_shared<OffsetVfsFile>(file, size, offset), offset); | 
					
						
							|  |  |  |     if (dec != nullptr) { | 
					
						
							|  |  |  |         auto npfs = std::make_shared<PartitionFilesystem>(std::move(dec)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (npfs->GetStatus() == Loader::ResultStatus::Success) { | 
					
						
							|  |  |  |             dirs.push_back(std::move(npfs)); | 
					
						
							|  |  |  |             if (IsDirectoryExeFS(dirs.back())) | 
					
						
							|  |  |  |                 exefs = dirs.back(); | 
					
						
							| 
									
										
										
										
											2019-01-15 15:56:32 -05:00
										 |  |  |             else if (IsDirectoryLogoPartition(dirs.back())) | 
					
						
							|  |  |  |                 logo = dirs.back(); | 
					
						
							| 
									
										
										
										
											2018-10-16 12:12:50 -04:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             if (has_rights_id) | 
					
						
							|  |  |  |                 status = Loader::ResultStatus::ErrorIncorrectTitlekeyOrTitlekek; | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |                 status = Loader::ResultStatus::ErrorIncorrectKeyAreaKey; | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         if (status != Loader::ResultStatus::Success) | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         if (has_rights_id) | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorIncorrectTitlekeyOrTitlekek; | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             status = Loader::ResultStatus::ErrorIncorrectKeyAreaKey; | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-04 14:57:21 -04:00
										 |  |  | u8 NCA::GetCryptoRevision() const { | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |     u8 master_key_id = header.crypto_type; | 
					
						
							|  |  |  |     if (header.crypto_type_2 > master_key_id) | 
					
						
							|  |  |  |         master_key_id = header.crypto_type_2; | 
					
						
							|  |  |  |     if (master_key_id > 0) | 
					
						
							|  |  |  |         --master_key_id; | 
					
						
							| 
									
										
										
										
											2018-08-04 14:57:21 -04:00
										 |  |  |     return master_key_id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  | std::optional<Core::Crypto::Key128> NCA::GetKeyAreaKey(NCASectionCryptoType type) const { | 
					
						
							| 
									
										
										
										
											2018-08-04 14:57:21 -04:00
										 |  |  |     const auto master_key_id = GetCryptoRevision(); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-29 20:47:33 -04:00
										 |  |  |     if (!keys.HasKey(Core::Crypto::S128KeyType::KeyArea, master_key_id, header.key_index)) | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2018-07-29 20:47:33 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |     std::vector<u8> key_area(header.key_area.begin(), header.key_area.end()); | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  |     Core::Crypto::AESCipher<Core::Crypto::Key128> cipher( | 
					
						
							|  |  |  |         keys.GetKey(Core::Crypto::S128KeyType::KeyArea, master_key_id, header.key_index), | 
					
						
							|  |  |  |         Core::Crypto::Mode::ECB); | 
					
						
							|  |  |  |     cipher.Transcode(key_area.data(), key_area.size(), key_area.data(), Core::Crypto::Op::Decrypt); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  |     Core::Crypto::Key128 out; | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |     if (type == NCASectionCryptoType::XTS) | 
					
						
							|  |  |  |         std::copy(key_area.begin(), key_area.begin() + 0x10, out.begin()); | 
					
						
							| 
									
										
										
										
											2018-08-25 19:01:46 -04:00
										 |  |  |     else if (type == NCASectionCryptoType::CTR || type == NCASectionCryptoType::BKTR) | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |         std::copy(key_area.begin() + 0x20, key_area.begin() + 0x30, out.begin()); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         LOG_CRITICAL(Crypto, "Called GetKeyAreaKey on invalid NCASectionCryptoType type={:02X}", | 
					
						
							|  |  |  |                      static_cast<u8>(type)); | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  |     u128 out_128{}; | 
					
						
							|  |  |  |     memcpy(out_128.data(), out.data(), 16); | 
					
						
							| 
									
										
										
										
											2018-10-05 09:19:35 -04:00
										 |  |  |     LOG_TRACE(Crypto, "called with crypto_rev={:02X}, kak_index={:02X}, key={:016X}{:016X}", | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |               master_key_id, header.key_index, out_128[1], out_128[0]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return out; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  | std::optional<Core::Crypto::Key128> NCA::GetTitlekey() { | 
					
						
							| 
									
										
										
										
											2018-08-04 14:57:21 -04:00
										 |  |  |     const auto master_key_id = GetCryptoRevision(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u128 rights_id{}; | 
					
						
							|  |  |  |     memcpy(rights_id.data(), header.rights_id.data(), 16); | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |     if (rights_id == u128{}) { | 
					
						
							|  |  |  |         status = Loader::ResultStatus::ErrorInvalidRightsID; | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-04 14:57:21 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     auto titlekey = keys.GetKey(Core::Crypto::S128KeyType::Titlekey, rights_id[1], rights_id[0]); | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |     if (titlekey == Core::Crypto::Key128{}) { | 
					
						
							|  |  |  |         status = Loader::ResultStatus::ErrorMissingTitlekey; | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!keys.HasKey(Core::Crypto::S128KeyType::Titlekek, master_key_id)) { | 
					
						
							|  |  |  |         status = Loader::ResultStatus::ErrorMissingTitlekek; | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-04 14:57:21 -04:00
										 |  |  |     Core::Crypto::AESCipher<Core::Crypto::Key128> cipher( | 
					
						
							|  |  |  |         keys.GetKey(Core::Crypto::S128KeyType::Titlekek, master_key_id), Core::Crypto::Mode::ECB); | 
					
						
							|  |  |  |     cipher.Transcode(titlekey.data(), titlekey.size(), titlekey.data(), Core::Crypto::Op::Decrypt); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return titlekey; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-16 12:08:15 -04:00
										 |  |  | VirtualFile NCA::Decrypt(const NCASectionHeader& s_header, VirtualFile in, u64 starting_offset) { | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |     if (!encrypted) | 
					
						
							|  |  |  |         return in; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-04 14:57:21 -04:00
										 |  |  |     switch (s_header.raw.header.crypto_type) { | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |     case NCASectionCryptoType::NONE: | 
					
						
							| 
									
										
										
										
											2019-06-19 22:01:41 -04:00
										 |  |  |         LOG_TRACE(Crypto, "called with mode=NONE"); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |         return in; | 
					
						
							|  |  |  |     case NCASectionCryptoType::CTR: | 
					
						
							| 
									
										
										
										
											2018-08-25 19:01:46 -04:00
										 |  |  |     // During normal BKTR decryption, this entire function is skipped. This is for the metadata,
 | 
					
						
							|  |  |  |     // which uses the same CTR as usual.
 | 
					
						
							|  |  |  |     case NCASectionCryptoType::BKTR: | 
					
						
							| 
									
										
										
										
											2019-06-19 22:01:41 -04:00
										 |  |  |         LOG_TRACE(Crypto, "called with mode=CTR, starting_offset={:016X}", starting_offset); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |             std::optional<Core::Crypto::Key128> key = {}; | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |             if (has_rights_id) { | 
					
						
							|  |  |  |                 status = Loader::ResultStatus::Success; | 
					
						
							| 
									
										
										
										
											2018-08-04 14:57:21 -04:00
										 |  |  |                 key = GetTitlekey(); | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |                 if (!key) { | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |                     if (status == Loader::ResultStatus::Success) | 
					
						
							|  |  |  |                         status = Loader::ResultStatus::ErrorMissingTitlekey; | 
					
						
							|  |  |  |                     return nullptr; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 key = GetKeyAreaKey(NCASectionCryptoType::CTR); | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |                 if (!key) { | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |                     status = Loader::ResultStatus::ErrorMissingKeyAreaKey; | 
					
						
							|  |  |  |                     return nullptr; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2018-08-04 14:57:21 -04:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-30 05:03:25 +01:00
										 |  |  |             auto out = std::make_shared<Core::Crypto::CTREncryptionLayer>(std::move(in), *key, | 
					
						
							|  |  |  |                                                                           starting_offset); | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  |             std::vector<u8> iv(16); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |             for (u8 i = 0; i < 8; ++i) | 
					
						
							| 
									
										
										
										
											2018-08-04 14:57:21 -04:00
										 |  |  |                 iv[i] = s_header.raw.section_ctr[0x8 - i - 1]; | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |             out->SetIV(iv); | 
					
						
							| 
									
										
										
										
											2018-07-29 20:47:33 -04:00
										 |  |  |             return std::static_pointer_cast<VfsFile>(out); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     case NCASectionCryptoType::XTS: | 
					
						
							| 
									
										
										
										
											2018-08-16 17:01:32 -04:00
										 |  |  |         // TODO(DarkLordZach): Find a test case for XTS-encrypted NCAs
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |     default: | 
					
						
							|  |  |  |         LOG_ERROR(Crypto, "called with unhandled crypto type={:02X}", | 
					
						
							| 
									
										
										
										
											2018-08-04 14:57:21 -04:00
										 |  |  |                   static_cast<u8>(s_header.raw.header.crypto_type)); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |         return nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | Loader::ResultStatus NCA::GetStatus() const { | 
					
						
							|  |  |  |     return status; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::vector<std::shared_ptr<VfsFile>> NCA::GetFiles() const { | 
					
						
							|  |  |  |     if (status != Loader::ResultStatus::Success) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     return files; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::vector<std::shared_ptr<VfsDirectory>> NCA::GetSubdirectories() const { | 
					
						
							|  |  |  |     if (status != Loader::ResultStatus::Success) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     return dirs; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string NCA::GetName() const { | 
					
						
							|  |  |  |     return file->GetName(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::shared_ptr<VfsDirectory> NCA::GetParentDirectory() const { | 
					
						
							|  |  |  |     return file->GetContainingDirectory(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NCAContentType NCA::GetType() const { | 
					
						
							|  |  |  |     return header.content_type; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u64 NCA::GetTitleId() const { | 
					
						
							| 
									
										
										
										
											2018-08-25 19:01:46 -04:00
										 |  |  |     if (is_update || status == Loader::ResultStatus::ErrorMissingBKTRBaseRomFS) | 
					
						
							|  |  |  |         return header.title_id | 0x800; | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  |     return header.title_id; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-25 19:01:46 -04:00
										 |  |  | bool NCA::IsUpdate() const { | 
					
						
							|  |  |  |     return is_update; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | VirtualFile NCA::GetRomFS() const { | 
					
						
							|  |  |  |     return romfs; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | VirtualDir NCA::GetExeFS() const { | 
					
						
							|  |  |  |     return exefs; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | VirtualFile NCA::GetBaseFile() const { | 
					
						
							|  |  |  |     return file; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 22:37:42 -04:00
										 |  |  | u64 NCA::GetBaseIVFCOffset() const { | 
					
						
							|  |  |  |     return ivfc_offset; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-15 15:56:32 -05:00
										 |  |  | VirtualDir NCA::GetLogoPartition() const { | 
					
						
							|  |  |  |     return logo; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | } // namespace FileSys
 |