/* 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 EthashAux.cpp * @author Gav Wood * @date 2014 */ #include "EthashAux.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include "BlockInfo.h" using namespace std; using namespace chrono; using namespace dev; using namespace eth; #define ETH_IGNORE_EXCEPTIONS(X) try { X; } catch (...) {} EthashAux* dev::eth::EthashAux::s_this = nullptr; EthashAux::~EthashAux() { } ethash_params EthashAux::params(BlockInfo const& _header) { return params((unsigned)_header.number); } ethash_params EthashAux::params(unsigned _n) { ethash_params p; p.cache_size = ethash_get_cachesize(_n); p.full_size = ethash_get_datasize(_n); return p; } h256 EthashAux::seedHash(unsigned _number) { unsigned epoch = _number / ETHASH_EPOCH_LENGTH; Guard l(get()->x_epochs); if (epoch >= get()->m_seedHashes.size()) { h256 ret; unsigned n = 0; if (!get()->m_seedHashes.empty()) { ret = get()->m_seedHashes.back(); n = get()->m_seedHashes.size() - 1; } get()->m_seedHashes.resize(epoch + 1); // cdebug << "Searching for seedHash of epoch " << epoch; for (; n <= epoch; ++n, ret = sha3(ret)) { get()->m_seedHashes[n] = ret; // cdebug << "Epoch" << n << "is" << ret; } } return get()->m_seedHashes[epoch]; } ethash_params EthashAux::params(h256 const& _seedHash) { Guard l(get()->x_epochs); unsigned epoch = 0; try { epoch = get()->m_epochs.at(_seedHash); } catch (...) { // cdebug << "Searching for seedHash " << _seedHash; for (h256 h; h != _seedHash && epoch < 2048; ++epoch, h = sha3(h), get()->m_epochs[h] = epoch) {} if (epoch == 2048) { std::ostringstream error; error << "apparent block number for " << _seedHash << " is too high; max is " << (ETHASH_EPOCH_LENGTH * 2048); throw std::invalid_argument(error.str()); } } return params(epoch * ETHASH_EPOCH_LENGTH); } void EthashAux::killCache(h256 const& _s) { RecursiveGuard l(x_this); m_lights.erase(_s); } EthashAux::LightType EthashAux::light(BlockInfo const& _header) { return light(_header.seedHash()); } EthashAux::LightType EthashAux::light(h256 const& _seedHash) { RecursiveGuard l(get()->x_this); LightType ret = get()->m_lights[_seedHash]; return ret ? ret : (get()->m_lights[_seedHash] = make_shared(_seedHash)); } EthashAux::LightAllocation::LightAllocation(h256 const& _seed) { auto p = params(_seed); size = p.cache_size; light = ethash_new_light(&p, _seed.data()); } EthashAux::LightAllocation::~LightAllocation() { ethash_delete_light(light); } EthashAux::FullType EthashAux::full(BlockInfo const& _header, bytesRef _dest, bool _createIfMissing) { return full(_header.seedHash(), _dest, _createIfMissing); } EthashAux::FullType EthashAux::full(h256 const& _seedHash, bytesRef _dest, bool _createIfMissing) { RecursiveGuard l(get()->x_this); FullType ret = get()->m_fulls[_seedHash].lock(); if (ret && _dest) { assert(ret->data.size() <= _dest.size()); ret->data.copyTo(_dest); return FullType(); } if (!ret) { // drop our last used cache sine we're allocating another 1GB. get()->m_lastUsedFull.reset(); try { boost::filesystem::create_directories(getDataDir("ethash")); } catch (...) {} auto info = rlpList(Ethash::revision(), _seedHash); std::string oldMemoFile = getDataDir("ethash") + "/full"; std::string memoFile = getDataDir("ethash") + "/full-R" + toString(ETHASH_REVISION) + "-" + toHex(_seedHash.ref().cropped(0, 8)); if (boost::filesystem::exists(oldMemoFile) && contents(oldMemoFile + ".info") == info) { // memofile valid - rename. boost::filesystem::rename(oldMemoFile, memoFile); } ETH_IGNORE_EXCEPTIONS(boost::filesystem::remove(oldMemoFile)); ETH_IGNORE_EXCEPTIONS(boost::filesystem::remove(oldMemoFile + ".info")); ethash_params p = params(_seedHash); assert(!_dest || _dest.size() >= p.full_size); // must be big enough. bytesRef r = contentsNew(memoFile, _dest); if (!r) { if (!_createIfMissing) return FullType(); // file didn't exist. if (_dest) // buffer was passed in - no insertion into cache nor need to allocate r = _dest; else r = bytesRef(new byte[p.full_size], p.full_size); ethash_prep_full(r.data(), &p, light(_seedHash)->light); writeFile(memoFile, r); } if (_dest) return FullType(); ret = make_shared(r); get()->m_fulls[_seedHash] = ret; } get()->m_lastUsedFull = ret; return ret; } Ethash::Result EthashAux::eval(BlockInfo const& _header, Nonce const& _nonce) { return eval(_header.seedHash(), _header.headerHash(WithoutNonce), _nonce); } Ethash::Result EthashAux::FullAllocation::compute(h256 const& _seedHash, h256 const& _headerHash, Nonce const& _nonce) const { ethash_return_value r; auto p = EthashAux::params(_seedHash); ethash_compute_full(&r, data.data(), &p, _headerHash.data(), (uint64_t)(u64)_nonce); return Ethash::Result{h256(r.result, h256::ConstructFromPointer), h256(r.mix_hash, h256::ConstructFromPointer)}; } Ethash::Result EthashAux::LightAllocation::compute(h256 const& _seedHash, h256 const& _headerHash, Nonce const& _nonce) const { ethash_return_value r; auto p = EthashAux::params(_seedHash); ethash_compute_light(&r, light, &p, _headerHash.data(), (uint64_t)(u64)_nonce); return Ethash::Result{h256(r.result, h256::ConstructFromPointer), h256(r.mix_hash, h256::ConstructFromPointer)}; } Ethash::Result EthashAux::eval(h256 const& _seedHash, h256 const& _headerHash, Nonce const& _nonce) { if (auto dag = EthashAux::get()->full(_seedHash, bytesRef(), false)) return dag->compute(_seedHash, _headerHash, _nonce); return EthashAux::get()->light(_seedHash)->compute(_seedHash, _headerHash, _nonce); }