1#[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 let num_elements = if bytes.len() % 31 == 0 {
22 bytes.len() / 31
23 } else {
24 bytes.len() / 31 + 1
25 };
26
27 let mut state = [Felt::zero(); STATE_WIDTH];
29
30 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 let chunk_len = chunk.len();
45 buf = [0u8; 32];
46 buf[..chunk_len].copy_from_slice(chunk);
47 if chunk_len < 31 {
49 buf[chunk_len] = 1;
50 }
51 }
52
53 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 Self::Digest::new(state[..DIGEST_SIZE].try_into().unwrap())
65 }
66
67 fn hash_field(elems: &[Felt]) -> Self::Digest {
68 let mut state = [Felt::zero(); STATE_WIDTH];
70
71 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 Self::Digest::new(state[..DIGEST_SIZE].try_into().unwrap())
84 }
85
86 fn merge(digests: &[Self::Digest; 2]) -> Self::Digest {
87 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 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 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 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 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 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}