anemoi/pallas/anemoi_2_1/
hasher.rs1#[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 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 AnemoiPallas_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 AnemoiPallas_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 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 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 "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 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 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 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}