osb/source/core/StarRandom.hpp

219 lines
6.7 KiB
C++
Raw Normal View History

2023-06-20 14:33:09 +10:00
#ifndef STAR_RANDOM_HPP
#define STAR_RANDOM_HPP
#include "StarStaticRandom.hpp"
#include "StarByteArray.hpp"
namespace Star {
STAR_EXCEPTION(RandomException, StarException);
// Deterministic random number source. Uses multiply-with-carry algorithm.
// Much higher quality than the predictable random number generators. Not
// thread safe (won't crash or anything, but might return less than optimal
// values).
class RandomSource {
public:
// Generates a RandomSource with a seed from Random::randu64()
RandomSource();
RandomSource(uint64_t seed);
// Re-initializes the random number generator using the given seed. It is
// exactly equivalent to constructing a new RandomSource, just using the same
// buffer.
void init();
void init(uint64_t seed);
void addEntropy();
void addEntropy(uint64_t seed);
uint32_t randu32();
uint64_t randu64();
int32_t randi32();
int64_t randi64();
// Generates values in the range [0.0, 1.0]
float randf();
// Generates values in the range [0.0, 1.0]
double randd();
// Random integer from [0, max], max must be >= 0
int64_t randInt(int64_t max);
uint64_t randUInt(uint64_t max);
// Random integer from [min, max]
int64_t randInt(int64_t min, int64_t max);
uint64_t randUInt(uint64_t min, uint64_t max);
float randf(float min, float max);
double randd(double min, double max);
bool randb();
// Generates values via normal distribution with box-muller algorithm
float nrandf(float stddev = 1.0f, float mean = 0.0f);
double nrandd(double stddev = 1.0, double mean = 0.0);
// Round a fractional value statistically towards the floor or ceiling. For
// example, if a value is 5.2, 80% of the time it will round to 5, but 20% of
// the time it will round to 6.
int64_t stochasticRound(double val);
void randBytes(char* buf, size_t len);
ByteArray randBytes(size_t len);
// Pick a random value out of a container
template <typename Container>
typename Container::value_type const& randFrom(Container const& container);
template <typename Container>
typename Container::value_type& randFrom(Container& container);
template <typename Container>
typename Container::value_type randValueFrom(Container const& container);
template <typename Container>
typename Container::value_type randValueFrom(Container const& container, typename Container::value_type const& defaultVal);
template <typename Container>
void shuffle(Container& container);
private:
uint32_t gen32();
uint32_t m_data[256];
uint32_t m_carry;
uint8_t m_index;
};
// Global static versions of the methods in RandomSource. It is not necessary
// to initialize the global RandomSource manually, it will be automatically
// initialized with a random seed on first use if it is not already initialized.
namespace Random {
void init();
void init(uint64_t seed);
void addEntropy();
void addEntropy(uint64_t seed);
uint32_t randu32();
uint64_t randu64();
int32_t randi32();
int64_t randi64();
float randf();
double randd();
long long randInt(long long max);
unsigned long long randUInt(unsigned long long max);
long long randInt(long long min, long long max);
unsigned long long randUInt(unsigned long long min, unsigned long long max);
float randf(float min, float max);
double randd(double min, double max);
bool randb();
float nrandf(float stddev = 1.0f, float mean = 0.0f);
double nrandd(double stddev = 1.0, double mean = 0.0);
int64_t stochasticRound(double val);
void randBytes(char* buf, size_t len);
ByteArray randBytes(size_t len);
template <typename Container>
typename Container::value_type const& randFrom(Container const& container);
template <typename Container>
typename Container::value_type& randFrom(Container& container);
template <typename Container>
typename Container::value_type randValueFrom(Container const& container);
template <typename Container>
typename Container::value_type randValueFrom(Container const& container, typename Container::value_type const& defaultVal);
template <typename Container>
void shuffle(Container& container);
}
template <typename Container>
typename Container::value_type const& RandomSource::randFrom(Container const& container) {
if (container.empty())
throw RandomException("Empty container in randFrom");
auto i = container.begin();
std::advance(i, randUInt(container.size() - 1));
return *i;
}
template <typename Container>
typename Container::value_type& RandomSource::randFrom(Container& container) {
if (container.empty())
throw RandomException("Empty container in randFrom");
auto i = container.begin();
std::advance(i, randUInt(container.size() - 1));
return *i;
}
template <typename Container>
typename Container::value_type const& Random::randFrom(Container const& container) {
if (container.empty())
throw RandomException("Empty container in randFrom");
auto i = container.begin();
std::advance(i, Random::randUInt(container.size() - 1));
return *i;
}
template <typename Container>
typename Container::value_type& Random::randFrom(Container& container) {
if (container.empty())
throw RandomException("Empty container in randFrom");
auto i = container.begin();
std::advance(i, Random::randUInt(container.size() - 1));
return *i;
}
template <typename Container>
typename Container::value_type RandomSource::randValueFrom(Container const& container) {
return randValueFrom(container, typename Container::value_type());
}
template <typename Container>
typename Container::value_type RandomSource::randValueFrom(
Container const& container, typename Container::value_type const& defaultVal) {
if (container.empty())
return defaultVal;
auto i = container.begin();
std::advance(i, randInt(container.size() - 1));
return *i;
}
template <typename Container>
void RandomSource::shuffle(Container& container) {
2023-06-25 20:01:32 +10:00
size_t max = container.size();
std::shuffle(container.begin(), container.end(), URBG<size_t>([this, max]() { return randUInt(max - 1); }));
2023-06-20 14:33:09 +10:00
}
template <typename Container>
typename Container::value_type Random::randValueFrom(Container const& container) {
return randValueFrom(container, typename Container::value_type());
}
template <typename Container>
typename Container::value_type Random::randValueFrom(
Container const& container, typename Container::value_type const& defaultVal) {
if (container.empty())
return defaultVal;
auto i = container.begin();
std::advance(i, Random::randInt(container.size() - 1));
return *i;
}
template <typename Container>
void Random::shuffle(Container& container) {
2023-06-25 20:01:32 +10:00
size_t max = container.size();
std::shuffle(container.begin(), container.end(), URBG<size_t>([max]() { return Random::randUInt(max - 1); }));
2023-06-20 14:33:09 +10:00
}
}
#endif