aboutsummaryrefslogtreecommitdiffstats
path: root/contrib/restricted/aws/s2n/pq-crypto/bike_r1/gf2x_mul.c
blob: 84a79589db8c662195f7c5208197b1eb6b53c8e2 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
/* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
 * SPDX-License-Identifier: Apache-2.0"
 *
 * Written by Nir Drucker and Shay Gueron,
 * AWS Cryptographic Algorithms Group.
 * (ndrucker@amazon.com, gueron@amazon.com)
 */

#include "cleanup.h"
#include "gf2x.h"
#include "gf2x_internal.h"
#include <stdlib.h>
#include <string.h>

#ifndef USE_OPENSSL_GF2M

// All the temporary data (which might hold secrets)
// is stored on a secure buffer, so that it can be easily cleaned up later.
// The secure buffer required is: 3n/2 (alah|blbh|tmp) in a recursive way.
// 3n/2 + 3n/4 + 3n/8 = 3(n/2 + n/4 + n/8) < 3n
#  define SECURE_BUFFER_SIZE (3 * R_PADDED_SIZE)

// Calculate number of uint64_t values needed to store SECURE_BUFFER_SIZE bytes. Rounding up to the next whole integer.
#  define SECURE_BUFFER_SIZE_64_BIT  ((SECURE_BUFFER_SIZE / sizeof(uint64_t)) + ((SECURE_BUFFER_SIZE % sizeof(uint64_t)) != 0))

// This functions assumes that n is even.
_INLINE_ void
karatzuba(OUT uint64_t *res,
          IN const uint64_t *a,
          IN const uint64_t *b,
          IN const uint64_t  n,
          uint64_t *         secure_buf)
{
  if(1 == n)
  {
    gf2x_mul_1x1(res, a[0], b[0]);
    return;
  }

  const uint64_t half_n = n >> 1;

  // Define pointers for the middle of each parameter
  // sepearting a=a_low and a_high (same for ba nd res)
  const uint64_t *a_high = a + half_n;
  const uint64_t *b_high = b + half_n;

  // Divide res into 4 parts res3|res2|res1|res in size n/2
  uint64_t *res1 = res + half_n;
  uint64_t *res2 = res1 + half_n;

  // All three parameters below are allocated on the secure buffer
  // All of them are in size half n
  uint64_t *alah = secure_buf;
  uint64_t *blbh = alah + half_n;
  uint64_t *tmp  = blbh + half_n;

  // Place the secure buffer ptr in the first free location,
  // so the recursive function can use it.
  secure_buf = tmp + half_n;

  // Calculate Z0 and store the result in res(low)
  karatzuba(res, a, b, half_n, secure_buf);

  // Calculate Z2 and store the result in res(high)
  karatzuba(res2, a_high, b_high, half_n, secure_buf);

  // Accomulate the results.
  karatzuba_add1(res, a, b, half_n, alah);

  // (a_low + a_high)(b_low + b_high) --> res1
  karatzuba(res1, alah, blbh, half_n, secure_buf);

  karatzuba_add2(res1, res2, res, tmp, half_n);
}

ret_t
gf2x_mod_mul(OUT uint64_t *res, IN const uint64_t *a, IN const uint64_t *b)
{
  bike_static_assert((R_PADDED_QW % 2 == 0), karatzuba_n_is_odd);

  ALIGN(sizeof(uint64_t)) uint64_t secure_buffer[SECURE_BUFFER_SIZE_64_BIT];

  /* make sure we have the correct size allocation. */
  bike_static_assert(sizeof(secure_buffer) % sizeof(uint64_t) == 0,
                     secure_buffer_not_eligable_for_uint64_t);

  karatzuba(res, a, b, R_PADDED_QW, (uint64_t *)secure_buffer);

  // This function implicitly assumes that the size of res is 2*R_PADDED_QW.
  red(res);

  secure_clean((uint8_t*)secure_buffer, sizeof(secure_buffer));

  return SUCCESS;
}

#endif // USE_OPENSSL_GF2M