anemoi/pallas/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::{AnemoiPallas_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 AnemoiPallas_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            AnemoiPallas_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            AnemoiPallas_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 AnemoiPallas_2_1 {
95    fn compress(elems: &[Felt]) -> Vec<Felt> {
96        assert!(elems.len() == STATE_WIDTH);
97
98        let mut state = elems.to_vec();
99        AnemoiPallas_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                "8786344908157484467702144443147728582570609298051356647558148413464066108428"
131            )],
132            vec![
133                MontFp!(
134                    "3703611713748744165229098306080671845129048809115003457470291999105546486325"
135                ),
136                MontFp!(
137                    "24153765309843050828623012216779077908319465245142980318434291969963354926334"
138                ),
139            ],
140            vec![
141                MontFp!(
142                    "358913411029015325318138375435731803790992617178880485013196553608252539481"
143                ),
144                MontFp!(
145                    "8230022665545053873997498884151270366890470487501907178234159557579737481432"
146                ),
147                MontFp!(
148                    "2047386461740345715599555656917867466628251271969723740633193819624724688488"
149                ),
150            ],
151            vec![
152                MontFp!(
153                    "26036341263213017331904003562532912502792005336457047110314507104201169419876"
154                ),
155                MontFp!(
156                    "24062649992427728021819494567227387940731985117940055830958150360565836942248"
157                ),
158                MontFp!(
159                    "8921365708708167109938294177011741430366329121282058520981996333836959700069"
160                ),
161                MontFp!(
162                    "7195157583406215915072568967368134796399291042697277049061440998502574921737"
163                ),
164            ],
165            vec![
166                MontFp!(
167                    "21356927064953552060503431555205407909741461873987532728641908180721700084840"
168                ),
169                MontFp!(
170                    "7966758083609056599281293435577685235629680655349403837025696737240329722986"
171                ),
172                MontFp!(
173                    "24135357023371509759002187273182309193982925096610998799260755998614247349066"
174                ),
175                MontFp!(
176                    "10588941192463807739722749656640339239064042571914912104463662343538225353648"
177                ),
178                MontFp!(
179                    "6839118740961792995837243286714037465343480861961456510297280599484025888117"
180                ),
181            ],
182            vec![
183                MontFp!(
184                    "24183859570943675199564758425190682354367562032419636289128612513970763668072"
185                ),
186                MontFp!(
187                    "18915354595390571159250786067138004725796825626596000701464917307766907388790"
188                ),
189                MontFp!(
190                    "23236759124405113083148363509694740555047269126008151654831625763404594146823"
191                ),
192                MontFp!(
193                    "8820782127262986192829346492854126768655317388934818603090790012122262224323"
194                ),
195                MontFp!(
196                    "15314973457919135327168897691453063865004877213365317319046997450929013432725"
197                ),
198                MontFp!(
199                    "26717381452276117819889657178273802186002762982479594599901892137785841519703"
200                ),
201            ],
202        ];
203
204        let output_data = [
205            [MontFp!(
206                "27921818220805746051987267737218857588847751174833938699888112568779408580714"
207            )],
208            [MontFp!(
209                "10003429244477285855747230809015634417193821429428555712832755192897533334106"
210            )],
211            [MontFp!(
212                "13290391632215095401717364136096553912180237160090779341925788216826661533240"
213            )],
214            [MontFp!(
215                "91993706971719007054866387319808123857126926105539303153205824718341931836"
216            )],
217            [MontFp!(
218                "4542202078280494022684418795562703153571828559807751715843580237564183190018"
219            )],
220            [MontFp!(
221                "6189149292975300181581176863094298311283007276406715688822500697364177384640"
222            )],
223            [MontFp!(
224                "20772154299187801558781683699482888401245070406146164128826035801084072271363"
225            )],
226            [MontFp!(
227                "12766169934011665030732729437530902277038872474141285443644059071867016845440"
228            )],
229            [MontFp!(
230                "7247086375400245676536480550203037779342035687559407087041004416219288636815"
231            )],
232            [MontFp!(
233                "26406354915723717036607769016580923969246177561932524143753201372077323307043"
234            )],
235        ];
236
237        for (input, expected) in input_data.iter().zip(output_data) {
238            assert_eq!(expected, AnemoiPallas_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                "27921818220805746051987267737218857588847751174833938699888112568779408580714"
255            )],
256            [MontFp!(
257                "10003429244477285855747230809015634417193821429428555712832755192897533334106"
258            )],
259            [MontFp!(
260                "13290391632215095401717364136096553912180237160090779341925788216826661533240"
261            )],
262            [MontFp!(
263                "91993706971719007054866387319808123857126926105539303153205824718341931836"
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, AnemoiPallas_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                "25339990266271823042514121456004268429320463644759496063425849229386076340157"
292            )],
293            [MontFp!(
294                "13925721908743741981737180373217062710291818499780666961262790653502874125673"
295            )],
296            [MontFp!(
297                "8161124099286768706096187391267581535071065397484800021294905474746837811202"
298            )],
299            [MontFp!(
300                "22464570152859425264804262779232945996568499466977613824889819037772248156041"
301            )],
302        ];
303
304        for (input, expected) in input_data.iter().zip(output_data) {
305            assert_eq!(expected.to_vec(), AnemoiPallas_2_1::compress(input));
306        }
307
308        for (input, expected) in input_data.iter().zip(output_data) {
309            assert_eq!(expected.to_vec(), AnemoiPallas_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                AnemoiPallas_2_1::merge(&[
316                    AnemoiDigest::new([input[0]]),
317                    AnemoiDigest::new([input[1]])
318                ])
319                .to_elements()
320            );
321        }
322    }
323}