|
@ -254,42 +254,71 @@ h256 crypto::kdf(Secret const& _priv, h256 const& _hash) |
|
|
return s; |
|
|
return s; |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
h256 Nonce::get(bool _commit) |
|
|
mutex Nonce::s_x; |
|
|
|
|
|
|
|
|
|
|
|
h256 Nonce::get() |
|
|
{ |
|
|
{ |
|
|
// todo: atomic efface bit, periodic save, kdf, rr, rng
|
|
|
// todo: atomic efface bit, periodic save, kdf, rr, rng
|
|
|
// todo: encrypt
|
|
|
// todo: encrypt
|
|
|
static h256 s_seed; |
|
|
Guard l(Nonce::s_x); |
|
|
static string s_seedFile(getDataDir() + "/seed"); |
|
|
return Nonce::singleton().next(); |
|
|
static mutex s_x; |
|
|
} |
|
|
Guard l(s_x); |
|
|
|
|
|
if (!s_seed) |
|
|
// get: Called for the first time: read seed hash from file (or generate)
|
|
|
{ |
|
|
// before destruction: increment current value and store
|
|
|
static Nonce s_nonce; |
|
|
|
|
|
bytes b = contents(s_seedFile); |
|
|
Nonce::~Nonce() |
|
|
|
|
|
{ |
|
|
|
|
|
Guard l(Nonce::s_x); |
|
|
|
|
|
// These might throw.
|
|
|
|
|
|
next(); |
|
|
|
|
|
commit(); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
Nonce& Nonce::singleton() |
|
|
|
|
|
{ |
|
|
|
|
|
static Nonce s; |
|
|
|
|
|
return s; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
void Nonce::initialiseIfNeeded() |
|
|
|
|
|
{ |
|
|
|
|
|
if (m_value) |
|
|
|
|
|
return; |
|
|
|
|
|
|
|
|
|
|
|
bytes b = contents(seedFile()); |
|
|
if (b.size() == 32) |
|
|
if (b.size() == 32) |
|
|
memcpy(s_seed.data(), b.data(), 32); |
|
|
memcpy(m_value.data(), b.data(), 32); |
|
|
else |
|
|
else |
|
|
{ |
|
|
{ |
|
|
// todo: replace w/entropy from user and system
|
|
|
// todo: replace w/entropy from user and system
|
|
|
std::mt19937_64 s_eng(time(0) + chrono::high_resolution_clock::now().time_since_epoch().count()); |
|
|
std::mt19937_64 s_eng(time(0) + chrono::high_resolution_clock::now().time_since_epoch().count()); |
|
|
std::uniform_int_distribution<uint16_t> d(0, 255); |
|
|
std::uniform_int_distribution<uint16_t> d(0, 255); |
|
|
for (unsigned i = 0; i < 32; ++i) |
|
|
for (unsigned i = 0; i < 32; ++i) |
|
|
s_seed[i] = (byte)d(s_eng); |
|
|
m_value[i] = byte(d(s_eng)); |
|
|
} |
|
|
} |
|
|
if (!s_seed) |
|
|
if (!m_value) |
|
|
BOOST_THROW_EXCEPTION(InvalidState()); |
|
|
BOOST_THROW_EXCEPTION(InvalidState()); |
|
|
|
|
|
|
|
|
// prevent seed reuse if process terminates abnormally
|
|
|
// prevent seed reuse if process terminates abnormally
|
|
|
try { writeFile(s_seedFile, bytes()); } catch (FileError const&) {} |
|
|
// this might throw
|
|
|
|
|
|
writeFile(seedFile(), bytes()); |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
h256 Nonce::next() |
|
|
|
|
|
{ |
|
|
|
|
|
initialiseIfNeeded(); |
|
|
|
|
|
m_value = sha3(m_value); |
|
|
|
|
|
return m_value; |
|
|
} |
|
|
} |
|
|
h256 prev(s_seed); |
|
|
|
|
|
sha3(prev.ref(), s_seed.ref()); |
|
|
void Nonce::commit() |
|
|
if (_commit) |
|
|
{ |
|
|
try { writeFile(s_seedFile, s_seed.asBytes()); } catch (FileError const&) {} |
|
|
// this might throw
|
|
|
return std::move(s_seed); |
|
|
writeFile(seedFile(), m_value.asBytes()); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
Nonce::~Nonce() |
|
|
string Nonce::seedFile() |
|
|
{ |
|
|
{ |
|
|
Nonce::get(true); |
|
|
return getDataDir() + "/seed"; |
|
|
} |
|
|
} |
|
|