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.
- checksums.yaml +4 -4
- checksums.yaml.gz.sig +0 -0
- data/.clang-format +54 -0
- data/.document +3 -3
- data/.rdoc_options +11 -0
- data/.rspec +2 -2
- data/.rubocop.yml +8 -1
- data/CHANGELOG.md +23 -0
- data/Gemfile +12 -0
- data/LICENSE.txt +1 -1
- data/README.md +185 -65
- data/Rakefile +12 -4
- data/certs/io+sha3@jsg.io.pem +26 -0
- data/doc/sha3.rb +83 -0
- data/ext/sha3/config.h +2 -2
- data/ext/sha3/digest.c +726 -169
- data/ext/sha3/digest.h +6 -35
- data/ext/sha3/extconf.rb +42 -38
- data/ext/sha3/kmac.c +504 -0
- data/ext/sha3/kmac.h +14 -0
- data/ext/sha3/lib/high/Keccak/KeccakDuplex.c +81 -0
- data/ext/sha3/lib/high/Keccak/KeccakDuplex.h +73 -0
- data/ext/sha3/lib/high/Keccak/KeccakDuplex.inc +201 -0
- data/ext/sha3/lib/high/Keccak/KeccakSponge.c +2 -18
- data/ext/sha3/lib/high/Keccak/KeccakSponge.h +4 -10
- data/ext/sha3/lib/high/Keccak/KeccakSponge.inc +27 -31
- data/ext/sha3/lib/high/Keccak/PRG/KeccakPRG.c +61 -0
- data/ext/sha3/lib/high/Keccak/PRG/KeccakPRG.h +67 -0
- data/ext/sha3/lib/high/Keccak/PRG/KeccakPRG.inc +128 -0
- data/ext/sha3/lib/high/Keccak/SP800-185/SP800-185.c +93 -0
- data/ext/sha3/lib/high/Keccak/SP800-185/SP800-185.h +599 -0
- data/ext/sha3/lib/high/Keccak/SP800-185/SP800-185.inc +573 -0
- data/ext/sha3/lib/high/common/Phases.h +25 -0
- data/ext/sha3/lib/low/KeccakP-1600/common/KeccakP-1600-64.macros +19 -9
- data/ext/sha3/lib/low/KeccakP-1600/ref-32bits/KeccakP-1600-SnP.h +18 -12
- data/ext/sha3/lib/low/KeccakP-1600/ref-32bits/KeccakP-1600-reference32BI.c +28 -36
- data/ext/sha3/lib/low/KeccakP-1600/ref-64bits/KeccakP-1600-SnP.h +18 -12
- data/ext/sha3/lib/low/KeccakP-1600/ref-64bits/KeccakP-1600-reference.c +28 -59
- data/ext/sha3/lib/low/common/PlSnP-Fallback.inc +291 -0
- data/ext/sha3/lib/low/common/SnP-Relaned.h +145 -0
- data/ext/sha3/sha3.c +28 -59
- data/ext/sha3/sha3.h +4 -13
- data/lib/constants.rb +5 -0
- data/lib/sha3.rb +25 -24
- data.tar.gz.sig +0 -0
- metadata +61 -127
- metadata.gz.sig +0 -0
- data/.yardopts +0 -1
- data/ChangeLog.rdoc +0 -27
- data/certs/johanns.pem +0 -25
- data/lib/sha3/doc.rb +0 -121
- data/lib/sha3/version.rb +0 -9
- data/sha3.gemspec +0 -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
|
-
|
546
|
-
|
547
|
-
|
548
|
-
|
549
|
-
|
550
|
-
|
551
|
-
|
552
|
-
|
553
|
-
|
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(
|
34
|
-
void KeccakP1600_AddByte(
|
35
|
-
void KeccakP1600_AddBytes(
|
36
|
-
void KeccakP1600_OverwriteBytes(
|
37
|
-
void KeccakP1600_OverwriteWithZeroes(
|
38
|
-
void KeccakP1600_Permute_Nrounds(
|
39
|
-
void KeccakP1600_Permute_12rounds(
|
40
|
-
void KeccakP1600_Permute_24rounds(
|
41
|
-
void KeccakP1600_ExtractBytes(const
|
42
|
-
void KeccakP1600_ExtractAndAddBytes(const
|
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
|