2020-04-15 20:06:41 -04:00
/*
* Copyright ( c ) 2020 DarkMatterCore
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms and conditions of the GNU General Public License ,
* version 2 , as published by the Free Software Foundation .
*
* This program is distributed in the hope it will be useful , but WITHOUT
* ANY WARRANTY ; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE . See the GNU General Public License for
* more details .
*
* You should have received a copy of the GNU General Public License
* along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
2020-04-11 01:28:26 -04:00
# pragma once
# ifndef __NCA_H__
# define __NCA_H__
# include <switch.h>
2020-04-20 06:39:41 -04:00
# include "tik.h"
2020-04-11 01:28:26 -04:00
# define NCA_HEADER_LENGTH 0x400
2020-04-15 16:50:07 -04:00
# define NCA_FS_HEADER_LENGTH 0x200
# define NCA_FS_HEADER_COUNT 4
# define NCA_FULL_HEADER_LENGTH (NCA_HEADER_LENGTH + (NCA_FS_HEADER_LENGTH * NCA_FS_HEADER_COUNT))
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
# define NCA_NCA0_MAGIC 0x4E434130 /* "NCA0" */
# define NCA_NCA2_MAGIC 0x4E434132 /* "NCA2" */
# define NCA_NCA3_MAGIC 0x4E434133 /* "NCA3" */
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
# define NCA_IVFC_MAGIC 0x49564643 /* "IVFC" */
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
# define NCA_BKTR_MAGIC 0x424B5452 /* "BKTR" */
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
# define NCA_FS_ENTRY_BLOCK_SIZE 0x200
2020-04-20 06:39:41 -04:00
# define NCA_FS_ENTRY_BLOCK_OFFSET(x) ((u64)(x) * NCA_FS_ENTRY_BLOCK_SIZE)
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
# define NCA_AES_XTS_SECTOR_SIZE 0x200
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
# define NCA_IVFC_BLOCK_SIZE(x) (1 << (x))
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
typedef enum {
NcaDistributionType_Download = 0 ,
NcaDistributionType_GameCard = 1
} NcaDistributionType ;
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
typedef enum {
NcaContentType_Program = 0 ,
NcaContentType_Meta = 1 ,
NcaContentType_Control = 2 ,
NcaContentType_Manual = 3 ,
NcaContentType_Data = 4 ,
NcaContentType_PublicData = 5
} NcaContentType ;
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
typedef enum {
NcaKeyGenerationOld_100_230 = 0 ,
NcaKeyGenerationOld_300 = 2
} NcaKeyGenerationOld ;
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
typedef enum {
NcaKeyAreaEncryptionKeyIndex_Application = 0 ,
NcaKeyAreaEncryptionKeyIndex_Ocean = 1 ,
NcaKeyAreaEncryptionKeyIndex_System = 2
} NcaKeyAreaEncryptionKeyIndex ;
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
/// 'NcaKeyGeneration_Latest' will always point to the last known key generation value.
2020-04-11 01:28:26 -04:00
typedef enum {
2020-04-15 16:50:07 -04:00
NcaKeyGeneration_301_302 = 3 ,
NcaKeyGeneration_400_410 = 4 ,
NcaKeyGeneration_500_510 = 5 ,
NcaKeyGeneration_600_610 = 6 ,
NcaKeyGeneration_620 = 7 ,
NcaKeyGeneration_700_801 = 8 ,
NcaKeyGeneration_810_811 = 9 ,
NcaKeyGeneration_900_901 = 10 ,
NcaKeyGeneration_910_920 = 11 ,
NcaKeyGeneration_Latest = NcaKeyGeneration_910_920
} NcaKeyGeneration ;
typedef struct {
u32 start_block_offset ; ///< Expressed in NCA_FS_ENTRY_BLOCK_SIZE blocks.
u32 end_block_offset ; ///< Expressed in NCA_FS_ENTRY_BLOCK_SIZE blocks.
u8 enable_entry ;
u8 reserved [ 0x7 ] ;
} NcaFsEntry ;
2020-04-11 01:28:26 -04:00
typedef struct {
2020-04-15 16:50:07 -04:00
u8 hash [ SHA256_HASH_SIZE ] ;
} NcaFsHash ;
2020-04-11 01:28:26 -04:00
typedef struct {
2020-04-15 16:50:07 -04:00
u8 key [ 0x10 ] ;
2020-04-19 18:44:22 -04:00
} NcaKey ;
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
typedef enum {
NcaFsType_RomFs = 0 ,
NcaFsType_PartitionFs = 1
} NcaFsType ;
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
typedef enum {
NcaHashType_Auto = 0 ,
NcaHashType_None = 1 ,
NcaHashType_HierarchicalSha256 = 2 , ///< Used by NcaFsType_PartitionFs.
NcaHashType_HierarchicalIntegrity = 3 ///< Used by NcaFsType_RomFs.
} NcaHashType ;
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
typedef enum {
NcaEncryptionType_Auto = 0 ,
NcaEncryptionType_None = 1 ,
NcaEncryptionType_AesXts = 2 ,
NcaEncryptionType_AesCtr = 3 ,
2020-04-20 06:39:41 -04:00
NcaEncryptionType_AesCtrEx = 4 ,
NcaEncryptionType_Nca0 = 5 ///< Only used to represent NCA0 FS section crypto - not actually used as a possible value for this field.
2020-04-15 16:50:07 -04:00
} NcaEncryptionType ;
2020-04-11 01:28:26 -04:00
typedef struct {
2020-04-15 16:50:07 -04:00
u64 offset ;
u64 size ;
} NcaHierarchicalSha256LayerInfo ;
2020-04-11 01:28:26 -04:00
2020-04-20 06:39:41 -04:00
/// Used for NcaFsType_PartitionFs and NCA0 NcaFsType_RomFsRomFS.
2020-04-11 01:28:26 -04:00
typedef struct {
2020-04-15 16:50:07 -04:00
u8 master_hash [ SHA256_HASH_SIZE ] ;
u32 hash_block_size ;
u32 layer_count ;
NcaHierarchicalSha256LayerInfo hash_data_layer_info ;
NcaHierarchicalSha256LayerInfo hash_target_layer_info ;
} NcaHierarchicalSha256 ;
2020-04-11 01:28:26 -04:00
typedef struct {
u64 offset ;
u64 size ;
2020-04-15 16:50:07 -04:00
u32 block_size ; ///< Use NCA_IVFC_CALC_BLOCK_SIZE to calculate the actual block size using this value.
u8 reserved [ 0x4 ] ;
} NcaHierarchicalIntegrityLayerInfo ;
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
/// Used for NcaFsType_RomFs.
2020-04-11 01:28:26 -04:00
typedef struct {
2020-04-15 16:50:07 -04:00
u32 magic ; ///< "IVFC".
u32 version ;
u32 master_hash_size ;
u32 layer_count ;
NcaHierarchicalIntegrityLayerInfo hash_data_layer_info [ 5 ] ;
NcaHierarchicalIntegrityLayerInfo hash_target_layer_info ;
u8 signature_salt [ 0x20 ] ;
u8 master_hash [ 0x20 ] ;
} NcaHierarchicalIntegrity ;
2020-04-11 01:28:26 -04:00
typedef struct {
union {
struct {
2020-04-20 06:39:41 -04:00
///< Used if hash_type == NcaHashType_HierarchicalSha256 (NcaFsType_PartitionFs and NCA0 NcaFsType_RomFs).
2020-04-15 16:50:07 -04:00
NcaHierarchicalSha256 hierarchical_sha256 ;
u8 reserved_1 [ 0xB0 ] ;
2020-04-11 01:28:26 -04:00
} ;
struct {
2020-04-15 16:50:07 -04:00
///< Used if hash_type == NcaHashType_HierarchicalIntegrity (NcaFsType_RomFs).
NcaHierarchicalIntegrity hierarchical_integrity ;
u8 reserved_2 [ 0x18 ] ;
2020-04-11 01:28:26 -04:00
} ;
} ;
2020-04-15 16:50:07 -04:00
} NcaHashInfo ;
2020-04-11 01:28:26 -04:00
typedef struct {
2020-04-15 16:50:07 -04:00
u32 magic ; ///< "BKTR".
u32 bucket_count ;
u32 entry_count ;
u8 reserved [ 0x4 ] ;
} NcaBucketTreeHeader ;
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
/// Only used for NcaEncryptionType_AesCtrEx (PatchRomFs).
2020-04-11 01:28:26 -04:00
typedef struct {
2020-04-15 16:50:07 -04:00
u64 indirect_offset ;
u64 indirect_size ;
NcaBucketTreeHeader indirect_header ;
u64 aes_ctr_ex_offset ;
u64 aes_ctr_ex_size ;
NcaBucketTreeHeader aes_ctr_ex_header ;
} NcaPatchInfo ;
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
/// Format unknown.
2020-04-11 01:28:26 -04:00
typedef struct {
2020-04-15 16:50:07 -04:00
u8 unknown [ 0x30 ] ;
} NcaSparseInfo ;
2020-04-11 01:28:26 -04:00
typedef struct {
2020-04-15 16:50:07 -04:00
u16 version ;
u8 fs_type ; ///< NcaFsType.
u8 hash_type ; ///< NcaHashType.
u8 encryption_type ; ///< NcaEncryptionType.
u8 reserved_1 [ 0x3 ] ;
NcaHashInfo hash_info ;
NcaPatchInfo patch_info ;
union {
u8 section_ctr [ 0x8 ] ;
struct {
u32 generation ;
u32 secure_value ;
} ;
} ;
NcaSparseInfo sparse_info ;
u8 reserved_2 [ 0x88 ] ;
} NcaFsHeader ;
typedef struct {
u8 main_signature [ 0x100 ] ; ///< RSA-PSS signature over header with fixed key.
u8 acid_signature [ 0x100 ] ; ///< RSA-PSS signature over header with key in NPDM.
u32 magic ; ///< "NCA0" / "NCA2" / "NCA3".
u8 distribution_type ; ///< NcaDistributionType.
u8 content_type ; ///< NcaContentType.
u8 key_generation_old ; ///< NcaKeyGenerationOld.
u8 kaek_index ; ///< NcaKeyAreaEncryptionKeyIndex.
u64 content_size ;
u64 program_id ;
u32 content_index ;
union {
u32 sdk_addon_version ;
struct {
u8 sdk_addon_revision ;
u8 sdk_addon_micro ;
u8 sdk_addon_minor ;
u8 sdk_addon_major ;
} ;
} ;
u8 key_generation ; ///< NcaKeyGeneration.
u8 main_signature_key_generation ;
u8 reserved_1 [ 0xE ] ;
FsRightsId rights_id ; ///< Used for titlekey crypto.
NcaFsEntry fs_entries [ 4 ] ; ///< Start and end offsets for each NCA FS section.
NcaFsHash fs_hashes [ 4 ] ; ///< SHA-256 hashes calculated over each NCA FS section header.
2020-04-19 18:44:22 -04:00
NcaKey encrypted_keys [ 4 ] ; ///< Only the encrypted key at index #2 is used. The other three are zero filled before the key area is encrypted.
2020-04-15 16:50:07 -04:00
u8 reserved_2 [ 0xC0 ] ;
NcaFsHeader fs_headers [ 4 ] ; /// NCA FS section headers.
} NcaHeader ;
2020-04-11 01:28:26 -04:00
2020-04-20 06:39:41 -04:00
typedef enum {
NcaVersion_Nca0 = 0 ,
NcaVersion_Nca2 = 1 ,
NcaVersion_Nca3 = 2
} NcaVersion ;
2020-04-11 01:28:26 -04:00
2020-04-20 06:39:41 -04:00
typedef enum {
NcaSectionType_PartitionFs = 0 , ///< NcaFsType_PartitionFs + NcaHashType_HierarchicalSha256.
NcaSectionType_RomFs = 1 , ///< NcaFsType_RomFs + NcaHashType_HierarchicalIntegrity.
NcaSectionType_PatchRomFs = 2 , ///< NcaFsType_RomFs + NcaHashType_HierarchicalIntegrity + NcaEncryptionType_AesCtrEx.
NcaSectionType_Nca0RomFs = 3 , ///< NcaFsType_RomFs + NcaHashType_HierarchicalSha256 + NcaVersion_Nca0.
NcaSectionType_Invalid = 4
} NcaSectionType ;
2020-04-11 01:28:26 -04:00
typedef struct {
2020-04-20 06:39:41 -04:00
void * nca_ctx ; ///< NcaContext. Used to perform NCA reads.
u8 section_num ;
2020-04-11 01:28:26 -04:00
u64 offset ;
u64 size ;
2020-04-20 06:39:41 -04:00
u8 section_type ; ///< NcaSectionType.
u8 encryption_type ; ///< NcaEncryptionType.
NcaFsHeader * header ;
bool use_xts ;
Aes128CtrContext ctr_ctx ;
Aes128XtsContext xts_ctx ;
u8 ctr [ 0x10 ] ; ///< Used to update the AES context IV based on the desired offset.
2020-04-15 16:50:07 -04:00
} NcaFsContext ;
typedef struct {
u8 storage_id ; ///< NcmStorageId.
NcmContentStorage * ncm_storage ; ///< Pointer to a NcmContentStorage instance. Used to read NCA data.
2020-04-16 06:13:11 -04:00
u64 gamecard_offset ; ///< Used to read NCA data from a gamecard using a FsStorage instance when storage_id == NcmStorageId_GameCard.
2020-04-15 16:50:07 -04:00
NcmContentId id ; ///< Also used to read NCA data.
char id_str [ 0x21 ] ;
u8 hash [ 0x20 ] ;
char hash_str [ 0x41 ] ;
u8 format_version ; ///< NcaVersion.
u8 type ; ///< NcmContentType. Retrieved from NcmContentInfo.
u64 size ; ///< Retrieved from NcmContentInfo.
u8 key_generation ; ///< NcaKeyGenerationOld / NcaKeyGeneration. Retrieved from the decrypted header.
u8 id_offset ; ///< Retrieved from NcmContentInfo.
bool rights_id_available ;
bool dirty_header ;
2020-04-20 06:39:41 -04:00
NcaHeader header ;
2020-04-15 16:50:07 -04:00
NcaFsContext fs_contexts [ 4 ] ;
2020-04-20 06:39:41 -04:00
NcaKey decrypted_keys [ 4 ] ;
u8 titlekey [ 0x10 ] ;
2020-04-15 16:50:07 -04:00
} NcaContext ;
2020-04-11 01:28:26 -04:00
2020-04-20 06:39:41 -04:00
/// Reads raw encrypted data from a NCA using an input NCA context.
/// 'storage_id', 'id', 'id_str' and 'size' elements must have been filled beforehand (e.g. using ncaProcessContent()).
2020-04-19 18:44:22 -04:00
/// If 'storage_id' != NcmStorageId_GameCard, the 'ncm_storage' element should point to a valid NcmContentStorage instance.
2020-04-20 06:39:41 -04:00
/// If 'storage_id' == NcmStorageId_GameCard, the 'gamecard_offset' element should hold a value greater than zero.
bool ncaReadContent ( NcaContext * ctx , void * out , u64 read_size , u64 offset ) ;
2020-04-19 18:44:22 -04:00
2020-04-20 06:39:41 -04:00
/// Generates a valid NCA context.
/// 'hash', 'type', 'size' and 'id_offset' elements can be retrieved from a NcmContentInfo object.
/// If the NCA holds a populated Rights ID field, and if the Ticket object pointed to by 'tik' hasn't been filled, the ticket and titlekey will be retrieved.
/// 'hfs_partition_type' is only necessary if 'storage_id' == NcmStorageId_GameCard.
bool ncaProcessContent ( NcaContext * out , Ticket * tik , u8 storage_id , NcmContentStorage * ncm_storage , const NcmContentId * id , const u8 * hash , u8 type , const u8 * size , u8 id_offset , u8 hfs_partition_type ) ;
2020-04-19 18:44:22 -04:00
2020-04-15 16:50:07 -04:00
static inline void ncaConvertNcmContentSizeToU64 ( const u8 * size , u64 * out )
{
if ( ! size | | ! out ) return ;
* out = 0 ;
memcpy ( out , size , 6 ) ;
}
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
static inline void ncaConvertU64ToNcmContentSize ( const u64 * size , u8 * out )
{
if ( ! size | | ! out ) return ;
memcpy ( out , size , 6 ) ;
}
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
static inline u8 ncaGetKeyGenerationValue ( NcaContext * ctx )
{
if ( ! ctx ) return 0 ;
return ( ctx - > header . key_generation > ctx - > header . key_generation_old ? ctx - > header . key_generation : ctx - > header . key_generation_old ) ;
}
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
static inline void ncaSetDownloadDistributionType ( NcaContext * ctx )
{
if ( ! ctx | | ctx - > header . distribution_type = = NcaDistributionType_Download ) return ;
ctx - > header . distribution_type = NcaDistributionType_Download ;
ctx - > dirty_header = true ;
}
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
static inline bool ncaCheckRightsIdAvailability ( NcaContext * ctx )
{
if ( ! ctx ) return false ;
bool rights_id_available = false ;
for ( u8 i = 0 ; i < 0x10 ; i + + )
{
if ( ctx - > header . rights_id . c [ i ] ! = 0 )
{
rights_id_available = true ;
break ;
}
}
return rights_id_available ;
}
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
static inline void ncaWipeRightsId ( NcaContext * ctx )
{
if ( ! ctx ) return ;
memset ( & ( ctx - > header . rights_id ) , 0 , sizeof ( FsRightsId ) ) ;
ctx - > dirty_header = true ;
}
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
bool ncaDecryptKeyArea ( NcaContext * nca_ctx ) ;
bool ncaEncryptKeyArea ( NcaContext * nca_ctx ) ;
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
bool ncaDecryptHeader ( NcaContext * ctx ) ;
bool ncaEncryptHeader ( NcaContext * ctx ) ;
2020-04-11 01:28:26 -04:00
2020-04-15 16:50:07 -04:00
# endif /* __NCA_H__ */