You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

114 lines
3.0 KiB

/*
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.
Foobar 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 Foobar. If not, see <http://www.gnu.org/licenses/>.
*/
/** @file BlockChain.cpp
* @author Gav Wood <i@gavwood.com>
* @date 2014
*/
#include "Common.h"
#include "BlockInfo.h"
#include "BlockChain.h"
using namespace std;
using namespace eth;
BlockChain::BlockChain()
{
// Initialise with the genesis as the last block on the longest chain.
m_lastBlockHash = m_genesisHash = BlockInfo::genesis().hash;
m_genesisBlock = BlockInfo::createGenesisBlock();
}
BlockChain::~BlockChain()
{
}
std::vector<u256> blockChain() const
{
// TODO: return the current valid block chain from most recent to genesis.
// TODO: arguments for specifying a set of early-ends
return std::vector<u256>();
}
void BlockChain::import(bytes const& _block)
{
11 years ago
BlockInfo bi;
try
{
// VERIFY: populates from the block and checks the block is internally coherent.
bi.populate(&_block);
bi.verifyInternals(&_block);
auto newHash = eth::sha3(_block);
11 years ago
// Check block doesn't already exist first!
if (m_numberAndParent.count(newHash))
return;
// Work out its number as the parent's number + 1
auto it = m_numberAndParent.find(bi.parentHash);
if (it == m_numberAndParent.end())
// We don't know the parent (yet) - discard for now. It'll get resent to us if we find out about its ancestry later on.
return;
bi.number = it->second.first + 1;
// CHECK ANCESTRY:
// TODO: check it hashes according to proof of work.
// TODO: check timestamp is after previous timestamp.
// TODO: check difficulty is correct given the two timestamps.
11 years ago
// Insert into DB
m_numberAndParent[newHash] = make_pair(bi.number, bi.parentHash);
m_children.insert(make_pair(bi.parentHash, newHash));
// TODO: put _block onto disk and load into cache.
// This might be the new last block; count back through ancestors to common shared ancestor and compare to current.
}
catch (...)
{
// Exit silently on exception(?)
return;
}
}
bytesConstRef BlockChain::block(u256 _hash) const
{
auto it = m_cache.find(_hash);
if (it == m_cache.end())
{
// Load block from disk.
pair<u256, std::shared_ptr<MappedBlock>> loaded;
it = m_cache.insert(loaded).first;
}
return it->second->data();
}
bytesConstRef BlockChain::lastBlock() const
{
if (m_lastBlockHash == m_genesisHash)
return bytesConstRef((bytes*)&m_genesisBlock);
return block(m_lastBlockHash);
}
u256 BlockChain::lastBlockNumber() const
{
if (m_lastBlockHash == m_genesisHash)
return 0;
return m_numberAndParent[m_lastBlockHash].first;
}