anemoi/vesta/anemoi_2_1/
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_2_1, Jive, Sponge};
8use super::{DIGEST_SIZE, STATE_WIDTH};
9use crate::traits::Anemoi;
10use ark_ff::PrimeField;
11
12use super::Felt;
13use super::{One, Zero};
14
15impl Sponge<Felt> for AnemoiVesta_2_1 {
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        // Initialize the internal hash state to all zeroes.
28        let mut state = [Felt::zero(); STATE_WIDTH];
29
30        // Absorption phase
31
32        // Break the string into 31-byte chunks, then convert each chunk into a field element,
33        // and absorb the element into the rate portion of the state. The conversion is
34        // guaranteed to succeed as we spare one last byte to ensure this can represent a valid
35        // element encoding.
36        let mut buf = [0u8; 32];
37        for (i, chunk) in bytes.chunks(31).enumerate() {
38            if i < num_elements - 1 {
39                buf[0..31].copy_from_slice(chunk);
40            } else {
41                // The last chunk may be smaller than the others, which requires a special handling.
42                // In this case, we also append a byte set to 1 to the end of the string, padding the
43                // sequence in a way that adding additional trailing zeros will yield a different hash.
44                let chunk_len = chunk.len();
45                buf = [0u8; 32];
46                buf[..chunk_len].copy_from_slice(chunk);
47                // [Different to paper]: We pad the last chunk with 1 to prevent length extension attack.
48                if chunk_len < 31 {
49                    buf[chunk_len] = 1;
50                }
51            }
52
53            // Convert the bytes into a field element and absorb it into the rate portion of the
54            // state. An Anemoi permutation is applied to the internal state if all the the rate
55            // registers have been filled with additional values. We then reset the insertion index.
56            state[0] += Felt::from_le_bytes_mod_order(&buf[..]);
57            AnemoiVesta_2_1::permutation(&mut state);
58        }
59        state[STATE_WIDTH - 1] += Felt::one();
60
61        // Squeezing phase
62
63        // Finally, return the first DIGEST_SIZE elements of the state.
64        Self::Digest::new(state[..DIGEST_SIZE].try_into().unwrap())
65    }
66
67    fn hash_field(elems: &[Felt]) -> Self::Digest {
68        // initialize state to all zeros
69        let mut state = [Felt::zero(); STATE_WIDTH];
70
71        // Absorption phase
72
73        for &element in elems.iter() {
74            state[0] += element;
75            AnemoiVesta_2_1::permutation(&mut state);
76        }
77
78        state[STATE_WIDTH - 1] += Felt::one();
79
80        // Squeezing phase
81
82        // Finally, return the first DIGEST_SIZE elements of the state.
83        Self::Digest::new(state[..DIGEST_SIZE].try_into().unwrap())
84    }
85
86    fn merge(digests: &[Self::Digest; 2]) -> Self::Digest {
87        // We use internally the Jive compression method, as compressing the digests
88        // through the Sponge construction would require two internal permutation calls.
89        let result = Self::compress(&Self::Digest::digests_to_elements(digests));
90        Self::Digest::new(result.try_into().unwrap())
91    }
92}
93
94impl Jive<Felt> for AnemoiVesta_2_1 {
95    fn compress(elems: &[Felt]) -> Vec<Felt> {
96        assert!(elems.len() == STATE_WIDTH);
97
98        let mut state = elems.to_vec();
99        AnemoiVesta_2_1::permutation(&mut state);
100
101        vec![state[0] + state[1] + elems[0] + elems[1]]
102    }
103
104    fn compress_k(elems: &[Felt], k: usize) -> Vec<Felt> {
105        // This instantiation only supports Jive-2 compression mode.
106        assert!(k == 2);
107
108        Self::compress(elems)
109    }
110}
111
112#[cfg(test)]
113mod tests {
114    #[cfg(not(feature = "std"))]
115    use alloc::vec;
116
117    use super::super::MontFp;
118    use super::*;
119    use ark_ff::BigInteger;
120
121    #[test]
122    fn test_anemoi_hash() {
123        // Generated from https://github.com/anemoi-hash/anemoi-hash/
124        let input_data = [
125            vec![Felt::zero(), Felt::zero()],
126            vec![Felt::one(), Felt::one()],
127            vec![Felt::zero(), Felt::one()],
128            vec![Felt::one(), Felt::zero()],
129            vec![MontFp!(
130                "23666212284893956228650649523052266029381274089955769026913903282248093405315"
131            )],
132            vec![
133                MontFp!(
134                    "3501078521102541220367678465136405970430548314172589887937956121704171293165"
135                ),
136                MontFp!(
137                    "21515037255667073903154594949079633786175296923471185077272491054680582398738"
138                ),
139            ],
140            vec![
141                MontFp!(
142                    "26245066117244349358562045231373743633930400064101767628209065741336089170284"
143                ),
144                MontFp!(
145                    "14006547378986078708925107537283765026889558540769120987741627559095518263486"
146                ),
147                MontFp!(
148                    "16319987390108120672751685067412732420038413171290841893567961226113025211588"
149                ),
150            ],
151            vec![
152                MontFp!(
153                    "24641144407733305733584202825717792208451054286148307900846544116227059877590"
154                ),
155                MontFp!(
156                    "4228524054177173784724834792175558882315579450456758510161598683145463027415"
157                ),
158                MontFp!(
159                    "25306022459096007145609287595234936859365285245289971425284966567494172328321"
160                ),
161                MontFp!(
162                    "8175349843958286223153744941633339765213483300995083554411569483944287197510"
163                ),
164            ],
165            vec![
166                MontFp!(
167                    "8740657817957640589599155262884611618719235709713726089073814688077491547535"
168                ),
169                MontFp!(
170                    "16595167152966320362582352064315089504560491581356149685807692878013337924241"
171                ),
172                MontFp!(
173                    "14064094898173722222310755363414374056362918969089607341720950961732912315011"
174                ),
175                MontFp!(
176                    "800124860363430545351049000046721525489884728528871381794803143175020182052"
177                ),
178                MontFp!(
179                    "17623916170046312159115432199378103392440393591499193444414499557262400227578"
180                ),
181            ],
182            vec![
183                MontFp!(
184                    "8559181616607833234020874084936434756111691716976821157781711259378342247349"
185                ),
186                MontFp!(
187                    "3712032813022748960353736644218860630753855350311785475557015515761931707021"
188                ),
189                MontFp!(
190                    "18244586813745203405426970765128722714527216813367626489683188870729460745061"
191                ),
192                MontFp!(
193                    "4625304081143527908085723806207105973411010421613082699340143544709821862785"
194                ),
195                MontFp!(
196                    "1579643740689782728856536923239854836285170264865496072329853825108973138439"
197                ),
198                MontFp!(
199                    "26615051575295251925147288423474991536248319820477111544037388634970756375021"
200                ),
201            ],
202        ];
203
204        let output_data = [
205            [MontFp!(
206                "3675563954394682696996536055507407419428734133672283731223142852773229935377"
207            )],
208            [MontFp!(
209                "2229117904311058771450486580798876451943541707571959402671744982352730393011"
210            )],
211            [MontFp!(
212                "7188444258209023690460633892327566210376620195026849798399800429667062149490"
213            )],
214            [MontFp!(
215                "9466600354778764410480247555459461306567613323441414865846416198947210643152"
216            )],
217            [MontFp!(
218                "22306538566530453062617248694580445489296260848593489687893794609592376195218"
219            )],
220            [MontFp!(
221                "23212306122676917261335277337636005824963471853164563343020044774792071907230"
222            )],
223            [MontFp!(
224                "14610922761737438776547487420770558826022222657127880833279093942376034354577"
225            )],
226            [MontFp!(
227                "4117277901691807903515683082396918651140029386316771349589349676343065811802"
228            )],
229            [MontFp!(
230                "25932136303950551788718747693050119626519481083317358740958528616374287701013"
231            )],
232            [MontFp!(
233                "13391622674438414062657793529140174149205584287960651726917252462804048527287"
234            )],
235        ];
236
237        for (input, expected) in input_data.iter().zip(output_data) {
238            assert_eq!(expected, AnemoiVesta_2_1::hash_field(input).to_elements());
239        }
240    }
241
242    #[test]
243    fn test_anemoi_hash_bytes() {
244        // Generated from https://github.com/anemoi-hash/anemoi-hash/
245        let input_data = [
246            vec![Felt::zero(), Felt::zero()],
247            vec![Felt::one(), Felt::one()],
248            vec![Felt::zero(), Felt::one()],
249            vec![Felt::one(), Felt::zero()],
250        ];
251
252        let output_data = [
253            [MontFp!(
254                "3675563954394682696996536055507407419428734133672283731223142852773229935377"
255            )],
256            [MontFp!(
257                "2229117904311058771450486580798876451943541707571959402671744982352730393011"
258            )],
259            [MontFp!(
260                "7188444258209023690460633892327566210376620195026849798399800429667062149490"
261            )],
262            [MontFp!(
263                "9466600354778764410480247555459461306567613323441414865846416198947210643152"
264            )],
265        ];
266
267        // The inputs can all be represented with at least 1 byte less than the field size,
268        // hence computing the Anemoi hash digest from the byte sequence yields the same
269        // result as treating the inputs as field elements.
270        for (input, expected) in input_data.iter().zip(output_data) {
271            let mut bytes = [0u8; 62];
272            bytes[0..31].copy_from_slice(&input[0].into_bigint().to_bytes_le()[0..31]);
273            bytes[31..62].copy_from_slice(&input[1].into_bigint().to_bytes_le()[0..31]);
274
275            assert_eq!(expected, AnemoiVesta_2_1::hash(&bytes).to_elements());
276        }
277    }
278
279    #[test]
280    fn test_anemoi_jive() {
281        // Generated from https://github.com/anemoi-hash/anemoi-hash/
282        let input_data = [
283            vec![Felt::zero(), Felt::zero()],
284            vec![Felt::one(), Felt::one()],
285            vec![Felt::zero(), Felt::one()],
286            vec![Felt::one(), Felt::zero()],
287        ];
288
289        let output_data = [
290            [MontFp!(
291                "25021493149962073135159779742595331542889244616612316327827500113201517573552"
292            )],
293            [MontFp!(
294                "1799222279508491238955156019299185816766170120519060796492407909371488003482"
295            )],
296            [MontFp!(
297                "3312133520551670415496942812397107034820036209574409986611422513836613460957"
298            )],
299            [MontFp!(
300                "3814795182979631662421649092648209217376074705260062533460857166180948182756"
301            )],
302        ];
303
304        for (input, expected) in input_data.iter().zip(output_data) {
305            assert_eq!(expected.to_vec(), AnemoiVesta_2_1::compress(input));
306        }
307
308        for (input, expected) in input_data.iter().zip(output_data) {
309            assert_eq!(expected.to_vec(), AnemoiVesta_2_1::compress_k(input, 2));
310        }
311
312        for (input, expected) in input_data.iter().zip(output_data) {
313            assert_eq!(
314                expected,
315                AnemoiVesta_2_1::merge(&[
316                    AnemoiDigest::new([input[0]]),
317                    AnemoiDigest::new([input[1]])
318                ])
319                .to_elements()
320            );
321        }
322    }
323}