ark_ec/hashing/curve_maps/
elligator2.rs1use crate::models::twisted_edwards::{MontCurveConfig, TECurveConfig};
2use ark_ff::{Field, One, Zero};
3use core::marker::PhantomData;
4
5use crate::{
6 hashing::{curve_maps::parity, map_to_curve_hasher::MapToCurve, HashToCurveError},
7 models::twisted_edwards::{Affine, Projective},
8};
9
10pub trait Elligator2Config: TECurveConfig + MontCurveConfig {
18 const Z: Self::BaseField;
23
24 const ONE_OVER_COEFF_B_SQUARE: Self::BaseField;
26
27 const COEFF_A_OVER_COEFF_B: Self::BaseField;
29}
30
31pub struct Elligator2Map<P: TECurveConfig>(PhantomData<fn() -> P>);
33
34impl<P: Elligator2Config> MapToCurve<Projective<P>> for Elligator2Map<P> {
35 fn check_parameters() -> Result<(), HashToCurveError> {
37 debug_assert!(
42 !P::Z.legendre().is_qr(),
43 "Z should be a quadratic non-residue for the Elligator2 map"
44 );
45
46 debug_assert_eq!(
47 P::ONE_OVER_COEFF_B_SQUARE,
48 <P as MontCurveConfig>::COEFF_B
49 .square()
50 .inverse()
51 .expect("B coefficient cannot be zero in Montgomery form"),
52 "ONE_OVER_COEFF_B_SQUARE is not equal to 1/COEFF_B^2 in Montgomery form"
53 );
54
55 debug_assert_eq!(
56 P::COEFF_A_OVER_COEFF_B,
57 <P as MontCurveConfig>::COEFF_A / <P as MontCurveConfig>::COEFF_B,
58 "COEFF_A_OVER_COEFF_B is not equal to COEFF_A/COEFF_B in Montgomery form"
59 );
60 Ok(())
61 }
62
63 fn map_to_curve(element: P::BaseField) -> Result<Affine<P>, HashToCurveError> {
65 let k = <P as MontCurveConfig>::COEFF_B;
79 let j_on_k = P::COEFF_A_OVER_COEFF_B;
80 let ksq_inv = P::ONE_OVER_COEFF_B_SQUARE;
81
82 let den_1 = <P::BaseField as One>::one() + P::Z * element.square();
83
84 let x1 = -j_on_k
85 / (if den_1.is_zero() {
86 <P::BaseField as One>::one()
87 } else {
88 den_1
89 });
90 let x1sq = x1.square();
91 let x1cb = x1sq * x1;
92 let gx1 = x1cb + j_on_k * x1sq + x1 * ksq_inv;
93
94 let x2 = -x1 - j_on_k;
95 let x2sq = x2.square();
96 let x2cb = x2sq * x2;
97 let gx2 = x2cb + j_on_k * x2sq + x2 * ksq_inv;
98
99 let (x, mut y, sgn0) = if gx1.legendre().is_qr() {
100 (
101 x1,
102 gx1.sqrt()
103 .expect("We have checked that gx1 is a quadratic residue. Q.E.D"),
104 true,
105 )
106 } else {
107 (
108 x2,
109 gx2.sqrt()
110 .expect("gx2 is a quadratic residue because gx1 is not. Q.E.D"),
111 false,
112 )
113 };
114
115 if parity(&y) != sgn0 {
116 y = -y;
117 }
118
119 let s = x * k;
120 let t = y * k;
121
122 let tv1 = s + <P::BaseField as One>::one();
129 let tv2 = tv1 * t;
130 let (v, w) = if tv2.is_zero() {
131 (<P::BaseField as Zero>::zero(), <P::BaseField as One>::one())
132 } else {
133 let tv2_inv = tv2
134 .inverse()
135 .expect("None zero element has inverse. Q.E.D.");
136 let v = tv2_inv * tv1 * s;
137 let w = tv2_inv * t * (s - <P::BaseField as One>::one());
138 (v, w)
139 };
140
141 let point_on_curve = Affine::<P>::new_unchecked(v, w);
142 debug_assert!(
143 point_on_curve.is_on_curve(),
144 "Elligator2 mapped to a point off the curve"
145 );
146 Ok(point_on_curve)
147 }
148}
149
150#[cfg(test)]
151mod test {
152 #[cfg(all(
153 target_has_atomic = "8",
154 target_has_atomic = "16",
155 target_has_atomic = "32",
156 target_has_atomic = "64",
157 target_has_atomic = "ptr"
158 ))]
159 type DefaultHasher = ahash::AHasher;
160
161 #[cfg(not(all(
162 target_has_atomic = "8",
163 target_has_atomic = "16",
164 target_has_atomic = "32",
165 target_has_atomic = "64",
166 target_has_atomic = "ptr"
167 )))]
168 type DefaultHasher = fnv::FnvHasher;
169
170 use crate::{
171 hashing::{map_to_curve_hasher::MapToCurveBasedHasher, HashToCurve},
172 CurveConfig,
173 };
174 use ark_ff::field_hashers::DefaultFieldHasher;
175 use ark_std::vec::*;
176
177 use super::*;
178 use ark_ff::{fields::Fp64, MontBackend, MontFp};
179 use hashbrown::HashMap;
180 use sha2::Sha256;
181
182 #[derive(ark_ff::MontConfig)]
183 #[modulus = "101"]
184 #[generator = "2"]
185 pub struct F101Config;
186 pub type F101 = Fp64<MontBackend<F101Config, 1>>;
187
188 #[derive(ark_ff::MontConfig)]
189 #[modulus = "11"]
190 #[generator = "2"]
191 pub struct F11Config;
192 pub type F11 = Fp64<MontBackend<F11Config, 1>>;
193
194 struct TestElligator2MapToCurveConfig;
195
196 impl CurveConfig for TestElligator2MapToCurveConfig {
197 const COFACTOR: &'static [u64] = &[8];
198
199 #[rustfmt::skip]
200 const COFACTOR_INV: F11 = MontFp!("7");
201
202 type BaseField = F101;
203 type ScalarField = F11;
204 }
205
206 impl TECurveConfig for TestElligator2MapToCurveConfig {
222 const COEFF_A: F101 = MontFp!("-1");
224
225 const COEFF_D: F101 = MontFp!("12");
227
228 const GENERATOR: Affine<TestElligator2MapToCurveConfig> =
229 Affine::<TestElligator2MapToCurveConfig>::new_unchecked(MontFp!("23"), MontFp!("24"));
230
231 type MontCurveConfig = TestElligator2MapToCurveConfig;
232 }
233
234 impl MontCurveConfig for TestElligator2MapToCurveConfig {
235 const COEFF_A: F101 = MontFp!("76");
237
238 const COEFF_B: F101 = MontFp!("23");
240
241 type TECurveConfig = TestElligator2MapToCurveConfig;
242 }
243
244 impl Elligator2Config for TestElligator2MapToCurveConfig {
252 const Z: F101 = MontFp!("2");
253 const ONE_OVER_COEFF_B_SQUARE: F101 = MontFp!("80");
254
255 const COEFF_A_OVER_COEFF_B: F101 = MontFp!("56");
256 }
257
258 #[test]
261 fn hash_arbitary_string_to_curve_elligator2() {
262 let test_elligator2_to_curve_hasher = MapToCurveBasedHasher::<
263 Projective<TestElligator2MapToCurveConfig>,
264 DefaultFieldHasher<Sha256, 128>,
265 Elligator2Map<TestElligator2MapToCurveConfig>,
266 >::new(&[1])
267 .unwrap();
268
269 let hash_result = test_elligator2_to_curve_hasher.hash(b"if you stick a Babel fish in your ear you can instantly understand anything said to you in any form of language.").expect("fail to hash the string to curve");
270
271 assert!(
272 hash_result.is_on_curve(),
273 "hash results into a point off the curve"
274 );
275 }
276
277 #[test]
281 fn map_field_to_curve_elligator2() {
282 Elligator2Map::<TestElligator2MapToCurveConfig>::check_parameters().unwrap();
283
284 let mut map_range: Vec<Affine<TestElligator2MapToCurveConfig>> = vec![];
285 for current_field_element in 0..101 {
288 map_range.push(
289 Elligator2Map::<TestElligator2MapToCurveConfig>::map_to_curve(F101::from(
290 current_field_element as u64,
291 ))
292 .unwrap(),
293 );
294 }
295
296 let mut counts =
297 HashMap::with_hasher(core::hash::BuildHasherDefault::<DefaultHasher>::default());
298
299 let mode = map_range
300 .iter()
301 .copied()
302 .max_by_key(|&n| {
303 let count = counts.entry(n).or_insert(0);
304 *count += 1;
305 *count
306 })
307 .unwrap();
308
309 assert!(
310 *counts.get(&mode).unwrap() != 101,
311 "a constant hash function is not good."
312 );
313 }
314}