2018-03-08 00:41:47 -07:00
|
|
|
use jubjub::{
|
|
|
|
JubjubEngine,
|
|
|
|
PrimeOrder,
|
|
|
|
edwards
|
|
|
|
};
|
|
|
|
|
|
|
|
use pairing::{
|
2018-05-17 00:19:57 -06:00
|
|
|
PrimeField
|
2018-03-08 00:41:47 -07:00
|
|
|
};
|
|
|
|
|
2018-03-05 17:58:34 -07:00
|
|
|
use blake2_rfc::blake2s::Blake2s;
|
2018-03-08 00:09:34 -07:00
|
|
|
use constants;
|
2018-03-05 18:08:49 -07:00
|
|
|
|
2018-03-06 22:26:03 -07:00
|
|
|
/// Produces a random point in the Jubjub curve.
|
|
|
|
/// The point is guaranteed to be prime order
|
|
|
|
/// and not the identity.
|
2017-12-18 11:00:10 -07:00
|
|
|
pub fn group_hash<E: JubjubEngine>(
|
|
|
|
tag: &[u8],
|
2018-03-05 19:21:41 -07:00
|
|
|
personalization: &[u8],
|
2017-12-18 11:00:10 -07:00
|
|
|
params: &E::Params
|
2018-01-29 08:56:58 -07:00
|
|
|
) -> Option<edwards::Point<E, PrimeOrder>>
|
2017-12-18 11:00:10 -07:00
|
|
|
{
|
2018-03-05 19:21:41 -07:00
|
|
|
assert_eq!(personalization.len(), 8);
|
|
|
|
|
2017-12-18 11:00:10 -07:00
|
|
|
// Check to see that scalar field is 255 bits
|
|
|
|
assert!(E::Fr::NUM_BITS == 255);
|
|
|
|
|
2018-03-05 19:21:41 -07:00
|
|
|
let mut h = Blake2s::with_params(32, &[], &[], personalization);
|
2018-03-08 00:09:34 -07:00
|
|
|
h.update(constants::GH_FIRST_BLOCK);
|
2018-03-05 17:58:34 -07:00
|
|
|
h.update(tag);
|
2018-05-17 00:19:57 -06:00
|
|
|
let h = h.finalize().as_ref().to_vec();
|
2017-12-18 11:00:10 -07:00
|
|
|
assert!(h.len() == 32);
|
|
|
|
|
2018-05-17 00:19:57 -06:00
|
|
|
match edwards::Point::<E, _>::read(&h[..], params) {
|
|
|
|
Ok(p) => {
|
2017-12-18 11:00:10 -07:00
|
|
|
let p = p.mul_by_cofactor(params);
|
|
|
|
|
2018-01-29 08:56:58 -07:00
|
|
|
if p != edwards::Point::zero() {
|
2017-12-22 02:57:34 -07:00
|
|
|
Some(p)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
2018-05-17 00:19:57 -06:00
|
|
|
},
|
|
|
|
Err(_) => None
|
2017-12-18 11:00:10 -07:00
|
|
|
}
|
|
|
|
}
|