mirror of
https://github.com/jakcron/nstool
synced 2024-11-15 02:06:40 +00:00
Update NCA code
This commit is contained in:
parent
e293b322c0
commit
ed2e0f462d
3 changed files with 201 additions and 74 deletions
|
@ -34,18 +34,11 @@ namespace nx
|
||||||
TYPE_DATA,
|
TYPE_DATA,
|
||||||
};
|
};
|
||||||
|
|
||||||
enum EncryptionType
|
enum KeyBankIndex
|
||||||
{
|
{
|
||||||
CRYPT_AUTO,
|
KEY_AESXTS_0,
|
||||||
CRYPT_NONE,
|
KEY_AESXTS_1,
|
||||||
CRYPT_AESCTR = 3
|
KEY_AESCTR,
|
||||||
};
|
|
||||||
|
|
||||||
enum EncryptionKeyIndex
|
|
||||||
{
|
|
||||||
KEY_UNUSED_0,
|
|
||||||
KEY_UNUSED_1,
|
|
||||||
KEY_DEFAULT,
|
|
||||||
KEY_UNUSED_3,
|
KEY_UNUSED_3,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -53,14 +46,12 @@ namespace nx
|
||||||
{
|
{
|
||||||
u64 offset;
|
u64 offset;
|
||||||
u64 size;
|
u64 size;
|
||||||
EncryptionType enc_type;
|
|
||||||
crypto::sha::sSha256Hash hash;
|
crypto::sha::sSha256Hash hash;
|
||||||
|
|
||||||
const sSection& operator=(const sSection& other)
|
const sSection& operator=(const sSection& other)
|
||||||
{
|
{
|
||||||
offset = other.offset;
|
offset = other.offset;
|
||||||
size = other.size;
|
size = other.size;
|
||||||
enc_type = other.enc_type;
|
|
||||||
hash = other.hash;
|
hash = other.hash;
|
||||||
|
|
||||||
return *this;
|
return *this;
|
||||||
|
@ -70,7 +61,6 @@ namespace nx
|
||||||
{
|
{
|
||||||
return (offset == other.offset) \
|
return (offset == other.offset) \
|
||||||
&& (size == other.size) \
|
&& (size == other.size) \
|
||||||
&& (enc_type == other.enc_type) \
|
|
||||||
&& (hash == other.hash);
|
&& (hash == other.hash);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -100,14 +90,16 @@ namespace nx
|
||||||
|
|
||||||
// variables
|
// variables
|
||||||
void clear();
|
void clear();
|
||||||
|
FormatVersion getFormatVersion() const;
|
||||||
|
void setFormatVersion(FormatVersion ver);
|
||||||
DistributionType getDistributionType() const;
|
DistributionType getDistributionType() const;
|
||||||
void setDistributionType(DistributionType type);
|
void setDistributionType(DistributionType type);
|
||||||
ContentType getContentType() const;
|
ContentType getContentType() const;
|
||||||
void setContentType(ContentType type);
|
void setContentType(ContentType type);
|
||||||
EncryptionType getEncryptionType() const;
|
byte_t getCryptoType() const;
|
||||||
void setEncryptionType(EncryptionType type);
|
void setCryptoType(byte_t type);
|
||||||
EncryptionKeyIndex getKeyIndex() const;
|
byte_t getKaekIndex() const;
|
||||||
void setKeyIndex(EncryptionKeyIndex index);
|
void setKaekIndex(byte_t index);
|
||||||
u64 getNcaSize() const;
|
u64 getNcaSize() const;
|
||||||
void setNcaSize(u64 size);
|
void setNcaSize(u64 size);
|
||||||
u64 getProgramId() const;
|
u64 getProgramId() const;
|
||||||
|
@ -123,7 +115,8 @@ namespace nx
|
||||||
|
|
||||||
private:
|
private:
|
||||||
const std::string kModuleName = "NCA_HEADER";
|
const std::string kModuleName = "NCA_HEADER";
|
||||||
const std::string kNcaSig = "NCA2";
|
const std::string kNca2Sig = "NCA2";
|
||||||
|
const std::string kNca3Sig = "NCA3";
|
||||||
static const size_t kSectionNum = 4;
|
static const size_t kSectionNum = 4;
|
||||||
static const size_t kAesKeyNum = 4;
|
static const size_t kAesKeyNum = 4;
|
||||||
static const u32 kDefaultSdkAddonVersion = 721920;
|
static const u32 kDefaultSdkAddonVersion = 721920;
|
||||||
|
@ -142,13 +135,15 @@ namespace nx
|
||||||
char signature[4];
|
char signature[4];
|
||||||
byte_t distribution_type;
|
byte_t distribution_type;
|
||||||
byte_t content_type;
|
byte_t content_type;
|
||||||
byte_t key_generation;
|
byte_t crypto_type; // KeyGeneration
|
||||||
byte_t key_area_encryption_key_index;
|
byte_t key_area_encryption_key_index;
|
||||||
le_uint64_t nca_size;
|
le_uint64_t nca_size;
|
||||||
le_uint64_t program_id;
|
le_uint64_t program_id;
|
||||||
le_uint32_t content_index;
|
le_uint32_t content_index;
|
||||||
le_uint32_t sdk_addon_version;
|
le_uint32_t sdk_addon_version;
|
||||||
byte_t reserved_2[0x20];
|
byte_t crypto_type_2;
|
||||||
|
byte_t reserved_2[0xf];
|
||||||
|
byte_t rights_id[0x10];
|
||||||
struct sNcaSection
|
struct sNcaSection
|
||||||
{
|
{
|
||||||
le_uint32_t start; // block units
|
le_uint32_t start; // block units
|
||||||
|
@ -166,10 +161,11 @@ namespace nx
|
||||||
fnd::MemoryBlob mBinaryBlob;
|
fnd::MemoryBlob mBinaryBlob;
|
||||||
|
|
||||||
// data
|
// data
|
||||||
|
FormatVersion mFormatVersion;
|
||||||
DistributionType mDistributionType;
|
DistributionType mDistributionType;
|
||||||
ContentType mContentType;
|
ContentType mContentType;
|
||||||
EncryptionType mEncryptionType;
|
byte_t mCryptoType;
|
||||||
EncryptionKeyIndex mKeyIndex;
|
byte_t mKaekIndex;
|
||||||
u64 mNcaSize;
|
u64 mNcaSize;
|
||||||
u64 mProgramId;
|
u64 mProgramId;
|
||||||
u32 mContentIndex;
|
u32 mContentIndex;
|
||||||
|
|
|
@ -8,15 +8,27 @@ void NcaHeader::exportBinary()
|
||||||
mBinaryBlob.alloc(sizeof(sNcaHeader));
|
mBinaryBlob.alloc(sizeof(sNcaHeader));
|
||||||
sNcaHeader* hdr = (sNcaHeader*)mBinaryBlob.getBytes();
|
sNcaHeader* hdr = (sNcaHeader*)mBinaryBlob.getBytes();
|
||||||
|
|
||||||
strncpy(hdr->signature, kNcaSig.c_str(), 4);
|
|
||||||
|
switch(mFormatVersion)
|
||||||
|
{
|
||||||
|
case (NCA2_FORMAT):
|
||||||
|
strncpy(hdr->signature, kNca2Sig.c_str(), 4);
|
||||||
|
break;
|
||||||
|
case (NCA3_FORMAT):
|
||||||
|
strncpy(hdr->signature, kNca3Sig.c_str(), 4);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw fnd::Exception(kModuleName, "Unsupported format version");
|
||||||
|
}
|
||||||
hdr->distribution_type = mDistributionType;
|
hdr->distribution_type = mDistributionType;
|
||||||
hdr->content_type = mContentType;
|
hdr->content_type = mContentType;
|
||||||
hdr->key_generation = mEncryptionType;
|
hdr->crypto_type = mCryptoType;
|
||||||
hdr->key_area_encryption_key_index = mKeyIndex;
|
hdr->key_area_encryption_key_index = mKaekIndex;
|
||||||
hdr->nca_size = mNcaSize;
|
hdr->nca_size = mNcaSize;
|
||||||
hdr->program_id = mProgramId;
|
hdr->program_id = mProgramId;
|
||||||
hdr->content_index = mContentIndex;
|
hdr->content_index = mContentIndex;
|
||||||
hdr->sdk_addon_version = mSdkAddonVersion;
|
hdr->sdk_addon_version = mSdkAddonVersion;
|
||||||
|
hdr->crypto_type_2 = 0;
|
||||||
|
|
||||||
// TODO: properly reconstruct NCA layout? atm in hands of user
|
// TODO: properly reconstruct NCA layout? atm in hands of user
|
||||||
|
|
||||||
|
@ -51,15 +63,23 @@ void NcaHeader::importBinary(const u8 * bytes, size_t len)
|
||||||
|
|
||||||
sNcaHeader* hdr = (sNcaHeader*)mBinaryBlob.getBytes();
|
sNcaHeader* hdr = (sNcaHeader*)mBinaryBlob.getBytes();
|
||||||
|
|
||||||
if (memcmp(hdr->signature, kNcaSig.c_str(), 4) != 0)
|
if (memcmp(hdr->signature, kNca2Sig.c_str(), 4) == 0)
|
||||||
|
{
|
||||||
|
mFormatVersion = NCA2_FORMAT;
|
||||||
|
}
|
||||||
|
else if (memcmp(hdr->signature, kNca3Sig.c_str(), 4) == 0)
|
||||||
|
{
|
||||||
|
mFormatVersion = NCA3_FORMAT;
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
throw fnd::Exception(kModuleName, "NCA header corrupt");
|
throw fnd::Exception(kModuleName, "NCA header corrupt");
|
||||||
}
|
}
|
||||||
|
|
||||||
mDistributionType = (DistributionType)hdr->distribution_type;
|
mDistributionType = (DistributionType)hdr->distribution_type;
|
||||||
mContentType = (ContentType)hdr->content_type;
|
mContentType = (ContentType)hdr->content_type;
|
||||||
mEncryptionType = (EncryptionType)hdr->key_generation;
|
mCryptoType = MAX(hdr->crypto_type, hdr->crypto_type_2);
|
||||||
mKeyIndex = (EncryptionKeyIndex)hdr->key_area_encryption_key_index;
|
mKaekIndex = hdr->key_area_encryption_key_index;
|
||||||
mNcaSize = *hdr->nca_size;
|
mNcaSize = *hdr->nca_size;
|
||||||
mProgramId = *hdr->program_id;
|
mProgramId = *hdr->program_id;
|
||||||
mContentIndex = *hdr->content_index;
|
mContentIndex = *hdr->content_index;
|
||||||
|
@ -73,21 +93,8 @@ void NcaHeader::importBinary(const u8 * bytes, size_t len)
|
||||||
// skip sections that don't exist
|
// skip sections that don't exist
|
||||||
if (*hdr->section[section].start == 0 && *hdr->section[section].end == 0) continue;
|
if (*hdr->section[section].start == 0 && *hdr->section[section].end == 0) continue;
|
||||||
|
|
||||||
EncryptionType encType = mEncryptionType;
|
|
||||||
if (encType == CRYPT_AUTO)
|
|
||||||
{
|
|
||||||
if (mContentType == TYPE_PROGRAM && section == SECTION_LOGO)
|
|
||||||
{
|
|
||||||
encType = CRYPT_NONE;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
encType = CRYPT_AESCTR;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// add high level struct
|
// add high level struct
|
||||||
mSections.addElement({ blockNumToSize(*hdr->section[section].start), blockNumToSize(hdr->section[section].end.get() - hdr->section[section].start.get()), encType, hdr->section_hash[section] });
|
mSections.addElement({ blockNumToSize(*hdr->section[section].start), blockNumToSize(hdr->section[section].end.get() - hdr->section[section].start.get()), hdr->section_hash[section] });
|
||||||
}
|
}
|
||||||
|
|
||||||
for (size_t i = 0; i < kAesKeyNum; i++)
|
for (size_t i = 0; i < kAesKeyNum; i++)
|
||||||
|
@ -98,10 +105,11 @@ void NcaHeader::importBinary(const u8 * bytes, size_t len)
|
||||||
|
|
||||||
void nx::NcaHeader::clear()
|
void nx::NcaHeader::clear()
|
||||||
{
|
{
|
||||||
|
mFormatVersion = NCA3_FORMAT;
|
||||||
mDistributionType = DIST_DOWNLOAD;
|
mDistributionType = DIST_DOWNLOAD;
|
||||||
mContentType = TYPE_PROGRAM;
|
mContentType = TYPE_PROGRAM;
|
||||||
mEncryptionType = CRYPT_AUTO;
|
mCryptoType = 0;
|
||||||
mKeyIndex = KEY_DEFAULT;
|
mKaekIndex = 0;
|
||||||
mNcaSize = 0;
|
mNcaSize = 0;
|
||||||
mProgramId = 0;
|
mProgramId = 0;
|
||||||
mContentIndex = 0;
|
mContentIndex = 0;
|
||||||
|
@ -110,6 +118,16 @@ void nx::NcaHeader::clear()
|
||||||
mEncAesKeys.clear();
|
mEncAesKeys.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
nx::NcaHeader::FormatVersion nx::NcaHeader::getFormatVersion() const
|
||||||
|
{
|
||||||
|
return mFormatVersion;
|
||||||
|
}
|
||||||
|
|
||||||
|
void nx::NcaHeader::setFormatVersion(FormatVersion version)
|
||||||
|
{
|
||||||
|
mFormatVersion = version;
|
||||||
|
}
|
||||||
|
|
||||||
nx::NcaHeader::DistributionType nx::NcaHeader::getDistributionType() const
|
nx::NcaHeader::DistributionType nx::NcaHeader::getDistributionType() const
|
||||||
{
|
{
|
||||||
return mDistributionType;
|
return mDistributionType;
|
||||||
|
@ -130,24 +148,24 @@ void nx::NcaHeader::setContentType(ContentType type)
|
||||||
mContentType = type;
|
mContentType = type;
|
||||||
}
|
}
|
||||||
|
|
||||||
nx::NcaHeader::EncryptionType nx::NcaHeader::getEncryptionType() const
|
byte_t nx::NcaHeader::getCryptoType() const
|
||||||
{
|
{
|
||||||
return mEncryptionType;
|
return mCryptoType;
|
||||||
}
|
}
|
||||||
|
|
||||||
void nx::NcaHeader::setEncryptionType(EncryptionType type)
|
void nx::NcaHeader::setCryptoType(byte_t type)
|
||||||
{
|
{
|
||||||
mEncryptionType = type;
|
mCryptoType = type;
|
||||||
}
|
}
|
||||||
|
|
||||||
nx::NcaHeader::EncryptionKeyIndex nx::NcaHeader::getKeyIndex() const
|
byte_t nx::NcaHeader::getKaekIndex() const
|
||||||
{
|
{
|
||||||
return mKeyIndex;
|
return mKaekIndex;
|
||||||
}
|
}
|
||||||
|
|
||||||
void nx::NcaHeader::setKeyIndex(EncryptionKeyIndex index)
|
void nx::NcaHeader::setKaekIndex(byte_t index)
|
||||||
{
|
{
|
||||||
mKeyIndex = index;
|
mKaekIndex = index;
|
||||||
}
|
}
|
||||||
|
|
||||||
u64 NcaHeader::getNcaSize() const
|
u64 NcaHeader::getNcaSize() const
|
||||||
|
@ -233,8 +251,8 @@ bool NcaHeader::isEqual(const NcaHeader & other) const
|
||||||
{
|
{
|
||||||
return (mDistributionType == other.mDistributionType) \
|
return (mDistributionType == other.mDistributionType) \
|
||||||
&& (mContentType == other.mContentType) \
|
&& (mContentType == other.mContentType) \
|
||||||
&& (mEncryptionType == other.mEncryptionType) \
|
&& (mCryptoType == other.mCryptoType) \
|
||||||
&& (mKeyIndex == other.mKeyIndex) \
|
&& (mKaekIndex == other.mKaekIndex) \
|
||||||
&& (mNcaSize == other.mNcaSize) \
|
&& (mNcaSize == other.mNcaSize) \
|
||||||
&& (mProgramId == other.mProgramId) \
|
&& (mProgramId == other.mProgramId) \
|
||||||
&& (mContentIndex == other.mContentIndex) \
|
&& (mContentIndex == other.mContentIndex) \
|
||||||
|
@ -254,8 +272,8 @@ void NcaHeader::copyFrom(const NcaHeader & other)
|
||||||
mBinaryBlob.clear();
|
mBinaryBlob.clear();
|
||||||
mDistributionType = other.mDistributionType;
|
mDistributionType = other.mDistributionType;
|
||||||
mContentType = other.mContentType;
|
mContentType = other.mContentType;
|
||||||
mEncryptionType = other.mEncryptionType;
|
mCryptoType = other.mCryptoType;
|
||||||
mKeyIndex = other.mKeyIndex;
|
mKaekIndex = other.mKaekIndex;
|
||||||
mNcaSize = other.mNcaSize;
|
mNcaSize = other.mNcaSize;
|
||||||
mProgramId = other.mProgramId;
|
mProgramId = other.mProgramId;
|
||||||
mContentIndex = other.mContentIndex;
|
mContentIndex = other.mContentIndex;
|
||||||
|
|
|
@ -13,6 +13,7 @@
|
||||||
|
|
||||||
const size_t kNcaSectorSize = nx::NcaHeader::kBlockSize;
|
const size_t kNcaSectorSize = nx::NcaHeader::kBlockSize;
|
||||||
|
|
||||||
|
inline size_t sectorToOffset(size_t sector_index) { return sector_index * kNcaSectorSize; }
|
||||||
|
|
||||||
void initNcaCtr(u8 ctr[crypto::aes::kAesBlockSize], u32 generation)
|
void initNcaCtr(u8 ctr[crypto::aes::kAesBlockSize], u32 generation)
|
||||||
{
|
{
|
||||||
|
@ -39,6 +40,25 @@ void xorData(const u8* a, const u8* b, u8* out, size_t len)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void decryptNcaHeader(byte_t header[0xc00], const u8* key1, const u8* key2)
|
||||||
|
{
|
||||||
|
byte_t tweak[crypto::aes::kAesBlockSize];
|
||||||
|
|
||||||
|
// decrypt main header
|
||||||
|
byte_t raw_hdr[kNcaSectorSize];
|
||||||
|
nx::NcaHeader hdr;
|
||||||
|
crypto::aes::AesXtsMakeTweak(tweak, 1);
|
||||||
|
crypto::aes::AesXtsDecryptSector(header + sectorToOffset(1), kNcaSectorSize, key1, key2, tweak, raw_hdr);
|
||||||
|
hdr.importBinary(raw_hdr, kNcaSectorSize);
|
||||||
|
|
||||||
|
// decrypt whole header
|
||||||
|
for (size_t i = 0; i < 6; i++)
|
||||||
|
{
|
||||||
|
crypto::aes::AesXtsMakeTweak(tweak, (i > 1 && hdr.getFormatVersion() == nx::NcaHeader::NCA2_FORMAT)? 0 : i);
|
||||||
|
crypto::aes::AesXtsDecryptSector(header + sectorToOffset(i), kNcaSectorSize, key1, key2, tweak, header + sectorToOffset(i));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void decryptNcaSectorXts(const fnd::MemoryBlob& nca, u8 out[kNcaSectorSize], size_t sector, const u8* key1, const u8* key2)
|
void decryptNcaSectorXts(const fnd::MemoryBlob& nca, u8 out[kNcaSectorSize], size_t sector, const u8* key1, const u8* key2)
|
||||||
{
|
{
|
||||||
u8 tweak[crypto::aes::kAesBlockSize];
|
u8 tweak[crypto::aes::kAesBlockSize];
|
||||||
|
@ -97,6 +117,12 @@ void dumpHxdStyleSector(u8* out, size_t len)
|
||||||
*/
|
*/
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::string kFormatVersionStr[]
|
||||||
|
{
|
||||||
|
"NCA2",
|
||||||
|
"NCA3"
|
||||||
|
};
|
||||||
|
|
||||||
std::string kDistributionTypeStr[]
|
std::string kDistributionTypeStr[]
|
||||||
{
|
{
|
||||||
"Download",
|
"Download",
|
||||||
|
@ -116,10 +142,54 @@ std::string kEncryptionTypeStr[]
|
||||||
{
|
{
|
||||||
"Auto",
|
"Auto",
|
||||||
"None",
|
"None",
|
||||||
"UNKNOWN_2",
|
"AesXts",
|
||||||
"AesCtr"
|
"AesCtr",
|
||||||
|
"BKTR"
|
||||||
};
|
};
|
||||||
|
|
||||||
|
std::string kHashTypeStr[]
|
||||||
|
{
|
||||||
|
"Auto",
|
||||||
|
"UNKNOWN_1",
|
||||||
|
"HierarchicalSha256",
|
||||||
|
"HierarchicalIntegrity"
|
||||||
|
};
|
||||||
|
|
||||||
|
std::string kFormatTypeStr[]
|
||||||
|
{
|
||||||
|
"RomFs",
|
||||||
|
"PartitionFs"
|
||||||
|
};
|
||||||
|
|
||||||
|
std::string kKaekIndexStr[]
|
||||||
|
{
|
||||||
|
"Application",
|
||||||
|
"Ocean",
|
||||||
|
"System"
|
||||||
|
};
|
||||||
|
|
||||||
|
enum EncryptionType
|
||||||
|
{
|
||||||
|
CRYPT_AUTO,
|
||||||
|
CRYPT_NONE,
|
||||||
|
CRYPT_AESXTS,
|
||||||
|
CRYPT_AESCTR,
|
||||||
|
CRYPT_BKTR
|
||||||
|
};
|
||||||
|
|
||||||
|
#pragma pack(push,1)
|
||||||
|
struct sNcaFsHeader
|
||||||
|
{
|
||||||
|
le_uint16_t version; // usually 0x0002
|
||||||
|
byte_t format_type; // RomFs(0x00), PartitionFs(0x01)
|
||||||
|
byte_t hash_type; // HashTypeAuto(0x00), HashTypeHierarchicalSha256(0x02), HashTypeHierarchicalIntegrity(0x03).RomFs uses (0x03) this is forced, PartitionFs uses (0x02).
|
||||||
|
byte_t encryption_type; // EncryptionTypeAuto(0x00), EncryptionTypeNone(0x01), EncryptionTypeAesCtr(0x03)
|
||||||
|
byte_t reserved[3];
|
||||||
|
};
|
||||||
|
#pragma pack(pop)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
int main(int argc, char** argv)
|
int main(int argc, char** argv)
|
||||||
{
|
{
|
||||||
if (argc < 2)
|
if (argc < 2)
|
||||||
|
@ -133,25 +203,45 @@ int main(int argc, char** argv)
|
||||||
fnd::MemoryBlob nca;
|
fnd::MemoryBlob nca;
|
||||||
fnd::io::readFile(argv[1], nca);
|
fnd::io::readFile(argv[1], nca);
|
||||||
|
|
||||||
u8 sector[kNcaSectorSize];
|
decryptNcaHeader(nca.getBytes(), crypto::aes::nx::dev::nca_header_key[0], crypto::aes::nx::dev::nca_header_key[1]);
|
||||||
|
//dumpHxdStyleSector(nca.getBytes(), 0xc00);
|
||||||
|
|
||||||
|
//u8 sector[kNcaSectorSize];
|
||||||
|
|
||||||
// nca test
|
// nca test
|
||||||
if (argc == 2 || argc == 3)
|
if (argc == 2 || argc == 3)
|
||||||
{
|
{
|
||||||
decryptNcaSectorXts(nca, sector, 1, crypto::aes::nx::dev::nca_header_key[0], crypto::aes::nx::dev::nca_header_key[1]);
|
//decryptNcaSectorXts(nca, sector, 1, crypto::aes::nx::dev::nca_header_key[0], crypto::aes::nx::dev::nca_header_key[1]);
|
||||||
|
|
||||||
nx::NcaHeader hdr;
|
nx::NcaHeader hdr;
|
||||||
hdr.importBinary(sector, kNcaSectorSize);
|
hdr.importBinary(nca.getBytes() + sectorToOffset(1), kNcaSectorSize);
|
||||||
|
|
||||||
printf("[NCA Header]\n");
|
printf("[NCA Header]\n");
|
||||||
|
printf(" Format Type: %s\n", kFormatVersionStr[hdr.getFormatVersion()].c_str());
|
||||||
printf(" Dist. Type: %s\n", kDistributionTypeStr[hdr.getDistributionType()].c_str());
|
printf(" Dist. Type: %s\n", kDistributionTypeStr[hdr.getDistributionType()].c_str());
|
||||||
printf(" Type: %s\n", kContentTypeStr[hdr.getContentType()].c_str());
|
printf(" Type: %s\n", kContentTypeStr[hdr.getContentType()].c_str());
|
||||||
printf(" Enc. Type: %s\n", kEncryptionTypeStr[hdr.getEncryptionType()].c_str());
|
printf(" Crypto Type: %d\n", hdr.getCryptoType());
|
||||||
printf(" KeyIndex: %d\n", hdr.getKeyIndex());
|
printf(" Kaek Index: %s (%d)\n", kKaekIndexStr[hdr.getKaekIndex()].c_str(), hdr.getKaekIndex());
|
||||||
printf(" Size: 0x%" PRIx64 "\n", hdr.getNcaSize());
|
printf(" Size: 0x%" PRIx64 "\n", hdr.getNcaSize());
|
||||||
printf(" ProgID: 0x%016" PRIx64 "\n", hdr.getProgramId());
|
printf(" ProgID: 0x%016" PRIx64 "\n", hdr.getProgramId());
|
||||||
printf(" Content. Idx: %" PRIu32 "\n", hdr.getContentIndex());
|
printf(" Content. Idx: %" PRIu32 "\n", hdr.getContentIndex());
|
||||||
printf(" SdkAddon Ver.: v%" PRIu32 "\n", hdr.getSdkAddonVersion());
|
uint32_t ver = hdr.getSdkAddonVersion();
|
||||||
|
printf(" SdkAddon Ver.: v%d.%d.%d.%d (v%" PRIu32 ")\n", (ver>>24 & 0xff),(ver>>16 & 0xff),(ver>>8 & 0xff),(ver>>0 & 0xff), ver);
|
||||||
|
printf(" Encrypted Key Area:\n");
|
||||||
|
for (size_t i = 0; i < hdr.getEncAesKeys().getSize(); i++)
|
||||||
|
{
|
||||||
|
printf(" %lu: ", i);
|
||||||
|
hexDump(hdr.getEncAesKeys()[i].key, crypto::aes::kAes128KeySize);
|
||||||
|
printf("\n");
|
||||||
|
/*
|
||||||
|
byte_t key[crypto::aes::kAes128KeySize];
|
||||||
|
crypto::aes::AesEcbDecrypt(hdr.getEncAesKeys()[i].key, crypto::aes::kAes128KeySize, crypto::aes::nx::dev::key_area_encryption_key_0, key);
|
||||||
|
printf(" dec: ", i);
|
||||||
|
hexDump(key, crypto::aes::kAes128KeySize);
|
||||||
|
printf("\n");
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
|
||||||
printf(" Sections:\n");
|
printf(" Sections:\n");
|
||||||
for (size_t i = 0; i < hdr.getSections().getSize(); i++)
|
for (size_t i = 0; i < hdr.getSections().getSize(); i++)
|
||||||
{
|
{
|
||||||
|
@ -161,19 +251,41 @@ int main(int argc, char** argv)
|
||||||
//printf(" End Blk: %" PRId32 "\n", section.end_blk);
|
//printf(" End Blk: %" PRId32 "\n", section.end_blk);
|
||||||
printf(" Offset: 0x%" PRIx64 "\n", section.offset);
|
printf(" Offset: 0x%" PRIx64 "\n", section.offset);
|
||||||
printf(" Size: 0x%" PRIx64 "\n", section.size);
|
printf(" Size: 0x%" PRIx64 "\n", section.size);
|
||||||
printf(" Enc. Type: %s\n", kEncryptionTypeStr[section.enc_type].c_str());
|
|
||||||
|
|
||||||
|
size_t sector_index = 1 + (hdr.getSections().getSize() - i);
|
||||||
|
|
||||||
|
byte_t hash[crypto::sha::kSha256HashLen];
|
||||||
|
crypto::sha::Sha256(nca.getBytes() + sectorToOffset(sector_index), kNcaSectorSize, hash);
|
||||||
|
if (section.hash.compare(hash) == false)
|
||||||
|
{
|
||||||
|
//throw fnd::Exception("ncatool", "NcaFsHeader has bad sha256 hash");
|
||||||
|
}
|
||||||
|
|
||||||
|
const sNcaFsHeader* fsHdr = (const sNcaFsHeader*)(nca.getBytes() + sectorToOffset(sector_index));
|
||||||
|
printf(" FsHeader:\n");
|
||||||
|
printf(" Version: 0x%d\n", fsHdr->version.get());
|
||||||
|
printf(" Format Type: %s\n", kFormatTypeStr[fsHdr->format_type].c_str());
|
||||||
|
printf(" Hash Type: %s\n", kHashTypeStr[fsHdr->hash_type].c_str());
|
||||||
|
printf(" Enc. Type: %s\n", kEncryptionTypeStr[fsHdr->encryption_type].c_str());
|
||||||
|
/*
|
||||||
printf(" Hash: ");
|
printf(" Hash: ");
|
||||||
hexDump(section.hash.bytes, crypto::sha::kSha256HashLen);
|
hexDump(section.hash.bytes, crypto::sha::kSha256HashLen);
|
||||||
printf("\n");
|
printf("\n");
|
||||||
}
|
byte_t hash[crypto::sha::kSha256HashLen];
|
||||||
printf(" Encrypted Body Keys:\n");
|
crypto::sha::Sha256(nca.getBytes() + sectorToOffset(sector_index), kNcaSectorSize, hash);
|
||||||
for (size_t i = 0; i < hdr.getEncAesKeys().getSize(); i++)
|
printf(" Hash: ");
|
||||||
{
|
hexDump(hash, crypto::sha::kSha256HashLen);
|
||||||
printf(" %lu: ", i);
|
|
||||||
hexDump(hdr.getEncAesKeys()[i].key, crypto::aes::kAes128KeySize);
|
|
||||||
printf("\n");
|
printf("\n");
|
||||||
|
*/
|
||||||
|
//dumpHxdStyleSector(nca.getBytes() + sectorToOffset(sector_index), 0x10);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef USE_OLD_CODE
|
||||||
if (argc == 3)
|
if (argc == 3)
|
||||||
{
|
{
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
|
@ -203,7 +315,8 @@ int main(int argc, char** argv)
|
||||||
dumpNcaSector(sect);
|
dumpNcaSector(sect);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
} catch (const fnd::Exception& e)
|
} catch (const fnd::Exception& e)
|
||||||
{
|
{
|
||||||
printf("%s\n",e.what());
|
printf("%s\n",e.what());
|
||||||
|
|
Loading…
Reference in a new issue