Coverage Report

Created: 2020-06-30 13:58

/src/botan/src/lib/hash/blake2/blake2b.cpp
Line
Count
Source (jump to first uncovered line)
1
/*
2
* BLAKE2b
3
* (C) 2016 cynecx
4
* (C) 2017 Jack Lloyd
5
*
6
* Botan is released under the Simplified BSD License (see license.txt)
7
*/
8
9
#include <botan/blake2b.h>
10
#include <botan/exceptn.h>
11
#include <botan/mem_ops.h>
12
#include <botan/loadstor.h>
13
#include <botan/rotate.h>
14
#include <algorithm>
15
16
namespace Botan {
17
18
namespace {
19
20
enum blake2b_constant {
21
  BLAKE2B_BLOCKBYTES = 128,
22
  BLAKE2B_IVU64COUNT = 8
23
};
24
25
const uint64_t blake2b_IV[BLAKE2B_IVU64COUNT] = {
26
   0x6a09e667f3bcc908, 0xbb67ae8584caa73b,
27
   0x3c6ef372fe94f82b, 0xa54ff53a5f1d36f1,
28
   0x510e527fade682d1, 0x9b05688c2b3e6c1f,
29
   0x1f83d9abfb41bd6b, 0x5be0cd19137e2179
30
};
31
32
}
33
34
BLAKE2b::BLAKE2b(size_t output_bits) :
35
   m_output_bits(output_bits),
36
   m_buffer(BLAKE2B_BLOCKBYTES),
37
   m_bufpos(0),
38
   m_H(BLAKE2B_IVU64COUNT)
39
0
   {
40
0
   if(output_bits == 0 || output_bits > 512 || output_bits % 8 != 0)
41
0
      {
42
0
      throw Invalid_Argument("Bad output bits size for BLAKE2b");
43
0
      }
44
0
45
0
   state_init();
46
0
   }
47
48
void BLAKE2b::state_init()
49
0
   {
50
0
   copy_mem(m_H.data(), blake2b_IV, BLAKE2B_IVU64COUNT);
51
0
   m_H[0] ^= 0x01010000 ^ static_cast<uint8_t>(output_length());
52
0
   m_T[0] = m_T[1] = 0;
53
0
   m_F[0] = m_F[1] = 0;
54
0
   m_bufpos = 0;
55
0
   }
56
57
namespace {
58
59
BOTAN_FORCE_INLINE void G(uint64_t& a, uint64_t& b, uint64_t& c, uint64_t& d,
60
                          uint64_t M0, uint64_t M1)
61
0
   {
62
0
   a = a + b + M0;
63
0
   d = rotr<32>(d ^ a);
64
0
   c = c + d;
65
0
   b = rotr<24>(b ^ c);
66
0
   a = a + b + M1;
67
0
   d = rotr<16>(d ^ a);
68
0
   c = c + d;
69
0
   b = rotr<63>(b ^ c);
70
0
   }
71
72
template<size_t i0, size_t i1, size_t i2, size_t i3, size_t i4, size_t i5, size_t i6, size_t i7,
73
         size_t i8, size_t i9, size_t iA, size_t iB, size_t iC, size_t iD, size_t iE, size_t iF>
74
BOTAN_FORCE_INLINE void ROUND(uint64_t* v, const uint64_t* M)
75
0
   {
76
0
   G(v[ 0], v[ 4], v[ 8], v[12], M[i0], M[i1]);
77
0
   G(v[ 1], v[ 5], v[ 9], v[13], M[i2], M[i3]);
78
0
   G(v[ 2], v[ 6], v[10], v[14], M[i4], M[i5]);
79
0
   G(v[ 3], v[ 7], v[11], v[15], M[i6], M[i7]);
80
0
   G(v[ 0], v[ 5], v[10], v[15], M[i8], M[i9]);
81
0
   G(v[ 1], v[ 6], v[11], v[12], M[iA], M[iB]);
82
0
   G(v[ 2], v[ 7], v[ 8], v[13], M[iC], M[iD]);
83
0
   G(v[ 3], v[ 4], v[ 9], v[14], M[iE], M[iF]);
84
0
   }
Unexecuted instantiation: blake2b.cpp:void Botan::(anonymous namespace)::ROUND<0ul, 1ul, 2ul, 3ul, 4ul, 5ul, 6ul, 7ul, 8ul, 9ul, 10ul, 11ul, 12ul, 13ul, 14ul, 15ul>(unsigned long*, unsigned long const*)
Unexecuted instantiation: blake2b.cpp:void Botan::(anonymous namespace)::ROUND<14ul, 10ul, 4ul, 8ul, 9ul, 15ul, 13ul, 6ul, 1ul, 12ul, 0ul, 2ul, 11ul, 7ul, 5ul, 3ul>(unsigned long*, unsigned long const*)
Unexecuted instantiation: blake2b.cpp:void Botan::(anonymous namespace)::ROUND<11ul, 8ul, 12ul, 0ul, 5ul, 2ul, 15ul, 13ul, 10ul, 14ul, 3ul, 6ul, 7ul, 1ul, 9ul, 4ul>(unsigned long*, unsigned long const*)
Unexecuted instantiation: blake2b.cpp:void Botan::(anonymous namespace)::ROUND<7ul, 9ul, 3ul, 1ul, 13ul, 12ul, 11ul, 14ul, 2ul, 6ul, 5ul, 10ul, 4ul, 0ul, 15ul, 8ul>(unsigned long*, unsigned long const*)
Unexecuted instantiation: blake2b.cpp:void Botan::(anonymous namespace)::ROUND<9ul, 0ul, 5ul, 7ul, 2ul, 4ul, 10ul, 15ul, 14ul, 1ul, 11ul, 12ul, 6ul, 8ul, 3ul, 13ul>(unsigned long*, unsigned long const*)
Unexecuted instantiation: blake2b.cpp:void Botan::(anonymous namespace)::ROUND<2ul, 12ul, 6ul, 10ul, 0ul, 11ul, 8ul, 3ul, 4ul, 13ul, 7ul, 5ul, 15ul, 14ul, 1ul, 9ul>(unsigned long*, unsigned long const*)
Unexecuted instantiation: blake2b.cpp:void Botan::(anonymous namespace)::ROUND<12ul, 5ul, 1ul, 15ul, 14ul, 13ul, 4ul, 10ul, 0ul, 7ul, 6ul, 3ul, 9ul, 2ul, 8ul, 11ul>(unsigned long*, unsigned long const*)
Unexecuted instantiation: blake2b.cpp:void Botan::(anonymous namespace)::ROUND<13ul, 11ul, 7ul, 14ul, 12ul, 1ul, 3ul, 9ul, 5ul, 0ul, 15ul, 4ul, 8ul, 6ul, 2ul, 10ul>(unsigned long*, unsigned long const*)
Unexecuted instantiation: blake2b.cpp:void Botan::(anonymous namespace)::ROUND<6ul, 15ul, 14ul, 9ul, 11ul, 3ul, 0ul, 8ul, 12ul, 2ul, 13ul, 7ul, 1ul, 4ul, 10ul, 5ul>(unsigned long*, unsigned long const*)
Unexecuted instantiation: blake2b.cpp:void Botan::(anonymous namespace)::ROUND<10ul, 2ul, 8ul, 4ul, 7ul, 6ul, 1ul, 5ul, 15ul, 11ul, 9ul, 14ul, 3ul, 12ul, 13ul, 0ul>(unsigned long*, unsigned long const*)
85
86
87
}
88
89
void BLAKE2b::compress(const uint8_t* input, size_t blocks, uint64_t increment)
90
0
   {
91
0
   for(size_t b = 0; b != blocks; ++b)
92
0
      {
93
0
      m_T[0] += increment;
94
0
      if(m_T[0] < increment)
95
0
         {
96
0
         m_T[1]++;
97
0
         }
98
0
99
0
      uint64_t M[16];
100
0
      uint64_t v[16];
101
0
      load_le(M, input, 16);
102
0
103
0
      input += BLAKE2B_BLOCKBYTES;
104
0
105
0
      for(size_t i = 0; i < 8; i++)
106
0
         v[i] = m_H[i];
107
0
      for(size_t i = 0; i != 8; ++i)
108
0
         v[i + 8] = blake2b_IV[i];
109
0
110
0
      v[12] ^= m_T[0];
111
0
      v[13] ^= m_T[1];
112
0
      v[14] ^= m_F[0];
113
0
      v[15] ^= m_F[1];
114
0
115
0
      ROUND< 0,  1,  2,  3,  4,  5,  6,  7,  8,  9, 10, 11, 12, 13, 14, 15>(v, M);
116
0
      ROUND<14, 10,  4,  8,  9, 15, 13,  6,  1, 12,  0,  2, 11,  7,  5,  3>(v, M);
117
0
      ROUND<11,  8, 12,  0,  5,  2, 15, 13, 10, 14,  3,  6,  7,  1,  9,  4>(v, M);
118
0
      ROUND< 7,  9,  3,  1, 13, 12, 11, 14,  2,  6,  5, 10,  4,  0, 15,  8>(v, M);
119
0
      ROUND< 9,  0,  5,  7,  2,  4, 10, 15, 14,  1, 11, 12,  6,  8,  3, 13>(v, M);
120
0
      ROUND< 2, 12,  6, 10,  0, 11,  8,  3,  4, 13,  7,  5, 15, 14,  1,  9>(v, M);
121
0
      ROUND<12,  5,  1, 15, 14, 13,  4, 10,  0,  7,  6,  3,  9,  2,  8, 11>(v, M);
122
0
      ROUND<13, 11,  7, 14, 12,  1,  3,  9,  5,  0, 15,  4,  8,  6,  2, 10>(v, M);
123
0
      ROUND< 6, 15, 14,  9, 11,  3,  0,  8, 12,  2, 13,  7,  1,  4, 10,  5>(v, M);
124
0
      ROUND<10,  2,  8,  4,  7,  6,  1,  5, 15, 11,  9, 14,  3, 12, 13,  0>(v, M);
125
0
      ROUND< 0,  1,  2,  3,  4,  5,  6,  7,  8,  9, 10, 11, 12, 13, 14, 15>(v, M);
126
0
      ROUND<14, 10,  4,  8,  9, 15, 13,  6,  1, 12,  0,  2, 11,  7,  5,  3>(v, M);
127
0
128
0
      for(size_t i = 0; i < 8; i++)
129
0
         {
130
0
         m_H[i] ^= v[i] ^ v[i + 8];
131
0
         }
132
0
      }
133
0
   }
134
135
void BLAKE2b::add_data(const uint8_t input[], size_t length)
136
0
   {
137
0
   if(length == 0)
138
0
      return;
139
0
140
0
   if(m_bufpos > 0)
141
0
      {
142
0
      if(m_bufpos < BLAKE2B_BLOCKBYTES)
143
0
         {
144
0
         const size_t take = std::min(BLAKE2B_BLOCKBYTES - m_bufpos, length);
145
0
         copy_mem(&m_buffer[m_bufpos], input, take);
146
0
         m_bufpos += take;
147
0
         length -= take;
148
0
         input += take;
149
0
         }
150
0
151
0
      if(m_bufpos == m_buffer.size() && length > 0)
152
0
         {
153
0
         compress(m_buffer.data(), 1, BLAKE2B_BLOCKBYTES);
154
0
         m_bufpos = 0;
155
0
         }
156
0
      }
157
0
158
0
   if(length > BLAKE2B_BLOCKBYTES)
159
0
      {
160
0
      const size_t full_blocks = ((length-1) / BLAKE2B_BLOCKBYTES);
161
0
      compress(input, full_blocks, BLAKE2B_BLOCKBYTES);
162
0
163
0
      input += full_blocks * BLAKE2B_BLOCKBYTES;
164
0
      length -= full_blocks * BLAKE2B_BLOCKBYTES;
165
0
      }
166
0
167
0
   if(length > 0)
168
0
      {
169
0
      copy_mem(&m_buffer[m_bufpos], input, length);
170
0
      m_bufpos += length;
171
0
      }
172
0
   }
173
174
void BLAKE2b::final_result(uint8_t output[])
175
0
   {
176
0
   if(m_bufpos != BLAKE2B_BLOCKBYTES)
177
0
      clear_mem(&m_buffer[m_bufpos], BLAKE2B_BLOCKBYTES - m_bufpos);
178
0
   m_F[0] = 0xFFFFFFFFFFFFFFFF;
179
0
   compress(m_buffer.data(), 1, m_bufpos);
180
0
   copy_out_vec_le(output, output_length(), m_H);
181
0
   state_init();
182
0
   }
183
184
std::string BLAKE2b::name() const
185
0
   {
186
0
   return "BLAKE2b(" + std::to_string(m_output_bits) + ")";
187
0
   }
188
189
HashFunction* BLAKE2b::clone() const
190
0
   {
191
0
   return new BLAKE2b(m_output_bits);
192
0
   }
193
194
std::unique_ptr<HashFunction> BLAKE2b::copy_state() const
195
0
   {
196
0
   return std::unique_ptr<HashFunction>(new BLAKE2b(*this));
197
0
   }
198
199
void BLAKE2b::clear()
200
0
   {
201
0
   zeroise(m_H);
202
0
   zeroise(m_buffer);
203
0
   m_bufpos = 0;
204
0
   state_init();
205
0
   }
206
207
}