mirror of
https://github.com/darkrenaissance/darkfi.git
synced 2026-04-28 03:00:18 -04:00
halo2: WIP Mint circuit
This commit is contained in:
1
examples/halo2/Cargo.lock
generated
1
examples/halo2/Cargo.lock
generated
@@ -171,6 +171,7 @@ dependencies = [
|
||||
"group",
|
||||
"halo2",
|
||||
"pasta_curves",
|
||||
"rand",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
|
||||
@@ -11,6 +11,7 @@ ff = "0.10"
|
||||
group = "0.10"
|
||||
pasta_curves = "0.1.2"
|
||||
bitvec = "0.22"
|
||||
rand = "0.8.4"
|
||||
|
||||
[dependencies.halo2]
|
||||
git = "https://github.com/zcash/halo2.git"
|
||||
|
||||
397
examples/halo2/src/bin/mint.rs
Normal file
397
examples/halo2/src/bin/mint.rs
Normal file
@@ -0,0 +1,397 @@
|
||||
use std::{convert::TryInto, time::Instant};
|
||||
|
||||
use group::{ff::Field, Curve, Group, GroupEncoding};
|
||||
use rand::rngs::OsRng;
|
||||
|
||||
use halo2::{
|
||||
arithmetic::{CurveAffine, CurveExt, FieldExt},
|
||||
circuit::{floor_planner, Layouter, SimpleFloorPlanner},
|
||||
dev::MockProver,
|
||||
pasta::{pallas, vesta, Ep, Eq, Fp, Fq},
|
||||
plonk,
|
||||
plonk::{
|
||||
Advice, Circuit, Column, ConstraintSystem, Error, Instance as InstanceColumn, Selector,
|
||||
},
|
||||
poly::{commitment, Rotation},
|
||||
transcript::{Blake2bRead, Blake2bWrite},
|
||||
};
|
||||
|
||||
use halo2_examples::circuit::gadget::{
|
||||
poseidon::{
|
||||
Hash as PoseidonHash, Pow5T3Chip as PoseidonChip, Pow5T3Config as PoseidonConfig,
|
||||
StateWord, Word,
|
||||
},
|
||||
utilities::{copy, CellValue, UtilitiesInstructions, Var},
|
||||
};
|
||||
use halo2_examples::primitives::poseidon::{ConstantLength, Hash, OrchardNullifier};
|
||||
|
||||
const K: u32 = 12;
|
||||
const VALUE_COMMITMENT_PERSONALIZATION: &str = "darkfi:Orchard-cv";
|
||||
const VALUE_COMMITMENT_V_BYTES: [u8; 1] = *b"v";
|
||||
const VALUE_COMMITMENT_R_BYTES: [u8; 1] = *b"r";
|
||||
|
||||
#[derive(Clone, Debug)]
|
||||
struct Config {
|
||||
primary: Column<InstanceColumn>,
|
||||
q_add: Selector,
|
||||
q_mul: Selector,
|
||||
advices: [Column<Advice>; 10],
|
||||
poseidon_config: PoseidonConfig<Fp>,
|
||||
}
|
||||
|
||||
#[derive(Default, Debug)]
|
||||
struct MintCircuit {
|
||||
pub_x: Option<Fp>, // x coordinate for pubkey
|
||||
pub_y: Option<Fp>, // y coordinate for pubkey
|
||||
value: Option<Fp>, // The value of this coin
|
||||
asset: Option<Fp>, // The asset ID
|
||||
serial: Option<Fp>, // Unique serial number corresponding to this coin
|
||||
coin_blind: Option<Fp>, // Random blinding factor for coin
|
||||
value_blind: Option<Fq>, // Random blinding factor for value commitment
|
||||
asset_blind: Option<Fq>, // Random blinding factor for the asset ID
|
||||
}
|
||||
|
||||
impl UtilitiesInstructions<Fp> for MintCircuit {
|
||||
type Var = CellValue<Fp>;
|
||||
}
|
||||
|
||||
impl Circuit<Fp> for MintCircuit {
|
||||
type Config = Config;
|
||||
type FloorPlanner = floor_planner::V1;
|
||||
//type FloorPlanner = SimpleFloorPlanner;
|
||||
|
||||
fn without_witnesses(&self) -> Self {
|
||||
Self::default()
|
||||
}
|
||||
|
||||
fn configure(meta: &mut ConstraintSystem<Fp>) -> Self::Config {
|
||||
let advices = [
|
||||
meta.advice_column(),
|
||||
meta.advice_column(),
|
||||
meta.advice_column(),
|
||||
meta.advice_column(),
|
||||
meta.advice_column(),
|
||||
meta.advice_column(),
|
||||
meta.advice_column(),
|
||||
meta.advice_column(),
|
||||
meta.advice_column(),
|
||||
meta.advice_column(),
|
||||
];
|
||||
|
||||
let q_add = meta.selector();
|
||||
let q_mul = meta.selector();
|
||||
|
||||
let primary = meta.instance_column();
|
||||
|
||||
meta.enable_equality(primary.into());
|
||||
|
||||
for advice in advices.iter() {
|
||||
meta.enable_equality((*advice).into());
|
||||
}
|
||||
|
||||
let lagrange_coeffs = [
|
||||
meta.fixed_column(),
|
||||
meta.fixed_column(),
|
||||
meta.fixed_column(),
|
||||
meta.fixed_column(),
|
||||
meta.fixed_column(),
|
||||
meta.fixed_column(),
|
||||
meta.fixed_column(),
|
||||
meta.fixed_column(),
|
||||
meta.fixed_column(),
|
||||
meta.fixed_column(),
|
||||
meta.fixed_column(),
|
||||
];
|
||||
|
||||
let rc_a = lagrange_coeffs[2..5].try_into().unwrap();
|
||||
let rc_b = lagrange_coeffs[5..8].try_into().unwrap();
|
||||
|
||||
meta.enable_constant(lagrange_coeffs[0]);
|
||||
|
||||
let poseidon_config = PoseidonChip::configure(
|
||||
meta,
|
||||
OrchardNullifier,
|
||||
advices[6..9].try_into().unwrap(),
|
||||
advices[5],
|
||||
rc_a,
|
||||
rc_b,
|
||||
);
|
||||
|
||||
Config {
|
||||
primary,
|
||||
q_add,
|
||||
q_mul,
|
||||
advices,
|
||||
poseidon_config,
|
||||
}
|
||||
}
|
||||
|
||||
fn synthesize(
|
||||
&self,
|
||||
config: Self::Config,
|
||||
mut layouter: impl Layouter<Fp>,
|
||||
) -> Result<(), Error> {
|
||||
let pub_x = self.load_private(
|
||||
layouter.namespace(|| "load pubkey x"),
|
||||
config.advices[0],
|
||||
self.pub_x,
|
||||
)?;
|
||||
let pub_y = self.load_private(
|
||||
layouter.namespace(|| "load pubkey y"),
|
||||
config.advices[0],
|
||||
self.pub_y,
|
||||
)?;
|
||||
let value = self.load_private(
|
||||
layouter.namespace(|| "load value"),
|
||||
config.advices[0],
|
||||
self.value,
|
||||
)?;
|
||||
let asset = self.load_private(
|
||||
layouter.namespace(|| "load asset"),
|
||||
config.advices[0],
|
||||
self.asset,
|
||||
)?;
|
||||
let serial = self.load_private(
|
||||
layouter.namespace(|| "load serial"),
|
||||
config.advices[0],
|
||||
self.serial,
|
||||
)?;
|
||||
let coin_blind = self.load_private(
|
||||
layouter.namespace(|| "load coin_blind"),
|
||||
config.advices[0],
|
||||
self.coin_blind,
|
||||
)?;
|
||||
|
||||
/*
|
||||
let value_blind = self.load_private(
|
||||
layouter.namespace(|| "load value_blind"),
|
||||
config.advices[0],
|
||||
self.value_blind,
|
||||
)?;
|
||||
let asset_blind = self.load_private(
|
||||
layouter.namespace(|| "load asset_blind"),
|
||||
config.advices[0],
|
||||
self.asset_blind,
|
||||
)?;
|
||||
*/
|
||||
|
||||
let mut coin = Fp::zero();
|
||||
|
||||
// TODO: See if the Poseidon gadget can somehow hash more than H(a,b)
|
||||
let messages = [[pub_x, pub_y], [value, asset], [serial, coin_blind]];
|
||||
//let messages = [[pub_x, pub_y], [value, asset]];
|
||||
//let messages = [[pub_x, pub_y]];
|
||||
|
||||
for msg in messages.iter() {
|
||||
let poseidon_message = layouter.assign_region(
|
||||
|| "load message",
|
||||
|mut region| {
|
||||
let mut message_word = |i: usize| {
|
||||
let val = msg[i].value();
|
||||
let var = region.assign_advice(
|
||||
|| format!("load message_{}", i),
|
||||
config.poseidon_config.state[i],
|
||||
0,
|
||||
|| val.ok_or(Error::SynthesisError),
|
||||
)?;
|
||||
region.constrain_equal(var, msg[i].cell())?;
|
||||
Ok(Word::<_, _, OrchardNullifier, 3, 2>::from_inner(
|
||||
StateWord::new(var, val),
|
||||
))
|
||||
};
|
||||
Ok([message_word(0)?, message_word(1)?])
|
||||
},
|
||||
)?;
|
||||
|
||||
let poseidon_hasher = PoseidonHash::init(
|
||||
PoseidonChip::construct(config.poseidon_config.clone()),
|
||||
layouter.namespace(|| "Poseidon init"),
|
||||
ConstantLength::<2>,
|
||||
)?;
|
||||
|
||||
let poseidon_output =
|
||||
poseidon_hasher.hash(layouter.namespace(|| "Poseidon hash"), poseidon_message)?;
|
||||
|
||||
let poseidon_output: CellValue<Fp> = poseidon_output.inner().into();
|
||||
|
||||
if !poseidon_output.value().is_none() {
|
||||
coin += poseidon_output.value().unwrap();
|
||||
}
|
||||
}
|
||||
|
||||
if coin != Fp::zero() {
|
||||
println!("circuit hash: {:?}", coin);
|
||||
}
|
||||
|
||||
// Constrain the coin C
|
||||
let hash = self.load_private(
|
||||
layouter.namespace(|| "load hash"),
|
||||
config.advices[0],
|
||||
Some(coin),
|
||||
)?;
|
||||
layouter.constrain_instance(hash.cell(), config.primary, 0)?;
|
||||
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
struct VerifyingKey {
|
||||
params: commitment::Params<vesta::Affine>,
|
||||
vk: plonk::VerifyingKey<vesta::Affine>,
|
||||
}
|
||||
|
||||
impl VerifyingKey {
|
||||
fn build() -> Self {
|
||||
let params = commitment::Params::new(K);
|
||||
let circuit: MintCircuit = Default::default();
|
||||
|
||||
let vk = plonk::keygen_vk(¶ms, &circuit).unwrap();
|
||||
|
||||
VerifyingKey { params, vk }
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
struct ProvingKey {
|
||||
params: commitment::Params<vesta::Affine>,
|
||||
pk: plonk::ProvingKey<vesta::Affine>,
|
||||
}
|
||||
|
||||
impl ProvingKey {
|
||||
fn build() -> Self {
|
||||
let params = commitment::Params::new(K);
|
||||
let circuit: MintCircuit = Default::default();
|
||||
|
||||
let vk = plonk::keygen_vk(¶ms, &circuit).unwrap();
|
||||
let pk = plonk::keygen_pk(¶ms, vk, &circuit).unwrap();
|
||||
|
||||
ProvingKey { params, pk }
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Clone, Debug)]
|
||||
struct Proof(Vec<u8>);
|
||||
|
||||
impl AsRef<[u8]> for Proof {
|
||||
fn as_ref(&self) -> &[u8] {
|
||||
&self.0
|
||||
}
|
||||
}
|
||||
|
||||
impl Proof {
|
||||
fn create(pk: &ProvingKey, circuits: &[MintCircuit], pubinputs: &[Fp]) -> Result<Self, Error> {
|
||||
let mut transcript = Blake2bWrite::<_, vesta::Affine, _>::init(vec![]);
|
||||
plonk::create_proof(
|
||||
&pk.params,
|
||||
&pk.pk,
|
||||
circuits,
|
||||
&[&[pubinputs]],
|
||||
&mut transcript,
|
||||
)?;
|
||||
Ok(Proof(transcript.finalize()))
|
||||
}
|
||||
|
||||
fn verify(&self, vk: &VerifyingKey, pubinputs: &[Fp]) -> Result<(), plonk::Error> {
|
||||
let msm = vk.params.empty_msm();
|
||||
let mut transcript = Blake2bRead::init(&self.0[..]);
|
||||
let guard = plonk::verify_proof(&vk.params, &vk.vk, msm, &[&[pubinputs]], &mut transcript)?;
|
||||
let msm = guard.clone().use_challenges();
|
||||
if msm.eval() {
|
||||
Ok(())
|
||||
} else {
|
||||
Err(Error::ConstraintSystemFailure)
|
||||
}
|
||||
}
|
||||
|
||||
// fn new(bytes: Vec<u8>) -> Self {
|
||||
// Proof(bytes)
|
||||
// }
|
||||
}
|
||||
|
||||
#[allow(non_snake_case)]
|
||||
fn pedersen_commitment(value: u64, blind: Fq) -> Ep {
|
||||
let hasher = Ep::hash_to_curve(VALUE_COMMITMENT_PERSONALIZATION);
|
||||
let V = hasher(&VALUE_COMMITMENT_V_BYTES);
|
||||
let R = hasher(&VALUE_COMMITMENT_R_BYTES);
|
||||
let value = Fq::from_u64(value);
|
||||
|
||||
V * value + R * blind
|
||||
}
|
||||
|
||||
fn main() {
|
||||
let pubkey = Ep::random(&mut OsRng);
|
||||
let coords = pubkey.to_affine().coordinates().unwrap();
|
||||
|
||||
let value = 110;
|
||||
let asset = 1;
|
||||
|
||||
let value_blind = Fq::random(&mut OsRng);
|
||||
let asset_blind = Fq::random(&mut OsRng);
|
||||
|
||||
let serial = Fp::random(&mut OsRng);
|
||||
let coin_blind = Fp::random(&mut OsRng);
|
||||
|
||||
let mut coin = Fp::zero();
|
||||
let messages = [
|
||||
[*coords.x(), *coords.y()],
|
||||
[Fp::from(value), Fp::from(asset)],
|
||||
[serial, coin_blind],
|
||||
];
|
||||
for msg in messages.iter() {
|
||||
coin += Hash::init(OrchardNullifier, ConstantLength::<2>).hash(*msg);
|
||||
}
|
||||
println!("outer hash: {:?}", coin);
|
||||
|
||||
let value_commit = pedersen_commitment(value, value_blind);
|
||||
let value_coords = value_commit.to_affine().coordinates().unwrap();
|
||||
|
||||
let asset_commit = pedersen_commitment(asset, asset_blind);
|
||||
let asset_coords = asset_commit.to_affine().coordinates().unwrap();
|
||||
|
||||
let mut public_inputs = vec![
|
||||
coin,
|
||||
*value_coords.x(),
|
||||
*value_coords.y(),
|
||||
*asset_coords.x(),
|
||||
*asset_coords.y(),
|
||||
];
|
||||
|
||||
let circuit = MintCircuit {
|
||||
pub_x: Some(*coords.x()),
|
||||
pub_y: Some(*coords.y()),
|
||||
value: Some(vesta::Scalar::from(value)),
|
||||
asset: Some(vesta::Scalar::from(asset)),
|
||||
serial: Some(serial),
|
||||
coin_blind: Some(coin_blind),
|
||||
value_blind: Some(value_blind),
|
||||
asset_blind: Some(asset_blind),
|
||||
};
|
||||
|
||||
// Valid MockProver
|
||||
let prover = MockProver::run(K, &circuit, vec![public_inputs.clone()]).unwrap();
|
||||
assert_eq!(prover.verify(), Ok(()));
|
||||
|
||||
// Add 1 to break the public inputs
|
||||
public_inputs[0] += Fp::one();
|
||||
// Invalid MockProver
|
||||
let prover = MockProver::run(K, &circuit, vec![public_inputs.clone()]).unwrap();
|
||||
assert!(prover.verify().is_err());
|
||||
|
||||
// Remove 1 to make the public inputs valid again
|
||||
public_inputs[0] -= Fp::one();
|
||||
|
||||
// Actual ZK proof
|
||||
let start = Instant::now();
|
||||
let vk = VerifyingKey::build();
|
||||
let pk = ProvingKey::build();
|
||||
println!("Setup: [{:?}]", start.elapsed());
|
||||
|
||||
let start = Instant::now();
|
||||
let proof = Proof::create(&pk, &[circuit], &public_inputs).unwrap();
|
||||
println!("Prove: [{:?}]", start.elapsed());
|
||||
|
||||
let start = Instant::now();
|
||||
assert!(proof.verify(&vk, &public_inputs).is_ok());
|
||||
println!("Verify: [{:?}]", start.elapsed());
|
||||
}
|
||||
Reference in New Issue
Block a user