/* 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 Miner.h * @author Gav Wood * @date 2015 */ #pragma once #include #include #include #include #include #include #include #include #include "EthashAux.h" #define MINER_WAIT_STATE_UNKNOWN 0 #define MINER_WAIT_STATE_WORK 1 #define MINER_WAIT_STATE_DAG 2 #define DAG_LOAD_MODE_PARALLEL 0 #define DAG_LOAD_MODE_SEQUENTIAL 1 #define DAG_LOAD_MODE_SINGLE 2 #define STRATUM_PROTOCOL_STRATUM 0 #define STRATUM_PROTOCOL_ETHPROXY 1 #define STRATUM_PROTOCOL_ETHEREUMSTRATUM 2 using namespace std; typedef struct { string host; string port; string user; string pass; } cred_t; namespace dev { namespace eth { enum class MinerType { CL, CUDA, Mixed }; /// Describes the progress of a mining operation. struct WorkingProgress { uint64_t hashes = 0; ///< Total number of hashes computed. uint64_t ms = 0; ///< Total number of milliseconds of mining thus far. uint64_t rate() const { return ms == 0 ? 0 : hashes * 1000 / ms; } }; inline std::ostream& operator<<(std::ostream& _out, WorkingProgress _p) { float mh = _p.rate() / 1000000.0f; char mhs[16]; sprintf(mhs, "%.2f", mh); _out << std::string(mhs) + "MH/s"; return _out; } class SolutionStats { public: void accepted() { accepts++; } void rejected() { rejects++; } void failed() { failures++; } void acceptedStale() { acceptedStales++; } void rejectedStale() { rejectedStales++; } void reset() { accepts = rejects = failures = acceptedStales = rejectedStales = 0; } unsigned getAccepts() { return accepts; } unsigned getRejects() { return rejects; } unsigned getFailures() { return failures; } unsigned getAcceptedStales() { return acceptedStales; } unsigned getRejectedStales() { return rejectedStales; } private: unsigned accepts = 0; unsigned rejects = 0; unsigned failures = 0; unsigned acceptedStales = 0; unsigned rejectedStales = 0; }; inline std::ostream& operator<<(std::ostream& os, SolutionStats s) { return os << "[A" << s.getAccepts() << "+" << s.getAcceptedStales() << ":R" << s.getRejects() << "+" << s.getRejectedStales() << ":F" << s.getFailures() << "]"; } class Miner; /** * @brief Class for hosting one or more Miners. * @warning Must be implemented in a threadsafe manner since it will be called from multiple * miner threads. */ class FarmFace { public: virtual ~FarmFace() = default; /** * @brief Called from a Miner to note a WorkPackage has a solution. * @param _p The solution. * @param _finder The miner that found it. * @return true iff the solution was good (implying that mining should be . */ virtual bool submitProof(Solution const& _p, Miner* _finder) = 0; }; /** * @brief A miner - a member and adoptee of the Farm. * @warning Not threadsafe. It is assumed Farm will synchronise calls to/from this class. */ class Miner { public: using ConstructionInfo = std::pair; Miner(ConstructionInfo const& _ci): m_farm(_ci.first), m_index(_ci.second) {} virtual ~Miner() {} // API FOR THE FARM TO CALL IN WITH void setWork(WorkPackage const& _work = WorkPackage()) { auto old = m_work; { Guard l(x_work); m_work = _work; } if (!!_work) { pause(); kickOff(); } else if (!_work && !!old) pause(); m_hashCount = 0; } uint64_t hashCount() const { return m_hashCount; } void resetHashCount() { m_hashCount = 0; } unsigned index() const { return m_index; } protected: // REQUIRED TO BE REIMPLEMENTED BY A SUBCLASS: /** * @brief Begin working on a given work package, discarding any previous work. * @param _work The package for which to find a solution. */ virtual void kickOff() = 0; /** * @brief No work left to be done. Pause until told to kickOff(). */ virtual void pause() = 0; // AVAILABLE FOR A SUBCLASS TO CALL: /** * @brief Notes that the Miner found a solution. * @param _s The solution. * @return true if the solution was correct and that the miner should pause. */ bool submitProof(Solution const& _s) { if (!m_farm) return true; if (m_farm->submitProof(_s, this)) { Guard l(x_work); m_work.reset(); return true; } return false; } WorkPackage const& work() const { Guard l(x_work); return m_work; } void accumulateHashes(unsigned _n) { m_hashCount += _n; } static unsigned s_dagLoadMode; static volatile unsigned s_dagLoadIndex; static unsigned s_dagCreateDevice; static volatile void* s_dagInHostMemory; private: FarmFace* m_farm = nullptr; unsigned m_index; uint64_t m_hashCount = 0; WorkPackage m_work; mutable Mutex x_work; }; } }