forked from lovesh/bulletproofs-r1cs-gadgets
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgadget_bound_check.rs
165 lines (127 loc) · 5.35 KB
/
gadget_bound_check.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
extern crate bulletproofs;
extern crate curve25519_dalek;
extern crate merlin;
extern crate rand;
use bulletproofs::r1cs::{ConstraintSystem, R1CSError, R1CSProof, Variable, Prover, Verifier};
use curve25519_dalek::scalar::Scalar;
use bulletproofs::{BulletproofGens, PedersenGens};
use curve25519_dalek::ristretto::CompressedRistretto;
use bulletproofs::r1cs::LinearCombination;
use merlin::Transcript;
use rand::{RngCore, CryptoRng};
use crate::r1cs_utils::{AllocatedQuantity, positive_no_gadget, constrain_lc_with_scalar};
pub fn bound_check_gadget<CS: ConstraintSystem>(
cs: &mut CS,
v: AllocatedQuantity,
a: AllocatedQuantity,
b: AllocatedQuantity,
max: u64,
min: u64,
bit_size: usize
) -> Result<(), R1CSError> {
// a = v - min
// b = max - v
// a + b = max - min
cs.constrain(v.variable - LinearCombination::from(min) - a.variable);
cs.constrain(LinearCombination::from(max) - v.variable - b.variable);
// Constrain a + b to be same as max - min.
constrain_lc_with_scalar::<CS>(cs, a.variable + b.variable, &Scalar::from(max - min));
// Constrain a in [0, 2^n)
assert!(positive_no_gadget(cs, a, bit_size).is_ok());
// Constrain b in [0, 2^n)
assert!(positive_no_gadget(cs, b, bit_size).is_ok());
Ok(())
}
/// Accepts the num for which the bounds have to proved and optionally the randomness used in committing to that number.
/// This randomness argument is accepted so that this can be used as a sub-protocol where the protocol on upper layer will create the commitment.
pub fn gen_proof_of_bounded_num<R: RngCore + CryptoRng>(val: u64, randomness: Option<Scalar>, lower: u64, upper: u64,
max_bits_in_val: usize, mut rng: &mut R, transcript_label: &'static [u8],
pc_gens: &PedersenGens, bp_gens: &BulletproofGens) -> Result<(R1CSProof, Vec<CompressedRistretto>), R1CSError> {
let a = val - lower;
let b = upper - val;
let mut comms = vec![];
// Prover makes a `ConstraintSystem` instance representing a range proof gadget
let mut prover_transcript = Transcript::new(transcript_label);
let mut prover = Prover::new(&pc_gens, &mut prover_transcript);
let (com_v, var_v) = prover.commit(val.into(), randomness.unwrap_or_else(|| Scalar::random(&mut rng)));
let quantity_v = AllocatedQuantity {
variable: var_v,
assignment: Some(val),
};
comms.push(com_v);
let (com_a, var_a) = prover.commit(a.into(), Scalar::random(&mut rng));
let quantity_a = AllocatedQuantity {
variable: var_a,
assignment: Some(a),
};
comms.push(com_a);
let (com_b, var_b) = prover.commit(b.into(), Scalar::random(&mut rng));
let quantity_b = AllocatedQuantity {
variable: var_b,
assignment: Some(b),
};
comms.push(com_b);
assert!(bound_check_gadget(&mut prover, quantity_v, quantity_a, quantity_b, upper, lower, max_bits_in_val).is_ok());
let proof = prover.prove(&bp_gens)?;
Ok((proof, comms))
}
pub fn verify_proof_of_bounded_num(lower: u64, upper: u64, max_bits_in_val: usize,
proof: R1CSProof, commitments: Vec<CompressedRistretto>,
transcript_label: &'static [u8], pc_gens: &PedersenGens, bp_gens: &BulletproofGens) -> Result<(), R1CSError> {
let mut verifier_transcript = Transcript::new(transcript_label);
let mut verifier = Verifier::new(&mut verifier_transcript);
let var_v = verifier.commit(commitments[0]);
let quantity_v = AllocatedQuantity {
variable: var_v,
assignment: None,
};
let var_a = verifier.commit(commitments[1]);
let quantity_a = AllocatedQuantity {
variable: var_a,
assignment: None,
};
let var_b = verifier.commit(commitments[2]);
let quantity_b = AllocatedQuantity {
variable: var_b,
assignment: None,
};
assert!(bound_check_gadget(&mut verifier, quantity_v, quantity_a, quantity_b, upper, lower, max_bits_in_val).is_ok());
verifier.verify(&proof, &pc_gens, &bp_gens)
}
#[cfg(test)]
mod tests {
use super::*;
use merlin::Transcript;
use rand::rngs::OsRng;
use rand::Rng;
fn bound_check(min: u64, max: u64, bit_size: usize) {
let mut rng = rand::thread_rng();
let v = rng.gen_range(min, max);
println!("v is {}", &v);
let randomness = Some(Scalar::random(&mut rng));
let pc_gens = PedersenGens::default();
let bp_gens = BulletproofGens::new(128, 1);
let label = b"BoundsTest";
let (proof, commitments) = gen_proof_of_bounded_num(v, randomness, min, max, bit_size, &mut rng, label, &pc_gens, &bp_gens).unwrap();
verify_proof_of_bounded_num(min, max, bit_size, proof, commitments, label, &pc_gens, &bp_gens).unwrap();
}
#[test]
fn test_bound_check_gadget_small_value() {
let min = 10;
let max = 100;
// Since max fits in 7 bits
let bit_size = 7;
bound_check(min, max, bit_size)
}
#[test]
fn test_bound_check_gadget_large_value() {
let min = std::u64::MAX/100001;
let max = std::u64::MAX/100000;
// Since max fits in 47 bits
let bit_size = 47;
bound_check(min, max, bit_size);
// check once more with a new minimum
let min = std::u64::MAX/100009;
bound_check(min, max, bit_size);
}
}