grin/core/src/pow/mod.rs

112 lines
3.6 KiB
Rust
Raw Normal View History

// 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
//! 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;
pub mod cuckoo;
2016-10-21 03:06:12 +03:00
use time;
use consensus::EASINESS;
use core::BlockHeader;
use core::hash::Hashed;
use core::target::Difficulty;
2016-10-21 03:06:12 +03:00
use pow::cuckoo::{Cuckoo, Miner, Error};
/// Validates the proof of work of a given header.
pub fn verify(bh: &BlockHeader) -> bool {
verify_size(bh, bh.cuckoo_len as u32)
2016-10-21 03:06:12 +03:00
}
/// Validates the proof of work of a given header, and that the proof of work
/// satisfies the requirements of the header.
pub fn verify_size(bh: &BlockHeader, cuckoo_sz: u32) -> bool {
// make sure the pow hash shows a difficulty at least as large as the target
// difficulty
if bh.difficulty > bh.pow.to_difficulty() {
2016-10-21 03:06:12 +03:00
return false;
}
Cuckoo::new(&bh.hash()[..], cuckoo_sz).verify(bh.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...
pub fn pow(bh: &mut BlockHeader, diff: Difficulty) -> Result<(), Error> {
let cuckoo_len = bh.cuckoo_len as u32;
pow_size(bh, 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).
pub fn pow20(bh: &mut BlockHeader, diff: Difficulty) -> Result<(), Error> {
pow_size(bh, diff, 20)
2016-10-21 03:06:12 +03:00
}
/// Actual pow function, takes an arbitrary pow size as input
pub fn pow_size(bh: &mut BlockHeader, diff: Difficulty, sizeshift: u32) -> Result<(), Error> {
let start_nonce = bh.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
let pow_hash = bh.hash();
2016-10-21 03:06:12 +03:00
// if we found a cycle (not guaranteed) and the proof hash is higher that the
// diff, we're all good
if let Ok(proof) = Miner::new(&pow_hash[..], EASINESS, sizeshift).mine() {
if proof.to_difficulty() >= diff {
bh.pow = proof;
return Ok(());
2016-10-21 03:06:12 +03:00
}
}
// otherwise increment the nonce
bh.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)
if bh.nonce == start_nonce {
bh.timestamp = time::at_utc(time::Timespec { sec: 0, nsec: 0 });
2016-10-21 03:06:12 +03:00
}
}
}
#[cfg(test)]
mod test {
use super::*;
use core::target::Difficulty;
2016-10-21 03:06:12 +03:00
use genesis;
#[test]
fn genesis_pow() {
let mut b = genesis::genesis();
b.header.nonce = 310;
pow20(&mut b.header, Difficulty::one()).unwrap();
assert!(b.header.nonce != 310);
assert!(b.header.pow.to_difficulty() >= Difficulty::one());
assert!(verify_size(&b.header, 20));
2016-10-21 03:06:12 +03:00
}
}