|
|
|
/*
|
|
|
|
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 <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
/** @file EthashCPUMiner.cpp
|
|
|
|
* @author Gav Wood <i@gavwood.com>
|
|
|
|
* @date 2014
|
|
|
|
*
|
|
|
|
* Determines the PoW algorithm.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "EthashCPUMiner.h"
|
|
|
|
#include <thread>
|
|
|
|
#include <chrono>
|
|
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
#if ETH_CPUID || !ETH_TRUE
|
|
|
|
#define HAVE_STDINT_H
|
|
|
|
#include <libcpuid/libcpuid.h>
|
|
|
|
#endif
|
|
|
|
using namespace std;
|
|
|
|
using namespace dev;
|
|
|
|
using namespace eth;
|
|
|
|
|
|
|
|
unsigned EthashCPUMiner::s_numInstances = 0;
|
|
|
|
|
|
|
|
#if ETH_CPUID || !ETH_TRUE
|
|
|
|
static string jsonEncode(map<string, string> const& _m)
|
|
|
|
{
|
|
|
|
string ret = "{";
|
|
|
|
|
|
|
|
for (auto const& i: _m)
|
|
|
|
{
|
|
|
|
string k = boost::replace_all_copy(boost::replace_all_copy(i.first, "\\", "\\\\"), "'", "\\'");
|
|
|
|
string v = boost::replace_all_copy(boost::replace_all_copy(i.second, "\\", "\\\\"), "'", "\\'");
|
|
|
|
if (ret.size() > 1)
|
|
|
|
ret += ", ";
|
|
|
|
ret += "\"" + k + "\":\"" + v + "\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret + "}";
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void EthashCPUMiner::workLoop()
|
|
|
|
{
|
|
|
|
auto tid = std::this_thread::get_id();
|
|
|
|
static std::mt19937_64 s_eng((time(0) + std::hash<decltype(tid)>()(tid)));
|
|
|
|
|
|
|
|
uint64_t tryNonce = (uint64_t)(u64)Nonce::random(s_eng);
|
|
|
|
ethash_return_value ethashReturn;
|
|
|
|
|
|
|
|
WorkPackage w = work();
|
|
|
|
|
|
|
|
EthashAux::FullType dag;
|
|
|
|
while (!shouldStop() && !dag)
|
|
|
|
{
|
|
|
|
while (!shouldStop() && EthashAux::computeFull(w.seedHash, true) != 100)
|
|
|
|
this_thread::sleep_for(chrono::milliseconds(500));
|
|
|
|
dag = EthashAux::full(w.seedHash, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
h256 boundary = w.boundary;
|
|
|
|
unsigned hashCount = 1;
|
|
|
|
for (; !shouldStop(); tryNonce++, hashCount++)
|
|
|
|
{
|
|
|
|
ethashReturn = ethash_full_compute(dag->full, *(ethash_h256_t*)w.headerHash.data(), tryNonce);
|
|
|
|
h256 value = h256((uint8_t*)ðashReturn.result, h256::ConstructFromPointer);
|
|
|
|
if (value <= boundary && submitProof(EthashProofOfWork::Solution{(h64)(u64)tryNonce, h256((uint8_t*)ðashReturn.mix_hash, h256::ConstructFromPointer)}))
|
|
|
|
break;
|
|
|
|
if (!(hashCount % 100))
|
|
|
|
accumulateHashes(100);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string EthashCPUMiner::platformInfo()
|
|
|
|
{
|
|
|
|
string baseline = toString(std::thread::hardware_concurrency()) + "-thread CPU";
|
|
|
|
#if ETH_CPUID || !ETH_TRUE
|
|
|
|
if (!cpuid_present())
|
|
|
|
return baseline;
|
|
|
|
struct cpu_raw_data_t raw;
|
|
|
|
struct cpu_id_t data;
|
|
|
|
if (cpuid_get_raw_data(&raw) < 0)
|
|
|
|
return baseline;
|
|
|
|
if (cpu_identify(&raw, &data) < 0)
|
|
|
|
return baseline;
|
|
|
|
map<string, string> m;
|
|
|
|
m["vendor"] = data.vendor_str;
|
|
|
|
m["codename"] = data.cpu_codename;
|
|
|
|
m["brand"] = data.brand_str;
|
|
|
|
m["L1 cache"] = toString(data.l1_data_cache);
|
|
|
|
m["L2 cache"] = toString(data.l2_cache);
|
|
|
|
m["L3 cache"] = toString(data.l3_cache);
|
|
|
|
m["cores"] = toString(data.num_cores);
|
|
|
|
m["threads"] = toString(data.num_logical_cpus);
|
|
|
|
m["clocknominal"] = toString(cpu_clock_by_os());
|
|
|
|
m["clocktested"] = toString(cpu_clock_measure(200, 0));
|
|
|
|
/*
|
|
|
|
printf(" MMX : %s\n", data.flags[CPU_FEATURE_MMX] ? "present" : "absent");
|
|
|
|
printf(" MMX-extended: %s\n", data.flags[CPU_FEATURE_MMXEXT] ? "present" : "absent");
|
|
|
|
printf(" SSE : %s\n", data.flags[CPU_FEATURE_SSE] ? "present" : "absent");
|
|
|
|
printf(" SSE2 : %s\n", data.flags[CPU_FEATURE_SSE2] ? "present" : "absent");
|
|
|
|
printf(" 3DNow! : %s\n", data.flags[CPU_FEATURE_3DNOW] ? "present" : "absent");
|
|
|
|
*/
|
|
|
|
return jsonEncode(m);
|
|
|
|
#else
|
|
|
|
return baseline;
|
|
|
|
#endif
|
|
|
|
}
|