mirror of
https://github.com/jakcron/nstool.git
synced 2024-12-22 18:55:29 +00:00
[nstool] Change processing of NCA key area.
This commit is contained in:
parent
cd00be2221
commit
0b0d546605
|
@ -233,6 +233,27 @@ void NcaProcess::generateNcaBodyEncryptionKeys()
|
||||||
byte_t masterkey_rev = nx::NcaUtils::getMasterKeyRevisionFromKeyGeneration(mHdr.getKeyGeneration());
|
byte_t masterkey_rev = nx::NcaUtils::getMasterKeyRevisionFromKeyGeneration(mHdr.getKeyGeneration());
|
||||||
byte_t keak_index = mHdr.getKaekIndex();
|
byte_t keak_index = mHdr.getKaekIndex();
|
||||||
|
|
||||||
|
// process key area
|
||||||
|
sKeys::sKeyAreaKey keak;
|
||||||
|
for (size_t i = 0; i < nx::nca::kAesKeyNum; i++)
|
||||||
|
{
|
||||||
|
if (mHdr.getEncAesKeys()[i] != zero_aesctr_key)
|
||||||
|
{
|
||||||
|
keak.index = (byte_t)i;
|
||||||
|
keak.enc = mHdr.getEncAesKeys()[i];
|
||||||
|
if (i < 4 && mKeyset->nca.key_area_key[keak_index][masterkey_rev] != zero_aesctr_key)
|
||||||
|
{
|
||||||
|
keak.decrypted = true;
|
||||||
|
nx::AesKeygen::generateKey(keak.dec.key, keak.enc.key, mKeyset->nca.key_area_key[keak_index][masterkey_rev].key);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
keak.decrypted = false;
|
||||||
|
}
|
||||||
|
mBodyKeys.keak_list.addElement(keak);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// set flag to indicate that the keys are not available
|
// set flag to indicate that the keys are not available
|
||||||
mBodyKeys.aes_ctr.isSet = false;
|
mBodyKeys.aes_ctr.isSet = false;
|
||||||
mBodyKeys.aes_xts.isSet = false;
|
mBodyKeys.aes_xts.isSet = false;
|
||||||
|
@ -260,15 +281,31 @@ void NcaProcess::generateNcaBodyEncryptionKeys()
|
||||||
// otherwise decrypt key area
|
// otherwise decrypt key area
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// if the key_area_key is available
|
crypto::aes::sAes128Key keak_aesctr_key = zero_aesctr_key;
|
||||||
if (mKeyset->nca.key_area_key[keak_index][masterkey_rev] != zero_aesctr_key)
|
crypto::aes::sAesXts128Key keak_aesxts_key = zero_aesxts_key;
|
||||||
|
for (size_t i = 0; i < mBodyKeys.keak_list.getSize(); i++)
|
||||||
{
|
{
|
||||||
nx::AesKeygen::generateKey(mBodyKeys.aes_ctr.var.key, mHdr.getEncAesKeys()[nx::nca::KEY_AESCTR].key, mKeyset->nca.key_area_key[keak_index][masterkey_rev].key);
|
if (mBodyKeys.keak_list[i].index == nx::nca::KEY_AESCTR && mBodyKeys.keak_list[i].decrypted)
|
||||||
mBodyKeys.aes_ctr.isSet = true;
|
{
|
||||||
|
keak_aesctr_key = mBodyKeys.keak_list[i].dec;
|
||||||
|
}
|
||||||
|
else if (mBodyKeys.keak_list[i].index == nx::nca::KEY_AESXTS_0 && mBodyKeys.keak_list[i].decrypted)
|
||||||
|
{
|
||||||
|
memcpy(keak_aesxts_key.key[0], mBodyKeys.keak_list[i].dec.key, sizeof(crypto::aes::sAes128Key));
|
||||||
|
}
|
||||||
|
else if (mBodyKeys.keak_list[i].index == nx::nca::KEY_AESXTS_1 && mBodyKeys.keak_list[i].decrypted)
|
||||||
|
{
|
||||||
|
memcpy(keak_aesxts_key.key[1], mBodyKeys.keak_list[i].dec.key, sizeof(crypto::aes::sAes128Key));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
nx::AesKeygen::generateKey(mBodyKeys.aes_xts.var.key[0], mHdr.getEncAesKeys()[nx::nca::KEY_AESXTS_0].key, mKeyset->nca.key_area_key[keak_index][masterkey_rev].key);
|
if (keak_aesctr_key != zero_aesctr_key)
|
||||||
nx::AesKeygen::generateKey(mBodyKeys.aes_xts.var.key[1], mHdr.getEncAesKeys()[nx::nca::KEY_AESXTS_1].key, mKeyset->nca.key_area_key[keak_index][masterkey_rev].key);
|
{
|
||||||
mBodyKeys.aes_xts.isSet = true;
|
mBodyKeys.aes_ctr = keak_aesctr_key;
|
||||||
|
}
|
||||||
|
if (keak_aesxts_key != zero_aesxts_key)
|
||||||
|
{
|
||||||
|
mBodyKeys.aes_xts = keak_aesxts_key;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -400,7 +437,6 @@ void NcaProcess::generatePartitionConfiguration()
|
||||||
}
|
}
|
||||||
catch (const fnd::Exception& e)
|
catch (const fnd::Exception& e)
|
||||||
{
|
{
|
||||||
printf("ugh\n");
|
|
||||||
info.fail_reason = std::string(e.error());
|
info.fail_reason = std::string(e.error());
|
||||||
if (info.reader != nullptr)
|
if (info.reader != nullptr)
|
||||||
delete info.reader;
|
delete info.reader;
|
||||||
|
@ -476,9 +512,6 @@ void NcaProcess::validateNcaSignatures()
|
||||||
|
|
||||||
void NcaProcess::displayHeader()
|
void NcaProcess::displayHeader()
|
||||||
{
|
{
|
||||||
crypto::aes::sAes128Key zero_key;
|
|
||||||
memset(zero_key.key, 0, sizeof(zero_key));
|
|
||||||
|
|
||||||
printf("[NCA Header]\n");
|
printf("[NCA Header]\n");
|
||||||
printf(" Format Type: %s\n", kFormatVersionStr[mHdr.getFormatVersion()].c_str());
|
printf(" Format Type: %s\n", kFormatVersionStr[mHdr.getFormatVersion()].c_str());
|
||||||
printf(" Dist. Type: %s\n", kDistributionTypeStr[mHdr.getDistributionType()].c_str());
|
printf(" Dist. Type: %s\n", kDistributionTypeStr[mHdr.getDistributionType()].c_str());
|
||||||
|
@ -493,16 +526,32 @@ void NcaProcess::displayHeader()
|
||||||
#undef _SPLIT_VER
|
#undef _SPLIT_VER
|
||||||
printf(" RightsId: ");
|
printf(" RightsId: ");
|
||||||
fnd::SimpleTextOutput::hexDump(mHdr.getRightsId(), nx::nca::kRightsIdLen);
|
fnd::SimpleTextOutput::hexDump(mHdr.getRightsId(), nx::nca::kRightsIdLen);
|
||||||
printf(" Key Area Keys: (Encrypted)\n");
|
|
||||||
for (size_t i = 0; i < mHdr.getEncAesKeys().getSize(); i++)
|
if (mBodyKeys.keak_list.getSize() > 0)
|
||||||
{
|
{
|
||||||
if (mHdr.getEncAesKeys()[i] != zero_key)
|
printf(" Key Area: \n");
|
||||||
|
printf(" <--------------------------------------------------------------------------->\n");
|
||||||
|
printf(" | IDX | ENCRYPTED KEY | DECRYPTED KEY |\n");
|
||||||
|
printf(" |-----|----------------------------------|----------------------------------|\n");
|
||||||
|
for (size_t i = 0; i < mBodyKeys.keak_list.getSize(); i++)
|
||||||
{
|
{
|
||||||
printf(" %2lu: ", i);
|
printf(" | %3lu | ", mBodyKeys.keak_list[i].index);
|
||||||
fnd::SimpleTextOutput::hexDump(mHdr.getEncAesKeys()[i].key, crypto::aes::kAes128KeySize);
|
|
||||||
|
for (size_t j = 0; j < 16; j++) printf("%02x", mBodyKeys.keak_list[i].enc.key[j]);
|
||||||
|
|
||||||
|
printf(" | ");
|
||||||
|
|
||||||
|
if (mBodyKeys.keak_list[i].decrypted)
|
||||||
|
for (size_t j = 0; j < 16; j++) printf("%02x", mBodyKeys.keak_list[i].dec.key[j]);
|
||||||
|
else
|
||||||
|
printf("<unable to decrypt> ");
|
||||||
|
|
||||||
|
printf(" |\n");
|
||||||
}
|
}
|
||||||
|
printf(" <--------------------------------------------------------------------------->\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
if (mBodyKeyList.getSize() > 0)
|
if (mBodyKeyList.getSize() > 0)
|
||||||
{
|
{
|
||||||
|
|
|
@ -55,6 +55,36 @@ private:
|
||||||
// crypto
|
// crypto
|
||||||
struct sKeys
|
struct sKeys
|
||||||
{
|
{
|
||||||
|
struct sKeyAreaKey
|
||||||
|
{
|
||||||
|
byte_t index;
|
||||||
|
bool decrypted;
|
||||||
|
crypto::aes::sAes128Key enc;
|
||||||
|
crypto::aes::sAes128Key dec;
|
||||||
|
|
||||||
|
void operator=(const sKeyAreaKey& other)
|
||||||
|
{
|
||||||
|
index = other.index;
|
||||||
|
decrypted = other.decrypted;
|
||||||
|
enc = other.enc;
|
||||||
|
dec = other.dec;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool operator==(const sKeyAreaKey& other) const
|
||||||
|
{
|
||||||
|
return (index == other.index) \
|
||||||
|
&& (decrypted == other.decrypted) \
|
||||||
|
&& (enc == other.enc) \
|
||||||
|
&& (dec == other.dec);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool operator!=(const sKeyAreaKey& other) const
|
||||||
|
{
|
||||||
|
return !(*this == other);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
fnd::List<sKeyAreaKey> keak_list;
|
||||||
|
|
||||||
sOptional<crypto::aes::sAes128Key> aes_ctr;
|
sOptional<crypto::aes::sAes128Key> aes_ctr;
|
||||||
sOptional<crypto::aes::sAesXts128Key> aes_xts;
|
sOptional<crypto::aes::sAesXts128Key> aes_xts;
|
||||||
} mBodyKeys;
|
} mBodyKeys;
|
||||||
|
|
Loading…
Reference in a new issue