| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | // Copyright 2018 yuzu emulator team
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | #include <array>
 | 
					
						
							| 
									
										
										
										
											2018-07-29 19:00:09 -04:00
										 |  |  | #include <type_traits>
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | #include <unordered_map>
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2018-07-28 14:28:14 -04:00
										 |  |  | #include <fmt/format.h>
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  | namespace Core::Crypto { | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  | using Key128 = std::array<u8, 0x10>; | 
					
						
							|  |  |  | using Key256 = std::array<u8, 0x20>; | 
					
						
							|  |  |  | using SHA256Hash = std::array<u8, 0x20>; | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | static_assert(sizeof(Key128) == 16, "Key128 must be 128 bytes big."); | 
					
						
							|  |  |  | static_assert(sizeof(Key256) == 32, "Key128 must be 128 bytes big."); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum class S256KeyType : u64 { | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  |     Header, //
 | 
					
						
							|  |  |  |     SDSave, //
 | 
					
						
							|  |  |  |     SDNCA,  //
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum class S128KeyType : u64 { | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  |     Master,        // f1=crypto revision
 | 
					
						
							|  |  |  |     Package1,      // f1=crypto revision
 | 
					
						
							|  |  |  |     Package2,      // f1=crypto revision
 | 
					
						
							|  |  |  |     Titlekek,      // f1=crypto revision
 | 
					
						
							|  |  |  |     ETicketRSAKek, //
 | 
					
						
							|  |  |  |     KeyArea,       // f1=crypto revision f2=type {app, ocean, system}
 | 
					
						
							|  |  |  |     SDSeed,        //
 | 
					
						
							|  |  |  |     Titlekey,      // f1=rights id LSB f2=rights id MSB
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum class KeyAreaKeyType : u8 { | 
					
						
							|  |  |  |     Application, | 
					
						
							|  |  |  |     Ocean, | 
					
						
							|  |  |  |     System, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename KeyType> | 
					
						
							|  |  |  | struct KeyIndex { | 
					
						
							|  |  |  |     KeyType type; | 
					
						
							|  |  |  |     u64 field1; | 
					
						
							|  |  |  |     u64 field2; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  |     std::string DebugInfo() const { | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |         u8 key_size = 16; | 
					
						
							| 
									
										
										
										
											2018-07-29 19:00:09 -04:00
										 |  |  |         if constexpr (std::is_same_v<KeyType, S256KeyType>) | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |             key_size = 32; | 
					
						
							|  |  |  |         return fmt::format("key_size={:02X}, key={:02X}, field1={:016X}, field2={:016X}", key_size, | 
					
						
							|  |  |  |                            static_cast<u8>(type), field1, field2); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // The following two (== and hash) are so KeyIndex can be a key in unordered_map
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename KeyType> | 
					
						
							|  |  |  | bool operator==(const KeyIndex<KeyType>& lhs, const KeyIndex<KeyType>& rhs) { | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  |     return std::tie(lhs.type, lhs.field1, lhs.field2) == std::tie(rhs.type, rhs.field1, rhs.field2); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  | template <typename KeyType> | 
					
						
							|  |  |  | bool operator!=(const KeyIndex<KeyType>& lhs, const KeyIndex<KeyType>& rhs) { | 
					
						
							|  |  |  |     return !operator==(lhs, rhs); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Core::Crypto
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace std { | 
					
						
							|  |  |  | template <typename KeyType> | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  | struct hash<Core::Crypto::KeyIndex<KeyType>> { | 
					
						
							|  |  |  |     size_t operator()(const Core::Crypto::KeyIndex<KeyType>& k) const { | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |         using std::hash; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return ((hash<u64>()(static_cast<u64>(k.type)) ^ (hash<u64>()(k.field1) << 1)) >> 1) ^ | 
					
						
							|  |  |  |                (hash<u64>()(k.field2) << 1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | } // namespace std
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  | namespace Core::Crypto { | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | std::array<u8, 0x10> operator"" _array16(const char* str, size_t len); | 
					
						
							|  |  |  | std::array<u8, 0x20> operator"" _array32(const char* str, size_t len); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  | class KeyManager { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     KeyManager(); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  |     bool HasKey(S128KeyType id, u64 field1 = 0, u64 field2 = 0) const; | 
					
						
							|  |  |  |     bool HasKey(S256KeyType id, u64 field1 = 0, u64 field2 = 0) const; | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  |     Key128 GetKey(S128KeyType id, u64 field1 = 0, u64 field2 = 0) const; | 
					
						
							|  |  |  |     Key256 GetKey(S256KeyType id, u64 field1 = 0, u64 field2 = 0) const; | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     void SetKey(S128KeyType id, Key128 key, u64 field1 = 0, u64 field2 = 0); | 
					
						
							|  |  |  |     void SetKey(S256KeyType id, Key256 key, u64 field1 = 0, u64 field2 = 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-30 12:46:23 -04:00
										 |  |  |     static bool KeyFileExists(bool title); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | private: | 
					
						
							|  |  |  |     std::unordered_map<KeyIndex<S128KeyType>, Key128> s128_keys; | 
					
						
							|  |  |  |     std::unordered_map<KeyIndex<S256KeyType>, Key256> s256_keys; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  |     bool dev_mode; | 
					
						
							|  |  |  |     void LoadFromFile(std::string_view filename, bool is_title_keys); | 
					
						
							| 
									
										
										
										
											2018-07-29 18:42:04 -04:00
										 |  |  |     void AttemptLoadKeyFile(std::string_view dir1, std::string_view dir2, std::string_view filename, | 
					
						
							|  |  |  |                             bool title); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-29 21:00:50 -04:00
										 |  |  |     static const std::unordered_map<std::string, KeyIndex<S128KeyType>> s128_file_id; | 
					
						
							|  |  |  |     static const std::unordered_map<std::string, KeyIndex<S256KeyType>> s256_file_id; | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  | } // namespace Core::Crypto
 |