// 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. //! Facade and handler for the rest of the blockchain implementation //! and mostly the chain pipeline. use std::collections::VecDeque; use std::sync::{Arc, Mutex, RwLock}; use secp::pedersen::Commitment; use core::core::{Block, BlockHeader, Output}; use core::core::target::Difficulty; use core::core::hash::Hash; use grin_store::Error::NotFoundErr; use pipe; use store; use sumtree; use types::*; use util::LOGGER; use core::global::{MiningParameterMode, MINING_PARAMETER_MODE}; const MAX_ORPHANS: usize = 20; /// Facade to the blockchain block processing pipeline and storage. Provides /// the current view of the UTXO set according to the chain state. Also /// maintains locking for the pipeline to avoid conflicting processing. pub struct Chain { store: Arc, adapter: Arc, head: Arc>, orphans: Arc>>, sumtrees: Arc>, // POW verification function pow_verifier: fn(&BlockHeader, u32) -> bool, } unsafe impl Sync for Chain {} unsafe impl Send for Chain {} impl Chain { /// Check whether the chain exists. If not, the call to 'init' will /// expect an already mined genesis block. This keeps the chain free /// from needing to know about the mining implementation pub fn chain_exists(db_root: String) -> bool { let chain_store = store::ChainKVStore::new(db_root).unwrap(); match chain_store.head() { Ok(_) => true, Err(NotFoundErr) => false, Err(_) => false, } } /// Initializes the blockchain and returns a new Chain instance. Does a /// check /// on the current chain head to make sure it exists and creates one based /// on /// the genesis block if necessary. pub fn init(db_root: String, adapter: Arc, gen_block: Option, pow_verifier: fn(&BlockHeader, u32) -> bool) -> Result { let chain_store = store::ChainKVStore::new(db_root.clone())?; // check if we have a head in store, otherwise the genesis block is it let head = match chain_store.head() { Ok(tip) => tip, Err(NotFoundErr) => { if let None = gen_block { return Err(Error::GenesisBlockRequired); } let gen = gen_block.unwrap(); chain_store.save_block(&gen)?; chain_store.setup_height(&gen.header)?; // saving a new tip based on genesis let tip = Tip::new(gen.hash()); chain_store.save_head(&tip)?; info!(LOGGER, "Saved genesis block with hash {}", gen.hash()); tip } Err(e) => return Err(Error::StoreErr(e)), }; let store = Arc::new(chain_store); let sumtrees = sumtree::SumTrees::open(db_root, store.clone())?; Ok(Chain { store: store, adapter: adapter, head: Arc::new(Mutex::new(head)), orphans: Arc::new(Mutex::new(VecDeque::with_capacity(MAX_ORPHANS + 1))), sumtrees: Arc::new(RwLock::new(sumtrees)), pow_verifier: pow_verifier, }) } /// Attempt to add a new block to the chain. Returns the new chain tip if it /// has been added to the longest chain, None if it's added to an (as of /// now) orphan chain. pub fn process_block(&self, b: Block, opts: Options) -> Result, Error> { let head = self.store.head().map_err(&Error::StoreErr)?; let ctx = self.ctx_from_head(head, opts); let res = pipe::process_block(&b, ctx); match res { Ok(Some(ref tip)) => { // block got accepted and extended the head, updating our head let chain_head = self.head.clone(); { let mut head = chain_head.lock().unwrap(); *head = tip.clone(); } self.check_orphans(); } Ok(None) => {} Err(Error::Orphan) => { let mut orphans = self.orphans.lock().unwrap(); orphans.push_front((opts, b)); orphans.truncate(MAX_ORPHANS); } Err(ref e) => { info!( LOGGER, "Rejected block {} at {} : {:?}", b.hash(), b.header.height, e ); } } res } /// Attempt to add a new header to the header chain. Only necessary during /// sync. pub fn process_block_header(&self, bh: &BlockHeader, opts: Options) -> Result, Error> { let head = self.store.get_header_head().map_err(&Error::StoreErr)?; let ctx = self.ctx_from_head(head, opts); pipe::process_block_header(bh, ctx) } fn ctx_from_head(&self, head: Tip, opts: Options) -> pipe::BlockContext { let opts_in = opts; let param_ref = MINING_PARAMETER_MODE.read().unwrap(); let opts_in = match *param_ref { MiningParameterMode::AutomatedTesting => opts_in | EASY_POW, MiningParameterMode::UserTesting => opts_in | EASY_POW, MiningParameterMode::Production => opts_in, }; pipe::BlockContext { opts: opts_in, store: self.store.clone(), adapter: self.adapter.clone(), head: head, pow_verifier: self.pow_verifier, sumtrees: self.sumtrees.clone(), } } /// Pop orphans out of the queue and check if we can now accept them. fn check_orphans(&self) { // first check how many we have to retry, unfort. we can't extend the lock // in the loop as it needs to be freed before going in process_block let orphan_count; { let orphans = self.orphans.lock().unwrap(); orphan_count = orphans.len(); } // pop each orphan and retry, if still orphaned, will be pushed again for _ in 0..orphan_count { let popped; { let mut orphans = self.orphans.lock().unwrap(); popped = orphans.pop_back(); } if let Some((opts, orphan)) = popped { let _process_result = self.process_block(orphan, opts); } } } /// Gets an unspent output from its commitment. With return None if the /// output doesn't exist or has been spent. This querying is done in a /// way that's consistent with the current chain state and more /// specifically the current winning fork. pub fn get_unspent(&self, output_ref: &Commitment) -> Result { let sumtrees = self.sumtrees.read().unwrap(); let is_unspent = sumtrees.is_unspent(output_ref)?; if is_unspent { self.store .get_output_by_commit(output_ref) .map_err(&Error::StoreErr) } else { Err(Error::OutputNotFound) } } /// Sets the sumtree roots on a brand new block by applying the block on the /// current sumtree state. pub fn set_sumtree_roots(&self, b: &mut Block) -> Result<(), Error> { let mut sumtrees = self.sumtrees.write().unwrap(); let roots = sumtree::extending(&mut sumtrees, |mut extension| { // apply the block on the sumtrees and check the resulting root extension.apply_block(b)?; extension.force_rollback(); Ok(extension.roots()) })?; b.header.utxo_root = roots.0.hash; b.header.range_proof_root = roots.1.hash; b.header.kernel_root = roots.2.hash; Ok(()) } /// Total difficulty at the head of the chain pub fn total_difficulty(&self) -> Difficulty { self.head.lock().unwrap().clone().total_difficulty } /// Get the tip that's also the head of the chain pub fn head(&self) -> Result { Ok(self.head.lock().unwrap().clone()) } /// Block header for the chain head pub fn head_header(&self) -> Result { self.store.head_header().map_err(&Error::StoreErr) } /// Gets a block header by hash pub fn get_block(&self, h: &Hash) -> Result { self.store.get_block(h).map_err(&Error::StoreErr) } /// Gets a block header by hash pub fn get_block_header(&self, h: &Hash) -> Result { self.store.get_block_header(h).map_err(&Error::StoreErr) } /// Gets the block header at the provided height pub fn get_header_by_height(&self, height: u64) -> Result { self.store .get_header_by_height(height) .map_err(&Error::StoreErr) } /// Gets the block header by the provided output commitment pub fn get_block_header_by_output_commit(&self, commit: &Commitment) -> Result { self.store .get_block_header_by_output_commit(commit) .map_err(&Error::StoreErr) } /// Get the tip of the header chain pub fn get_header_head(&self) -> Result { self.store.get_header_head().map_err(&Error::StoreErr) } /// Builds an iterator on blocks starting from the current chain head and /// running backward. Specialized to return information pertaining to block /// difficulty calculation (timestamp and previous difficulties). pub fn difficulty_iter(&self) -> store::DifficultyIter { let head = self.head.lock().unwrap(); store::DifficultyIter::from(head.last_block_h, self.store.clone()) } }