2016-10-22 21:35:48 +03:00
|
|
|
// Copyright 2016 The Grin Developers
|
|
|
|
//
|
|
|
|
// 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
|
|
|
//! The proof of work needs to strike a balance between fast header
|
|
|
|
//! verification to avoid DoS attacks and difficulty for block verifiers to
|
|
|
|
//! build new blocks. In addition, mining new blocks should also be as
|
|
|
|
//! difficult on high end custom-made hardware (ASICs) as on commodity hardware
|
2016-10-22 01:02:20 +03:00
|
|
|
//! or smartphones. For this reason we use Cuckoo Cycle (see the cuckoo
|
2016-10-21 03:06:12 +03:00
|
|
|
//! module for more information).
|
|
|
|
//!
|
|
|
|
//! Note that this miner implementation is here mostly for tests and
|
|
|
|
//! reference. It's not optimized for speed.
|
|
|
|
|
|
|
|
mod siphash;
|
2016-11-27 23:31:15 +03:00
|
|
|
pub mod cuckoo;
|
2016-10-21 03:06:12 +03:00
|
|
|
|
|
|
|
use time;
|
|
|
|
|
2016-11-16 01:37:49 +03:00
|
|
|
use consensus::EASINESS;
|
2016-11-10 02:51:24 +03:00
|
|
|
use core::{Block, Proof};
|
2016-10-23 01:08:53 +03:00
|
|
|
use core::hash::{Hash, Hashed};
|
2016-12-27 02:39:31 +03:00
|
|
|
use core::target::Difficulty;
|
2016-10-21 03:06:12 +03:00
|
|
|
use pow::cuckoo::{Cuckoo, Miner, Error};
|
|
|
|
|
|
|
|
use ser;
|
2016-11-01 04:44:11 +03:00
|
|
|
use ser::{Writeable, Writer};
|
2016-10-21 03:06:12 +03:00
|
|
|
|
|
|
|
/// Validates the proof of work of a given header.
|
2016-11-17 01:08:46 +03:00
|
|
|
pub fn verify(b: &Block) -> bool {
|
|
|
|
verify_size(b, b.header.cuckoo_len as u32)
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2016-12-21 04:32:19 +03:00
|
|
|
pub fn verify_size(b: &Block, cuckoo_sz: u32) -> bool {
|
2016-12-27 02:39:31 +03:00
|
|
|
// make sure the pow hash shows a difficulty at least as large as the target
|
|
|
|
// difficulty
|
|
|
|
if b.header.difficulty > b.header.pow.to_difficulty() {
|
2016-10-21 03:06:12 +03:00
|
|
|
return false;
|
|
|
|
}
|
2017-01-09 23:07:38 +03:00
|
|
|
Cuckoo::new(b.hash().to_slice(), cuckoo_sz).verify(b.header.pow, EASINESS as u64)
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Runs a naive single-threaded proof of work computation over the provided
|
|
|
|
/// block, until the required difficulty target is reached. May take a
|
|
|
|
/// while for a low target...
|
2017-01-09 23:07:38 +03:00
|
|
|
pub fn pow(b: &mut Block, diff: Difficulty) -> Result<(), Error> {
|
2017-01-10 02:16:44 +03:00
|
|
|
let cuckoo_len = b.header.cuckoo_len as u32;
|
2017-01-09 23:07:38 +03:00
|
|
|
pow_size(b, diff, cuckoo_len)
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Same as default pow function but uses the much easier Cuckoo20 (mostly for
|
|
|
|
/// tests).
|
2017-01-09 23:07:38 +03:00
|
|
|
pub fn pow20(b: &mut Block, diff: Difficulty) -> Result<(), Error> {
|
2016-12-27 02:39:31 +03:00
|
|
|
pow_size(b, diff, 20)
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2017-01-09 23:07:38 +03:00
|
|
|
pub fn pow_size(b: &mut Block, diff: Difficulty, sizeshift: u32) -> Result<(), Error> {
|
|
|
|
let start_nonce = b.header.nonce;
|
2016-10-21 03:06:12 +03:00
|
|
|
|
|
|
|
// try to find a cuckoo cycle on that header hash
|
|
|
|
loop {
|
|
|
|
// can be trivially optimized by avoiding re-serialization every time but this
|
|
|
|
// is not meant as a fast miner implementation
|
2017-01-09 23:07:38 +03:00
|
|
|
let pow_hash = b.hash();
|
2016-10-21 03:06:12 +03:00
|
|
|
|
2016-12-27 02:39:31 +03:00
|
|
|
// if we found a cycle (not guaranteed) and the proof hash is higher that the
|
|
|
|
// diff, we're all good
|
2016-10-21 03:06:12 +03:00
|
|
|
if let Ok(proof) = Miner::new(pow_hash.to_slice(), EASINESS, sizeshift).mine() {
|
2016-12-27 02:39:31 +03:00
|
|
|
if proof.to_difficulty() >= diff {
|
2017-01-10 02:16:44 +03:00
|
|
|
b.header.pow = proof;
|
2017-01-09 23:07:38 +03:00
|
|
|
return Ok(());
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// otherwise increment the nonce
|
2017-01-09 23:07:38 +03:00
|
|
|
b.header.nonce += 1;
|
2016-10-21 03:06:12 +03:00
|
|
|
|
|
|
|
// and if we're back where we started, update the time (changes the hash as
|
|
|
|
// well)
|
2017-01-09 23:07:38 +03:00
|
|
|
if b.header.nonce == start_nonce {
|
2017-01-10 02:16:44 +03:00
|
|
|
b.header.timestamp = time::at_utc(time::Timespec { sec: 0, nsec: 0 });
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
use super::*;
|
2016-11-01 04:44:11 +03:00
|
|
|
use core::Proof;
|
2016-12-27 02:39:31 +03:00
|
|
|
use core::target::Difficulty;
|
2016-10-21 03:06:12 +03:00
|
|
|
use genesis;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn genesis_pow() {
|
|
|
|
let mut b = genesis::genesis();
|
2017-01-10 02:16:44 +03:00
|
|
|
b.header.nonce = 310;
|
2017-01-09 23:07:38 +03:00
|
|
|
pow20(&mut b, Difficulty::one()).unwrap();
|
|
|
|
assert!(b.header.nonce != 310);
|
|
|
|
assert!(b.header.pow.to_difficulty() >= Difficulty::one());
|
|
|
|
assert!(verify_size(&b, 20));
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
}
|