/* This file is part of cpp-ethereum. cpp-ethereum 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. cpp-ethereum 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 cpp-ethereum. If not, see . */ /** @file BlockChain.h * @author Gav Wood * @date 2014 */ #pragma once #include #include #include #include #include #include #include #include #include #include #include #include "BlockDetails.h" #include "Account.h" #include "Transaction.h" #include "BlockQueue.h" #include "VerifiedBlock.h" #include "State.h" namespace std { template <> struct hash> { size_t operator()(pair const& _x) const { return hash()(_x.first) ^ hash()(_x.second); } }; } namespace dev { class OverlayDB; namespace eth { static const h256s NullH256s; class State; struct AlreadyHaveBlock: virtual Exception {}; struct UnknownParent: virtual Exception {}; struct FutureTime: virtual Exception {}; struct TransientError: virtual Exception {}; struct BlockChainChat: public LogChannel { static const char* name(); static const int verbosity = 5; }; struct BlockChainNote: public LogChannel { static const char* name(); static const int verbosity = 3; }; struct BlockChainWarn: public LogChannel { static const char* name(); static const int verbosity = 1; }; struct BlockChainDebug: public LogChannel { static const char* name(); static const int verbosity = 0; }; // TODO: Move all this Genesis stuff into Genesis.h/.cpp std::unordered_map const& genesisState(); ldb::Slice toSlice(h256 const& _h, unsigned _sub = 0); ldb::Slice toSlice(uint64_t _n, unsigned _sub = 0); using BlocksHash = std::unordered_map; using TransactionHashes = h256s; using UncleHashes = h256s; enum { ExtraDetails = 0, ExtraBlockHash, ExtraTransactionAddress, ExtraLogBlooms, ExtraReceipts, ExtraBlocksBlooms }; using ProgressCallback = std::function; using StateDefinition = std::unordered_map; class VersionChecker { public: VersionChecker(std::string const& _dbPath, h256 const& _genesisHash); }; /** * @brief Implements the blockchain database. All data this gives is disk-backed. * @threadsafe */ class BlockChain { public: /// Doesn't open the database - if you want it open it's up to you to subclass this and open it /// in the constructor there. BlockChain(bytes const& _genesisBlock, StateDefinition const& _genesisState, std::string const& _path); ~BlockChain(); /// Reopen everything. virtual void reopen(WithExisting _we = WithExisting::Trust, ProgressCallback const& _pc = ProgressCallback()) { close(); open(m_genesisBlock, m_genesisState, m_dbPath); openDatabase(m_dbPath, _we, _pc); } /// (Potentially) renders invalid existing bytesConstRef returned by lastBlock. /// To be called from main loop every 100ms or so. void process(); /// Sync the chain with any incoming blocks. All blocks should, if processed in order. /// @returns fresh blocks, dead blocks and true iff there are additional blocks to be processed waiting. std::tuple sync(BlockQueue& _bq, OverlayDB const& _stateDB, unsigned _max); /// Attempt to import the given block directly into the CanonBlockChain and sync with the state DB. /// @returns the block hashes of any blocks that came into/went out of the canonical block chain. std::pair attemptImport(bytes const& _block, OverlayDB const& _stateDB, ImportRequirements::value _ir = ImportRequirements::Everything) noexcept; /// Import block into disk-backed DB /// @returns the block hashes of any blocks that came into/went out of the canonical block chain. ImportRoute import(bytes const& _block, OverlayDB const& _stateDB, ImportRequirements::value _ir = ImportRequirements::Everything); ImportRoute import(VerifiedBlockRef const& _block, OverlayDB const& _db, ImportRequirements::value _ir = ImportRequirements::Everything); /// Returns true if the given block is known (though not necessarily a part of the canon chain). bool isKnown(h256 const& _hash) const; /// Get the partial-header of a block (or the most recent mined if none given). Thread-safe. BlockInfo info(h256 const& _hash) const { return BlockInfo(headerData(_hash), CheckNothing, _hash, HeaderData); } BlockInfo info() const { return info(currentHash()); } /// Get a block (RLP format) for the given hash (or the most recent mined if none given). Thread-safe. bytes block(h256 const& _hash) const; bytes block() const { return block(currentHash()); } /// Get a block (RLP format) for the given hash (or the most recent mined if none given). Thread-safe. bytes headerData(h256 const& _hash) const; bytes headerData() const { return headerData(currentHash()); } /// Get the familial details concerning a block (or the most recent mined if none given). Thread-safe. BlockDetails details(h256 const& _hash) const { return queryExtras(_hash, m_details, x_details, NullBlockDetails); } BlockDetails details() const { return details(currentHash()); } /// Get the transactions' log blooms of a block (or the most recent mined if none given). Thread-safe. BlockLogBlooms logBlooms(h256 const& _hash) const { return queryExtras(_hash, m_logBlooms, x_logBlooms, NullBlockLogBlooms); } BlockLogBlooms logBlooms() const { return logBlooms(currentHash()); } /// Get the transactions' receipts of a block (or the most recent mined if none given). Thread-safe. /// receipts are given in the same order are in the same order as the transactions BlockReceipts receipts(h256 const& _hash) const { return queryExtras(_hash, m_receipts, x_receipts, NullBlockReceipts); } BlockReceipts receipts() const { return receipts(currentHash()); } /// Get the transaction by block hash and index; TransactionReceipt transactionReceipt(h256 const& _blockHash, unsigned _i) const { return receipts(_blockHash).receipts[_i]; } /// Get the transaction receipt by transaction hash. Thread-safe. TransactionReceipt transactionReceipt(h256 const& _transactionHash) const { TransactionAddress ta = queryExtras(_transactionHash, m_transactionAddresses, x_transactionAddresses, NullTransactionAddress); if (!ta) return bytesConstRef(); return transactionReceipt(ta.blockHash, ta.index); } /// Get a list of transaction hashes for a given block. Thread-safe. TransactionHashes transactionHashes(h256 const& _hash) const { auto b = block(_hash); RLP rlp(b); h256s ret; for (auto t: rlp[1]) ret.push_back(sha3(t.data())); return ret; } TransactionHashes transactionHashes() const { return transactionHashes(currentHash()); } /// Get a list of uncle hashes for a given block. Thread-safe. UncleHashes uncleHashes(h256 const& _hash) const { auto b = block(_hash); RLP rlp(b); h256s ret; for (auto t: rlp[2]) ret.push_back(sha3(t.data())); return ret; } UncleHashes uncleHashes() const { return uncleHashes(currentHash()); } /// Get the hash for a given block's number. h256 numberHash(unsigned _i) const { if (!_i) return genesisHash(); return queryExtras(_i, m_blockHashes, x_blockHashes, NullBlockHash).value; } /// Get the last N hashes for a given block. (N is determined by the LastHashes type.) LastHashes lastHashes() const { return lastHashes(number()); } LastHashes lastHashes(unsigned _i) const; /** Get the block blooms for a number of blocks. Thread-safe. * @returns the object pertaining to the blocks: * level 0: * 0x, 0x + 1, .. (1x - 1) * 1x, 1x + 1, .. (2x - 1) * ... * (255x .. (256x - 1)) * level 1: * 0x .. (1x - 1), 1x .. (2x - 1), ..., (255x .. (256x - 1)) * 256x .. (257x - 1), 257x .. (258x - 1), ..., (511x .. (512x - 1)) * ... * level n, index i, offset o: * i * (x ^ n) + o * x ^ (n - 1) */ BlocksBlooms blocksBlooms(unsigned _level, unsigned _index) const { return blocksBlooms(chunkId(_level, _index)); } BlocksBlooms blocksBlooms(h256 const& _chunkId) const { return queryExtras(_chunkId, m_blocksBlooms, x_blocksBlooms, NullBlocksBlooms); } void clearBlockBlooms(unsigned _begin, unsigned _end); LogBloom blockBloom(unsigned _number) const { return blocksBlooms(chunkId(0, _number / c_bloomIndexSize)).blooms[_number % c_bloomIndexSize]; } std::vector withBlockBloom(LogBloom const& _b, unsigned _earliest, unsigned _latest) const; std::vector withBlockBloom(LogBloom const& _b, unsigned _earliest, unsigned _latest, unsigned _topLevel, unsigned _index) const; /// Returns true if transaction is known. Thread-safe bool isKnownTransaction(h256 const& _transactionHash) const { TransactionAddress ta = queryExtras(_transactionHash, m_transactionAddresses, x_transactionAddresses, NullTransactionAddress); return !!ta; } /// Get a transaction from its hash. Thread-safe. bytes transaction(h256 const& _transactionHash) const { TransactionAddress ta = queryExtras(_transactionHash, m_transactionAddresses, x_transactionAddresses, NullTransactionAddress); if (!ta) return bytes(); return transaction(ta.blockHash, ta.index); } std::pair transactionLocation(h256 const& _transactionHash) const { TransactionAddress ta = queryExtras(_transactionHash, m_transactionAddresses, x_transactionAddresses, NullTransactionAddress); if (!ta) return std::pair(h256(), 0); return std::make_pair(ta.blockHash, ta.index); } /// Get a block's transaction (RLP format) for the given block hash (or the most recent mined if none given) & index. Thread-safe. bytes transaction(h256 const& _blockHash, unsigned _i) const { bytes b = block(_blockHash); return RLP(b)[1][_i].data().toBytes(); } bytes transaction(unsigned _i) const { return transaction(currentHash(), _i); } /// Get all transactions from a block. std::vector transactions(h256 const& _blockHash) const { bytes b = block(_blockHash); std::vector ret; for (auto const& i: RLP(b)[1]) ret.push_back(i.data().toBytes()); return ret; } std::vector transactions() const { return transactions(currentHash()); } /// Get a number for the given hash (or the most recent mined if none given). Thread-safe. unsigned number(h256 const& _hash) const { return details(_hash).number; } unsigned number() const { return m_lastBlockNumber; } /// Get a given block (RLP format). Thread-safe. h256 currentHash() const { ReadGuard l(x_lastBlockHash); return m_lastBlockHash; } /// Get the hash of the genesis block. Thread-safe. h256 genesisHash() const { return m_genesisHash; } /// Get all blocks not allowed as uncles given a parent (i.e. featured as uncles/main in parent, parent + 1, ... parent + @a _generations). /// @returns set including the header-hash of every parent (including @a _parent) up to and including generation + @a _generations /// togther with all their quoted uncles. h256Hash allKinFrom(h256 const& _parent, unsigned _generations) const; /// Run through database and verify all blocks by reevaluating. /// Will call _progress with the progress in this operation first param done, second total. void rebuild(std::string const& _path, ProgressCallback const& _progress = std::function(), bool _prepPoW = false); /// Alter the head of the chain to some prior block along it. void rewind(unsigned _newHead); /// Rescue the database. void rescue(OverlayDB& _db); /** @returns a tuple of: * - an vector of hashes of all blocks between @a _from and @a _to, all blocks are ordered first by a number of * blocks that are parent-to-child, then two sibling blocks, then a number of blocks that are child-to-parent; * - the block hash of the latest common ancestor of both blocks; * - the index where the latest common ancestor of both blocks would either be found or inserted, depending * on whether it is included. * * @param _common if true, include the common ancestor in the returned vector. * @param _pre if true, include all block hashes running from @a _from until the common ancestor in the returned vector. * @param _post if true, include all block hashes running from the common ancestor until @a _to in the returned vector. * * e.g. if the block tree is 3a -> 2a -> 1a -> g and 2b -> 1b -> g (g is genesis, *a, *b are competing chains), * then: * @code * treeRoute(3a, 2b, false) == make_tuple({ 3a, 2a, 1a, 1b, 2b }, g, 3); * treeRoute(2a, 1a, false) == make_tuple({ 2a, 1a }, 1a, 1) * treeRoute(1a, 2a, false) == make_tuple({ 1a, 2a }, 1a, 0) * treeRoute(1b, 2a, false) == make_tuple({ 1b, 1a, 2a }, g, 1) * treeRoute(3a, 2b, true) == make_tuple({ 3a, 2a, 1a, g, 1b, 2b }, g, 3); * treeRoute(2a, 1a, true) == make_tuple({ 2a, 1a }, 1a, 1) * treeRoute(1a, 2a, true) == make_tuple({ 1a, 2a }, 1a, 0) * treeRoute(1b, 2a, true) == make_tuple({ 1b, g, 1a, 2a }, g, 1) * @endcode */ std::tuple treeRoute(h256 const& _from, h256 const& _to, bool _common = true, bool _pre = true, bool _post = true) const; struct Statistics { unsigned memBlocks; unsigned memDetails; unsigned memLogBlooms; unsigned memReceipts; unsigned memTransactionAddresses; unsigned memBlockHashes; unsigned memTotal() const { return memBlocks + memDetails + memLogBlooms + memReceipts + memTransactionAddresses + memBlockHashes; } }; /// @returns statistics about memory usage. Statistics usage(bool _freshen = false) const { if (_freshen) updateStats(); return m_lastStats; } /// Deallocate unused data. void garbageCollect(bool _force = false); /// Change the function that is called with a bad block. template void setOnBad(T const& _t) { m_onBad = _t; } /// Get a pre-made genesis State object. State genesisState(OverlayDB const& _db); /// Verify block and prepare it for enactment virtual VerifiedBlockRef verifyBlock(bytesConstRef _block, std::function const& _onBad, ImportRequirements::value _ir) const = 0; protected: static h256 chunkId(unsigned _level, unsigned _index) { return h256(_index * 0xff + _level); } /// Initialise everything and ready for openning the database. // TODO: rename to init void open(bytes const& _genesisBlock, std::unordered_map const& _genesisState, std::string const& _path); /// Open the database. // TODO: rename to open. unsigned openDatabase(std::string const& _path, WithExisting _we); /// Finalise everything and close the database. void close(); /// Open the database, rebuilding if necessary. void openDatabase(std::string const& _path, WithExisting _we, ProgressCallback const& _pc) { if (openDatabase(_path, _we) != c_minorProtocolVersion || _we == WithExisting::Verify) rebuild(_path, _pc); } template T queryExtras(K const& _h, std::unordered_map& _m, boost::shared_mutex& _x, T const& _n, ldb::DB* _extrasDB = nullptr) const { { ReadGuard l(_x); auto it = _m.find(_h); if (it != _m.end()) return it->second; } std::string s; (_extrasDB ? _extrasDB : m_extrasDB)->Get(m_readOptions, toSlice(_h, N), &s); if (s.empty()) return _n; noteUsed(_h, N); WriteGuard l(_x); auto ret = _m.insert(std::make_pair(_h, T(RLP(s)))); return ret.first->second; } template T queryExtras(h256 const& _h, std::unordered_map& _m, boost::shared_mutex& _x, T const& _n, ldb::DB* _extrasDB = nullptr) const { return queryExtras(_h, _m, _x, _n, _extrasDB); } void checkConsistency(); /// The caches of the disk DB and their locks. mutable SharedMutex x_blocks; mutable BlocksHash m_blocks; mutable SharedMutex x_details; mutable BlockDetailsHash m_details; mutable SharedMutex x_logBlooms; mutable BlockLogBloomsHash m_logBlooms; mutable SharedMutex x_receipts; mutable BlockReceiptsHash m_receipts; mutable SharedMutex x_transactionAddresses; mutable TransactionAddressHash m_transactionAddresses; mutable SharedMutex x_blockHashes; mutable BlockHashHash m_blockHashes; mutable SharedMutex x_blocksBlooms; mutable BlocksBloomsHash m_blocksBlooms; using CacheID = std::pair; mutable Mutex x_cacheUsage; mutable std::deque> m_cacheUsage; mutable std::unordered_set m_inUse; void noteUsed(h256 const& _h, unsigned _extra = (unsigned)-1) const; void noteUsed(uint64_t const& _h, unsigned _extra = (unsigned)-1) const { (void)_h; (void)_extra; } // don't note non-hash types std::chrono::system_clock::time_point m_lastCollection; void noteCanonChanged() const { Guard l(x_lastLastHashes); m_lastLastHashes.clear(); } mutable Mutex x_lastLastHashes; mutable LastHashes m_lastLastHashes; mutable unsigned m_lastLastHashesNumber = (unsigned)-1; void updateStats() const; mutable Statistics m_lastStats; /// The disk DBs. Thread-safe, so no need for locks. ldb::DB* m_blocksDB; ldb::DB* m_extrasDB; /// Hash of the last (valid) block on the longest chain. mutable boost::shared_mutex x_lastBlockHash; h256 m_lastBlockHash; unsigned m_lastBlockNumber = 0; /// Genesis block info. h256 m_genesisHash; bytes m_genesisBlock; std::unordered_map m_genesisState; ldb::ReadOptions m_readOptions; ldb::WriteOptions m_writeOptions; std::function m_onBad; ///< Called if we have a block that doesn't verify. std::string m_dbPath; friend std::ostream& operator<<(std::ostream& _out, BlockChain const& _bc); }; template class FullBlockChain: public BlockChain { public: using BlockHeader = typename Sealer::BlockHeader; FullBlockChain(bytes const& _genesisBlock, StateDefinition const& _genesisState, std::string const& _path, WithExisting _we, ProgressCallback const& _pc = ProgressCallback()): BlockChain(_genesisBlock, _genesisState, _path) { openDatabase(_path, _we, _pc); } /// Get the header of a block (or the most recent mined if none given). Thread-safe. typename Sealer::BlockHeader header(h256 const& _hash) const { return typename Sealer::BlockHeader(headerData(_hash), IgnoreSeal, _hash, HeaderData); } typename Sealer::BlockHeader header() const { return header(currentHash()); } virtual VerifiedBlockRef verifyBlock(bytesConstRef _block, std::function const& _onBad, ImportRequirements::value _ir) const override { VerifiedBlockRef res; BlockHeader h; try { h = BlockHeader(_block, (_ir & ImportRequirements::ValidSeal) ? Strictness::CheckEverything : Strictness::QuickNonce); h.verifyInternals(_block); if ((_ir & ImportRequirements::Parent) != 0) { bytes parentHeader(headerData(h.parentHash())); if (parentHeader.empty()) BOOST_THROW_EXCEPTION(InvalidParentHash()); h.verifyParent(typename Sealer::BlockHeader(parentHeader, IgnoreSeal, h.parentHash(), HeaderData)); } res.info = static_cast(h); } catch (Exception& ex) { ex << errinfo_phase(1); ex << errinfo_now(time(0)); ex << errinfo_block(_block.toBytes()); // only populate extraData if we actually managed to extract it. otherwise, // we might be clobbering the existing one. if (!h.extraData().empty()) ex << errinfo_extraData(h.extraData()); if (_onBad) _onBad(ex); throw; } RLP r(_block); unsigned i = 0; if (_ir && ImportRequirements::UncleBasic) for (auto const& uncle: r[2]) { BlockHeader h; try { h.populateFromHeader(RLP(uncle.data()), (_ir & ImportRequirements::UncleSeals) ? Strictness::CheckEverything : Strictness::IgnoreSeal); } catch (Exception& ex) { ex << errinfo_phase(1); ex << errinfo_uncleIndex(i); ex << errinfo_now(time(0)); ex << errinfo_block(_block.toBytes()); // only populate extraData if we actually managed to extract it. otherwise, // we might be clobbering the existing one. if (!h.extraData().empty()) ex << errinfo_extraData(h.extraData()); if (_onBad) _onBad(ex); throw; } ++i; } i = 0; if (_ir && ImportRequirements::TransactionBasic) for (RLP const& tr: r[1]) { bytesConstRef d = tr.data(); try { res.transactions.push_back(Transaction(d, (_ir & ImportRequirements::TransactionSignatures) ? CheckTransaction::Everything : CheckTransaction::None)); } catch (Exception& ex) { ex << errinfo_phase(1); ex << errinfo_transactionIndex(i); ex << errinfo_transaction(d.toBytes()); ex << errinfo_block(_block.toBytes()); // only populate extraData if we actually managed to extract it. otherwise, // we might be clobbering the existing one. if (!h.extraData().empty()) ex << errinfo_extraData(h.extraData()); if (_onBad) _onBad(ex); throw; } ++i; } res.block = bytesConstRef(_block); return res; } protected: /// Constructor for derived classes to use when they'll open the chain db afterwards. FullBlockChain(bytes const& _genesisBlock, StateDefinition const& _genesisState, std::string const& _path): BlockChain(_genesisBlock, _genesisState, _path) {} }; std::ostream& operator<<(std::ostream& _out, BlockChain const& _bc); } }