Gav Wood
11 years ago
3 changed files with 122 additions and 0 deletions
@ -0,0 +1,86 @@ |
|||||
|
#include <boost/detail/endian.hpp> |
||||
|
#include <chrono> |
||||
|
#include <array> |
||||
|
#include <sha3.h> |
||||
|
#include <random> |
||||
|
#include "Common.h" |
||||
|
#include "Dagger.h" |
||||
|
using namespace std; |
||||
|
using namespace std::chrono; |
||||
|
using namespace eth; |
||||
|
|
||||
|
Dagger::Dagger(u256 _hash): m_hash(_hash) |
||||
|
{ |
||||
|
} |
||||
|
|
||||
|
Dagger::~Dagger() |
||||
|
{ |
||||
|
} |
||||
|
|
||||
|
u256 Dagger::search(uint _msTimeout, u256 _diff) |
||||
|
{ |
||||
|
static mt19937_64 s_engine((std::random_device())()); |
||||
|
u256 bound = (u256)((bigint(1) << 256) / _diff); |
||||
|
|
||||
|
auto start = steady_clock::now(); |
||||
|
|
||||
|
while (steady_clock::now() - start < milliseconds(_msTimeout)) |
||||
|
for (uint sp = std::uniform_int_distribution<uint>()(s_engine), j = 0; j < 1000; ++j, ++sp) |
||||
|
if (eval(sp) < bound) |
||||
|
return sp; |
||||
|
return 0; |
||||
|
} |
||||
|
|
||||
|
template <class _T> |
||||
|
inline void update(_T& _sha, u256 const& _value) |
||||
|
{ |
||||
|
std::array<byte, 32> buf; |
||||
|
toBigEndian(_value, buf); |
||||
|
_sha.Update(buf.data(), 32); |
||||
|
} |
||||
|
|
||||
|
template <class _T> |
||||
|
inline u256 get(_T& _sha) |
||||
|
{ |
||||
|
byte buf[32]; |
||||
|
_sha.TruncatedFinal(buf, 32); |
||||
|
return fromBigEndian<u256>(bytesConstRef(buf, 32)); |
||||
|
} |
||||
|
|
||||
|
u256 Dagger::node(uint_fast32_t _L, uint_fast32_t _i) const |
||||
|
{ |
||||
|
if (!_L && !_i) |
||||
|
return m_hash; |
||||
|
u256 m = (_L == 9) ? 16 : 3; |
||||
|
CryptoPP::SHA3_256 bsha; |
||||
|
for (uint_fast32_t k = 0; k < m; ++k) |
||||
|
{ |
||||
|
CryptoPP::SHA3_256 sha; |
||||
|
update(sha, m_hash); |
||||
|
update(sha, m_xn); |
||||
|
update(sha, (u256)_L); |
||||
|
update(sha, (u256)_i); |
||||
|
update(sha, (u256)k); |
||||
|
uint_fast32_t pk = (uint_fast32_t)get(sha) & ((1 << ((_L - 1) * 3)) - 1); |
||||
|
update(bsha, node(_L - 1, pk)); |
||||
|
} |
||||
|
return get(bsha); |
||||
|
} |
||||
|
|
||||
|
u256 Dagger::eval(u256 _N) |
||||
|
{ |
||||
|
m_xn = _N >> 26; // with xn = floor(n / 2^26) -> assuming this is with xn = floor(N / 2^26)
|
||||
|
CryptoPP::SHA3_256 bsha; |
||||
|
for (uint_fast32_t k = 0; k < 4; ++k) |
||||
|
{ |
||||
|
//sha256(D || xn || i || k) -> sha256(D || xn || k) - there's no 'i' here!
|
||||
|
CryptoPP::SHA3_256 sha; |
||||
|
update(sha, m_hash); |
||||
|
update(sha, m_xn); |
||||
|
update(sha, _N); |
||||
|
update(sha, (u256)k); |
||||
|
uint_fast32_t pk = (uint_fast32_t)get(sha) & 0x1ffffff; // mod 8^8 * 2 [ == mod 2^25 ?! ] [ == & ((1 << 25) - 1) ] [ == & 0x1ffffff ]
|
||||
|
update(bsha, node(9, pk)); |
||||
|
} |
||||
|
return get(bsha); |
||||
|
} |
@ -0,0 +1,26 @@ |
|||||
|
#pragma once |
||||
|
|
||||
|
#include "Common.h" |
||||
|
|
||||
|
namespace eth |
||||
|
{ |
||||
|
|
||||
|
/// Functions are not re-entrant. If you want to multi-thread, then use different classes for each thread.
|
||||
|
class Dagger |
||||
|
{ |
||||
|
public: |
||||
|
Dagger(u256 _hash); |
||||
|
~Dagger(); |
||||
|
|
||||
|
u256 node(uint_fast32_t _L, uint_fast32_t _i) const; |
||||
|
u256 eval(u256 _N); |
||||
|
u256 search(uint _msTimeout, u256 _diff); |
||||
|
|
||||
|
private: |
||||
|
u256 m_hash; |
||||
|
u256 m_xn; |
||||
|
}; |
||||
|
|
||||
|
} |
||||
|
|
||||
|
|
Loading…
Reference in new issue