sha3 1.0.5 → 2.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (54) hide show
  1. checksums.yaml +4 -4
  2. checksums.yaml.gz.sig +0 -0
  3. data/.clang-format +54 -0
  4. data/.document +3 -3
  5. data/.rdoc_options +11 -0
  6. data/.rspec +2 -2
  7. data/.rubocop.yml +8 -1
  8. data/CHANGELOG.md +23 -0
  9. data/Gemfile +12 -0
  10. data/LICENSE.txt +1 -1
  11. data/README.md +185 -65
  12. data/Rakefile +12 -4
  13. data/certs/io+sha3@jsg.io.pem +26 -0
  14. data/doc/sha3.rb +83 -0
  15. data/ext/sha3/config.h +2 -2
  16. data/ext/sha3/digest.c +726 -169
  17. data/ext/sha3/digest.h +6 -35
  18. data/ext/sha3/extconf.rb +42 -38
  19. data/ext/sha3/kmac.c +504 -0
  20. data/ext/sha3/kmac.h +14 -0
  21. data/ext/sha3/lib/high/Keccak/KeccakDuplex.c +81 -0
  22. data/ext/sha3/lib/high/Keccak/KeccakDuplex.h +73 -0
  23. data/ext/sha3/lib/high/Keccak/KeccakDuplex.inc +201 -0
  24. data/ext/sha3/lib/high/Keccak/KeccakSponge.c +2 -18
  25. data/ext/sha3/lib/high/Keccak/KeccakSponge.h +4 -10
  26. data/ext/sha3/lib/high/Keccak/KeccakSponge.inc +27 -31
  27. data/ext/sha3/lib/high/Keccak/PRG/KeccakPRG.c +61 -0
  28. data/ext/sha3/lib/high/Keccak/PRG/KeccakPRG.h +67 -0
  29. data/ext/sha3/lib/high/Keccak/PRG/KeccakPRG.inc +128 -0
  30. data/ext/sha3/lib/high/Keccak/SP800-185/SP800-185.c +93 -0
  31. data/ext/sha3/lib/high/Keccak/SP800-185/SP800-185.h +599 -0
  32. data/ext/sha3/lib/high/Keccak/SP800-185/SP800-185.inc +573 -0
  33. data/ext/sha3/lib/high/common/Phases.h +25 -0
  34. data/ext/sha3/lib/low/KeccakP-1600/common/KeccakP-1600-64.macros +19 -9
  35. data/ext/sha3/lib/low/KeccakP-1600/ref-32bits/KeccakP-1600-SnP.h +18 -12
  36. data/ext/sha3/lib/low/KeccakP-1600/ref-32bits/KeccakP-1600-reference32BI.c +28 -36
  37. data/ext/sha3/lib/low/KeccakP-1600/ref-64bits/KeccakP-1600-SnP.h +18 -12
  38. data/ext/sha3/lib/low/KeccakP-1600/ref-64bits/KeccakP-1600-reference.c +28 -59
  39. data/ext/sha3/lib/low/common/PlSnP-Fallback.inc +291 -0
  40. data/ext/sha3/lib/low/common/SnP-Relaned.h +145 -0
  41. data/ext/sha3/sha3.c +28 -59
  42. data/ext/sha3/sha3.h +4 -13
  43. data/lib/constants.rb +5 -0
  44. data/lib/sha3.rb +25 -24
  45. data.tar.gz.sig +0 -0
  46. metadata +61 -127
  47. metadata.gz.sig +0 -0
  48. data/.yardopts +0 -1
  49. data/ChangeLog.rdoc +0 -27
  50. data/certs/johanns.pem +0 -25
  51. data/lib/sha3/doc.rb +0 -121
  52. data/lib/sha3/version.rb +0 -9
  53. data/sha3.gemspec +0 -54
  54. data/tests.sh +0 -29
@@ -0,0 +1,573 @@
1
+ /*
2
+ The eXtended Keccak Code Package (XKCP)
3
+ https://github.com/XKCP/XKCP
4
+
5
+ Keccak, designed by Guido Bertoni, Joan Daemen, Michaël Peeters and Gilles Van Assche.
6
+
7
+ Implementation by Ronny Van Keer, hereby denoted as "the implementer".
8
+
9
+ For more information, feedback or questions, please refer to the Keccak Team website:
10
+ https://keccak.team/
11
+
12
+ To the extent possible under law, the implementer has waived all copyright
13
+ and related or neighboring rights to the source code in this file.
14
+ http://creativecommons.org/publicdomain/zero/1.0/
15
+ */
16
+
17
+ #define JOIN0(a, b) a ## b
18
+ #define JOIN(a, b) JOIN0(a, b)
19
+
20
+ #define capacity (2*security)
21
+ #define capacityInBytes (capacity/8)
22
+ #define capacityInLanes (capacityInBytes/laneSize)
23
+ #define rate (1600-capacity)
24
+ #define rateInBytes (rate/8)
25
+ #define rateInLanes (rateInBytes/laneSize)
26
+
27
+
28
+ #define cSHAKE JOIN(cSHAKE,security)
29
+ #define cSHAKE_Initialize JOIN(cSHAKE,_Initialize)
30
+ #define cSHAKE_Update JOIN(cSHAKE,_Update)
31
+ #define cSHAKE_Final JOIN(cSHAKE,_Final)
32
+ #define cSHAKE_Squeeze JOIN(cSHAKE,_Squeeze)
33
+
34
+ int cSHAKE_Initialize(cSHAKE_Instance *csk, BitLength outputBitLen, const BitSequence *name, BitLength nameBitLen, const BitSequence *customization, BitLength customBitLen)
35
+ {
36
+ unsigned char encbuf[sizeof(BitLength)+1];
37
+
38
+ /* Only full bytes are supported for 'name', otherwise customization string would have to be shifted before absorbing */
39
+ if ((nameBitLen & 7) != 0)
40
+ return 1;
41
+ if (KeccakWidth1600_SpongeInitialize(&csk->sponge, rate, capacity) != 0)
42
+ return 1;
43
+ csk->lastByteBitLen = 0;
44
+ csk->lastByteValue = 0;
45
+ csk->fixedOutputLength = outputBitLen;
46
+ csk->phase = ABSORBING;
47
+
48
+ if ((nameBitLen == 0) && (customBitLen == 0))
49
+ csk->emptyNameCustom = 1;
50
+ else
51
+ {
52
+ csk->emptyNameCustom = 0;
53
+
54
+ /* Absorb bytepad(.., rate) */
55
+ if (KeccakWidth1600_SpongeAbsorb(&csk->sponge, encbuf, left_encode(encbuf, rateInBytes)) != 0)
56
+ return 1;
57
+
58
+ /* Absorb encode_string(name) */
59
+ if (KeccakWidth1600_SpongeAbsorb(&csk->sponge, encbuf, left_encode(encbuf, nameBitLen)) != 0)
60
+ return 1;
61
+ if (KeccakWidth1600_SpongeAbsorb(&csk->sponge, name, nameBitLen / 8) != 0)
62
+ return 1;
63
+
64
+ /* Absorb encode_string(customization) */
65
+ if (KeccakWidth1600_SpongeAbsorb(&csk->sponge, encbuf, left_encode(encbuf, customBitLen)) != 0)
66
+ return 1;
67
+ if (KeccakWidth1600_SpongeAbsorb(&csk->sponge, customization, (customBitLen + 7) / 8) != 0) /* allowed to be a bit string, as zero padding is following */
68
+ return 1;
69
+
70
+ /* Zero padding up to rate */
71
+ if ( csk->sponge.byteIOIndex != 0 ) {
72
+ csk->sponge.byteIOIndex = rateInBytes - 1;
73
+ encbuf[0] = 0;
74
+ return KeccakWidth1600_SpongeAbsorb(&csk->sponge, encbuf, 1);
75
+ }
76
+ }
77
+ return 0;
78
+ }
79
+
80
+ int cSHAKE_Update(cSHAKE_Instance *csk, const BitSequence *input, BitLength inputBitLen)
81
+ {
82
+
83
+ if (csk->phase != ABSORBING)
84
+ return 1;
85
+ if (csk->lastByteBitLen != 0) /* check if previous call input were full bytes */
86
+ return 1;
87
+ csk->lastByteBitLen = inputBitLen & 7;
88
+ if(csk->lastByteBitLen != 0)
89
+ csk->lastByteValue = input[inputBitLen / 8] & ((1 << csk->lastByteBitLen) - 1); /* strip unwanted bits */
90
+ return KeccakWidth1600_SpongeAbsorb(&csk->sponge, input, inputBitLen / 8);
91
+ }
92
+
93
+ int cSHAKE_Final(cSHAKE_Instance *csk, BitSequence *output)
94
+ {
95
+ unsigned short delimitedLastBytes;
96
+ unsigned char delimitedSuffix;
97
+
98
+ if (csk->phase != ABSORBING)
99
+ return 1;
100
+
101
+ /* Concatenate the last few input bits with those of the suffix */
102
+ if (csk->emptyNameCustom != 0)
103
+ delimitedLastBytes = (unsigned short)(csk->lastByteValue | (0x1F << csk->lastByteBitLen));
104
+ else
105
+ delimitedLastBytes = (unsigned short)(csk->lastByteValue | (0x04 << csk->lastByteBitLen)); /* Suffix '04': 2 zero bits '00' */
106
+ if ((delimitedLastBytes >> 8) == 0) {
107
+ delimitedSuffix = (unsigned char)delimitedLastBytes;
108
+ }
109
+ else {
110
+ unsigned char oneByte[1];
111
+ oneByte[0] = (unsigned char)delimitedLastBytes;
112
+ if(KeccakWidth1600_SpongeAbsorb(&csk->sponge, oneByte, 1) != 0)
113
+ return 1;
114
+ delimitedSuffix = (unsigned char)(delimitedLastBytes >> 8);
115
+ }
116
+ if (KeccakWidth1600_SpongeAbsorbLastFewBits(&csk->sponge, delimitedSuffix) != 0)
117
+ return 1;
118
+ csk->phase = SQUEEZING;
119
+ if ( csk->fixedOutputLength != 0 ) {
120
+ if(cSHAKE_Squeeze(csk, output, csk->fixedOutputLength) != 0)
121
+ return 1;
122
+ csk->phase = FINAL;
123
+ }
124
+ return 0;
125
+ }
126
+
127
+ int cSHAKE_Squeeze(cSHAKE_Instance *csk, BitSequence *output, BitLength outputBitLen)
128
+ {
129
+ if (csk->phase != SQUEEZING)
130
+ return 1;
131
+ if(KeccakWidth1600_SpongeSqueeze(&csk->sponge, output, (outputBitLen + 7) / 8) != 0)
132
+ return 1;
133
+ if ((outputBitLen & 7) !=0) {
134
+ output[outputBitLen / 8] &= (1 << (outputBitLen & 7)) - 1; /* clear unwanted bits */
135
+ csk->phase = FINAL; /* only last output can have an non complete byte, block nexts calls */
136
+ }
137
+ return 0;
138
+ }
139
+
140
+ int cSHAKE( const BitSequence *input, BitLength inputBitLen, BitSequence *output, BitLength outputBitLen,
141
+ const BitSequence *name, BitLength nameBitLen, const BitSequence *customization, BitLength customBitLen )
142
+ {
143
+ cSHAKE_Instance csk;
144
+
145
+ if (outputBitLen == 0)
146
+ return 1;
147
+ if (cSHAKE_Initialize(&csk, outputBitLen, name, nameBitLen, customization, customBitLen) != 0)
148
+ return 1;
149
+ if (cSHAKE_Update(&csk, input, inputBitLen) != 0)
150
+ return 1;
151
+ return cSHAKE_Final(&csk, output);
152
+ }
153
+
154
+ /* ------------------------------------------------------------------------- */
155
+
156
+ #define KMAC JOIN(KMAC,security)
157
+ #define KMAC_Initialize JOIN(KMAC,_Initialize)
158
+ #define KMAC_Update JOIN(KMAC,_Update)
159
+ #define KMAC_Final JOIN(KMAC,_Final)
160
+ #define KMAC_Squeeze JOIN(KMAC,_Squeeze)
161
+
162
+ int KMAC_Initialize(KMAC_Instance *km, const BitSequence *key, BitLength keyBitLen, BitLength outputBitLen, const BitSequence *customization, BitLength customBitLen)
163
+ {
164
+ BitLength bufferLen;
165
+ BitLength keyByteLen;
166
+ BitSequence buffer[rateInBytes];
167
+
168
+ if (cSHAKE_Initialize(&km->csi, outputBitLen, (const BitSequence*)"KMAC", 4*8, customization, customBitLen) != 0)
169
+ return 1;
170
+ km->outputBitLen = outputBitLen;
171
+
172
+ /* bytepad(encode_string(k)) */
173
+ bufferLen = left_encode(buffer, rateInBytes);
174
+ bufferLen += left_encode(buffer + bufferLen, keyBitLen);
175
+ if (cSHAKE_Update(&km->csi, buffer, bufferLen*8) != 0)
176
+ return 1;
177
+ keyByteLen = (keyBitLen + 7) / 8;
178
+ if (cSHAKE_Update(&km->csi, key, keyByteLen*8) != 0)
179
+ return 1;
180
+ bufferLen = (bufferLen + keyByteLen) % rateInBytes; /* zero padding */
181
+ if (bufferLen != 0) {
182
+ bufferLen = rateInBytes - bufferLen;
183
+ memset(buffer, 0, bufferLen);
184
+ if (cSHAKE_Update(&km->csi, buffer, bufferLen*8) != 0)
185
+ return 1;
186
+ }
187
+ return 0;
188
+ }
189
+
190
+ int KMAC_Update(KMAC_Instance *km, const BitSequence *input, BitLength inputBitLen)
191
+ {
192
+ if ((inputBitLen & 7) != 0) /* Only full bytes are supported */
193
+ return 1;
194
+ return cSHAKE_Update(&km->csi, input, inputBitLen);
195
+ }
196
+
197
+ int KMAC_Final(KMAC_Instance *km, BitSequence *output)
198
+ {
199
+ unsigned char encbuf[sizeof(BitLength)+1];
200
+
201
+ if (cSHAKE_Update(&km->csi, encbuf, right_encode(encbuf, km->outputBitLen)*8) != 0)
202
+ return 1;
203
+ return cSHAKE_Final(&km->csi, output);
204
+ }
205
+
206
+ int KMAC_Squeeze(KMAC_Instance *km, BitSequence *output, BitLength outputBitLen)
207
+ {
208
+ return cSHAKE_Squeeze(&km->csi, output, outputBitLen);
209
+ }
210
+
211
+ int KMAC(const BitSequence *key, BitLength keyBitLen, const BitSequence *input, BitLength inputBitLen,
212
+ BitSequence *output, BitLength outputBitLen, const BitSequence *customization, BitLength customBitLen)
213
+ {
214
+ KMAC_Instance km;
215
+
216
+ if (outputBitLen == 0)
217
+ return 1;
218
+ if (KMAC_Initialize(&km, key, keyBitLen, outputBitLen, customization, customBitLen) != 0)
219
+ return 1;
220
+ if (KMAC_Update(&km, input, inputBitLen) != 0)
221
+ return 1;
222
+ return KMAC_Final(&km, output);
223
+ }
224
+
225
+ #undef KMAC_Initialize
226
+ #undef KMAC_Update
227
+ #undef KMAC_Final
228
+ #undef KMAC_Squeeze
229
+ #undef KMAC
230
+
231
+ /* ------------------------------------------------------------------------- */
232
+
233
+ #define ParallelHash JOIN(ParallelHash,security)
234
+ #define ParallelHash_Initialize JOIN(ParallelHash,_Initialize)
235
+ #define ParallelHash_Update JOIN(ParallelHash,_Update)
236
+ #define ParallelHash_Final JOIN(ParallelHash,_Final)
237
+ #define ParallelHash_Squeeze JOIN(ParallelHash,_Squeeze)
238
+
239
+ #define ParallelSpongeFastLoop( Parallellism ) \
240
+ while ( inputByteLen >= Parallellism * phi->blockLen ) { \
241
+ KeccakP1600times##Parallellism##_states states; \
242
+ unsigned char intermediate[Parallellism*capacityInBytes]; \
243
+ size_t localBlockLen = phi->blockLen; \
244
+ const unsigned char * localInput = input; \
245
+ unsigned int i; \
246
+ size_t fastLoopOffset; \
247
+ \
248
+ KeccakP1600times##Parallellism##_StaticInitialize(); \
249
+ KeccakP1600times##Parallellism##_InitializeAll(&states); \
250
+ fastLoopOffset = KeccakF1600times##Parallellism##_FastLoop_Absorb(&states, rateInLanes, phi->blockLen / laneSize, rateInLanes, localInput, Parallellism * phi->blockLen); \
251
+ localBlockLen -= fastLoopOffset; \
252
+ localInput += fastLoopOffset; \
253
+ for ( i = 0; i < Parallellism; ++i, localInput += phi->blockLen ) { \
254
+ KeccakP1600times##Parallellism##_AddBytes(&states, i, localInput, 0, localBlockLen); \
255
+ KeccakP1600times##Parallellism##_AddByte(&states, i, suffix, localBlockLen); \
256
+ KeccakP1600times##Parallellism##_AddByte(&states, i, 0x80, rateInBytes-1); \
257
+ } \
258
+ KeccakP1600times##Parallellism##_PermuteAll_24rounds(&states); \
259
+ input += Parallellism * phi->blockLen; \
260
+ inputByteLen -= Parallellism * phi->blockLen; \
261
+ KeccakP1600times##Parallellism##_ExtractLanesAll(&states, intermediate, capacityInLanes, capacityInLanes ); \
262
+ if (KeccakWidth1600_SpongeAbsorb(&phi->finalNode, intermediate, Parallellism * capacityInBytes) != 0) return 1; \
263
+ }
264
+
265
+ #define ParallelSpongeLoop( Parallellism ) \
266
+ while ( inputByteLen >= Parallellism * phi->blockLen ) { \
267
+ KeccakP1600times##Parallellism##_states states; \
268
+ unsigned char intermediate[Parallellism*capacityInBytes]; \
269
+ size_t localBlockLen = phi->blockLen; \
270
+ const unsigned char * localInput = input; \
271
+ unsigned int i; \
272
+ \
273
+ KeccakP1600times##Parallellism##_StaticInitialize(); \
274
+ KeccakP1600times##Parallellism##_InitializeAll(&states); \
275
+ while(localBlockLen >= rateInBytes) { \
276
+ KeccakP1600times##Parallellism##_AddLanesAll(&states, localInput, rateInLanes, phi->blockLen / laneSize); \
277
+ KeccakP1600times##Parallellism##_PermuteAll_24rounds(&states); \
278
+ localBlockLen -= rateInBytes; \
279
+ localInput += rateInBytes; \
280
+ } \
281
+ for ( i = 0; i < Parallellism; ++i, localInput += phi->blockLen ) { \
282
+ KeccakP1600times##Parallellism##_AddBytes(&states, i, localInput, 0, localBlockLen); \
283
+ KeccakP1600times##Parallellism##_AddByte(&states, i, suffix, localBlockLen); \
284
+ KeccakP1600times##Parallellism##_AddByte(&states, i, 0x80, rateInBytes-1); \
285
+ } \
286
+ KeccakP1600times##Parallellism##_PermuteAll_24rounds(&states); \
287
+ input += Parallellism * phi->blockLen; \
288
+ inputByteLen -= Parallellism * phi->blockLen; \
289
+ KeccakP1600times##Parallellism##_ExtractLanesAll(&states, intermediate, capacityInLanes, capacityInLanes ); \
290
+ if (KeccakWidth1600_SpongeAbsorb(&phi->finalNode, intermediate, Parallellism * capacityInBytes) != 0) return 1; \
291
+ }
292
+
293
+ int ParallelHash_Initialize(ParallelHash_Instance *phi, size_t blockByteLen,
294
+ BitLength outputBitLen, const BitSequence *customization, BitLength customBitLen)
295
+ {
296
+ size_t t;
297
+ unsigned char encbuf[sizeof(size_t)+1];
298
+
299
+ if ( blockByteLen < laneSize) /* blockLen must be greater than or equal to lane size */
300
+ return 1;
301
+ for ( t = blockByteLen; t > 1; t >>= 1 ) /* blockLen (in bytes) must be a power of two */
302
+ if ( (t & 1) && (t != 1) ) /* bit0 set and other bits unset */
303
+ return 1;
304
+ if (KeccakWidth1600_SpongeInitialize(&phi->finalNode, rate, capacity) != 0)
305
+ return 1;
306
+ phi->fixedOutputLength = outputBitLen;
307
+ phi->blockLen = blockByteLen;
308
+ phi->queueAbsorbedLen = 0;
309
+ phi->totalInputSize = 0;
310
+ phi->phase = ABSORBING;
311
+
312
+ /* Absorb bytepad(.., rate) */
313
+ if (KeccakWidth1600_SpongeAbsorb(&phi->finalNode, encbuf, left_encode(encbuf, rateInBytes)) != 0)
314
+ return 1;
315
+
316
+ /* Absorb string_encode("ParallelHash") */
317
+ if (KeccakWidth1600_SpongeAbsorb(&phi->finalNode, (const BitSequence*)"\x01\x60" "ParallelHash", 14) != 0)
318
+ return 1;
319
+
320
+ /* Absorb string_encode(customization) */
321
+ if (KeccakWidth1600_SpongeAbsorb(&phi->finalNode, encbuf, left_encode(encbuf, customBitLen)) != 0)
322
+ return 1;
323
+ if (KeccakWidth1600_SpongeAbsorb(&phi->finalNode, customization, (customBitLen + 7) / 8) != 0)
324
+ return 1;
325
+
326
+ /* Zero padding up to rate */
327
+ if ( phi->finalNode.byteIOIndex != 0 ) {
328
+ phi->finalNode.byteIOIndex = rateInBytes - 1;
329
+ encbuf[0] = 0;
330
+ if (KeccakWidth1600_SpongeAbsorb(&phi->finalNode, encbuf, 1) != 0)
331
+ return 1;
332
+ }
333
+
334
+ /* Absorb B */
335
+ if (KeccakWidth1600_SpongeAbsorb(&phi->finalNode, encbuf, left_encode(encbuf, blockByteLen)) != 0)
336
+ return 1;
337
+ return 0;
338
+ }
339
+
340
+ int ParallelHash_Update(ParallelHash_Instance *phi, const BitSequence *input, BitLength inputBitLen)
341
+ {
342
+ size_t inputByteLen;
343
+
344
+ if (phi->phase != ABSORBING)
345
+ return 1;
346
+ if ((inputBitLen & 7) != 0) /* Only full bytes are supported */
347
+ return 1;
348
+ phi->totalInputSize += inputBitLen;
349
+ inputByteLen = inputBitLen / 8;
350
+ if ( phi->queueAbsorbedLen != 0 ) {
351
+ /* There is data in the queue, absorb further in queue until full */
352
+ size_t len = (inputByteLen < (phi->blockLen - phi->queueAbsorbedLen)) ? inputByteLen : (phi->blockLen - phi->queueAbsorbedLen);
353
+ if (KeccakWidth1600_SpongeAbsorb(&phi->queueNode, input, len) != 0)
354
+ return 1;
355
+ input += len;
356
+ inputByteLen -= len;
357
+ phi->queueAbsorbedLen += len;
358
+ if ( phi->queueAbsorbedLen == phi->blockLen ) {
359
+ unsigned char intermediate[capacityInBytes];
360
+ phi->queueAbsorbedLen = 0;
361
+ if (KeccakWidth1600_SpongeAbsorbLastFewBits(&phi->queueNode, suffix) != 0)
362
+ return 1;
363
+ if (KeccakWidth1600_SpongeSqueeze(&phi->queueNode, intermediate, capacityInBytes) != 0)
364
+ return 1;
365
+ if (KeccakWidth1600_SpongeAbsorb(&phi->finalNode, intermediate, capacityInBytes) != 0)
366
+ return 1;
367
+ }
368
+ }
369
+
370
+ #if defined(KeccakP1600times8_implementation)
371
+ #if defined(KeccakF1600times8_FastLoop_supported)
372
+ ParallelSpongeFastLoop( 8 )
373
+ #else
374
+ ParallelSpongeLoop( 8 )
375
+ #endif
376
+ #endif
377
+
378
+ #if defined(KeccakP1600times4_implementation)
379
+ #if defined(KeccakF1600times4_FastLoop_supported)
380
+ ParallelSpongeFastLoop( 4 )
381
+ #else
382
+ ParallelSpongeLoop( 4 )
383
+ #endif
384
+ #endif
385
+
386
+ #if defined(KeccakP1600times2_implementation)
387
+ #if defined(KeccakF1600times2_FastLoop_supported)
388
+ ParallelSpongeFastLoop( 2 )
389
+ #else
390
+ ParallelSpongeLoop( 2 )
391
+ #endif
392
+ #endif
393
+
394
+ while ( inputByteLen > 0 ) {
395
+ size_t len = (inputByteLen < phi->blockLen) ? inputByteLen : phi->blockLen;
396
+ if (KeccakWidth1600_SpongeInitialize(&phi->queueNode, rate, capacity) != 0)
397
+ return 1;
398
+ if (KeccakWidth1600_SpongeAbsorb(&phi->queueNode, input, len) != 0)
399
+ return 1;
400
+ input += len;
401
+ inputByteLen -= len;
402
+ if ( len == phi->blockLen ) {
403
+ unsigned char intermediate[capacityInBytes];
404
+ if (KeccakWidth1600_SpongeAbsorbLastFewBits(&phi->queueNode, suffix) != 0)
405
+ return 1;
406
+ if (KeccakWidth1600_SpongeSqueeze(&phi->queueNode, intermediate, capacityInBytes) != 0)
407
+ return 1;
408
+ if (KeccakWidth1600_SpongeAbsorb(&phi->finalNode, intermediate, capacityInBytes) != 0)
409
+ return 1;
410
+ }
411
+ else
412
+ phi->queueAbsorbedLen = len;
413
+ }
414
+
415
+ return 0;
416
+ }
417
+
418
+ int ParallelHash_Final(ParallelHash_Instance *phi, BitSequence * output)
419
+ {
420
+ unsigned char encbuf[sizeof(size_t)+1];
421
+ size_t nBlocks;
422
+
423
+ if (phi->phase != ABSORBING)
424
+ return 1;
425
+ if ( phi->queueAbsorbedLen != 0 ) {
426
+ /* There is data in the queue */
427
+ unsigned char intermediate[capacityInBytes];
428
+ if (KeccakWidth1600_SpongeAbsorbLastFewBits(&phi->queueNode, suffix) != 0)
429
+ return 1;
430
+ if (KeccakWidth1600_SpongeSqueeze(&phi->queueNode, intermediate, capacityInBytes) != 0)
431
+ return 1;
432
+ if (KeccakWidth1600_SpongeAbsorb(&phi->finalNode, intermediate, capacityInBytes) != 0)
433
+ return 1;
434
+ }
435
+
436
+ nBlocks = (phi->totalInputSize / 8 + phi->blockLen - 1) / phi->blockLen;
437
+ if (KeccakWidth1600_SpongeAbsorb(&phi->finalNode, encbuf, right_encode(encbuf, nBlocks)) != 0) /* Absorb number of blocks */
438
+ return 1;
439
+
440
+ if (KeccakWidth1600_SpongeAbsorb(&phi->finalNode, encbuf, right_encode(encbuf, phi->fixedOutputLength)) != 0) /* Absorb output length in bits */
441
+ return 1;
442
+
443
+ if (KeccakWidth1600_SpongeAbsorbLastFewBits(&phi->finalNode, 0x04) != 0) /* Absorb 2 zero bits '00' */
444
+ return 1;
445
+ phi->phase = SQUEEZING;
446
+ if ( phi->fixedOutputLength != 0 ) {
447
+ if (ParallelHash_Squeeze(phi, output, phi->fixedOutputLength) != 0)
448
+ return 1;
449
+ phi->phase = FINAL;
450
+ }
451
+ return 0;
452
+ }
453
+
454
+ int ParallelHash_Squeeze(ParallelHash_Instance *phi, BitSequence *output, BitLength outputBitLen)
455
+ {
456
+ if (phi->phase != SQUEEZING)
457
+ return 1;
458
+ if (KeccakWidth1600_SpongeSqueeze(&phi->finalNode, output, (outputBitLen + 7) / 8) != 0)
459
+ return 1;
460
+ if ((outputBitLen & 7) !=0) {
461
+ output[outputBitLen / 8] &= (1 << (outputBitLen & 7)) - 1; /* clear unwanted bits */
462
+ phi->phase = FINAL; /* only last output can have an non complete byte, block nexts calls */
463
+ }
464
+ return 0;
465
+ }
466
+
467
+ int ParallelHash( const BitSequence *input, BitLength inputBitLen, size_t blockByteLen,
468
+ BitSequence *output, BitLength outputBitLen, const BitSequence *customization, BitLength customBitLen)
469
+ {
470
+ ParallelHash_Instance phi;
471
+
472
+ if (outputBitLen == 0)
473
+ return 1;
474
+ if (ParallelHash_Initialize(&phi, blockByteLen, outputBitLen, customization, customBitLen) != 0)
475
+ return 1;
476
+ if (ParallelHash_Update(&phi, input, inputBitLen) != 0)
477
+ return 1;
478
+ return ParallelHash_Final(&phi, output);
479
+ }
480
+
481
+ #undef ParallelHash_Initialize
482
+ #undef ParallelHash_Update
483
+ #undef ParallelHash_Final
484
+ #undef ParallelHash_Squeeze
485
+ #undef ParallelHash
486
+
487
+ #undef ParallelSpongeFastLoop
488
+ #undef ParallelSpongeLoop
489
+
490
+ /* ------------------------------------------------------------------------- */
491
+
492
+ #define TupleHash JOIN(TupleHash,security)
493
+ #define TupleHash_Initialize JOIN(TupleHash,_Initialize)
494
+ #define TupleHash_Update JOIN(TupleHash,_Update)
495
+ #define TupleHash_Final JOIN(TupleHash,_Final)
496
+ #define TupleHash_Squeeze JOIN(TupleHash,_Squeeze)
497
+
498
+ int TupleHash_Initialize(TupleHash_Instance *thi, BitLength outputBitLen,
499
+ const BitSequence *customization, BitLength customBitLen)
500
+ {
501
+ if (cSHAKE_Initialize(&thi->csi, outputBitLen, (const BitSequence*)"TupleHash", 9*8, customization, customBitLen) != 0)
502
+ return 1;
503
+ thi->outputBitLen = outputBitLen;
504
+ return 0;
505
+ }
506
+
507
+ int TupleHash_Update(TupleHash_Instance *thi, const TupleElement *tuple, size_t numberOfElements)
508
+ {
509
+ unsigned char encbuf[sizeof(BitLength)+1];
510
+
511
+ while (numberOfElements-- != 0) {
512
+ if ((tuple->inputBitLen & 7) != 0) /* Only full bytes are supported */
513
+ return 1;
514
+ if (cSHAKE_Update(&thi->csi, encbuf, left_encode(encbuf, tuple->inputBitLen)*8) != 0)
515
+ return 1;
516
+ if (cSHAKE_Update(&thi->csi, tuple->input, tuple->inputBitLen) != 0)
517
+ return 1;
518
+ ++tuple;
519
+ }
520
+ return 0;
521
+ }
522
+
523
+ int TupleHash_Final(TupleHash_Instance *thi, BitSequence * output)
524
+ {
525
+ unsigned char encbuf[sizeof(BitLength)+1];
526
+
527
+ if (cSHAKE_Update(&thi->csi, encbuf, right_encode(encbuf, thi->outputBitLen)*8) != 0)
528
+ return 1;
529
+ return cSHAKE_Final(&thi->csi, output);
530
+ }
531
+
532
+ int TupleHash_Squeeze(TupleHash_Instance *thi, BitSequence *output, BitLength outputBitLen)
533
+ {
534
+ return cSHAKE_Squeeze(&thi->csi, output, outputBitLen);
535
+ }
536
+
537
+ int TupleHash( const TupleElement *tuple, size_t numberOfElements,
538
+ BitSequence *output, BitLength outputBitLen, const BitSequence *customization, BitLength customBitLen)
539
+ {
540
+ TupleHash_Instance thi;
541
+
542
+ if (outputBitLen == 0)
543
+ return 1;
544
+ if (TupleHash_Initialize(&thi, outputBitLen, customization, customBitLen) != 0)
545
+ return 1;
546
+ if (TupleHash_Update(&thi, tuple, numberOfElements) != 0)
547
+ return 1;
548
+ return TupleHash_Final(&thi, output);
549
+ }
550
+
551
+ #undef TupleHash_Initialize
552
+ #undef TupleHash_Update
553
+ #undef TupleHash_Final
554
+ #undef TupleHash_Squeeze
555
+ #undef TupleHash
556
+
557
+ /* ------------------------------------------------------------------------- */
558
+
559
+ #undef JOIN0
560
+ #undef JOIN
561
+
562
+ #undef capacity
563
+ #undef capacityInBytes
564
+ #undef capacityInLanes
565
+ #undef rate
566
+ #undef rateInBytes
567
+ #undef rateInLanes
568
+
569
+ #undef cSHAKE_Initialize
570
+ #undef cSHAKE_Update
571
+ #undef cSHAKE_Final
572
+ #undef cSHAKE_Squeeze
573
+ #undef cSHAKE
@@ -0,0 +1,25 @@
1
+ /*
2
+ The eXtended Keccak Code Package (XKCP)
3
+ https://github.com/XKCP/XKCP
4
+
5
+ Implementation by Ronny Van Keer, hereby denoted as "the implementer".
6
+
7
+ For more information, feedback or questions, please refer to the Keccak Team website:
8
+ https://keccak.team/
9
+
10
+ To the extent possible under law, the implementer has waived all copyright
11
+ and related or neighboring rights to the source code in this file.
12
+ http://creativecommons.org/publicdomain/zero/1.0/
13
+ */
14
+
15
+ #ifndef _Phases_h_
16
+ #define _Phases_h_
17
+
18
+ typedef enum {
19
+ NOT_INITIALIZED,
20
+ ABSORBING,
21
+ FINAL,
22
+ SQUEEZING
23
+ } KCP_Phases;
24
+
25
+ #endif
@@ -542,15 +542,25 @@ http://creativecommons.org/publicdomain/zero/1.0/
542
542
  #if (PLATFORM_BYTE_ORDER == IS_LITTLE_ENDIAN)
543
543
  #define HTOLE64(x) (x)
544
544
  #else
545
- #define HTOLE64(x) (\
546
- ((x & 0xff00000000000000ull) >> 56) | \
547
- ((x & 0x00ff000000000000ull) >> 40) | \
548
- ((x & 0x0000ff0000000000ull) >> 24) | \
549
- ((x & 0x000000ff00000000ull) >> 8) | \
550
- ((x & 0x00000000ff000000ull) << 8) | \
551
- ((x & 0x0000000000ff0000ull) << 24) | \
552
- ((x & 0x000000000000ff00ull) << 40) | \
553
- ((x & 0x00000000000000ffull) << 56))
545
+ /*
546
+ * Big Endian platforms macro to swap data.
547
+ *
548
+ * NOTE: we cannot directly use the 64-bit input
549
+ * of the following macro because of possible alignment
550
+ * constraints (on some platforms such as Sparc or MIPS,
551
+ * dereferencing an uint64_t on a buffer not aligned on the
552
+ * word size will induce a 'bus error'). Instead, we read and
553
+ * swap the data byte per byte.
554
+ */
555
+ #define HTOLE64(x) \
556
+ ( ((uint64_t) (((uint8_t*)&(x))[ 7]) << 56 ) \
557
+ | ((uint64_t) (((uint8_t*)&(x))[ 6]) << 48 ) \
558
+ | ((uint64_t) (((uint8_t*)&(x))[ 5]) << 40 ) \
559
+ | ((uint64_t) (((uint8_t*)&(x))[ 4]) << 32 ) \
560
+ | ((uint64_t) (((uint8_t*)&(x))[ 3]) << 24 ) \
561
+ | ((uint64_t) (((uint8_t*)&(x))[ 2]) << 16 ) \
562
+ | ((uint64_t) (((uint8_t*)&(x))[ 1]) << 8 ) \
563
+ | ((uint64_t) (((uint8_t*)&(x))[ 0])))
554
564
  #endif
555
565
 
556
566
  #define addInput(X, input, laneCount) \
@@ -21,24 +21,30 @@ Please refer to SnP-documentation.h for more details.
21
21
  #ifndef _KeccakP_1600_SnP_h_
22
22
  #define _KeccakP_1600_SnP_h_
23
23
 
24
+ #include <stdint.h>
25
+
26
+ typedef struct {
27
+ uint32_t A[50];
28
+ } KeccakP1600_plain32_state;
29
+
30
+ typedef KeccakP1600_plain32_state KeccakP1600_state;
31
+
24
32
  #define KeccakP1600_implementation "32-bit bit-interleaved reference implementation"
25
- #define KeccakP1600_stateSizeInBytes 200
26
- #define KeccakP1600_stateAlignment 4
27
33
 
28
34
  #ifdef KeccakReference
29
35
  void KeccakP1600_StaticInitialize( void );
30
36
  #else
31
37
  #define KeccakP1600_StaticInitialize()
32
38
  #endif
33
- void KeccakP1600_Initialize(void *state);
34
- void KeccakP1600_AddByte(void *state, unsigned char data, unsigned int offset);
35
- void KeccakP1600_AddBytes(void *state, const unsigned char *data, unsigned int offset, unsigned int length);
36
- void KeccakP1600_OverwriteBytes(void *state, const unsigned char *data, unsigned int offset, unsigned int length);
37
- void KeccakP1600_OverwriteWithZeroes(void *state, unsigned int byteCount);
38
- void KeccakP1600_Permute_Nrounds(void *state, unsigned int nrounds);
39
- void KeccakP1600_Permute_12rounds(void *state);
40
- void KeccakP1600_Permute_24rounds(void *state);
41
- void KeccakP1600_ExtractBytes(const void *state, unsigned char *data, unsigned int offset, unsigned int length);
42
- void KeccakP1600_ExtractAndAddBytes(const void *state, const unsigned char *input, unsigned char *output, unsigned int offset, unsigned int length);
39
+ void KeccakP1600_Initialize(KeccakP1600_plain32_state *state);
40
+ void KeccakP1600_AddByte(KeccakP1600_plain32_state *state, unsigned char data, unsigned int offset);
41
+ void KeccakP1600_AddBytes(KeccakP1600_plain32_state *state, const unsigned char *data, unsigned int offset, unsigned int length);
42
+ void KeccakP1600_OverwriteBytes(KeccakP1600_plain32_state *state, const unsigned char *data, unsigned int offset, unsigned int length);
43
+ void KeccakP1600_OverwriteWithZeroes(KeccakP1600_plain32_state *state, unsigned int byteCount);
44
+ void KeccakP1600_Permute_Nrounds(KeccakP1600_plain32_state *state, unsigned int nrounds);
45
+ void KeccakP1600_Permute_12rounds(KeccakP1600_plain32_state *state);
46
+ void KeccakP1600_Permute_24rounds(KeccakP1600_plain32_state *state);
47
+ void KeccakP1600_ExtractBytes(const KeccakP1600_plain32_state *state, unsigned char *data, unsigned int offset, unsigned int length);
48
+ void KeccakP1600_ExtractAndAddBytes(const KeccakP1600_plain32_state *state, const unsigned char *input, unsigned char *output, unsigned int offset, unsigned int length);
43
49
 
44
50
  #endif