Files
ChameleonUltra/software/src/nested.c
T
2022-12-22 12:43:05 +08:00

278 lines
6.1 KiB
C

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <inttypes.h>
#include <ctype.h>
#include "crapto1.h"
#include "parity.h"
#if WIN32
#include "windows.h"
#else
#include "unistd.h"
#endif
#define MEM_CHUNK 10000
#define TRY_KEYS 50
typedef struct {
uint64_t key;
int count;
} countKeys;
typedef struct {
uint32_t ntp;
uint32_t ks1;
} NtpKs1;
typedef struct {
NtpKs1* pNK;
uint32_t authuid;
uint64_t* keys;
uint32_t keyCount;
uint32_t startPos;
uint32_t endPos;
} RecPar;
int compar_int(const void* a, const void* b) {
return (*(uint64_t*)b - *(uint64_t*)a);
}
// Compare countKeys structure
int compar_special_int(const void* a, const void* b) {
return (((countKeys*)b)->count - ((countKeys*)a)->count);
}
// keys qsort and unique.
countKeys* uniqsort(uint64_t* possibleKeys, uint32_t size) {
unsigned int i, j = 0;
int count = 0;
countKeys* our_counts;
qsort(possibleKeys, size, sizeof(uint64_t), compar_int);
our_counts = calloc(size, sizeof(countKeys));
if (our_counts == NULL) {
printf("Memory allocation error for our_counts");
exit(EXIT_FAILURE);
}
for (i = 0; i < size; i++) {
if (possibleKeys[i + 1] == possibleKeys[i]) {
count++;
}
else {
our_counts[j].key = possibleKeys[i];
our_counts[j].count = count;
j++;
count = 0;
}
}
qsort(our_counts, j, sizeof(countKeys), compar_special_int);
return (our_counts);
}
uint32_t atoui(const char* str)
{
uint32_t result = 0, i = 0;
char* tmp = NULL;
for (i = 0; isspace(str[i]) && i < strlen(str); i++)
;
tmp = str + i;
while (*tmp)
{
result = result * 10 + *tmp - '0';
tmp++;
}
return result;
}
// nested decrypt
static void nested_revover(RecPar* rp) {
struct Crypto1State* revstate, * revstate_start = NULL;
uint64_t lfsr = 0;
uint32_t i, kcount = 0;
rp->keyCount = 0;
rp->keys = NULL;
for (i = rp->startPos; i < rp->endPos; i++) {
uint32_t nt_probe = rp->pNK[i].ntp;
uint32_t ks1 = rp->pNK[i].ks1;
// And finally recover the first 32 bits of the key
revstate = lfsr_recovery32(ks1, nt_probe ^ rp->authuid);
if (revstate_start == NULL) {
revstate_start = revstate;
}
while ((revstate->odd != 0x0) || (revstate->even != 0x0)) {
lfsr_rollback_word(revstate, nt_probe ^ rp->authuid, 0);
crypto1_get_lfsr(revstate, &lfsr);
// Allocate a new space for keys
if (((kcount % MEM_CHUNK) == 0) || (kcount >= rp->keyCount)) {
rp->keyCount += MEM_CHUNK;
// printf("New chunk by %d, sizeof %lu\n", kcount, key_count * sizeof(uint64_t));
void* tmp = realloc(rp->keys, rp->keyCount * sizeof(uint64_t));
if (tmp == NULL) {
printf("Memory allocation error for pk->possibleKeys");
// exit(EXIT_FAILURE);
rp->keyCount = 0;
return;
}
rp->keys = (uint64_t*)tmp;
}
rp->keys[kcount] = lfsr;
kcount++;
revstate++;
}
free(revstate_start);
revstate_start = NULL;
}
// Truncate
if (kcount != 0) {
rp->keyCount = --kcount;
void* tmp = (uint64_t*)realloc(rp->keys, rp->keyCount * sizeof(uint64_t));
if (tmp == NULL) {
printf("Memory allocation error for pk->possibleKeys");
// exit(EXIT_FAILURE);
rp->keyCount = 0;
return;
}
rp->keys = tmp;
return;
}
rp->keyCount = 0;
return;
}
uint64_t* nested(NtpKs1* pNK, uint32_t sizePNK, uint32_t authuid, uint32_t* keyCount) {
*keyCount = 0;
uint32_t i;
RecPar* pRPs = malloc(sizeof(RecPar));
if (pRPs == NULL) {
return NULL;
}
pRPs->pNK = pNK;
pRPs->authuid = authuid;
pRPs->startPos = 0;
pRPs->endPos = sizePNK;
// start recover
nested_revover(pRPs);
*keyCount = pRPs->keyCount;
uint64_t* keys = NULL;
if (*keyCount != 0) {
keys = malloc(*keyCount * sizeof(uint64_t));
if (keys != NULL) {
memcpy(keys, pRPs->keys, pRPs->keyCount * sizeof(uint64_t));
free(pRPs->keys);
}
}
free(pRPs);
countKeys* ck = uniqsort(keys, *keyCount);
free(keys);
keys = (uint64_t*)NULL;
*keyCount = 0;
if (ck != NULL) {
for (i = 0; i < TRY_KEYS; i++) {
// We don't known this key, try to break it
// This key can be found here two or more times
if (ck[i].count > 0) {
*keyCount += 1;
void* tmp = realloc(keys, sizeof(uint64_t) * (*keyCount));
if (tmp != NULL) {
keys = tmp;
keys[*keyCount - 1] = ck[i].key;
}
else {
printf("Cannot allocate memory for keys on merge.");
free(keys);
break;
}
}
}
}
else {
printf("Cannot allocate memory for ck on uniqsort.");
}
return keys;
}
// Return 1 if the nonce is invalid else return 0
static uint8_t valid_nonce(uint32_t Nt, uint32_t NtEnc, uint32_t Ks1, uint8_t* parity) {
return (
(oddparity8((Nt >> 24) & 0xFF) == ((parity[0]) ^ oddparity8((NtEnc >> 24) & 0xFF) ^ BIT(Ks1, 16))) && \
(oddparity8((Nt >> 16) & 0xFF) == ((parity[1]) ^ oddparity8((NtEnc >> 16) & 0xFF) ^ BIT(Ks1, 8))) && \
(oddparity8((Nt >> 8) & 0xFF) == ((parity[2]) ^ oddparity8((NtEnc >> 8) & 0xFF) ^ BIT(Ks1, 0)))
) ? 1 : 0;
}
int main(int argc, char* const argv[]) {
NtpKs1* pNK = NULL;
uint32_t i, j, m;
uint32_t nt1, nt2, nttest, ks1, dist;
uint8_t par_int;
uint8_t par_arr[3] = { 0x00 };
uint32_t authuid = atoui(argv[1]); // uid
dist = atoui(argv[2]); // dist
// process all args.
for (i = 3, j = 0; i < argc; i += 3) {
// nt + par
nt1 = atoui(argv[i]);
nt2 = atoui(argv[i + 1]);
par_int = atoui(argv[i + 2]);
if (par_int != 0) {
for (m = 0; m < 3; m++) {
par_arr[m] = (par_int >> m) & 0x01;
}
}
else {
memset(par_arr, 0, 3);
}
// Try to recover the keystream1
nttest = prng_successor(nt1, dist - 14);
for (m = dist - 14; m <= dist + 14; m += 1) {
ks1 = nt2 ^ nttest;
if (valid_nonce(nttest, nt2, ks1, par_arr)) {
++j;
// append to list
void* tmp = realloc(pNK, sizeof(NtpKs1) * j);
if (tmp == NULL) {
goto error;
}
pNK = tmp;
pNK[j - 1].ntp = nttest;
pNK[j - 1].ks1 = ks1;
}
nttest = prng_successor(nttest, 1);
}
}
uint32_t keyCount = 0;
uint64_t* keys = nested(pNK, j, authuid, &keyCount);
if (keyCount > 0) {
for (i = 0; i < keyCount; i++) {
printf("Key%d: %llx\r\n", i + 1, keys[i]);
fflush(stdout);
}
}
fflush(stdout);
free(keys);
exit(EXIT_SUCCESS);
error:
exit(EXIT_FAILURE);
}