2018-11-23 17:21:38 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 Denys Vlasenko
|
|
|
|
*
|
|
|
|
* Licensed under GPLv2, see file LICENSE in this source tree.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "tls.h"
|
|
|
|
|
|
|
|
typedef uint8_t byte;
|
|
|
|
typedef uint32_t word32;
|
|
|
|
#define XMEMSET memset
|
|
|
|
#define XMEMCPY memcpy
|
|
|
|
|
2018-11-23 18:31:26 +01:00
|
|
|
/* from wolfssl-3.15.3/wolfcrypt/src/aes.c */
|
2018-11-23 17:21:38 +01:00
|
|
|
|
2018-11-23 18:31:26 +01:00
|
|
|
static ALWAYS_INLINE void FlattenSzInBits(byte* buf, word32 sz)
|
2018-11-23 17:21:38 +01:00
|
|
|
{
|
|
|
|
/* Multiply the sz by 8 */
|
2018-11-23 18:31:26 +01:00
|
|
|
//bbox: these sizes are never even close to 2^32/8
|
|
|
|
// word32 szHi = (sz >> (8*sizeof(sz) - 3));
|
2018-11-23 17:21:38 +01:00
|
|
|
sz <<= 3;
|
|
|
|
|
|
|
|
/* copy over the words of the sz into the destination buffer */
|
2018-11-23 18:31:26 +01:00
|
|
|
// buf[0] = (szHi >> 24) & 0xff;
|
|
|
|
// buf[1] = (szHi >> 16) & 0xff;
|
|
|
|
// buf[2] = (szHi >> 8) & 0xff;
|
|
|
|
// buf[3] = szHi & 0xff;
|
2018-11-23 18:55:15 +01:00
|
|
|
*(uint32_t*)(buf + 0) = 0;
|
2018-11-23 18:31:26 +01:00
|
|
|
// buf[4] = (sz >> 24) & 0xff;
|
|
|
|
// buf[5] = (sz >> 16) & 0xff;
|
|
|
|
// buf[6] = (sz >> 8) & 0xff;
|
|
|
|
// buf[7] = sz & 0xff;
|
2018-11-23 18:55:15 +01:00
|
|
|
*(uint32_t*)(buf + 4) = SWAP_BE32(sz);
|
2018-11-23 17:21:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void RIGHTSHIFTX(byte* x)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int carryOut = 0;
|
|
|
|
int carryIn = 0;
|
|
|
|
int borrow = x[15] & 0x01;
|
|
|
|
|
|
|
|
for (i = 0; i < AES_BLOCK_SIZE; i++) {
|
|
|
|
carryOut = x[i] & 0x01;
|
|
|
|
x[i] = (x[i] >> 1) | (carryIn ? 0x80 : 0);
|
|
|
|
carryIn = carryOut;
|
|
|
|
}
|
|
|
|
if (borrow) x[0] ^= 0xE1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void GMULT(byte* X, byte* Y)
|
|
|
|
{
|
2018-11-24 14:08:29 +01:00
|
|
|
byte Z[AES_BLOCK_SIZE] ALIGNED_long;
|
|
|
|
byte V[AES_BLOCK_SIZE] ALIGNED_long;
|
2018-11-23 17:21:38 +01:00
|
|
|
int i, j;
|
|
|
|
|
|
|
|
XMEMSET(Z, 0, AES_BLOCK_SIZE);
|
|
|
|
XMEMCPY(V, X, AES_BLOCK_SIZE);
|
|
|
|
for (i = 0; i < AES_BLOCK_SIZE; i++)
|
|
|
|
{
|
|
|
|
byte y = Y[i];
|
|
|
|
for (j = 0; j < 8; j++)
|
|
|
|
{
|
|
|
|
if (y & 0x80) {
|
2018-11-24 14:08:29 +01:00
|
|
|
xorbuf_aligned_AES_BLOCK_SIZE(Z, V);
|
2018-11-23 17:21:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
RIGHTSHIFTX(V);
|
|
|
|
y = y << 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
XMEMCPY(X, Z, AES_BLOCK_SIZE);
|
|
|
|
}
|
|
|
|
|
2018-11-23 18:31:26 +01:00
|
|
|
//bbox:
|
2018-11-23 19:00:12 +01:00
|
|
|
// for TLS AES-GCM, a (which is AAD) is always 13 bytes long, and bbox code provides
|
2018-11-23 18:31:26 +01:00
|
|
|
// extra 3 zeroed bytes, making it a[16], or a[AES_BLOCK_SIZE].
|
2018-11-23 19:00:12 +01:00
|
|
|
// Resulting auth tag in s[] is also always AES_BLOCK_SIZE bytes.
|
2018-11-23 18:31:26 +01:00
|
|
|
//
|
|
|
|
// This allows some simplifications.
|
2018-11-24 13:47:44 +01:00
|
|
|
#define aSz 13
|
2018-11-23 18:31:26 +01:00
|
|
|
#define sSz AES_BLOCK_SIZE
|
|
|
|
void FAST_FUNC aesgcm_GHASH(byte* h,
|
|
|
|
const byte* a, //unsigned aSz,
|
|
|
|
const byte* c, unsigned cSz,
|
|
|
|
byte* s //, unsigned sSz
|
|
|
|
)
|
2018-11-23 17:21:38 +01:00
|
|
|
{
|
2018-11-24 14:08:29 +01:00
|
|
|
byte x[AES_BLOCK_SIZE] ALIGNED_long;
|
2018-11-25 14:03:59 +01:00
|
|
|
// byte scratch[AES_BLOCK_SIZE] ALIGNED_long;
|
|
|
|
unsigned blocks, partial;
|
2018-11-23 17:21:38 +01:00
|
|
|
//was: byte* h = aes->H;
|
|
|
|
|
2018-11-23 18:31:26 +01:00
|
|
|
//XMEMSET(x, 0, AES_BLOCK_SIZE);
|
2018-11-23 17:21:38 +01:00
|
|
|
|
|
|
|
/* Hash in A, the Additional Authentication Data */
|
2018-11-23 18:31:26 +01:00
|
|
|
// if (aSz != 0 && a != NULL) {
|
|
|
|
// blocks = aSz / AES_BLOCK_SIZE;
|
|
|
|
// partial = aSz % AES_BLOCK_SIZE;
|
|
|
|
// while (blocks--) {
|
|
|
|
//xorbuf(x, a, AES_BLOCK_SIZE);
|
|
|
|
XMEMCPY(x, a, AES_BLOCK_SIZE);// memcpy(x,a) = memset(x,0)+xorbuf(x,a)
|
2018-11-23 17:21:38 +01:00
|
|
|
GMULT(x, h);
|
2018-11-23 18:31:26 +01:00
|
|
|
// a += AES_BLOCK_SIZE;
|
|
|
|
// }
|
|
|
|
// if (partial != 0) {
|
|
|
|
// XMEMSET(scratch, 0, AES_BLOCK_SIZE);
|
|
|
|
// XMEMCPY(scratch, a, partial);
|
|
|
|
// xorbuf(x, scratch, AES_BLOCK_SIZE);
|
|
|
|
// GMULT(x, h);
|
|
|
|
// }
|
|
|
|
// }
|
2018-11-23 17:21:38 +01:00
|
|
|
|
|
|
|
/* Hash in C, the Ciphertext */
|
2018-11-23 18:31:26 +01:00
|
|
|
if (cSz != 0 /*&& c != NULL*/) {
|
2018-11-23 17:21:38 +01:00
|
|
|
blocks = cSz / AES_BLOCK_SIZE;
|
|
|
|
partial = cSz % AES_BLOCK_SIZE;
|
|
|
|
while (blocks--) {
|
2018-11-25 12:01:44 +01:00
|
|
|
if (BB_UNALIGNED_MEMACCESS_OK) // c is not guaranteed to be aligned
|
|
|
|
xorbuf_aligned_AES_BLOCK_SIZE(x, c);
|
|
|
|
else
|
|
|
|
xorbuf(x, c, AES_BLOCK_SIZE);
|
2018-11-23 17:21:38 +01:00
|
|
|
GMULT(x, h);
|
|
|
|
c += AES_BLOCK_SIZE;
|
|
|
|
}
|
|
|
|
if (partial != 0) {
|
2018-11-23 19:07:05 +01:00
|
|
|
//XMEMSET(scratch, 0, AES_BLOCK_SIZE);
|
|
|
|
//XMEMCPY(scratch, c, partial);
|
|
|
|
//xorbuf(x, scratch, AES_BLOCK_SIZE);
|
2018-11-24 14:08:29 +01:00
|
|
|
xorbuf(x, c, partial);//same result as above
|
2018-11-23 17:21:38 +01:00
|
|
|
GMULT(x, h);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Hash in the lengths of A and C in bits */
|
2018-11-25 14:03:59 +01:00
|
|
|
//FlattenSzInBits(&scratch[0], aSz);
|
|
|
|
//FlattenSzInBits(&scratch[8], cSz);
|
|
|
|
//xorbuf_aligned_AES_BLOCK_SIZE(x, scratch);
|
|
|
|
// simpler:
|
|
|
|
#define P32(v) ((uint32_t*)v)
|
|
|
|
//P32(x)[0] ^= 0;
|
|
|
|
P32(x)[1] ^= SWAP_BE32(aSz * 8);
|
|
|
|
//P32(x)[2] ^= 0;
|
|
|
|
P32(x)[3] ^= SWAP_BE32(cSz * 8);
|
|
|
|
#undef P32
|
|
|
|
|
2018-11-23 17:21:38 +01:00
|
|
|
GMULT(x, h);
|
|
|
|
|
|
|
|
/* Copy the result into s. */
|
|
|
|
XMEMCPY(s, x, sSz);
|
|
|
|
}
|