1#[cfg(not(feature = "std"))]
4use alloc::vec::Vec;
5
6use super::digest::AnemoiDigest;
7use super::{AnemoiVesta_4_3, Jive, Sponge};
8use super::{DIGEST_SIZE, NUM_COLUMNS, RATE_WIDTH, STATE_WIDTH};
9use crate::traits::Anemoi;
10use ark_ff::PrimeField;
11
12use super::Felt;
13use super::{One, Zero};
14
15impl Sponge<Felt> for AnemoiVesta_4_3 {
16 type Digest = AnemoiDigest;
17
18 fn hash(bytes: &[u8]) -> Self::Digest {
19 let num_elements = if bytes.len() % 31 == 0 {
22 bytes.len() / 31
23 } else {
24 bytes.len() / 31 + 1
25 };
26
27 let sigma = if num_elements % RATE_WIDTH == 0 {
28 Felt::one()
29 } else {
30 Felt::zero()
31 };
32
33 let mut state = [Felt::zero(); STATE_WIDTH];
35
36 let mut i = 0;
43 let mut num_hashed = 0;
44 let mut buf = [0u8; 32];
45 for chunk in bytes.chunks(31) {
46 if num_hashed + i < num_elements - 1 {
47 buf[..31].copy_from_slice(chunk);
48 } else {
49 let chunk_len = chunk.len();
53 buf = [0u8; 32];
54 buf[..chunk_len].copy_from_slice(chunk);
55 if chunk_len < 31 {
57 buf[chunk_len] = 1;
58 }
59 }
60
61 state[i] += Felt::from_le_bytes_mod_order(&buf[..]);
65 i += 1;
66 if i % RATE_WIDTH == 0 {
67 AnemoiVesta_4_3::permutation(&mut state);
68 i = 0;
69 num_hashed += RATE_WIDTH;
70 }
71 }
72
73 state[STATE_WIDTH - 1] += sigma;
75
76 if sigma.is_zero() {
82 state[i] += Felt::one();
83 AnemoiVesta_4_3::permutation(&mut state);
84 }
85
86 Self::Digest::new(state[..DIGEST_SIZE].try_into().unwrap())
90 }
91
92 fn hash_field(elems: &[Felt]) -> Self::Digest {
93 let mut state = [Felt::zero(); STATE_WIDTH];
95
96 let sigma = if elems.len() % RATE_WIDTH == 0 {
97 Felt::one()
98 } else {
99 Felt::zero()
100 };
101
102 let mut i = 0;
103 for &element in elems.iter() {
104 state[i] += element;
105 i += 1;
106 if i % RATE_WIDTH == 0 {
107 AnemoiVesta_4_3::permutation(&mut state);
108 i = 0;
109 }
110 }
111
112 state[STATE_WIDTH - 1] += sigma;
114
115 if sigma.is_zero() {
121 state[i] += Felt::one();
122 AnemoiVesta_4_3::permutation(&mut state);
123 }
124
125 Self::Digest::new(state[..DIGEST_SIZE].try_into().unwrap())
128 }
129
130 fn merge(digests: &[Self::Digest; 2]) -> Self::Digest {
131 let mut state = [Felt::zero(); STATE_WIDTH];
133
134 state[0..DIGEST_SIZE].copy_from_slice(digests[0].as_elements());
137 state[DIGEST_SIZE..2 * DIGEST_SIZE].copy_from_slice(digests[0].as_elements());
138
139 AnemoiVesta_4_3::permutation(&mut state);
141
142 Self::Digest::new(state[..DIGEST_SIZE].try_into().unwrap())
143 }
144}
145
146impl Jive<Felt> for AnemoiVesta_4_3 {
147 fn compress(elems: &[Felt]) -> Vec<Felt> {
148 assert!(elems.len() == STATE_WIDTH);
149
150 let mut state = elems.to_vec();
151 AnemoiVesta_4_3::permutation(&mut state);
152
153 let mut result = [Felt::zero(); NUM_COLUMNS];
154 for (i, r) in result.iter_mut().enumerate() {
155 *r = elems[i] + elems[i + NUM_COLUMNS] + state[i] + state[i + NUM_COLUMNS];
156 }
157
158 result.to_vec()
159 }
160
161 fn compress_k(elems: &[Felt], k: usize) -> Vec<Felt> {
162 assert!(elems.len() == STATE_WIDTH);
163 assert!(STATE_WIDTH % k == 0);
164 assert!(k % 2 == 0);
165
166 let mut state = elems.to_vec();
167 AnemoiVesta_4_3::permutation(&mut state);
168
169 let mut result = vec![Felt::zero(); STATE_WIDTH / k];
170 let c = result.len();
171 for (i, r) in result.iter_mut().enumerate() {
172 for j in 0..k {
173 *r += elems[i + c * j] + state[i + c * j];
174 }
175 }
176
177 result
178 }
179}
180
181#[cfg(test)]
182mod tests {
183 #[cfg(not(feature = "std"))]
184 use alloc::vec;
185
186 use super::super::MontFp;
187 use super::*;
188 use ark_ff::BigInteger;
189
190 #[test]
191 fn test_anemoi_hash() {
192 let input_data = [
194 vec![Felt::zero(), Felt::zero(), Felt::zero(), Felt::zero()],
195 vec![Felt::one(), Felt::one(), Felt::one(), Felt::one()],
196 vec![Felt::zero(), Felt::zero(), Felt::one(), Felt::one()],
197 vec![Felt::one(), Felt::one(), Felt::zero(), Felt::zero()],
198 vec![MontFp!(
199 "7764516216146978186543732804836859542970724356478521491256477355492627411676"
200 )],
201 vec![
202 MontFp!(
203 "22907694841645630585206426872085849258295842985677302589412328525870756066394"
204 ),
205 MontFp!(
206 "4169673177469938832983408323556363511312758627908452193980748118915671330837"
207 ),
208 ],
209 vec![
210 MontFp!(
211 "15435395061201999509908033209911731823908765952074391516984626065576002144425"
212 ),
213 MontFp!(
214 "1373703704887686828762179144173833609201564828560942020455143134110608444233"
215 ),
216 MontFp!(
217 "20363545585638743235928574358837257410927557124964422516999296287451458583899"
218 ),
219 ],
220 vec![
221 MontFp!(
222 "19737755197119505596565023771317041889892926723277350338870313434841836325558"
223 ),
224 MontFp!(
225 "23729531450816876722219741944080547049408566868054130749873492469628373649772"
226 ),
227 MontFp!(
228 "26204513837294252303065523757180069529491792275684413266576931716040028415701"
229 ),
230 MontFp!(
231 "5497113521591738232212478403300478363119804014193035879001932507184179361880"
232 ),
233 ],
234 vec![
235 MontFp!(
236 "13375370216302311521700013725347609399532334479246690464800191396923778241218"
237 ),
238 MontFp!(
239 "14876027157593517139229423836996423271630362084655756694521458791463669911260"
240 ),
241 MontFp!(
242 "16702498949380500451475191924085103265198600790916501051205760468419176003508"
243 ),
244 MontFp!(
245 "6695341102210164665179608171900587634238989653216410510668285866345844659563"
246 ),
247 MontFp!(
248 "17339489542647227971818078732159861692512722239006820520684287465025064770535"
249 ),
250 ],
251 vec![
252 MontFp!(
253 "18371458419042225327196408930995066412765059814499030652366683114741124920534"
254 ),
255 MontFp!(
256 "26104863697817533244632757166247186677246954963079946787997558864637706562787"
257 ),
258 MontFp!(
259 "12059940180902035024688447118770018806865296733534891123984848371927300652705"
260 ),
261 MontFp!(
262 "176760369206591843207298018372745619374153399069720199050588194776740716420"
263 ),
264 MontFp!(
265 "7879750185791152904736606040999994226397391884337383165443171129026430613343"
266 ),
267 MontFp!(
268 "21922030067662672999471329324457178633313324829063788722213944047268018712948"
269 ),
270 ],
271 ];
272
273 let output_data = [
274 [MontFp!(
275 "18477755575328750816885850798387952933259583428511618735955991488059329852802"
276 )],
277 [MontFp!(
278 "1089622155824349342136353788060036422274330799810677835411793924529134549007"
279 )],
280 [MontFp!(
281 "23383407270625247748646649290487223541381602019790071288068181519850316119688"
282 )],
283 [MontFp!(
284 "15333143339905011434961662026430877746311759859432025806436665255558283094312"
285 )],
286 [MontFp!(
287 "7957074743442196816441071837512213774788799178524178087337708054612561105779"
288 )],
289 [MontFp!(
290 "27964596164958766866855168409189386623161234288265058577280237576522926353764"
291 )],
292 [MontFp!(
293 "8832233859471293253974233791949469443578850234406402147920527938156237262199"
294 )],
295 [MontFp!(
296 "15732110983354407176919337835090149963515384873765009451440597278637567592368"
297 )],
298 [MontFp!(
299 "10330960395688575039826453108111489685452424322682532413106886720241532044898"
300 )],
301 [MontFp!(
302 "28601285983493688442854476172888368799474301316045325162155888974354765051381"
303 )],
304 ];
305
306 for (input, expected) in input_data.iter().zip(output_data) {
307 assert_eq!(expected, AnemoiVesta_4_3::hash_field(input).to_elements());
308 }
309 }
310
311 #[test]
312 fn test_anemoi_hash_bytes() {
313 let input_data = [
315 vec![Felt::zero(); 4],
316 vec![Felt::one(); 4],
317 vec![Felt::zero(), Felt::zero(), Felt::one(), Felt::one()],
318 vec![Felt::one(), Felt::one(), Felt::zero(), Felt::zero()],
319 ];
320
321 let output_data = [
322 [MontFp!(
323 "18477755575328750816885850798387952933259583428511618735955991488059329852802"
324 )],
325 [MontFp!(
326 "1089622155824349342136353788060036422274330799810677835411793924529134549007"
327 )],
328 [MontFp!(
329 "23383407270625247748646649290487223541381602019790071288068181519850316119688"
330 )],
331 [MontFp!(
332 "15333143339905011434961662026430877746311759859432025806436665255558283094312"
333 )],
334 ];
335
336 for (input, expected) in input_data.iter().zip(output_data) {
340 let mut bytes = [0u8; 124];
341 bytes[0..31].copy_from_slice(&input[0].into_bigint().to_bytes_le()[0..31]);
342 bytes[31..62].copy_from_slice(&input[1].into_bigint().to_bytes_le()[0..31]);
343 bytes[62..93].copy_from_slice(&input[2].into_bigint().to_bytes_le()[0..31]);
344 bytes[93..124].copy_from_slice(&input[3].into_bigint().to_bytes_le()[0..31]);
345
346 assert_eq!(expected, AnemoiVesta_4_3::hash(&bytes).to_elements());
347 }
348 }
349
350 #[test]
351 fn test_anemoi_jive() {
352 let input_data = [
354 vec![Felt::zero(), Felt::zero(), Felt::zero(), Felt::zero()],
355 vec![Felt::one(), Felt::one(), Felt::one(), Felt::one()],
356 vec![Felt::zero(), Felt::zero(), Felt::one(), Felt::one()],
357 vec![Felt::one(), Felt::one(), Felt::zero(), Felt::zero()],
358 ];
359
360 let output_data = [
361 [
362 MontFp!(
363 "17814074569133619376001583724909369837639006754012602478069266346597252318717"
364 ),
365 MontFp!(
366 "4645324191354213145361406141970260204294326396494284352043313444215651448956"
367 ),
368 ],
369 [
370 MontFp!(
371 "24893920820224764770810638096592977772418417663317024760390036326700338198854"
372 ),
373 MontFp!(
374 "15860435497577179224758206597683167799910420450474978252299280484428162460455"
375 ),
376 ],
377 [
378 MontFp!(
379 "27126766676817950878771190712975642236437683248501595320269350109633958938314"
380 ),
381 MontFp!(
382 "14615425120361368767589882074890977200866828715767578149241602916394899244813"
383 ),
384 ],
385 [
386 MontFp!(
387 "16280243636991791422017348691987967083568430057992986467671369808716501319725"
388 ),
389 MontFp!(
390 "12714680685026350363019468164077713565648590311659840387446374044983187263140"
391 ),
392 ],
393 ];
394
395 for (input, expected) in input_data.iter().zip(output_data) {
396 assert_eq!(expected.to_vec(), AnemoiVesta_4_3::compress(input));
397 }
398
399 for (input, expected) in input_data.iter().zip(output_data) {
400 assert_eq!(expected.to_vec(), AnemoiVesta_4_3::compress_k(input, 2));
401 }
402
403 let input_data = [
404 vec![Felt::zero(), Felt::zero(), Felt::zero(), Felt::zero()],
405 vec![Felt::one(), Felt::one(), Felt::one(), Felt::one()],
406 vec![Felt::zero(), Felt::zero(), Felt::one(), Felt::one()],
407 vec![Felt::one(), Felt::one(), Felt::zero(), Felt::zero()],
408 ];
409
410 let output_data = [
411 [MontFp!(
412 "22459398760487832521362989866879630041933333150506886830112579790812903767673"
413 )],
414 [MontFp!(
415 "11806334008472895139676098442104168608965781631850355633009574062735137711212"
416 )],
417 [MontFp!(
418 "12794169487850270790468326535694642473941455482327526089831210277635495235030"
419 )],
420 [MontFp!(
421 "46902012689092929144070603893703685853963887711179475438001105306325634768"
422 )],
423 ];
424
425 for (input, expected) in input_data.iter().zip(output_data) {
426 assert_eq!(expected.to_vec(), AnemoiVesta_4_3::compress_k(input, 4));
427 }
428 }
429}