/* 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 Ethasher.cpp * @author Gav Wood * @date 2014 */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include "BlockInfo.h" #include "Ethasher.h" using namespace std; using namespace chrono; using namespace dev; using namespace eth; Ethasher* dev::eth::Ethasher::s_this = nullptr; Ethasher::~Ethasher() { while (!m_lights.empty()) killCache(m_lights.begin()->first); } void Ethasher::killCache(h256 const& _s) { RecursiveGuard l(x_this); if (m_lights.count(_s)) { ethash_delete_light(m_lights.at(_s)); m_lights.erase(_s); } } void const* Ethasher::light(BlockInfo const& _header) { RecursiveGuard l(x_this); if (_header.number > c_ethashEpochLength * 2048) { std::ostringstream error; error << "block number is too high; max is " << c_ethashEpochLength * 2048 << "(was " << _header.number << ")"; throw std::invalid_argument( error.str() ); } if (!m_lights.count(_header.seedHash())) { ethash_params p = params((unsigned)_header.number); m_lights[_header.seedHash()] = ethash_new_light(&p, _header.seedHash().data()); } return m_lights[_header.seedHash()]; } #define IGNORE_EXCEPTIONS(X) try { X; } catch (...) {} bytesConstRef Ethasher::full(BlockInfo const& _header) { RecursiveGuard l(x_this); if (!m_fulls.count(_header.seedHash())) { // @memoryleak @bug place it on a pile for deletion - perhaps use shared_ptr. /* if (!m_fulls.empty()) { delete [] m_fulls.begin()->second.data(); m_fulls.erase(m_fulls.begin()); }*/ try { boost::filesystem::create_directories(getDataDir("ethash")); } catch (...) {} auto info = rlpList(c_ethashRevision, _header.seedHash()); std::string oldMemoFile = getDataDir("ethash") + "/full"; std::string memoFile = getDataDir("ethash") + "/full-R" + toString(c_ethashRevision) + "-" + toHex(_header.seedHash().ref().cropped(0, 8)); if (boost::filesystem::exists(oldMemoFile) && contents(oldMemoFile + ".info") == info) { // memofile valid - rename. boost::filesystem::rename(oldMemoFile, memoFile); } IGNORE_EXCEPTIONS(boost::filesystem::remove(oldMemoFile)); IGNORE_EXCEPTIONS(boost::filesystem::remove(oldMemoFile + ".info")); m_fulls[_header.seedHash()] = contentsNew(memoFile); if (!m_fulls[_header.seedHash()]) { ethash_params p = params((unsigned)_header.number); m_fulls[_header.seedHash()] = bytesRef(new byte[p.full_size], p.full_size); auto c = light(_header); ethash_prep_full(m_fulls[_header.seedHash()].data(), &p, c); writeFile(memoFile, m_fulls[_header.seedHash()]); } } return m_fulls[_header.seedHash()]; } ethash_params Ethasher::params(BlockInfo const& _header) { return params((unsigned)_header.number); } void Ethasher::readFull(BlockInfo const& _header, void* _dest) { if (!m_fulls.count(_header.seedHash())) { // @memoryleak @bug place it on a pile for deletion - perhaps use shared_ptr. /* if (!m_fulls.empty()) { delete [] m_fulls.begin()->second.data(); m_fulls.erase(m_fulls.begin()); }*/ try { boost::filesystem::create_directories(getDataDir("ethash")); } catch (...) {} auto info = rlpList(c_ethashRevision, _header.seedHash()); std::string oldMemoFile = getDataDir("ethash") + "/full"; std::string memoFile = getDataDir("ethash") + "/full-R" + toString(c_ethashRevision) + "-" + toHex(_header.seedHash().ref().cropped(0, 8)); if (boost::filesystem::exists(oldMemoFile) && contents(oldMemoFile + ".info") == info) { // memofile valid - rename. boost::filesystem::rename(oldMemoFile, memoFile); } IGNORE_EXCEPTIONS(boost::filesystem::remove(oldMemoFile)); IGNORE_EXCEPTIONS(boost::filesystem::remove(oldMemoFile + ".info")); ethash_params p = params((unsigned)_header.number); bytesRef r = contentsNew(memoFile, bytesRef((byte*)_dest, p.full_size)); if (!r) { auto c = light(_header); ethash_prep_full(_dest, &p, c); writeFile(memoFile, bytesConstRef((byte*)_dest, p.full_size)); } } } ethash_params Ethasher::params(unsigned _n) { ethash_params p; p.cache_size = ethash_get_cachesize(_n); p.full_size = ethash_get_datasize(_n); return p; } bool Ethasher::verify(BlockInfo const& _header) { if (_header.number >= c_ethashEpochLength * 2048) return false; h256 boundary = u256((bigint(1) << 256) / _header.difficulty); bool quick = ethash_quick_check_difficulty( _header.headerHash(WithoutNonce).data(), (uint64_t)(u64)_header.nonce, _header.mixHash.data(), boundary.data()); #if !ETH_DEBUG if (!quick) return false; #endif auto result = eval(_header); bool slow = result.value <= boundary && result.mixHash == _header.mixHash; #if ETH_DEBUG if (!quick && slow) { cwarn << "WARNING: evaluated result gives true whereas ethash_quick_check_difficulty gives false."; cwarn << "headerHash:" << _header.headerHash(WithoutNonce); cwarn << "nonce:" << _header.nonce; cwarn << "mixHash:" << _header.mixHash; cwarn << "difficulty:" << _header.difficulty; cwarn << "boundary:" << boundary; cwarn << "result.value:" << result.value; cwarn << "result.mixHash:" << result.mixHash; } #endif return slow; } Ethasher::Result Ethasher::eval(BlockInfo const& _header, Nonce const& _nonce) { auto p = Ethasher::params(_header); ethash_return_value r; if (Ethasher::get()->m_fulls.count(_header.seedHash())) ethash_compute_full(&r, Ethasher::get()->full(_header).data(), &p, _header.headerHash(WithoutNonce).data(), (uint64_t)(u64)_nonce); else ethash_compute_light(&r, Ethasher::get()->light(_header), &p, _header.headerHash(WithoutNonce).data(), (uint64_t)(u64)_nonce); // cdebug << "Ethasher::eval sha3(cache):" << sha3(Ethasher::get()->cache(_header)) << "hh:" << _header.headerHash(WithoutNonce) << "nonce:" << _nonce << " => " << h256(r.result, h256::ConstructFromPointer); return Result{h256(r.result, h256::ConstructFromPointer), h256(r.mix_hash, h256::ConstructFromPointer)}; }