mirror of
https://github.com/SciresM/hactool.git
synced 2024-11-27 06:20:35 +00:00
2567d516d5
-Not all pertitent information is printed out. -Extraction is pretty much supported, though. -TODO: CERT stuff, decryption, title ID fetching...
135 lines
5.5 KiB
C
135 lines
5.5 KiB
C
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include "rsa.h"
|
|
#include "sha.h"
|
|
#include "utils.h"
|
|
#include "types.h"
|
|
|
|
#define RSA_2048_BYTES 0x100
|
|
#define RSA_2048_BITS (RSA_2048_BYTES*8)
|
|
|
|
/* Perform an RSA-PSS verify operation on data, with signature and N. */
|
|
int rsa2048_pss_verify(const void *data, size_t len, const unsigned char *signature, const unsigned char *modulus) {
|
|
mbedtls_mpi signature_mpi;
|
|
mbedtls_mpi modulus_mpi;
|
|
mbedtls_mpi e_mpi;
|
|
mbedtls_mpi message_mpi;
|
|
|
|
mbedtls_mpi_init(&signature_mpi);
|
|
mbedtls_mpi_init(&modulus_mpi);
|
|
mbedtls_mpi_init(&e_mpi);
|
|
mbedtls_mpi_init(&message_mpi);
|
|
mbedtls_mpi_lset(&message_mpi, RSA_2048_BITS);
|
|
|
|
unsigned char m_buf[RSA_2048_BYTES];
|
|
unsigned char h_buf[0x24];
|
|
const unsigned char E[3] = {1, 0, 1};
|
|
|
|
mbedtls_mpi_read_binary(&e_mpi, E, 3);
|
|
mbedtls_mpi_read_binary(&signature_mpi, signature, RSA_2048_BYTES);
|
|
mbedtls_mpi_read_binary(&modulus_mpi, modulus, RSA_2048_BYTES);
|
|
mbedtls_mpi_exp_mod(&message_mpi, &signature_mpi, &e_mpi, &modulus_mpi, NULL);
|
|
|
|
if (mbedtls_mpi_write_binary(&message_mpi, m_buf, RSA_2048_BYTES) != 0) {
|
|
FATAL_ERROR("Failed to export exponentiated RSA message!");
|
|
}
|
|
|
|
mbedtls_mpi_free(&signature_mpi);
|
|
mbedtls_mpi_free(&modulus_mpi);
|
|
mbedtls_mpi_free(&e_mpi);
|
|
mbedtls_mpi_free(&message_mpi);
|
|
|
|
/* There's no automated PSS verification as far as I can tell. */
|
|
if (m_buf[RSA_2048_BYTES-1] != 0xBC) {
|
|
return false;
|
|
}
|
|
|
|
memset(h_buf, 0, 0x24);
|
|
memcpy(h_buf, m_buf + RSA_2048_BYTES - 0x20 - 0x1, 0x20);
|
|
|
|
/* Decrypt maskedDB. Should MGF1 be its own function? */
|
|
unsigned char seed = 0;
|
|
unsigned char mgf1_buf[0x20];
|
|
for (unsigned int ofs = 0; ofs < RSA_2048_BYTES - 0x20 - 1; ofs += 0x20) {
|
|
h_buf[0x23] = seed++;
|
|
sha256_hash_buffer(mgf1_buf, h_buf, 0x24);
|
|
for (unsigned int i = ofs; i < ofs + 0x20 && i < RSA_2048_BYTES - 0x20 - 1; i++) {
|
|
m_buf[i] ^= mgf1_buf[i - ofs];
|
|
}
|
|
}
|
|
|
|
m_buf[0] &= 0x7F; /* Constant lmask for rsa-2048-pss. */
|
|
|
|
/* Validate DB. */
|
|
for (unsigned int i = 0; i < RSA_2048_BYTES - 0x20 - 0x20 - 1 - 1; i++) {
|
|
if (m_buf[i] != 0) {
|
|
return false;
|
|
}
|
|
}
|
|
if (m_buf[RSA_2048_BYTES - 0x20 - 0x20 - 1 - 1] != 1) {
|
|
return false;
|
|
}
|
|
|
|
/* Check hash correctness. */
|
|
unsigned char validate_buf[8 + 0x20 + 0x20];
|
|
unsigned char validate_hash[0x20];
|
|
memset(validate_buf, 0, 0x48);
|
|
sha256_hash_buffer(&validate_buf[8], data, len);
|
|
memcpy(&validate_buf[0x28], &m_buf[RSA_2048_BYTES - 0x20 - 0x20 - 1], 0x20);
|
|
sha256_hash_buffer(validate_hash, validate_buf, 0x48);
|
|
return memcmp(h_buf, validate_hash, 0x20) == 0;
|
|
}
|
|
|
|
/* Perform an RSA-PKCS1 verify operation on data, with signature and N. */
|
|
int rsa2048_pkcs1_verify(const void *data, size_t len, const unsigned char *signature, const unsigned char *modulus) {
|
|
mbedtls_mpi signature_mpi;
|
|
mbedtls_mpi modulus_mpi;
|
|
mbedtls_mpi e_mpi;
|
|
mbedtls_mpi message_mpi;
|
|
|
|
mbedtls_mpi_init(&signature_mpi);
|
|
mbedtls_mpi_init(&modulus_mpi);
|
|
mbedtls_mpi_init(&e_mpi);
|
|
mbedtls_mpi_init(&message_mpi);
|
|
mbedtls_mpi_lset(&message_mpi, RSA_2048_BITS);
|
|
|
|
unsigned char m_buf[RSA_2048_BYTES];
|
|
unsigned char h_buf[0x20];
|
|
const unsigned char E[3] = {1, 0, 1};
|
|
|
|
mbedtls_mpi_read_binary(&e_mpi, E, 3);
|
|
mbedtls_mpi_read_binary(&signature_mpi, signature, RSA_2048_BYTES);
|
|
mbedtls_mpi_read_binary(&modulus_mpi, modulus, RSA_2048_BYTES);
|
|
mbedtls_mpi_exp_mod(&message_mpi, &signature_mpi, &e_mpi, &modulus_mpi, NULL);
|
|
|
|
if (mbedtls_mpi_write_binary(&message_mpi, m_buf, RSA_2048_BYTES) != 0) {
|
|
FATAL_ERROR("Failed to export exponentiated RSA message!");
|
|
}
|
|
|
|
mbedtls_mpi_free(&signature_mpi);
|
|
mbedtls_mpi_free(&modulus_mpi);
|
|
mbedtls_mpi_free(&e_mpi);
|
|
mbedtls_mpi_free(&message_mpi);
|
|
|
|
/* For RSA-2048, this prefix is just a constant. */
|
|
const unsigned char pkcs1_hash_prefix[0xE0] = {
|
|
0x00, 0x01, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x30, 0x31, 0x30,
|
|
0x0D, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x01, 0x05, 0x00, 0x04, 0x20
|
|
};
|
|
|
|
sha256_hash_buffer(h_buf, data, len);
|
|
|
|
return memcmp(pkcs1_hash_prefix, m_buf, 0xE0) == 0 && memcmp(&m_buf[0xE0], h_buf, 0x20) == 0;
|
|
} |