2018-03-05 22:33:44 +03:00
|
|
|
// Copyright 2018 The Grin Developers
|
2016-10-22 21:35:48 +03:00
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2016-10-21 03:06:12 +03:00
|
|
|
//! Implementation of Cuckoo Cycle designed by John Tromp. Ported to Rust from
|
|
|
|
//! the C and Java code at https://github.com/tromp/cuckoo. Note that only the
|
|
|
|
//! simple miner is included, mostly for testing purposes. John Tromp's Tomato
|
|
|
|
//! miner will be much faster in almost every environment.
|
|
|
|
|
2018-10-03 13:39:16 +03:00
|
|
|
use pow::common::{self, CuckooParams, Edge, EdgeType};
|
2018-09-28 13:53:14 +03:00
|
|
|
use pow::error::{Error, ErrorKind};
|
|
|
|
use pow::num::ToPrimitive;
|
|
|
|
use pow::{PoWContext, Proof};
|
|
|
|
|
2016-10-21 03:06:12 +03:00
|
|
|
use std::cmp;
|
2018-06-01 22:41:26 +03:00
|
|
|
use std::collections::HashSet;
|
2016-10-21 03:06:12 +03:00
|
|
|
|
|
|
|
const MAXPATHLEN: usize = 8192;
|
|
|
|
|
2017-04-10 09:17:23 +03:00
|
|
|
/// Cuckoo cycle context
|
2018-09-28 13:53:14 +03:00
|
|
|
pub struct CuckooContext<T>
|
|
|
|
where
|
|
|
|
T: EdgeType,
|
|
|
|
{
|
2018-10-03 13:39:16 +03:00
|
|
|
params: CuckooParams<T>,
|
2018-09-28 13:53:14 +03:00
|
|
|
graph: Vec<T>,
|
|
|
|
_max_sols: u32,
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
impl<T> PoWContext<T> for CuckooContext<T>
|
|
|
|
where
|
|
|
|
T: EdgeType,
|
|
|
|
{
|
|
|
|
fn new(
|
|
|
|
edge_bits: u8,
|
|
|
|
proof_size: usize,
|
|
|
|
easiness_pct: u32,
|
|
|
|
max_sols: u32,
|
|
|
|
) -> Result<Box<Self>, Error> {
|
|
|
|
Ok(Box::new(CuckooContext::<T>::new_impl(
|
|
|
|
edge_bits,
|
|
|
|
proof_size,
|
|
|
|
easiness_pct,
|
|
|
|
max_sols,
|
|
|
|
)?))
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
fn set_header_nonce(
|
|
|
|
&mut self,
|
|
|
|
header: Vec<u8>,
|
|
|
|
nonce: Option<u32>,
|
|
|
|
solve: bool,
|
|
|
|
) -> Result<(), Error> {
|
|
|
|
self.set_header_nonce_impl(header, nonce, solve)
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
fn find_cycles(&mut self) -> Result<Vec<Proof>, Error> {
|
|
|
|
self.find_cycles_impl()
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
fn verify(&self, proof: &Proof) -> Result<(), Error> {
|
|
|
|
self.verify_impl(proof)
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
impl<T> CuckooContext<T>
|
|
|
|
where
|
|
|
|
T: EdgeType,
|
|
|
|
{
|
|
|
|
/// Create a new cuckoo context with given parameters
|
|
|
|
pub fn new_impl(
|
|
|
|
edge_bits: u8,
|
|
|
|
proof_size: usize,
|
|
|
|
easiness_pct: u32,
|
|
|
|
max_sols: u32,
|
|
|
|
) -> Result<CuckooContext<T>, Error> {
|
2018-10-03 13:39:16 +03:00
|
|
|
let params = CuckooParams::new(edge_bits, proof_size, easiness_pct, false)?;
|
|
|
|
let num_edges = params.num_edges as usize;
|
2018-09-28 13:53:14 +03:00
|
|
|
Ok(CuckooContext {
|
2018-10-03 13:39:16 +03:00
|
|
|
params: params,
|
|
|
|
graph: vec![T::zero(); num_edges + 1],
|
2018-09-28 13:53:14 +03:00
|
|
|
_max_sols: max_sols,
|
|
|
|
})
|
|
|
|
}
|
2016-10-21 03:06:12 +03:00
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
fn reset(&mut self) -> Result<(), Error> {
|
2018-10-03 13:39:16 +03:00
|
|
|
self.graph = vec![T::zero(); self.params.num_edges as usize + 1];
|
2018-09-28 13:53:14 +03:00
|
|
|
Ok(())
|
2018-06-23 00:59:56 +03:00
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
/// Set the header and optional nonce in the last part of the header
|
|
|
|
/// and create siphash keys
|
|
|
|
pub fn set_header_nonce_impl(
|
|
|
|
&mut self,
|
|
|
|
header: Vec<u8>,
|
|
|
|
nonce: Option<u32>,
|
|
|
|
solve: bool,
|
|
|
|
) -> Result<(), Error> {
|
2018-10-03 13:39:16 +03:00
|
|
|
self.params.reset_header_nonce(header, nonce)?;
|
2018-09-28 13:53:14 +03:00
|
|
|
if solve {
|
|
|
|
self.reset()?;
|
2018-07-01 01:36:38 +03:00
|
|
|
}
|
2018-09-28 13:53:14 +03:00
|
|
|
Ok(())
|
2018-04-24 11:18:24 +03:00
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
/// Generates a node in the cuckoo graph generated from our seed. A node is
|
|
|
|
/// simply materialized as a u64 from a nonce and an offset (generally 0 or
|
|
|
|
/// 1).
|
|
|
|
fn new_node(&self, edge: T, uorv: u64) -> Result<T, Error> {
|
2018-10-03 13:39:16 +03:00
|
|
|
self.params.sipnode(edge, uorv, true)
|
2018-09-28 13:53:14 +03:00
|
|
|
}
|
2016-10-21 03:06:12 +03:00
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
/// Creates a new edge in the cuckoo graph generated by our seed from a
|
|
|
|
/// nonce. Generates two node coordinates from the nonce and links them
|
|
|
|
/// together.
|
|
|
|
fn new_edge(&self, nonce: T) -> Result<Edge<T>, Error> {
|
|
|
|
Ok(Edge {
|
|
|
|
u: self.new_node(nonce, 0)?,
|
|
|
|
v: self.new_node(nonce, 1)?,
|
|
|
|
})
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
fn path(&self, mut u: T, us: &mut [T]) -> Result<T, Error> {
|
2016-10-21 03:06:12 +03:00
|
|
|
let mut nu = 0;
|
2018-09-28 13:53:14 +03:00
|
|
|
while u != T::zero() {
|
2016-10-21 03:06:12 +03:00
|
|
|
nu += 1;
|
|
|
|
if nu >= MAXPATHLEN {
|
|
|
|
while nu != 0 && us[(nu - 1) as usize] != u {
|
|
|
|
nu -= 1;
|
|
|
|
}
|
2018-09-28 13:53:14 +03:00
|
|
|
return Err(ErrorKind::Path)?;
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
us[nu as usize] = u;
|
2018-09-28 13:53:14 +03:00
|
|
|
u = self.graph[to_usize!(u)];
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
2018-09-28 13:53:14 +03:00
|
|
|
Ok(to_edge!(nu))
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
fn update_graph(
|
|
|
|
&mut self,
|
|
|
|
mut nu: usize,
|
|
|
|
us: &[T],
|
|
|
|
mut nv: usize,
|
|
|
|
vs: &[T],
|
|
|
|
) -> Result<(), Error> {
|
2016-10-21 03:06:12 +03:00
|
|
|
if nu < nv {
|
|
|
|
while nu != 0 {
|
|
|
|
nu -= 1;
|
2018-09-28 13:53:14 +03:00
|
|
|
self.graph[to_usize!(us[nu + 1])] = us[nu];
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
2018-09-28 13:53:14 +03:00
|
|
|
self.graph[to_usize!(us[0])] = vs[0];
|
2016-10-21 03:06:12 +03:00
|
|
|
} else {
|
|
|
|
while nv != 0 {
|
|
|
|
nv -= 1;
|
2018-09-28 13:53:14 +03:00
|
|
|
self.graph[to_usize!(vs[nv + 1])] = vs[nv];
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
2018-09-28 13:53:14 +03:00
|
|
|
self.graph[to_usize!(vs[0])] = us[0];
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
2018-09-28 13:53:14 +03:00
|
|
|
Ok(())
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
fn find_sol(
|
|
|
|
&mut self,
|
|
|
|
mut nu: usize,
|
|
|
|
us: &[T],
|
|
|
|
mut nv: usize,
|
|
|
|
vs: &[T],
|
|
|
|
) -> Result<Vec<T>, Error> {
|
2016-10-21 03:06:12 +03:00
|
|
|
if us[nu] == vs[nv] {
|
|
|
|
let min = cmp::min(nu, nv);
|
|
|
|
nu -= min;
|
|
|
|
nv -= min;
|
|
|
|
while us[nu] != vs[nv] {
|
|
|
|
nu += 1;
|
|
|
|
nv += 1;
|
|
|
|
}
|
2018-10-03 13:39:16 +03:00
|
|
|
if nu + nv + 1 == self.params.proof_size {
|
2018-09-28 13:53:14 +03:00
|
|
|
self.solution(&us, nu as u64, &vs, nv as u64)
|
2016-10-21 03:06:12 +03:00
|
|
|
} else {
|
2018-09-28 13:53:14 +03:00
|
|
|
Err(ErrorKind::InvalidCycle(nu + nv + 1))?
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
} else {
|
2018-09-28 13:53:14 +03:00
|
|
|
Err(ErrorKind::NoCycle)?
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
fn solution(&mut self, us: &[T], mut nu: u64, vs: &[T], mut nv: u64) -> Result<Vec<T>, Error> {
|
2016-10-21 03:06:12 +03:00
|
|
|
let mut cycle = HashSet::new();
|
2018-09-28 13:53:14 +03:00
|
|
|
cycle.insert(Edge { u: us[0], v: vs[0] });
|
2016-10-21 03:06:12 +03:00
|
|
|
while nu != 0 {
|
|
|
|
// u's in even position; v's in odd
|
2018-09-28 13:53:14 +03:00
|
|
|
nu = nu - 1;
|
2016-10-21 03:06:12 +03:00
|
|
|
cycle.insert(Edge {
|
2018-09-28 13:53:14 +03:00
|
|
|
u: us[((nu + 1) & !1) as usize],
|
|
|
|
v: us[(nu | 1) as usize],
|
2016-10-21 03:06:12 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
while nv != 0 {
|
|
|
|
// u's in odd position; v's in even
|
|
|
|
nv -= 1;
|
|
|
|
cycle.insert(Edge {
|
2018-09-28 13:53:14 +03:00
|
|
|
u: vs[(nv | 1) as usize],
|
|
|
|
v: vs[((nv + 1) & !1) as usize],
|
2016-10-21 03:06:12 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
let mut n = 0;
|
2018-10-03 13:39:16 +03:00
|
|
|
let mut sol = vec![T::zero(); self.params.proof_size];
|
|
|
|
for nonce in 0..to_usize!(self.params.easiness) {
|
2018-09-28 13:53:14 +03:00
|
|
|
let edge = self.new_edge(to_edge!(nonce))?;
|
2016-10-21 03:06:12 +03:00
|
|
|
if cycle.contains(&edge) {
|
2018-09-28 13:53:14 +03:00
|
|
|
sol[n] = to_edge!(nonce);
|
2016-10-21 03:06:12 +03:00
|
|
|
n += 1;
|
|
|
|
cycle.remove(&edge);
|
|
|
|
}
|
|
|
|
}
|
2018-10-03 13:39:16 +03:00
|
|
|
return if n == self.params.proof_size {
|
2018-09-28 13:53:14 +03:00
|
|
|
Ok(sol)
|
2016-10-21 03:06:12 +03:00
|
|
|
} else {
|
2018-09-28 13:53:14 +03:00
|
|
|
Err(ErrorKind::NoCycle)?
|
2016-10-21 03:06:12 +03:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
/// Searches for a solution (simple implementation)
|
|
|
|
pub fn find_cycles_impl(&mut self) -> Result<Vec<Proof>, Error> {
|
|
|
|
let mut us = [T::zero(); MAXPATHLEN];
|
|
|
|
let mut vs = [T::zero(); MAXPATHLEN];
|
2018-10-03 13:39:16 +03:00
|
|
|
for nonce in 0..to_usize!(self.params.easiness) {
|
2018-09-28 13:53:14 +03:00
|
|
|
us[0] = self.new_node(to_edge!(nonce), 0)?;
|
|
|
|
vs[0] = self.new_node(to_edge!(nonce), 1)?;
|
|
|
|
let u = self.graph[to_usize!(us[0])];
|
|
|
|
let v = self.graph[to_usize!(vs[0])];
|
|
|
|
if us[0] == T::zero() {
|
|
|
|
continue; // ignore duplicate edges
|
|
|
|
}
|
|
|
|
let nu = to_usize!(self.path(u, &mut us)?);
|
|
|
|
let nv = to_usize!(self.path(v, &mut vs)?);
|
|
|
|
|
|
|
|
let sol = self.find_sol(nu, &us, nv, &vs);
|
|
|
|
match sol {
|
|
|
|
Ok(s) => {
|
|
|
|
let mut proof = Proof::new(map_vec!(s.to_vec(), |&n| n.to_u64().unwrap_or(0)));
|
2018-10-03 13:39:16 +03:00
|
|
|
proof.cuckoo_sizeshift = self.params.edge_bits;
|
2018-09-28 13:53:14 +03:00
|
|
|
return Ok(vec![proof]);
|
|
|
|
}
|
|
|
|
Err(e) => match e.kind() {
|
|
|
|
ErrorKind::InvalidCycle(_) => continue,
|
|
|
|
ErrorKind::NoCycle => self.update_graph(nu, &us, nv, &vs)?,
|
|
|
|
_ => return Err(e),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(ErrorKind::NoSolution)?
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Assuming increasing nonces all smaller than easiness, verifies the
|
|
|
|
/// nonces form a cycle in a Cuckoo graph. Each nonce generates an edge, we
|
|
|
|
/// build the nodes on both side of that edge and count the connections.
|
|
|
|
pub fn verify_impl(&self, proof: &Proof) -> Result<(), Error> {
|
2018-10-03 13:39:16 +03:00
|
|
|
let easiness = to_u64!(self.params.easiness);
|
2018-09-28 13:53:14 +03:00
|
|
|
let nonces = &proof.nonces;
|
|
|
|
let mut us = vec![T::zero(); proof.proof_size()];
|
|
|
|
let mut vs = vec![T::zero(); proof.proof_size()];
|
|
|
|
for n in 0..proof.proof_size() {
|
|
|
|
if nonces[n] >= easiness || (n != 0 && nonces[n] <= nonces[n - 1]) {
|
|
|
|
return Err(ErrorKind::Verification("edge wrong size".to_owned()))?;
|
|
|
|
}
|
|
|
|
us[n] = self.new_node(to_edge!(nonces[n]), 0)?;
|
|
|
|
vs[n] = self.new_node(to_edge!(nonces[n]), 1)?;
|
|
|
|
}
|
|
|
|
let mut i = 0;
|
|
|
|
let mut count = proof.proof_size();
|
|
|
|
loop {
|
|
|
|
let mut j = i;
|
|
|
|
for k in 0..proof.proof_size() {
|
|
|
|
// find unique other j with same vs[j]
|
|
|
|
if k != i && vs[k] == vs[i] {
|
|
|
|
if j != i {
|
|
|
|
return Err(ErrorKind::Verification("".to_owned()))?;
|
|
|
|
}
|
|
|
|
j = k;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if j == i {
|
|
|
|
return Err(ErrorKind::Verification("".to_owned()))?;
|
|
|
|
}
|
|
|
|
i = j;
|
|
|
|
for k in 0..proof.proof_size() {
|
|
|
|
// find unique other i with same us[i]
|
|
|
|
if k != j && us[k] == us[j] {
|
|
|
|
if i != j {
|
|
|
|
return Err(ErrorKind::Verification("".to_owned()))?;
|
|
|
|
}
|
|
|
|
i = k;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if i == j {
|
|
|
|
return Err(ErrorKind::Verification("".to_owned()))?;
|
|
|
|
}
|
|
|
|
count -= 2;
|
|
|
|
if i == 0 {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
match count == 0 {
|
|
|
|
true => Ok(()),
|
|
|
|
false => Err(ErrorKind::Verification("Invalid solution".to_owned()))?,
|
|
|
|
}
|
|
|
|
}
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
use super::*;
|
|
|
|
|
2018-06-29 20:41:28 +03:00
|
|
|
static V1: [u64; 42] = [
|
2018-03-04 03:19:54 +03:00
|
|
|
0x3bbd, 0x4e96, 0x1013b, 0x1172b, 0x1371b, 0x13e6a, 0x1aaa6, 0x1b575, 0x1e237, 0x1ee88,
|
|
|
|
0x22f94, 0x24223, 0x25b4f, 0x2e9f3, 0x33b49, 0x34063, 0x3454a, 0x3c081, 0x3d08e, 0x3d863,
|
|
|
|
0x4285a, 0x42f22, 0x43122, 0x4b853, 0x4cd0c, 0x4f280, 0x557d5, 0x562cf, 0x58e59, 0x59a62,
|
|
|
|
0x5b568, 0x644b9, 0x657e9, 0x66337, 0x6821c, 0x7866f, 0x7e14b, 0x7ec7c, 0x7eed7, 0x80643,
|
|
|
|
0x8628c, 0x8949e,
|
2017-11-01 02:32:33 +03:00
|
|
|
];
|
2018-06-29 20:41:28 +03:00
|
|
|
static V2: [u64; 42] = [
|
2018-03-04 03:19:54 +03:00
|
|
|
0x5e3a, 0x8a8b, 0x103d8, 0x1374b, 0x14780, 0x16110, 0x1b571, 0x1c351, 0x1c826, 0x28228,
|
|
|
|
0x2909f, 0x29516, 0x2c1c4, 0x334eb, 0x34cdd, 0x38a2c, 0x3ad23, 0x45ac5, 0x46afe, 0x50f43,
|
|
|
|
0x51ed6, 0x52ddd, 0x54a82, 0x5a46b, 0x5dbdb, 0x60f6f, 0x60fcd, 0x61c78, 0x63899, 0x64dab,
|
|
|
|
0x6affc, 0x6b569, 0x72639, 0x73987, 0x78806, 0x7b98e, 0x7c7d7, 0x7ddd4, 0x7fa88, 0x8277c,
|
|
|
|
0x832d9, 0x8ba6f,
|
2017-11-01 02:32:33 +03:00
|
|
|
];
|
2018-06-29 20:41:28 +03:00
|
|
|
static V3: [u64; 42] = [
|
2018-03-04 03:19:54 +03:00
|
|
|
0x308b, 0x9004, 0x91fc, 0x983e, 0x9d67, 0xa293, 0xb4cb, 0xb6c8, 0xccc8, 0xdddc, 0xf04d,
|
|
|
|
0x1372f, 0x16ec9, 0x17b61, 0x17d03, 0x1e3bc, 0x1fb0f, 0x29e6e, 0x2a2ca, 0x2a719, 0x3a078,
|
|
|
|
0x3b7cc, 0x3c71d, 0x40daa, 0x43e17, 0x46adc, 0x4b359, 0x4c3aa, 0x4ce92, 0x4d06e, 0x51140,
|
|
|
|
0x565ac, 0x56b1f, 0x58a8b, 0x5e410, 0x5e607, 0x5ebb5, 0x5f8ae, 0x7aeac, 0x7b902, 0x7d6af,
|
|
|
|
0x7f400,
|
2017-11-01 02:32:33 +03:00
|
|
|
];
|
2016-10-21 03:06:12 +03:00
|
|
|
// cuckoo28 at 50% edges of letter 'u'
|
2018-06-29 20:41:28 +03:00
|
|
|
static V4: [u64; 42] = [
|
2018-03-04 03:19:54 +03:00
|
|
|
0xf7243, 0x11f130, 0x193812, 0x23b565, 0x279ac3, 0x69b270, 0xe0778f, 0xef51fc, 0x10bf6e8,
|
|
|
|
0x13ccf7d, 0x1551177, 0x1b6cfd2, 0x1f872c3, 0x2075681, 0x2e23ccc, 0x2e4c0aa, 0x2f607f1,
|
|
|
|
0x3007eeb, 0x3407e9a, 0x35423f9, 0x39e48bf, 0x45e3bf6, 0x46aa484, 0x47c0fe1, 0x4b1d5a6,
|
|
|
|
0x4bae0ba, 0x4dfdbaf, 0x5048eda, 0x537da6b, 0x5402887, 0x56b8897, 0x5bd8e8b, 0x622de20,
|
|
|
|
0x62be5ce, 0x62d538e, 0x6464518, 0x650a6d5, 0x66ec4fa, 0x66f9476, 0x6b1e5f6, 0x6fd5d88,
|
|
|
|
0x701f37b,
|
2017-11-01 02:32:33 +03:00
|
|
|
];
|
2016-10-21 03:06:12 +03:00
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
#[test]
|
|
|
|
fn cuckoo_context() {
|
|
|
|
let ret = mine20_vectors::<u32>();
|
|
|
|
if let Err(r) = ret {
|
|
|
|
panic!("mine20_vectors u32: Error: {}", r);
|
|
|
|
}
|
|
|
|
let ret = mine20_vectors::<u64>();
|
|
|
|
if let Err(r) = ret {
|
|
|
|
panic!("mine20_vectors u64: Error: {}", r);
|
|
|
|
}
|
|
|
|
let ret = validate20_vectors::<u32>();
|
|
|
|
if let Err(r) = ret {
|
|
|
|
panic!("validate20_vectors u32: Error: {}", r);
|
|
|
|
}
|
|
|
|
let ret = validate20_vectors::<u64>();
|
|
|
|
if let Err(r) = ret {
|
|
|
|
panic!("validate20_vectors u64: Error: {}", r);
|
|
|
|
}
|
|
|
|
let ret = validate_fail::<u32>();
|
|
|
|
if let Err(r) = ret {
|
|
|
|
panic!("validate_fail u32: Error: {}", r);
|
|
|
|
}
|
|
|
|
let ret = validate_fail::<u64>();
|
|
|
|
if let Err(r) = ret {
|
|
|
|
panic!("validate_fail u64: Error: {}", r);
|
|
|
|
}
|
|
|
|
let ret = mine16_validate::<u32>();
|
|
|
|
if let Err(r) = ret {
|
|
|
|
panic!("mine16_validate u32: Error: {}", r);
|
|
|
|
}
|
|
|
|
let ret = mine16_validate::<u64>();
|
|
|
|
if let Err(r) = ret {
|
|
|
|
panic!("mine16_validate u64: Error: {}", r);
|
|
|
|
}
|
2018-06-23 00:59:56 +03:00
|
|
|
}
|
|
|
|
|
2018-06-13 19:03:34 +03:00
|
|
|
/// Find a 42-cycle on Cuckoo20 at 75% easiness and verify against a few
|
2016-10-21 03:06:12 +03:00
|
|
|
/// known cycle proofs
|
|
|
|
/// generated by other implementations.
|
2018-09-28 13:53:14 +03:00
|
|
|
fn mine20_vectors<T>() -> Result<(), Error>
|
|
|
|
where
|
|
|
|
T: EdgeType,
|
|
|
|
{
|
|
|
|
let mut cuckoo_ctx = CuckooContext::<T>::new(20, 42, 75, 10)?;
|
|
|
|
cuckoo_ctx.set_header_nonce([49].to_vec(), None, true)?;
|
|
|
|
let res = cuckoo_ctx.find_cycles()?;
|
2018-06-30 07:07:24 +03:00
|
|
|
let mut proof = Proof::new(V1.to_vec());
|
|
|
|
proof.cuckoo_sizeshift = 20;
|
2018-09-28 13:53:14 +03:00
|
|
|
assert_eq!(proof, res[0]);
|
2016-10-21 03:06:12 +03:00
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
let mut cuckoo_ctx = CuckooContext::<T>::new(20, 42, 70, 10)?;
|
|
|
|
cuckoo_ctx.set_header_nonce([50].to_vec(), None, true)?;
|
|
|
|
let res = cuckoo_ctx.find_cycles()?;
|
2018-06-30 07:07:24 +03:00
|
|
|
let mut proof = Proof::new(V2.to_vec());
|
|
|
|
proof.cuckoo_sizeshift = 20;
|
2018-09-28 13:53:14 +03:00
|
|
|
assert_eq!(proof, res[0]);
|
2016-10-21 03:06:12 +03:00
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
//re-use context
|
|
|
|
cuckoo_ctx.set_header_nonce([51].to_vec(), None, true)?;
|
|
|
|
let res = cuckoo_ctx.find_cycles()?;
|
2018-06-30 07:07:24 +03:00
|
|
|
let mut proof = Proof::new(V3.to_vec());
|
|
|
|
proof.cuckoo_sizeshift = 20;
|
2018-09-28 13:53:14 +03:00
|
|
|
assert_eq!(proof, res[0]);
|
|
|
|
Ok(())
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
fn validate20_vectors<T>() -> Result<(), Error>
|
|
|
|
where
|
|
|
|
T: EdgeType,
|
|
|
|
{
|
|
|
|
let mut cuckoo_ctx = CuckooContext::<T>::new(20, 42, 75, 10)?;
|
|
|
|
cuckoo_ctx.set_header_nonce([49].to_vec(), None, false)?;
|
|
|
|
assert!(cuckoo_ctx.verify(&Proof::new(V1.to_vec().clone())).is_ok());
|
|
|
|
let mut cuckoo_ctx = CuckooContext::<T>::new(20, 42, 70, 10)?;
|
|
|
|
cuckoo_ctx.set_header_nonce([50].to_vec(), None, false)?;
|
|
|
|
assert!(cuckoo_ctx.verify(&Proof::new(V2.to_vec().clone())).is_ok());
|
|
|
|
cuckoo_ctx.set_header_nonce([51].to_vec(), None, false)?;
|
|
|
|
assert!(cuckoo_ctx.verify(&Proof::new(V3.to_vec().clone())).is_ok());
|
|
|
|
Ok(())
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
fn validate_fail<T>() -> Result<(), Error>
|
|
|
|
where
|
|
|
|
T: EdgeType,
|
|
|
|
{
|
|
|
|
// edge checks
|
|
|
|
let mut cuckoo_ctx = CuckooContext::<T>::new(20, 42, 75, 10)?;
|
|
|
|
cuckoo_ctx.set_header_nonce([49].to_vec(), None, false)?;
|
2016-10-21 03:06:12 +03:00
|
|
|
// edge checks
|
2018-09-28 13:53:14 +03:00
|
|
|
assert!(!cuckoo_ctx.verify(&Proof::new(vec![0; 42])).is_ok());
|
|
|
|
assert!(!cuckoo_ctx.verify(&Proof::new(vec![0xffff; 42])).is_ok());
|
2016-10-21 03:06:12 +03:00
|
|
|
// wrong data for proof
|
2018-09-28 13:53:14 +03:00
|
|
|
cuckoo_ctx.set_header_nonce([50].to_vec(), None, false)?;
|
|
|
|
assert!(!cuckoo_ctx.verify(&Proof::new(V1.to_vec().clone())).is_ok());
|
2017-09-29 21:44:25 +03:00
|
|
|
let mut test_header = [0; 32];
|
|
|
|
test_header[0] = 24;
|
2018-09-28 13:53:14 +03:00
|
|
|
let mut cuckoo_ctx = CuckooContext::<T>::new(20, 42, 50, 10)?;
|
|
|
|
cuckoo_ctx.set_header_nonce(test_header.to_vec(), None, false)?;
|
|
|
|
assert!(!cuckoo_ctx.verify(&Proof::new(V4.to_vec().clone())).is_ok());
|
|
|
|
Ok(())
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2018-09-28 13:53:14 +03:00
|
|
|
fn mine16_validate<T>() -> Result<(), Error>
|
|
|
|
where
|
|
|
|
T: EdgeType,
|
|
|
|
{
|
2016-10-21 03:06:12 +03:00
|
|
|
for n in 1..5 {
|
|
|
|
let h = [n; 32];
|
2018-09-28 13:53:14 +03:00
|
|
|
let mut cuckoo_ctx = CuckooContext::<T>::new(16, 42, 75, 10)?;
|
|
|
|
cuckoo_ctx.set_header_nonce(h.to_vec(), None, false)?;
|
|
|
|
let res = cuckoo_ctx.find_cycles()?;
|
|
|
|
assert!(cuckoo_ctx.verify(&res[0]).is_ok())
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
2018-09-28 13:53:14 +03:00
|
|
|
Ok(())
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
}
|