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
|
|
|
//! 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.
|
|
|
|
|
2017-08-22 21:23:54 +03:00
|
|
|
#![deny(non_upper_case_globals)]
|
|
|
|
#![deny(non_camel_case_types)]
|
|
|
|
#![deny(non_snake_case)]
|
|
|
|
#![deny(unused_mut)]
|
|
|
|
#![warn(missing_docs)]
|
|
|
|
|
|
|
|
extern crate blake2_rfc as blake2;
|
|
|
|
#[macro_use]
|
|
|
|
extern crate lazy_static;
|
2017-11-01 02:32:33 +03:00
|
|
|
extern crate rand;
|
2017-08-22 21:23:54 +03:00
|
|
|
extern crate serde;
|
|
|
|
#[macro_use]
|
|
|
|
extern crate serde_derive;
|
2017-11-01 02:32:33 +03:00
|
|
|
#[macro_use]
|
|
|
|
extern crate slog;
|
|
|
|
extern crate time;
|
2017-08-22 21:23:54 +03:00
|
|
|
|
|
|
|
extern crate grin_core as core;
|
2017-10-12 19:56:44 +03:00
|
|
|
extern crate grin_util as util;
|
2017-08-22 21:23:54 +03:00
|
|
|
|
2018-03-13 21:10:45 +03:00
|
|
|
// Re-export (mostly for stat collection)
|
|
|
|
pub extern crate cuckoo_miner as cuckoo_;
|
|
|
|
pub use cuckoo_ as cuckoo_miner;
|
2017-08-22 21:23:54 +03:00
|
|
|
|
2018-01-19 03:53:53 +03:00
|
|
|
mod siphash;
|
2017-08-22 21:23:54 +03:00
|
|
|
pub mod plugin;
|
2016-11-27 23:31:15 +03:00
|
|
|
pub mod cuckoo;
|
2017-08-22 21:23:54 +03:00
|
|
|
pub mod types;
|
2016-10-21 03:06:12 +03:00
|
|
|
|
2017-08-22 21:23:54 +03:00
|
|
|
use core::consensus;
|
|
|
|
use core::core::BlockHeader;
|
|
|
|
use core::core::Proof;
|
|
|
|
use core::core::target::Difficulty;
|
|
|
|
use core::global;
|
|
|
|
use core::genesis;
|
|
|
|
use cuckoo::{Cuckoo, Error};
|
2017-07-11 20:11:03 +03:00
|
|
|
|
|
|
|
/// Should be implemented by anything providing mining services
|
|
|
|
///
|
|
|
|
|
|
|
|
pub trait MiningWorker {
|
2017-07-20 17:22:40 +03:00
|
|
|
/// This only sets parameters and does initialisation work now
|
2017-11-01 02:32:33 +03:00
|
|
|
fn new(ease: u32, sizeshift: u32, proof_size: usize) -> Self
|
|
|
|
where
|
|
|
|
Self: Sized;
|
2017-08-10 03:54:10 +03:00
|
|
|
|
2017-07-20 17:22:40 +03:00
|
|
|
/// Actually perform a mining attempt on the given input and
|
|
|
|
/// return a proof if found
|
2017-07-11 20:11:03 +03:00
|
|
|
fn mine(&mut self, header: &[u8]) -> Result<Proof, Error>;
|
|
|
|
}
|
|
|
|
|
2017-04-10 09:17:23 +03:00
|
|
|
/// Validates the proof of work of a given header, and that the proof of work
|
|
|
|
/// satisfies the requirements of the header.
|
2017-02-08 00:48:11 +03:00
|
|
|
pub fn verify_size(bh: &BlockHeader, cuckoo_sz: u32) -> bool {
|
2018-03-16 22:04:31 +03:00
|
|
|
Cuckoo::new(&bh.pre_pow_hash()[..], cuckoo_sz)
|
|
|
|
.verify(bh.pow.clone(), consensus::EASINESS as u64)
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
/// Mines a genesis block, using the config specified miner if specified.
|
2017-11-16 00:49:15 +03:00
|
|
|
/// Otherwise, uses the internal miner
|
2017-11-20 20:35:52 +03:00
|
|
|
pub fn mine_genesis_block(
|
|
|
|
miner_config: Option<types::MinerConfig>,
|
|
|
|
) -> Result<core::core::Block, Error> {
|
2018-01-19 20:48:18 +03:00
|
|
|
let mut gen = genesis::genesis_testnet2();
|
2018-03-19 22:23:58 +03:00
|
|
|
if global::is_user_testing_mode() {
|
|
|
|
gen = genesis::genesis_dev();
|
|
|
|
gen.header.timestamp = time::now();
|
|
|
|
}
|
2018-03-15 22:16:34 +03:00
|
|
|
|
|
|
|
// total_difficulty on the genesis header *is* the difficulty of that block
|
|
|
|
let genesis_difficulty = gen.header.total_difficulty.clone();
|
2017-09-29 21:44:25 +03:00
|
|
|
|
2017-08-22 21:23:54 +03:00
|
|
|
let sz = global::sizeshift() as u32;
|
|
|
|
let proof_size = global::proofsize();
|
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
let mut miner: Box<MiningWorker> = match miner_config {
|
2018-03-09 20:16:31 +03:00
|
|
|
Some(c) => if c.enable_mining {
|
2017-11-01 02:32:33 +03:00
|
|
|
let mut p = plugin::PluginMiner::new(consensus::EASINESS, sz, proof_size);
|
|
|
|
p.init(c.clone());
|
|
|
|
Box::new(p)
|
|
|
|
} else {
|
|
|
|
Box::new(cuckoo::Miner::new(consensus::EASINESS, sz, proof_size))
|
|
|
|
},
|
2017-08-22 21:23:54 +03:00
|
|
|
None => Box::new(cuckoo::Miner::new(consensus::EASINESS, sz, proof_size)),
|
|
|
|
};
|
2018-03-15 22:16:34 +03:00
|
|
|
pow_size(&mut *miner, &mut gen.header, genesis_difficulty, sz as u32).unwrap();
|
2017-11-20 20:35:52 +03:00
|
|
|
Ok(gen)
|
2017-08-22 21:23:54 +03:00
|
|
|
}
|
|
|
|
|
2017-09-29 21:44:25 +03:00
|
|
|
/// Runs a proof of work computation over the provided block using the provided
|
2017-11-16 00:49:15 +03:00
|
|
|
/// Mining Worker, until the required difficulty target is reached. May take a
|
|
|
|
/// while for a low target...
|
2017-11-01 02:32:33 +03:00
|
|
|
pub fn pow_size<T: MiningWorker + ?Sized>(
|
|
|
|
miner: &mut T,
|
|
|
|
bh: &mut BlockHeader,
|
|
|
|
diff: Difficulty,
|
|
|
|
_: u32,
|
|
|
|
) -> Result<(), Error> {
|
2017-02-08 00:48:11 +03:00
|
|
|
let start_nonce = bh.nonce;
|
2016-10-21 03:06:12 +03:00
|
|
|
|
2018-03-04 03:19:54 +03:00
|
|
|
// set the nonce for faster solution finding in user testing
|
|
|
|
if bh.height == 0 && global::is_user_testing_mode() {
|
|
|
|
bh.nonce = global::get_genesis_nonce();
|
2017-08-22 21:23:54 +03:00
|
|
|
}
|
|
|
|
|
2018-03-04 03:19:54 +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
|
2018-03-16 22:04:31 +03:00
|
|
|
let pow_hash = bh.pre_pow_hash();
|
2018-03-04 03:19:54 +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.mine(&pow_hash[..]) {
|
|
|
|
if proof.clone().to_difficulty() >= diff {
|
|
|
|
bh.pow = proof.clone();
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// otherwise increment the nonce
|
|
|
|
bh.nonce += 1;
|
|
|
|
|
|
|
|
// 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::*;
|
2017-08-10 03:54:10 +03:00
|
|
|
use global;
|
2017-08-22 21:23:54 +03:00
|
|
|
use core::core::target::Difficulty;
|
|
|
|
use core::genesis;
|
2017-11-16 00:49:15 +03:00
|
|
|
use core::global::ChainTypes;
|
2016-10-21 03:06:12 +03:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn genesis_pow() {
|
2017-11-16 00:49:15 +03:00
|
|
|
global::set_mining_mode(ChainTypes::AutomatedTesting);
|
|
|
|
let mut b = genesis::genesis_dev();
|
|
|
|
b.header.nonce = 485;
|
2017-09-29 21:44:25 +03:00
|
|
|
let mut internal_miner = cuckoo::Miner::new(
|
|
|
|
consensus::EASINESS,
|
|
|
|
global::sizeshift() as u32,
|
|
|
|
global::proofsize(),
|
|
|
|
);
|
|
|
|
pow_size(
|
|
|
|
&mut internal_miner,
|
|
|
|
&mut b.header,
|
2018-01-27 10:48:53 +03:00
|
|
|
Difficulty::one(),
|
2017-09-29 21:44:25 +03:00
|
|
|
global::sizeshift() as u32,
|
|
|
|
).unwrap();
|
2017-01-09 23:07:38 +03:00
|
|
|
assert!(b.header.nonce != 310);
|
2018-01-27 10:48:53 +03:00
|
|
|
assert!(b.header.pow.clone().to_difficulty() >= Difficulty::one());
|
2017-08-09 19:40:23 +03:00
|
|
|
assert!(verify_size(&b.header, global::sizeshift() as u32));
|
2016-10-21 03:06:12 +03:00
|
|
|
}
|
|
|
|
}
|