2014-06-26 08:41:53 -04:00
|
|
|
// Copyright (c) 2009-2010 Satoshi Nakamoto
|
2020-04-16 13:14:08 -04:00
|
|
|
// Copyright (c) 2009-2020 The Bitcoin Core developers
|
2014-12-13 01:09:33 -03:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
2014-06-26 08:41:53 -04:00
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
|
|
|
#ifndef BITCOIN_RANDOM_H
|
|
|
|
#define BITCOIN_RANDOM_H
|
|
|
|
|
2017-11-09 21:57:53 -03:00
|
|
|
#include <crypto/chacha20.h>
|
|
|
|
#include <crypto/common.h>
|
2022-01-31 09:29:33 -03:00
|
|
|
#include <span.h>
|
2017-11-09 21:57:53 -03:00
|
|
|
#include <uint256.h>
|
2014-06-26 08:41:53 -04:00
|
|
|
|
2021-12-14 07:15:10 -03:00
|
|
|
#include <chrono>
|
2019-06-12 15:19:02 -04:00
|
|
|
#include <cstdint>
|
2018-03-20 23:10:39 -03:00
|
|
|
#include <limits>
|
2014-06-26 08:41:53 -04:00
|
|
|
|
2019-01-13 15:51:17 -03:00
|
|
|
/**
|
|
|
|
* Overall design of the RNG and entropy sources.
|
|
|
|
*
|
|
|
|
* We maintain a single global 256-bit RNG state for all high-quality randomness.
|
|
|
|
* The following (classes of) functions interact with that state by mixing in new
|
|
|
|
* entropy, and optionally extracting random output from it:
|
|
|
|
*
|
|
|
|
* - The GetRand*() class of functions, as well as construction of FastRandomContext objects,
|
|
|
|
* perform 'fast' seeding, consisting of mixing in:
|
|
|
|
* - A stack pointer (indirectly committing to calling thread and call stack)
|
|
|
|
* - A high-precision timestamp (rdtsc when available, c++ high_resolution_clock otherwise)
|
2019-01-24 23:40:02 -03:00
|
|
|
* - 64 bits from the hardware RNG (rdrand) when available.
|
2019-01-13 15:51:17 -03:00
|
|
|
* These entropy sources are very fast, and only designed to protect against situations
|
|
|
|
* where a VM state restore/copy results in multiple systems with the same randomness.
|
|
|
|
* FastRandomContext on the other hand does not protect against this once created, but
|
|
|
|
* is even faster (and acceptable to use inside tight loops).
|
|
|
|
*
|
|
|
|
* - The GetStrongRand*() class of function perform 'slow' seeding, including everything
|
|
|
|
* that fast seeding includes, but additionally:
|
|
|
|
* - OS entropy (/dev/urandom, getrandom(), ...). The application will terminate if
|
|
|
|
* this entropy source fails.
|
|
|
|
* - Another high-precision timestamp (indirectly committing to a benchmark of all the
|
|
|
|
* previous sources).
|
|
|
|
* These entropy sources are slower, but designed to make sure the RNG state contains
|
|
|
|
* fresh data that is unpredictable to attackers.
|
|
|
|
*
|
2019-11-18 11:58:10 -03:00
|
|
|
* - RandAddPeriodic() seeds everything that fast seeding includes, but additionally:
|
|
|
|
* - A high-precision timestamp
|
|
|
|
* - Dynamic environment data (performance monitoring, ...)
|
|
|
|
* - Strengthen the entropy for 10 ms using repeated SHA512.
|
|
|
|
* This is run once every minute.
|
2019-01-13 15:51:17 -03:00
|
|
|
*
|
|
|
|
* On first use of the RNG (regardless of what function is called first), all entropy
|
|
|
|
* sources used in the 'slow' seeder are included, but also:
|
2019-01-24 23:40:02 -03:00
|
|
|
* - 256 bits from the hardware RNG (rdseed or rdrand) when available.
|
2019-11-18 11:58:10 -03:00
|
|
|
* - Dynamic environment data (performance monitoring, ...)
|
|
|
|
* - Static environment data
|
2019-01-21 18:25:14 -03:00
|
|
|
* - Strengthen the entropy for 100 ms using repeated SHA512.
|
2019-01-13 15:51:17 -03:00
|
|
|
*
|
|
|
|
* When mixing in new entropy, H = SHA512(entropy || old_rng_state) is computed, and
|
|
|
|
* (up to) the first 32 bytes of H are produced as output, while the last 32 bytes
|
|
|
|
* become the new RNG state.
|
|
|
|
*/
|
|
|
|
|
2014-06-26 08:41:53 -04:00
|
|
|
/**
|
2018-12-13 23:37:29 -03:00
|
|
|
* Generate random data via the internal PRNG.
|
|
|
|
*
|
|
|
|
* These functions are designed to be fast (sub microsecond), but do not necessarily
|
|
|
|
* meaningfully add entropy to the PRNG state.
|
|
|
|
*
|
|
|
|
* Thread-safe.
|
2014-06-26 08:41:53 -04:00
|
|
|
*/
|
2022-01-31 09:29:33 -03:00
|
|
|
void GetRandBytes(Span<unsigned char> bytes) noexcept;
|
2020-04-27 09:25:05 -04:00
|
|
|
/** Generate a uniform random integer in the range [0..range). Precondition: range > 0 */
|
2018-12-17 20:11:33 -03:00
|
|
|
uint64_t GetRand(uint64_t nMax) noexcept;
|
2020-04-27 09:25:05 -04:00
|
|
|
/** Generate a uniform random duration in the range [0..max). Precondition: max.count() > 0 */
|
|
|
|
template <typename D>
|
|
|
|
D GetRandomDuration(typename std::common_type<D>::type max) noexcept
|
|
|
|
// Having the compiler infer the template argument from the function argument
|
|
|
|
// is dangerous, because the desired return value generally has a different
|
|
|
|
// type than the function argument. So std::common_type is used to force the
|
|
|
|
// call site to specify the type of the return value.
|
|
|
|
{
|
|
|
|
assert(max.count() > 0);
|
|
|
|
return D{GetRand(max.count())};
|
|
|
|
};
|
|
|
|
constexpr auto GetRandMicros = GetRandomDuration<std::chrono::microseconds>;
|
|
|
|
constexpr auto GetRandMillis = GetRandomDuration<std::chrono::milliseconds>;
|
2021-12-14 07:15:10 -03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a timestamp in the future sampled from an exponential distribution
|
|
|
|
* (https://en.wikipedia.org/wiki/Exponential_distribution). This distribution
|
|
|
|
* is memoryless and should be used for repeated network events (e.g. sending a
|
|
|
|
* certain type of message) to minimize leaking information to observers.
|
|
|
|
*
|
2022-01-29 19:19:44 -03:00
|
|
|
* The probability of an event occurring before time x is 1 - e^-(x/a) where a
|
2021-12-14 07:15:10 -03:00
|
|
|
* is the average interval between events.
|
|
|
|
* */
|
2020-04-15 19:06:59 -04:00
|
|
|
std::chrono::microseconds GetExponentialRand(std::chrono::microseconds now, std::chrono::seconds average_interval);
|
2021-12-14 07:15:10 -03:00
|
|
|
|
2018-12-17 20:11:33 -03:00
|
|
|
int GetRandInt(int nMax) noexcept;
|
|
|
|
uint256 GetRandHash() noexcept;
|
2014-06-26 08:41:53 -04:00
|
|
|
|
2017-05-22 16:01:52 -04:00
|
|
|
/**
|
2018-12-13 23:37:29 -03:00
|
|
|
* Gather entropy from various sources, feed it into the internal PRNG, and
|
|
|
|
* generate random data using it.
|
|
|
|
*
|
|
|
|
* This function will cause failure whenever the OS RNG fails.
|
|
|
|
*
|
|
|
|
* Thread-safe.
|
2017-05-22 16:01:52 -04:00
|
|
|
*/
|
2022-01-31 09:29:33 -03:00
|
|
|
void GetStrongRandBytes(Span<unsigned char> bytes) noexcept;
|
2017-05-22 16:01:52 -04:00
|
|
|
|
2016-04-16 07:25:12 -03:00
|
|
|
/**
|
2019-10-29 15:55:59 -03:00
|
|
|
* Gather entropy from various expensive sources, and feed them to the PRNG state.
|
2018-12-13 23:37:29 -03:00
|
|
|
*
|
|
|
|
* Thread-safe.
|
2016-04-16 07:25:12 -03:00
|
|
|
*/
|
2019-11-18 12:21:28 -03:00
|
|
|
void RandAddPeriodic() noexcept;
|
2016-04-16 07:25:12 -03:00
|
|
|
|
2019-11-23 13:42:23 -03:00
|
|
|
/**
|
|
|
|
* Gathers entropy from the low bits of the time at which events occur. Should
|
|
|
|
* be called with a uint32_t describing the event at the time an event occurs.
|
|
|
|
*
|
|
|
|
* Thread-safe.
|
|
|
|
*/
|
2019-12-04 20:14:43 -03:00
|
|
|
void RandAddEvent(const uint32_t event_info) noexcept;
|
2019-11-23 13:42:23 -03:00
|
|
|
|
2014-06-26 08:41:53 -04:00
|
|
|
/**
|
2016-10-13 11:19:20 -03:00
|
|
|
* Fast randomness source. This is seeded once with secure random data, but
|
2018-12-13 23:37:29 -03:00
|
|
|
* is completely deterministic and does not gather more entropy after that.
|
|
|
|
*
|
2016-10-13 11:19:20 -03:00
|
|
|
* This class is not thread-safe.
|
2014-06-26 08:41:53 -04:00
|
|
|
*/
|
2020-03-28 05:56:38 -03:00
|
|
|
class FastRandomContext
|
|
|
|
{
|
2017-02-15 22:45:22 -03:00
|
|
|
private:
|
|
|
|
bool requires_seed;
|
|
|
|
ChaCha20 rng;
|
|
|
|
|
|
|
|
unsigned char bytebuf[64];
|
|
|
|
int bytebuf_size;
|
|
|
|
|
|
|
|
uint64_t bitbuf;
|
|
|
|
int bitbuf_size;
|
|
|
|
|
|
|
|
void RandomSeed();
|
|
|
|
|
|
|
|
void FillByteBuffer()
|
|
|
|
{
|
|
|
|
if (requires_seed) {
|
|
|
|
RandomSeed();
|
|
|
|
}
|
2019-03-01 13:03:58 -03:00
|
|
|
rng.Keystream(bytebuf, sizeof(bytebuf));
|
2017-02-15 22:45:22 -03:00
|
|
|
bytebuf_size = sizeof(bytebuf);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FillBitBuffer()
|
|
|
|
{
|
|
|
|
bitbuf = rand64();
|
|
|
|
bitbuf_size = 64;
|
|
|
|
}
|
|
|
|
|
2016-10-13 11:19:20 -03:00
|
|
|
public:
|
2018-12-17 20:11:33 -03:00
|
|
|
explicit FastRandomContext(bool fDeterministic = false) noexcept;
|
2016-10-13 11:19:20 -03:00
|
|
|
|
2017-02-15 22:45:22 -03:00
|
|
|
/** Initialize with explicit seed (only for testing) */
|
2018-12-17 20:11:33 -03:00
|
|
|
explicit FastRandomContext(const uint256& seed) noexcept;
|
2017-02-15 22:45:22 -03:00
|
|
|
|
2018-10-31 19:41:13 -03:00
|
|
|
// Do not permit copying a FastRandomContext (move it, or create a new one to get reseeded).
|
|
|
|
FastRandomContext(const FastRandomContext&) = delete;
|
|
|
|
FastRandomContext(FastRandomContext&&) = delete;
|
|
|
|
FastRandomContext& operator=(const FastRandomContext&) = delete;
|
|
|
|
|
|
|
|
/** Move a FastRandomContext. If the original one is used again, it will be reseeded. */
|
|
|
|
FastRandomContext& operator=(FastRandomContext&& from) noexcept;
|
|
|
|
|
2017-02-15 22:45:22 -03:00
|
|
|
/** Generate a random 64-bit integer. */
|
2018-12-17 20:11:33 -03:00
|
|
|
uint64_t rand64() noexcept
|
2017-02-15 22:45:22 -03:00
|
|
|
{
|
|
|
|
if (bytebuf_size < 8) FillByteBuffer();
|
|
|
|
uint64_t ret = ReadLE64(bytebuf + 64 - bytebuf_size);
|
|
|
|
bytebuf_size -= 8;
|
|
|
|
return ret;
|
2016-10-13 11:19:20 -03:00
|
|
|
}
|
|
|
|
|
2017-02-15 22:45:22 -03:00
|
|
|
/** Generate a random (bits)-bit integer. */
|
2020-03-28 05:56:38 -03:00
|
|
|
uint64_t randbits(int bits) noexcept
|
|
|
|
{
|
2017-02-15 22:45:22 -03:00
|
|
|
if (bits == 0) {
|
|
|
|
return 0;
|
|
|
|
} else if (bits > 32) {
|
|
|
|
return rand64() >> (64 - bits);
|
|
|
|
} else {
|
|
|
|
if (bitbuf_size < bits) FillBitBuffer();
|
|
|
|
uint64_t ret = bitbuf & (~(uint64_t)0 >> (64 - bits));
|
|
|
|
bitbuf >>= bits;
|
|
|
|
bitbuf_size -= bits;
|
|
|
|
return ret;
|
|
|
|
}
|
2017-02-17 21:32:16 -03:00
|
|
|
}
|
|
|
|
|
2020-03-28 05:56:38 -03:00
|
|
|
/** Generate a random integer in the range [0..range).
|
|
|
|
* Precondition: range > 0.
|
|
|
|
*/
|
2018-12-17 20:11:33 -03:00
|
|
|
uint64_t randrange(uint64_t range) noexcept
|
2017-02-25 17:16:58 -03:00
|
|
|
{
|
2019-10-28 20:42:39 -03:00
|
|
|
assert(range);
|
2017-02-25 17:16:58 -03:00
|
|
|
--range;
|
|
|
|
int bits = CountBits(range);
|
|
|
|
while (true) {
|
|
|
|
uint64_t ret = randbits(bits);
|
|
|
|
if (ret <= range) return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-02 15:04:31 -03:00
|
|
|
/** Generate random bytes. */
|
|
|
|
std::vector<unsigned char> randbytes(size_t len);
|
|
|
|
|
2017-02-15 22:45:22 -03:00
|
|
|
/** Generate a random 32-bit integer. */
|
2018-12-17 20:11:33 -03:00
|
|
|
uint32_t rand32() noexcept { return randbits(32); }
|
2017-02-15 22:45:22 -03:00
|
|
|
|
2017-05-02 15:04:31 -03:00
|
|
|
/** generate a random uint256. */
|
2018-12-17 20:11:33 -03:00
|
|
|
uint256 rand256() noexcept;
|
2017-05-02 15:04:31 -03:00
|
|
|
|
2017-02-15 22:45:22 -03:00
|
|
|
/** Generate a random boolean. */
|
2018-12-17 20:11:33 -03:00
|
|
|
bool randbool() noexcept { return randbits(1); }
|
2018-03-20 23:10:39 -03:00
|
|
|
|
|
|
|
// Compatibility with the C++11 UniformRandomBitGenerator concept
|
|
|
|
typedef uint64_t result_type;
|
|
|
|
static constexpr uint64_t min() { return 0; }
|
|
|
|
static constexpr uint64_t max() { return std::numeric_limits<uint64_t>::max(); }
|
2018-12-17 20:11:33 -03:00
|
|
|
inline uint64_t operator()() noexcept { return rand64(); }
|
2016-10-13 11:19:20 -03:00
|
|
|
};
|
2014-06-26 08:41:53 -04:00
|
|
|
|
2018-10-31 19:42:03 -03:00
|
|
|
/** More efficient than using std::shuffle on a FastRandomContext.
|
|
|
|
*
|
|
|
|
* This is more efficient as std::shuffle will consume entropy in groups of
|
|
|
|
* 64 bits at the time and throw away most.
|
|
|
|
*
|
|
|
|
* This also works around a bug in libstdc++ std::shuffle that may cause
|
|
|
|
* type::operator=(type&&) to be invoked on itself, which the library's
|
|
|
|
* debug mode detects and panics on. This is a known issue, see
|
|
|
|
* https://stackoverflow.com/questions/22915325/avoiding-self-assignment-in-stdshuffle
|
|
|
|
*/
|
2020-03-28 05:56:38 -03:00
|
|
|
template <typename I, typename R>
|
2018-10-31 19:42:03 -03:00
|
|
|
void Shuffle(I first, I last, R&& rng)
|
|
|
|
{
|
|
|
|
while (first != last) {
|
|
|
|
size_t j = rng.randrange(last - first);
|
|
|
|
if (j) {
|
|
|
|
using std::swap;
|
|
|
|
swap(*first, *(first + j));
|
|
|
|
}
|
|
|
|
++first;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-22 03:38:42 -03:00
|
|
|
/* Number of random bytes returned by GetOSRand.
|
|
|
|
* When changing this constant make sure to change all call sites, and make
|
|
|
|
* sure that the underlying OS APIs for all platforms support the number.
|
|
|
|
* (many cap out at 256 bytes).
|
|
|
|
*/
|
2017-11-09 17:06:49 -03:00
|
|
|
static const int NUM_OS_RANDOM_BYTES = 32;
|
2017-02-21 13:36:37 -03:00
|
|
|
|
|
|
|
/** Get 32 bytes of system entropy. Do not use this in application code: use
|
|
|
|
* GetStrongRandBytes instead.
|
|
|
|
*/
|
2020-03-28 05:56:38 -03:00
|
|
|
void GetOSRand(unsigned char* ent32);
|
2017-02-21 13:36:37 -03:00
|
|
|
|
2017-02-22 04:02:50 -03:00
|
|
|
/** Check that OS randomness is available and returning the requested number
|
|
|
|
* of bytes.
|
|
|
|
*/
|
|
|
|
bool Random_SanityCheck();
|
|
|
|
|
2018-12-17 21:48:21 -03:00
|
|
|
/**
|
|
|
|
* Initialize global RNG state and log any CPU features that are used.
|
|
|
|
*
|
|
|
|
* Calling this function is optional. RNG state will be initialized when first
|
|
|
|
* needed if it is not called.
|
|
|
|
*/
|
2017-05-09 19:13:00 -03:00
|
|
|
void RandomInit();
|
|
|
|
|
2014-06-26 08:41:53 -04:00
|
|
|
#endif // BITCOIN_RANDOM_H
|