anemoi/vesta/anemoi_4_3/
hasher.rs

1//! Sponge trait implementation for Anemoi
2
3#[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        // Compute the number of field elements required to represent this
20        // sequence of bytes.
21        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        // Initialize the internal hash state to all zeroes.
34        let mut state = [Felt::zero(); STATE_WIDTH];
35
36        // Absorption phase
37
38        // Break the string into 31-byte chunks, then convert each chunk into a field element,
39        // and absorb the element into the rate portion of the state. The conversion is
40        // guaranteed to succeed as we spare one last byte to ensure this can represent a valid
41        // element encoding.
42        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                // The last chunk may be smaller than the others, which requires a special handling.
50                // In this case, we also append a byte set to 1 to the end of the string, padding the
51                // sequence in a way that adding additional trailing zeros will yield a different hash.
52                let chunk_len = chunk.len();
53                buf = [0u8; 32];
54                buf[..chunk_len].copy_from_slice(chunk);
55                // [Different to paper]: We pad the last chunk with 1 to prevent length extension attack.
56                if chunk_len < 31 {
57                    buf[chunk_len] = 1;
58                }
59            }
60
61            // Convert the bytes into a field element and absorb it into the rate portion of the
62            // state. An Anemoi permutation is applied to the internal state if all the the rate
63            // registers have been filled with additional values. We then reset the insertion index.
64            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        // We then add sigma to the last register of the capacity.
74        state[STATE_WIDTH - 1] += sigma;
75
76        // If the message length is not a multiple of RATE_WIDTH, we append 1 to the rate cell
77        // next to the one where we previously appended the last message element. This is
78        // guaranted to be in the rate registers (i.e. to not require an extra permutation before
79        // adding this constant) if sigma is equal to zero. We then apply a final Anemoi permutation
80        // to the whole state.
81        if sigma.is_zero() {
82            state[i] += Felt::one();
83            AnemoiVesta_4_3::permutation(&mut state);
84        }
85
86        // Squeezing phase
87
88        // Finally, return the first DIGEST_SIZE elements of the state.
89        Self::Digest::new(state[..DIGEST_SIZE].try_into().unwrap())
90    }
91
92    fn hash_field(elems: &[Felt]) -> Self::Digest {
93        // initialize state to all zeros
94        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        // We then add sigma to the last register of the capacity.
113        state[STATE_WIDTH - 1] += sigma;
114
115        // If the message length is not a multiple of RATE_WIDTH, we append 1 to the rate cell
116        // next to the one where we previously appended the last message element. This is
117        // guaranted to be in the rate registers (i.e. to not require an extra permutation before
118        // adding this constant) if sigma is equal to zero. We then apply a final Anemoi permutation
119        // to the whole state.
120        if sigma.is_zero() {
121            state[i] += Felt::one();
122            AnemoiVesta_4_3::permutation(&mut state);
123        }
124
125        // Squeezing phase
126
127        Self::Digest::new(state[..DIGEST_SIZE].try_into().unwrap())
128    }
129
130    fn merge(digests: &[Self::Digest; 2]) -> Self::Digest {
131        // initialize state to all zeros
132        let mut state = [Felt::zero(); STATE_WIDTH];
133
134        // 2*DIGEST_SIZE < RATE_SIZE so we can safely store
135        // the digests into the rate registers at once
136        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        // Apply internal Anemoi permutation
140        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        // Generated from https://github.com/anemoi-hash/anemoi-hash/
193        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        // Generated from https://github.com/anemoi-hash/anemoi-hash/
314        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        // The inputs can all be represented with at least 1 byte less than the field size,
337        // hence computing the Anemoi hash digest from the byte sequence yields the same
338        // result as treating the inputs as field elements.
339        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        // Generated from https://github.com/anemoi-hash/anemoi-hash/
353        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}