mirror of
				https://github.com/PabloMK7/citra.git
				synced 2025-10-31 05:40:04 +00:00 
			
		
		
		
	FileSys/cia: add ticket parsing
This commit is contained in:
		
							parent
							
								
									b70e2bce58
								
							
						
					
					
						commit
						df77491938
					
				
					 8 changed files with 178 additions and 27 deletions
				
			
		|  | @ -50,6 +50,7 @@ add_library(core STATIC | ||||||
|     file_sys/archive_source_sd_savedata.h |     file_sys/archive_source_sd_savedata.h | ||||||
|     file_sys/archive_systemsavedata.cpp |     file_sys/archive_systemsavedata.cpp | ||||||
|     file_sys/archive_systemsavedata.h |     file_sys/archive_systemsavedata.h | ||||||
|  |     file_sys/cia_common.h | ||||||
|     file_sys/cia_container.cpp |     file_sys/cia_container.cpp | ||||||
|     file_sys/cia_container.h |     file_sys/cia_container.h | ||||||
|     file_sys/directory_backend.h |     file_sys/directory_backend.h | ||||||
|  | @ -68,6 +69,8 @@ add_library(core STATIC | ||||||
|     file_sys/romfs_reader.h |     file_sys/romfs_reader.h | ||||||
|     file_sys/savedata_archive.cpp |     file_sys/savedata_archive.cpp | ||||||
|     file_sys/savedata_archive.h |     file_sys/savedata_archive.h | ||||||
|  |     file_sys/ticket.cpp | ||||||
|  |     file_sys/ticket.h | ||||||
|     file_sys/title_metadata.cpp |     file_sys/title_metadata.cpp | ||||||
|     file_sys/title_metadata.h |     file_sys/title_metadata.h | ||||||
|     frontend/applets/default_applets.cpp |     frontend/applets/default_applets.cpp | ||||||
|  |  | ||||||
							
								
								
									
										40
									
								
								src/core/file_sys/cia_common.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								src/core/file_sys/cia_common.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,40 @@ | ||||||
|  | // Copyright 2018 Citra Emulator Project
 | ||||||
|  | // Licensed under GPLv2 or any later version
 | ||||||
|  | // Refer to the license.txt file included.
 | ||||||
|  | 
 | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include "common/assert.h" | ||||||
|  | #include "common/common_types.h" | ||||||
|  | 
 | ||||||
|  | namespace FileSys { | ||||||
|  | 
 | ||||||
|  | enum TMDSignatureType : u32 { | ||||||
|  |     Rsa4096Sha1 = 0x10000, | ||||||
|  |     Rsa2048Sha1 = 0x10001, | ||||||
|  |     EllipticSha1 = 0x10002, | ||||||
|  |     Rsa4096Sha256 = 0x10003, | ||||||
|  |     Rsa2048Sha256 = 0x10004, | ||||||
|  |     EcdsaSha256 = 0x10005 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | inline u32 GetSignatureSize(u32 signature_type) { | ||||||
|  |     switch (signature_type) { | ||||||
|  |     case Rsa4096Sha1: | ||||||
|  |     case Rsa4096Sha256: | ||||||
|  |         return 0x200; | ||||||
|  | 
 | ||||||
|  |     case Rsa2048Sha1: | ||||||
|  |     case Rsa2048Sha256: | ||||||
|  |         return 0x100; | ||||||
|  | 
 | ||||||
|  |     case EllipticSha1: | ||||||
|  |     case EcdsaSha256: | ||||||
|  |         return 0x3C; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     UNREACHABLE(); | ||||||
|  |     return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace FileSys
 | ||||||
|  | @ -124,6 +124,11 @@ Loader::ResultStatus CIAContainer::LoadHeader(const std::vector<u8>& header_data | ||||||
|     return Loader::ResultStatus::Success; |     return Loader::ResultStatus::Success; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | Loader::ResultStatus CIAContainer::LoadTicket(const std::vector<u8>& ticket_data, | ||||||
|  |                                               std::size_t offset) { | ||||||
|  |     return cia_ticket.Load(ticket_data, offset); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| Loader::ResultStatus CIAContainer::LoadTitleMetadata(const std::vector<u8>& tmd_data, | Loader::ResultStatus CIAContainer::LoadTitleMetadata(const std::vector<u8>& tmd_data, | ||||||
|                                                      std::size_t offset) { |                                                      std::size_t offset) { | ||||||
|     return cia_tmd.Load(tmd_data, offset); |     return cia_tmd.Load(tmd_data, offset); | ||||||
|  | @ -139,6 +144,10 @@ Loader::ResultStatus CIAContainer::LoadMetadata(const std::vector<u8>& meta_data | ||||||
|     return Loader::ResultStatus::Success; |     return Loader::ResultStatus::Success; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | const Ticket& CIAContainer::GetTicket() const { | ||||||
|  |     return cia_ticket; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| const TitleMetadata& CIAContainer::GetTitleMetadata() const { | const TitleMetadata& CIAContainer::GetTitleMetadata() const { | ||||||
|     return cia_tmd; |     return cia_tmd; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -10,6 +10,7 @@ | ||||||
| #include <vector> | #include <vector> | ||||||
| #include "common/common_types.h" | #include "common/common_types.h" | ||||||
| #include "common/swap.h" | #include "common/swap.h" | ||||||
|  | #include "core/file_sys/ticket.h" | ||||||
| #include "core/file_sys/title_metadata.h" | #include "core/file_sys/title_metadata.h" | ||||||
| 
 | 
 | ||||||
| namespace Loader { | namespace Loader { | ||||||
|  | @ -44,9 +45,11 @@ public: | ||||||
| 
 | 
 | ||||||
|     // Load parts of CIAs (for CIAs streamed in)
 |     // Load parts of CIAs (for CIAs streamed in)
 | ||||||
|     Loader::ResultStatus LoadHeader(const std::vector<u8>& header_data, std::size_t offset = 0); |     Loader::ResultStatus LoadHeader(const std::vector<u8>& header_data, std::size_t offset = 0); | ||||||
|  |     Loader::ResultStatus LoadTicket(const std::vector<u8>& ticket_data, std::size_t offset = 0); | ||||||
|     Loader::ResultStatus LoadTitleMetadata(const std::vector<u8>& tmd_data, std::size_t offset = 0); |     Loader::ResultStatus LoadTitleMetadata(const std::vector<u8>& tmd_data, std::size_t offset = 0); | ||||||
|     Loader::ResultStatus LoadMetadata(const std::vector<u8>& meta_data, std::size_t offset = 0); |     Loader::ResultStatus LoadMetadata(const std::vector<u8>& meta_data, std::size_t offset = 0); | ||||||
| 
 | 
 | ||||||
|  |     const Ticket& GetTicket() const; | ||||||
|     const TitleMetadata& GetTitleMetadata() const; |     const TitleMetadata& GetTitleMetadata() const; | ||||||
|     std::array<u64, 0x30>& GetDependencies(); |     std::array<u64, 0x30>& GetDependencies(); | ||||||
|     u32 GetCoreVersion() const; |     u32 GetCoreVersion() const; | ||||||
|  | @ -99,6 +102,7 @@ private: | ||||||
| 
 | 
 | ||||||
|     Header cia_header; |     Header cia_header; | ||||||
|     Metadata cia_metadata; |     Metadata cia_metadata; | ||||||
|  |     Ticket cia_ticket; | ||||||
|     TitleMetadata cia_tmd; |     TitleMetadata cia_tmd; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										56
									
								
								src/core/file_sys/ticket.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								src/core/file_sys/ticket.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,56 @@ | ||||||
|  | // Copyright 2018 Citra Emulator Project
 | ||||||
|  | // Licensed under GPLv2 or any later version
 | ||||||
|  | // Refer to the license.txt file included.
 | ||||||
|  | 
 | ||||||
|  | #include <algorithm> | ||||||
|  | #include <cryptopp/aes.h> | ||||||
|  | #include <cryptopp/modes.h> | ||||||
|  | #include "common/alignment.h" | ||||||
|  | #include "core/file_sys/cia_common.h" | ||||||
|  | #include "core/file_sys/ticket.h" | ||||||
|  | #include "core/hw/aes/key.h" | ||||||
|  | #include "core/loader/loader.h" | ||||||
|  | 
 | ||||||
|  | namespace FileSys { | ||||||
|  | 
 | ||||||
|  | Loader::ResultStatus Ticket::Load(const std::vector<u8> file_data, std::size_t offset) { | ||||||
|  |     std::size_t total_size = static_cast<std::size_t>(file_data.size() - offset); | ||||||
|  |     if (total_size < sizeof(u32)) | ||||||
|  |         return Loader::ResultStatus::Error; | ||||||
|  | 
 | ||||||
|  |     std::memcpy(&signature_type, &file_data[offset], sizeof(u32)); | ||||||
|  | 
 | ||||||
|  |     // Signature lengths are variable, and the body follows the signature
 | ||||||
|  |     u32 signature_size = GetSignatureSize(signature_type); | ||||||
|  | 
 | ||||||
|  |     // The ticket body start position is rounded to the nearest 0x40 after the signature
 | ||||||
|  |     std::size_t body_start = Common::AlignUp(signature_size + sizeof(u32), 0x40); | ||||||
|  |     std::size_t body_end = body_start + sizeof(Body); | ||||||
|  | 
 | ||||||
|  |     if (total_size < body_end) | ||||||
|  |         return Loader::ResultStatus::Error; | ||||||
|  | 
 | ||||||
|  |     // Read signature + ticket body
 | ||||||
|  |     ticket_signature.resize(signature_size); | ||||||
|  |     memcpy(ticket_signature.data(), &file_data[offset + sizeof(u32)], signature_size); | ||||||
|  |     memcpy(&ticket_body, &file_data[offset + body_start], sizeof(Body)); | ||||||
|  | 
 | ||||||
|  |     return Loader::ResultStatus::Success; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | boost::optional<std::array<u8, 16>> Ticket::GetTitleKey() const { | ||||||
|  |     HW::AES::InitKeys(); | ||||||
|  |     std::array<u8, 16> ctr{}; | ||||||
|  |     std::memcpy(ctr.data(), &ticket_body.title_id, sizeof(u64)); | ||||||
|  |     HW::AES::SelectCommonKeyIndex(ticket_body.common_key_index); | ||||||
|  |     if (!HW::AES::IsNormalKeyAvailable(HW::AES::KeySlotID::TicketCommonKey)) { | ||||||
|  |         return boost::none; | ||||||
|  |     } | ||||||
|  |     auto key = HW::AES::GetNormalKey(HW::AES::KeySlotID::TicketCommonKey); | ||||||
|  |     auto title_key = ticket_body.title_key; | ||||||
|  |     CryptoPP::CBC_Mode<CryptoPP::AES>::Decryption{key.data(), key.size(), ctr.data()}.ProcessData( | ||||||
|  |         title_key.data(), title_key.data(), title_key.size()); | ||||||
|  |     return title_key; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace FileSys
 | ||||||
							
								
								
									
										58
									
								
								src/core/file_sys/ticket.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								src/core/file_sys/ticket.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,58 @@ | ||||||
|  | // Copyright 2018 Citra Emulator Project
 | ||||||
|  | // Licensed under GPLv2 or any later version
 | ||||||
|  | // Refer to the license.txt file included.
 | ||||||
|  | 
 | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include <array> | ||||||
|  | #include <string> | ||||||
|  | #include <vector> | ||||||
|  | #include <boost/optional.hpp> | ||||||
|  | #include "common/common_funcs.h" | ||||||
|  | #include "common/common_types.h" | ||||||
|  | #include "common/swap.h" | ||||||
|  | 
 | ||||||
|  | namespace Loader { | ||||||
|  | enum class ResultStatus; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | namespace FileSys { | ||||||
|  | class Ticket { | ||||||
|  | public: | ||||||
|  | #pragma pack(push, 1) | ||||||
|  |     struct Body { | ||||||
|  |         std::array<u8, 0x40> issuer; | ||||||
|  |         std::array<u8, 0x3C> ecc_public_key; | ||||||
|  |         u8 version; | ||||||
|  |         u8 ca_crl_version; | ||||||
|  |         u8 signer_crl_version; | ||||||
|  |         std::array<u8, 0x10> title_key; | ||||||
|  |         INSERT_PADDING_BYTES(1); | ||||||
|  |         u64_be ticket_id; | ||||||
|  |         u32_be console_id; | ||||||
|  |         u64_be title_id; | ||||||
|  |         INSERT_PADDING_BYTES(2); | ||||||
|  |         u16_be ticket_title_version; | ||||||
|  |         INSERT_PADDING_BYTES(8); | ||||||
|  |         u8 license_type; | ||||||
|  |         u8 common_key_index; | ||||||
|  |         INSERT_PADDING_BYTES(0x2A); | ||||||
|  |         u32_be eshop_account_id; | ||||||
|  |         INSERT_PADDING_BYTES(1); | ||||||
|  |         u8 audit; | ||||||
|  |         INSERT_PADDING_BYTES(0x42); | ||||||
|  |         std::array<u8, 0x40> limits; | ||||||
|  |         std::array<u8, 0xAC> content_index; | ||||||
|  |     }; | ||||||
|  |     static_assert(sizeof(Body) == 0x210, "Ticket body structure size is wrong"); | ||||||
|  | #pragma pack(pop) | ||||||
|  | 
 | ||||||
|  |     Loader::ResultStatus Load(const std::vector<u8> file_data, std::size_t offset = 0); | ||||||
|  |     boost::optional<std::array<u8, 16>> GetTitleKey() const; | ||||||
|  | 
 | ||||||
|  | private: | ||||||
|  |     Body ticket_body; | ||||||
|  |     u32_be signature_type; | ||||||
|  |     std::vector<u8> ticket_signature; | ||||||
|  | }; | ||||||
|  | } // namespace FileSys
 | ||||||
|  | @ -7,6 +7,7 @@ | ||||||
| #include "common/alignment.h" | #include "common/alignment.h" | ||||||
| #include "common/file_util.h" | #include "common/file_util.h" | ||||||
| #include "common/logging/log.h" | #include "common/logging/log.h" | ||||||
|  | #include "core/file_sys/cia_common.h" | ||||||
| #include "core/file_sys/title_metadata.h" | #include "core/file_sys/title_metadata.h" | ||||||
| #include "core/loader/loader.h" | #include "core/loader/loader.h" | ||||||
| 
 | 
 | ||||||
|  | @ -15,24 +16,6 @@ | ||||||
| 
 | 
 | ||||||
| namespace FileSys { | namespace FileSys { | ||||||
| 
 | 
 | ||||||
| static u32 GetSignatureSize(u32 signature_type) { |  | ||||||
|     switch (signature_type) { |  | ||||||
|     case Rsa4096Sha1: |  | ||||||
|     case Rsa4096Sha256: |  | ||||||
|         return 0x200; |  | ||||||
| 
 |  | ||||||
|     case Rsa2048Sha1: |  | ||||||
|     case Rsa2048Sha256: |  | ||||||
|         return 0x100; |  | ||||||
| 
 |  | ||||||
|     case EllipticSha1: |  | ||||||
|     case EcdsaSha256: |  | ||||||
|         return 0x3C; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return 0; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| Loader::ResultStatus TitleMetadata::Load(const std::string& file_path) { | Loader::ResultStatus TitleMetadata::Load(const std::string& file_path) { | ||||||
|     FileUtil::IOFile file(file_path, "rb"); |     FileUtil::IOFile file(file_path, "rb"); | ||||||
|     if (!file.IsOpen()) |     if (!file.IsOpen()) | ||||||
|  | @ -188,6 +171,12 @@ u64 TitleMetadata::GetContentSizeByIndex(u16 index) const { | ||||||
|     return tmd_chunks[index].size; |     return tmd_chunks[index].size; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | std::array<u8, 16> TitleMetadata::GetContentCTRByIndex(u16 index) const { | ||||||
|  |     std::array<u8, 16> ctr{}; | ||||||
|  |     std::memcpy(ctr.data(), &tmd_chunks[index].index, sizeof(u16)); | ||||||
|  |     return ctr; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void TitleMetadata::SetTitleID(u64 title_id) { | void TitleMetadata::SetTitleID(u64 title_id) { | ||||||
|     tmd_body.title_id = title_id; |     tmd_body.title_id = title_id; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -19,15 +19,6 @@ enum class ResultStatus; | ||||||
| 
 | 
 | ||||||
| namespace FileSys { | namespace FileSys { | ||||||
| 
 | 
 | ||||||
| enum TMDSignatureType : u32 { |  | ||||||
|     Rsa4096Sha1 = 0x10000, |  | ||||||
|     Rsa2048Sha1 = 0x10001, |  | ||||||
|     EllipticSha1 = 0x10002, |  | ||||||
|     Rsa4096Sha256 = 0x10003, |  | ||||||
|     Rsa2048Sha256 = 0x10004, |  | ||||||
|     EcdsaSha256 = 0x10005 |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| enum TMDContentTypeFlag : u16 { | enum TMDContentTypeFlag : u16 { | ||||||
|     Encrypted = 1 << 0, |     Encrypted = 1 << 0, | ||||||
|     Disc = 1 << 2, |     Disc = 1 << 2, | ||||||
|  | @ -108,6 +99,7 @@ public: | ||||||
|     u32 GetContentIDByIndex(u16 index) const; |     u32 GetContentIDByIndex(u16 index) const; | ||||||
|     u16 GetContentTypeByIndex(u16 index) const; |     u16 GetContentTypeByIndex(u16 index) const; | ||||||
|     u64 GetContentSizeByIndex(u16 index) const; |     u64 GetContentSizeByIndex(u16 index) const; | ||||||
|  |     std::array<u8, 16> GetContentCTRByIndex(u16 index) const; | ||||||
| 
 | 
 | ||||||
|     void SetTitleID(u64 title_id); |     void SetTitleID(u64 title_id); | ||||||
|     void SetTitleType(u32 type); |     void SetTitleType(u32 type); | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue