2018-01-27 10:48:53 +03:00
|
|
|
// Copyright 2018 The Grin Developers
|
2016-11-11 03:02:47 +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.
|
|
|
|
|
|
|
|
//! All the rules required for a cryptocurrency to have reach consensus across
|
|
|
|
//! the whole network are complex and hard to completely isolate. Some can be
|
|
|
|
//! simple parameters (like block reward), others complex algorithms (like
|
|
|
|
//! Merkle sum trees or reorg rules). However, as long as they're simple
|
2016-11-14 22:21:18 +03:00
|
|
|
//! enough, consensus-relevant constants and short functions should be kept
|
2016-11-11 03:02:47 +03:00
|
|
|
//! here.
|
|
|
|
|
2017-06-19 18:59:56 +03:00
|
|
|
use std::fmt;
|
2017-11-14 03:45:10 +03:00
|
|
|
use std::cmp::max;
|
2016-11-16 01:37:49 +03:00
|
|
|
|
2016-12-27 02:39:31 +03:00
|
|
|
use core::target::Difficulty;
|
2018-01-27 10:48:53 +03:00
|
|
|
use global;
|
2016-11-16 01:37:49 +03:00
|
|
|
|
2017-11-15 04:14:07 +03:00
|
|
|
/// A grin is divisible to 10^9, following the SI prefixes
|
2017-10-22 13:56:55 +03:00
|
|
|
pub const GRIN_BASE: u64 = 1_000_000_000;
|
2017-11-15 04:14:07 +03:00
|
|
|
/// Milligrin, a thousand of a grin
|
|
|
|
pub const MILLI_GRIN: u64 = GRIN_BASE / 1_000;
|
|
|
|
/// Microgrin, a thousand of a milligrin
|
|
|
|
pub const MICRO_GRIN: u64 = MILLI_GRIN / 1_000;
|
|
|
|
/// Nanogrin, smallest unit, takes a billion to make a grin
|
|
|
|
pub const NANO_GRIN: u64 = 1;
|
2017-10-22 13:56:55 +03:00
|
|
|
|
2018-01-07 07:21:50 +03:00
|
|
|
/// The block subsidy amount, one grin per second on average
|
|
|
|
pub const REWARD: u64 = 60 * GRIN_BASE;
|
2016-11-11 03:02:47 +03:00
|
|
|
|
2017-10-05 10:23:04 +03:00
|
|
|
/// Actual block reward for a given total fee amount
|
|
|
|
pub fn reward(fee: u64) -> u64 {
|
2018-03-13 05:39:22 +03:00
|
|
|
REWARD + fee
|
2017-10-05 10:23:04 +03:00
|
|
|
}
|
|
|
|
|
2017-09-12 20:24:24 +03:00
|
|
|
/// Number of blocks before a coinbase matures and can be spent
|
2017-10-04 20:44:22 +03:00
|
|
|
pub const COINBASE_MATURITY: u64 = 1_000;
|
2017-09-12 20:24:24 +03:00
|
|
|
|
2016-11-16 01:37:49 +03:00
|
|
|
/// Block interval, in seconds, the network will tune its next_target for. Note
|
|
|
|
/// that we may reduce this value in the future as we get more data on mining
|
|
|
|
/// with Cuckoo Cycle, networks improve and block propagation is optimized
|
|
|
|
/// (adjusting the reward accordingly).
|
2017-08-03 19:57:55 +03:00
|
|
|
pub const BLOCK_TIME_SEC: u64 = 60;
|
2016-11-11 03:02:47 +03:00
|
|
|
|
|
|
|
/// Cuckoo-cycle proof size (cycle length)
|
|
|
|
pub const PROOFSIZE: usize = 42;
|
|
|
|
|
2017-06-19 18:59:56 +03:00
|
|
|
/// Default Cuckoo Cycle size shift used for mining and validating.
|
|
|
|
pub const DEFAULT_SIZESHIFT: u8 = 30;
|
2016-11-16 01:37:49 +03:00
|
|
|
|
2016-11-11 03:02:47 +03:00
|
|
|
/// Default Cuckoo Cycle easiness, high enough to have good likeliness to find
|
|
|
|
/// a solution.
|
|
|
|
pub const EASINESS: u32 = 50;
|
|
|
|
|
2017-06-18 06:17:26 +03:00
|
|
|
/// Default number of blocks in the past when cross-block cut-through will start
|
|
|
|
/// happening. Needs to be long enough to not overlap with a long reorg.
|
|
|
|
/// Rational
|
|
|
|
/// behind the value is the longest bitcoin fork was about 30 blocks, so 5h. We
|
|
|
|
/// add an order of magnitude to be safe and round to 48h of blocks to make it
|
|
|
|
/// easier to reason about.
|
|
|
|
pub const CUT_THROUGH_HORIZON: u32 = 48 * 3600 / (BLOCK_TIME_SEC as u32);
|
|
|
|
|
|
|
|
/// The maximum size we're willing to accept for any message. Enforced by the
|
|
|
|
/// peer-to-peer networking layer only for DoS protection.
|
|
|
|
pub const MAX_MSG_LEN: u64 = 20_000_000;
|
|
|
|
|
2017-10-06 06:34:35 +03:00
|
|
|
/// Weight of an input when counted against the max block weigth capacity
|
|
|
|
pub const BLOCK_INPUT_WEIGHT: usize = 1;
|
|
|
|
|
|
|
|
/// Weight of an output when counted against the max block weight capacity
|
|
|
|
pub const BLOCK_OUTPUT_WEIGHT: usize = 10;
|
|
|
|
|
|
|
|
/// Weight of a kernel when counted against the max block weight capacity
|
|
|
|
pub const BLOCK_KERNEL_WEIGHT: usize = 2;
|
|
|
|
|
|
|
|
/// Total maximum block weight
|
|
|
|
pub const MAX_BLOCK_WEIGHT: usize = 80_000;
|
|
|
|
|
2017-11-20 03:59:07 +03:00
|
|
|
/// Maximum inputs for a block (issue#261)
|
|
|
|
/// Hundreds of inputs + 1 output might be slow to validate (issue#258)
|
|
|
|
pub const MAX_BLOCK_INPUTS: usize = 300_000; // soft fork down when too_high
|
|
|
|
|
2018-03-17 01:23:10 +03:00
|
|
|
/// Maximum inputs for a transaction
|
2018-03-17 01:26:36 +03:00
|
|
|
pub const MAX_TX_INPUTS: u64 = 2048;
|
2018-03-16 22:02:22 +03:00
|
|
|
|
2018-03-17 01:23:10 +03:00
|
|
|
/// Maximum outputs for a transaction
|
2018-03-16 22:02:22 +03:00
|
|
|
pub const MAX_TX_OUTPUTS: u64 = 500; // wallet uses 500 as max
|
|
|
|
|
2018-03-17 01:23:10 +03:00
|
|
|
/// Maximum kernels for a transaction
|
2018-03-16 22:02:22 +03:00
|
|
|
pub const MAX_TX_KERNELS: u64 = 2048;
|
|
|
|
|
2017-10-06 06:34:35 +03:00
|
|
|
/// Whether a block exceeds the maximum acceptable weight
|
|
|
|
pub fn exceeds_weight(input_len: usize, output_len: usize, kernel_len: usize) -> bool {
|
2017-11-01 02:32:33 +03:00
|
|
|
input_len * BLOCK_INPUT_WEIGHT + output_len * BLOCK_OUTPUT_WEIGHT
|
2018-03-04 03:19:54 +03:00
|
|
|
+ kernel_len * BLOCK_KERNEL_WEIGHT > MAX_BLOCK_WEIGHT || input_len > MAX_BLOCK_INPUTS
|
2017-10-06 06:34:35 +03:00
|
|
|
}
|
|
|
|
|
2017-10-10 03:08:17 +03:00
|
|
|
/// Fork every 250,000 blocks for first 2 years, simple number and just a
|
|
|
|
/// little less than 6 months.
|
|
|
|
pub const HARD_FORK_INTERVAL: u64 = 250_000;
|
|
|
|
|
|
|
|
/// Check whether the block version is valid at a given height, implements
|
|
|
|
/// 6 months interval scheduled hard forks for the first 2 years.
|
|
|
|
pub fn valid_header_version(height: u64, version: u16) -> bool {
|
|
|
|
// uncomment below as we go from hard fork to hard fork
|
|
|
|
if height <= HARD_FORK_INTERVAL && version == 1 {
|
|
|
|
true
|
|
|
|
/* } else if height <= 2 * HARD_FORK_INTERVAL && version == 2 {
|
|
|
|
true */
|
|
|
|
/* } else if height <= 3 * HARD_FORK_INTERVAL && version == 3 {
|
|
|
|
true */
|
|
|
|
/* } else if height <= 4 * HARD_FORK_INTERVAL && version == 4 {
|
|
|
|
true */
|
|
|
|
/* } else if height > 4 * HARD_FORK_INTERVAL && version > 4 {
|
|
|
|
true */
|
|
|
|
} else {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-20 17:22:40 +03:00
|
|
|
/// Time window in blocks to calculate block time median
|
2017-08-03 19:57:55 +03:00
|
|
|
pub const MEDIAN_TIME_WINDOW: u64 = 11;
|
2017-06-18 06:17:26 +03:00
|
|
|
|
2018-02-07 20:23:48 +03:00
|
|
|
/// Index at half the desired median
|
2018-01-27 10:48:53 +03:00
|
|
|
pub const MEDIAN_TIME_INDEX: u64 = MEDIAN_TIME_WINDOW / 2;
|
|
|
|
|
2017-07-20 17:22:40 +03:00
|
|
|
/// Number of blocks used to calculate difficulty adjustments
|
2018-01-27 10:48:53 +03:00
|
|
|
pub const DIFFICULTY_ADJUST_WINDOW: u64 = 60;
|
2017-06-18 06:17:26 +03:00
|
|
|
|
2017-07-20 17:22:40 +03:00
|
|
|
/// Average time span of the difficulty adjustment window
|
2017-08-03 19:57:55 +03:00
|
|
|
pub const BLOCK_TIME_WINDOW: u64 = DIFFICULTY_ADJUST_WINDOW * BLOCK_TIME_SEC;
|
2017-06-18 06:17:26 +03:00
|
|
|
|
2017-11-14 03:45:10 +03:00
|
|
|
/// Maximum size time window used for difficulty adjustments
|
2018-01-27 10:48:53 +03:00
|
|
|
pub const UPPER_TIME_BOUND: u64 = BLOCK_TIME_WINDOW * 2;
|
2017-06-18 06:17:26 +03:00
|
|
|
|
2017-11-14 03:45:10 +03:00
|
|
|
/// Minimum size time window used for difficulty adjustments
|
2018-01-27 10:48:53 +03:00
|
|
|
pub const LOWER_TIME_BOUND: u64 = BLOCK_TIME_WINDOW / 2;
|
|
|
|
|
|
|
|
/// Dampening factor to use for difficulty adjustment
|
|
|
|
pub const DAMP_FACTOR: u64 = 3;
|
|
|
|
|
|
|
|
/// The initial difficulty at launch. This should be over-estimated
|
|
|
|
/// and difficulty should come down at launch rather than up
|
|
|
|
/// Currently grossly over-estimated at 10% of current
|
|
|
|
/// ethereum GPUs (assuming 1GPU can solve a block at diff 1
|
|
|
|
/// in one block interval)
|
|
|
|
pub const INITIAL_DIFFICULTY: u64 = 1_000_000;
|
2017-06-18 06:17:26 +03:00
|
|
|
|
2017-12-22 20:15:44 +03:00
|
|
|
/// Consensus errors
|
|
|
|
#[derive(Clone, Debug, PartialEq)]
|
|
|
|
pub enum Error {
|
|
|
|
/// Inputs/outputs/kernels must be sorted lexicographically.
|
|
|
|
SortError,
|
|
|
|
}
|
|
|
|
|
2017-06-19 18:59:56 +03:00
|
|
|
/// Error when computing the next difficulty adjustment.
|
2017-06-18 06:17:26 +03:00
|
|
|
#[derive(Debug, Clone)]
|
2017-06-19 18:59:56 +03:00
|
|
|
pub struct TargetError(pub String);
|
|
|
|
|
|
|
|
impl fmt::Display for TargetError {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
write!(f, "Error computing new difficulty: {}", self.0)
|
|
|
|
}
|
2017-06-18 06:17:26 +03:00
|
|
|
}
|
2017-06-19 18:59:56 +03:00
|
|
|
|
|
|
|
/// Computes the proof-of-work difficulty that the next block should comply
|
|
|
|
/// with. Takes an iterator over past blocks, from latest (highest height) to
|
|
|
|
/// oldest (lowest height). The iterator produces pairs of timestamp and
|
|
|
|
/// difficulty for each block.
|
|
|
|
///
|
|
|
|
/// The difficulty calculation is based on both Digishield and GravityWave
|
|
|
|
/// family of difficulty computation, coming to something very close to Zcash.
|
|
|
|
/// The refence difficulty is an average of the difficulty over a window of
|
2018-02-07 20:23:48 +03:00
|
|
|
/// DIFFICULTY_ADJUST_WINDOW blocks. The corresponding timespan is calculated
|
|
|
|
/// by using the difference between the median timestamps at the beginning
|
2018-01-27 10:48:53 +03:00
|
|
|
/// and the end of the window.
|
2017-06-19 18:59:56 +03:00
|
|
|
pub fn next_difficulty<T>(cursor: T) -> Result<Difficulty, TargetError>
|
2017-09-29 21:44:25 +03:00
|
|
|
where
|
|
|
|
T: IntoIterator<Item = Result<(u64, Difficulty), TargetError>>,
|
2017-06-18 06:17:26 +03:00
|
|
|
{
|
2018-01-27 10:48:53 +03:00
|
|
|
// Create vector of difficulty data running from earliest
|
|
|
|
// to latest, and pad with simulated pre-genesis data to allow earlier
|
|
|
|
// adjustment if there isn't enough window data
|
|
|
|
// length will be DIFFICULTY_ADJUST_WINDOW+MEDIAN_TIME_WINDOW
|
|
|
|
let diff_data = global::difficulty_data_to_vector(cursor);
|
|
|
|
|
|
|
|
// Obtain the median window for the earlier time period
|
2018-03-30 21:21:06 +03:00
|
|
|
// the first MEDIAN_TIME_WINDOW elements
|
2018-03-04 03:19:54 +03:00
|
|
|
let mut window_earliest: Vec<u64> = diff_data
|
|
|
|
.iter()
|
2018-01-27 10:48:53 +03:00
|
|
|
.take(MEDIAN_TIME_WINDOW as usize)
|
|
|
|
.map(|n| n.clone().unwrap().0)
|
|
|
|
.collect();
|
2018-03-30 21:21:06 +03:00
|
|
|
// pick median
|
|
|
|
window_earliest.sort();
|
|
|
|
let earliest_ts = window_earliest[MEDIAN_TIME_INDEX as usize];
|
2018-01-27 10:48:53 +03:00
|
|
|
|
|
|
|
// Obtain the median window for the latest time period
|
2018-03-30 21:21:06 +03:00
|
|
|
// i.e. the last MEDIAN_TIME_WINDOW elements
|
2018-03-04 03:19:54 +03:00
|
|
|
let mut window_latest: Vec<u64> = diff_data
|
|
|
|
.iter()
|
2018-01-27 10:48:53 +03:00
|
|
|
.skip(DIFFICULTY_ADJUST_WINDOW as usize)
|
|
|
|
.map(|n| n.clone().unwrap().0)
|
|
|
|
.collect();
|
2018-03-30 21:21:06 +03:00
|
|
|
// pick median
|
2018-01-27 10:48:53 +03:00
|
|
|
window_latest.sort();
|
|
|
|
let latest_ts = window_latest[MEDIAN_TIME_INDEX as usize];
|
|
|
|
|
2018-03-30 21:21:06 +03:00
|
|
|
// median time delta
|
2018-01-27 10:48:53 +03:00
|
|
|
let ts_delta = latest_ts - earliest_ts;
|
|
|
|
|
2018-03-30 21:21:06 +03:00
|
|
|
// Get the difficulty sum of the last DIFFICULTY_ADJUST_WINDOW elements
|
|
|
|
let diff_sum = diff_data
|
|
|
|
.iter()
|
|
|
|
.skip(MEDIAN_TIME_WINDOW as usize)
|
|
|
|
.fold(0, |sum, d| sum + d.clone().unwrap().1.into_num());
|
|
|
|
|
|
|
|
// Apply dampening except when difficulty is near 1
|
|
|
|
let ts_damp = if diff_sum < DAMP_FACTOR * DIFFICULTY_ADJUST_WINDOW {
|
|
|
|
ts_delta
|
|
|
|
} else {
|
2018-04-23 21:55:25 +03:00
|
|
|
(1 * ts_delta + (DAMP_FACTOR - 1) * BLOCK_TIME_WINDOW) / DAMP_FACTOR
|
2018-01-27 10:48:53 +03:00
|
|
|
};
|
2017-06-18 06:17:26 +03:00
|
|
|
|
|
|
|
// Apply time bounds
|
|
|
|
let adj_ts = if ts_damp < LOWER_TIME_BOUND {
|
|
|
|
LOWER_TIME_BOUND
|
|
|
|
} else if ts_damp > UPPER_TIME_BOUND {
|
|
|
|
UPPER_TIME_BOUND
|
|
|
|
} else {
|
|
|
|
ts_damp
|
|
|
|
};
|
|
|
|
|
2018-03-30 21:21:06 +03:00
|
|
|
// AVOID BREAKING CONSENSUS FOR NOW WITH OLD DOUBLE TRUNCATION CALC
|
|
|
|
let difficulty = (diff_sum / DIFFICULTY_ADJUST_WINDOW) * BLOCK_TIME_WINDOW / adj_ts;
|
|
|
|
// EVENTUALLY BREAK CONSENSUS WITH THIS IMPROVED SINGLE TRUNCATION DIFF CALC
|
|
|
|
// let difficulty = diff_sum * BLOCK_TIME_SEC / adj_ts;
|
2018-01-27 10:48:53 +03:00
|
|
|
|
2018-03-30 21:21:06 +03:00
|
|
|
Ok(Difficulty::from_num(max(difficulty, 1)))
|
2017-06-18 06:17:26 +03:00
|
|
|
}
|
|
|
|
|
2017-12-22 20:15:44 +03:00
|
|
|
/// Consensus rule that collections of items are sorted lexicographically.
|
2017-10-13 20:23:18 +03:00
|
|
|
pub trait VerifySortOrder<T> {
|
|
|
|
/// Verify a collection of items is sorted as required.
|
2017-12-22 20:15:44 +03:00
|
|
|
fn verify_sort_order(&self) -> Result<(), Error>;
|
2017-10-13 20:23:18 +03:00
|
|
|
}
|