2017-01-25 18:51:41 +01:00
|
|
|
// Copyright 2015-2017 Parity Technologies (UK) Ltd.
|
2016-02-05 13:40:41 +01:00
|
|
|
// This file is part of Parity.
|
|
|
|
|
|
|
|
// Parity is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
|
|
|
// Parity is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2016-01-17 12:00:34 +01:00
|
|
|
//! Ethash implementation
|
|
|
|
//! See https://github.com/ethereum/wiki/wiki/Ethash
|
2016-02-18 01:47:47 +01:00
|
|
|
extern crate primal;
|
2016-01-18 12:49:39 +01:00
|
|
|
extern crate sha3;
|
2016-07-13 19:59:59 +02:00
|
|
|
extern crate parking_lot;
|
|
|
|
|
2016-01-30 23:02:55 +01:00
|
|
|
#[macro_use]
|
|
|
|
extern crate log;
|
2016-01-17 12:00:34 +01:00
|
|
|
mod compute;
|
|
|
|
|
2016-02-18 22:35:14 +01:00
|
|
|
use std::mem;
|
2016-01-17 12:00:34 +01:00
|
|
|
use compute::Light;
|
2016-09-28 23:31:59 +02:00
|
|
|
pub use compute::{ETHASH_EPOCH_LENGTH, H256, ProofOfWork, SeedHashCompute, quick_get_difficulty, slow_get_seedhash};
|
2016-01-17 12:00:34 +01:00
|
|
|
|
2016-07-13 19:59:59 +02:00
|
|
|
use std::sync::Arc;
|
|
|
|
use parking_lot::Mutex;
|
2016-01-17 12:00:34 +01:00
|
|
|
|
2016-02-18 22:35:14 +01:00
|
|
|
struct LightCache {
|
|
|
|
recent_epoch: Option<u64>,
|
|
|
|
recent: Option<Arc<Light>>,
|
|
|
|
prev_epoch: Option<u64>,
|
|
|
|
prev: Option<Arc<Light>>,
|
|
|
|
}
|
|
|
|
|
2016-02-29 19:30:13 +01:00
|
|
|
/// Light/Full cache manager.
|
2016-01-17 12:00:34 +01:00
|
|
|
pub struct EthashManager {
|
2016-02-18 22:35:14 +01:00
|
|
|
cache: Mutex<LightCache>,
|
2016-01-17 12:00:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl EthashManager {
|
|
|
|
/// Create a new new instance of ethash manager
|
|
|
|
pub fn new() -> EthashManager {
|
2016-02-18 01:47:47 +01:00
|
|
|
EthashManager {
|
2016-02-18 22:35:14 +01:00
|
|
|
cache: Mutex::new(LightCache {
|
|
|
|
recent_epoch: None,
|
|
|
|
recent: None,
|
|
|
|
prev_epoch: None,
|
|
|
|
prev: None,
|
|
|
|
}),
|
2016-01-17 12:00:34 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Calculate the light client data
|
|
|
|
/// `block_number` - Block number to check
|
|
|
|
/// `light` - The light client handler
|
|
|
|
/// `header_hash` - The header hash to pack into the mix
|
|
|
|
/// `nonce` - The nonce to pack into the mix
|
|
|
|
pub fn compute_light(&self, block_number: u64, header_hash: &H256, nonce: u64) -> ProofOfWork {
|
|
|
|
let epoch = block_number / ETHASH_EPOCH_LENGTH;
|
2016-01-30 23:02:55 +01:00
|
|
|
let light = {
|
2016-07-13 19:59:59 +02:00
|
|
|
let mut lights = self.cache.lock();
|
2016-02-18 22:35:14 +01:00
|
|
|
let light = match lights.recent_epoch.clone() {
|
|
|
|
Some(ref e) if *e == epoch => lights.recent.clone(),
|
|
|
|
_ => match lights.prev_epoch.clone() {
|
|
|
|
Some(e) if e == epoch => {
|
2016-10-28 16:10:30 +02:00
|
|
|
// don't swap if recent is newer.
|
|
|
|
if lights.recent_epoch > lights.prev_epoch {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
// swap
|
|
|
|
let t = lights.prev_epoch;
|
|
|
|
lights.prev_epoch = lights.recent_epoch;
|
|
|
|
lights.recent_epoch = t;
|
|
|
|
let t = lights.prev.clone();
|
|
|
|
lights.prev = lights.recent.clone();
|
|
|
|
lights.recent = t;
|
|
|
|
lights.recent.clone()
|
|
|
|
}
|
2016-02-18 22:35:14 +01:00
|
|
|
}
|
|
|
|
_ => None,
|
2016-04-13 00:06:02 +02:00
|
|
|
},
|
2016-02-18 22:35:14 +01:00
|
|
|
};
|
|
|
|
match light {
|
2016-01-30 23:02:55 +01:00
|
|
|
None => {
|
|
|
|
let light = match Light::from_file(block_number) {
|
|
|
|
Ok(light) => Arc::new(light),
|
2016-02-18 01:47:47 +01:00
|
|
|
Err(e) => {
|
2016-01-30 23:02:55 +01:00
|
|
|
debug!("Light cache file not found for {}:{}", block_number, e);
|
|
|
|
let light = Light::new(block_number);
|
|
|
|
if let Err(e) = light.to_file() {
|
|
|
|
warn!("Light cache file write error: {}", e);
|
|
|
|
}
|
|
|
|
Arc::new(light)
|
|
|
|
}
|
|
|
|
};
|
2016-02-18 22:35:14 +01:00
|
|
|
lights.prev_epoch = mem::replace(&mut lights.recent_epoch, Some(epoch));
|
|
|
|
lights.prev = mem::replace(&mut lights.recent, Some(light.clone()));
|
2016-01-30 23:02:55 +01:00
|
|
|
light
|
2016-01-21 16:48:37 +01:00
|
|
|
}
|
2016-04-13 00:06:02 +02:00
|
|
|
Some(light) => light,
|
2016-01-17 12:00:34 +01:00
|
|
|
}
|
2016-01-30 23:02:55 +01:00
|
|
|
};
|
|
|
|
light.compute(header_hash, nonce)
|
2016-01-17 12:00:34 +01:00
|
|
|
}
|
|
|
|
}
|
2016-02-18 22:35:14 +01:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_lru() {
|
|
|
|
let ethash = EthashManager::new();
|
|
|
|
let hash = [0u8; 32];
|
|
|
|
ethash.compute_light(1, &hash, 1);
|
|
|
|
ethash.compute_light(50000, &hash, 1);
|
2016-07-13 19:59:59 +02:00
|
|
|
assert_eq!(ethash.cache.lock().recent_epoch.unwrap(), 1);
|
|
|
|
assert_eq!(ethash.cache.lock().prev_epoch.unwrap(), 0);
|
2016-02-18 22:35:14 +01:00
|
|
|
ethash.compute_light(1, &hash, 1);
|
2016-07-13 19:59:59 +02:00
|
|
|
assert_eq!(ethash.cache.lock().recent_epoch.unwrap(), 0);
|
|
|
|
assert_eq!(ethash.cache.lock().prev_epoch.unwrap(), 1);
|
2016-02-18 22:35:14 +01:00
|
|
|
ethash.compute_light(70000, &hash, 1);
|
2016-07-13 19:59:59 +02:00
|
|
|
assert_eq!(ethash.cache.lock().recent_epoch.unwrap(), 2);
|
|
|
|
assert_eq!(ethash.cache.lock().prev_epoch.unwrap(), 0);
|
2016-02-18 22:35:14 +01:00
|
|
|
}
|