X-Git-Url: http://bilbo.iut-bm.univ-fcomte.fr/pub/gitweb/simgrid.git/blobdiff_plain/a80d47a1e761b8b2286a25cfb835c5f7d8a9496f..ab5814bd4ea6117ecf9ac2df328333e89eb45f5f:/src/xbt/random.cpp diff --git a/src/xbt/random.cpp b/src/xbt/random.cpp index 539340bb67..f63dd2f2d1 100644 --- a/src/xbt/random.cpp +++ b/src/xbt/random.cpp @@ -1,4 +1,4 @@ -/* Copyright (c) 2019-2020. The SimGrid Team. All rights reserved. */ +/* Copyright (c) 2019-2023. The SimGrid Team. All rights reserved. */ /* This program is free software; you can redistribute it and/or modify it * under the terms of the license (GNU LGPL) which comes with this package. */ @@ -15,69 +15,71 @@ XBT_LOG_EXTERNAL_CATEGORY(xbt); XBT_LOG_NEW_DEFAULT_SUBCATEGORY(xbt_random, xbt, "Random"); -namespace simgrid { -namespace xbt { -namespace random { +namespace simgrid::xbt::random { -bool Random::read_state(std::string filename) +bool Random::read_state(const std::string& filename) { std::ifstream file(filename); - if (file) { - file >> mt19937_gen; - return true; - } else { - XBT_WARN("Could not open %s and thus not save the RNG state.", filename.c_str()); - return false; - } + file >> mt19937_gen; + file.close(); + if (file.fail()) + XBT_WARN("Could not save the RNG state to file %s.", filename.c_str()); + return not file.fail(); } -bool Random::write_state(std::string filename) +bool Random::write_state(const std::string& filename) const { std::ofstream file(filename); - if (file) { - file << mt19937_gen; - return true; - } else { - XBT_WARN("Could not open %s and thus not read the RNG state.", filename.c_str()); - return false; - } + file << mt19937_gen; + file.close(); + if (file.fail()) + XBT_WARN("Could not read the RNG state from file %s.", filename.c_str()); + return not file.fail(); } int StdRandom::uniform_int(int min, int max) { - std::uniform_int_distribution<> dist(min, max); + std::uniform_int_distribution dist(min, max); return dist(mt19937_gen); } double StdRandom::uniform_real(double min, double max) { - std::uniform_real_distribution<> dist(min, max); + std::uniform_real_distribution dist(min, max); return dist(mt19937_gen); } double StdRandom::exponential(double lambda) { - std::exponential_distribution<> dist(lambda); + std::exponential_distribution dist(lambda); return dist(mt19937_gen); } double StdRandom::normal(double mean, double sd) { - std::normal_distribution<> dist(mean, sd); + std::normal_distribution dist(mean, sd); return dist(mt19937_gen); } int XbtRandom::uniform_int(int min, int max) { - unsigned long range = max - min + 1; + // The casts to unsigned are here to ensure that the value of range is correctly calculated, even when greater than + // INT_MAX. See the corresponding unit tests for examples. + unsigned long range = static_cast(max) - static_cast(min); xbt_assert(min <= max, "The minimum value for the uniform integer distribution must not be greater than the maximum value"); - xbt_assert(range > 0, "Overflow in the uniform integer distribution, please use a smaller range."); + xbt_assert(range <= decltype(mt19937_gen)::max(), + "Overflow in the uniform integer distribution, please use a smaller range."); + if (range == decltype(mt19937_gen)::max()) + return static_cast(mt19937_gen() + min); + + ++range; + unsigned long limit = decltype(mt19937_gen)::max() - decltype(mt19937_gen)::max() % range; unsigned long value; do { value = mt19937_gen(); - } while (value >= decltype(mt19937_gen)::max() - decltype(mt19937_gen)::max() % range); - return value % range + min; + } while (value >= limit); + return static_cast(value % range + min); } double XbtRandom::uniform_real(double min, double max) @@ -88,7 +90,7 @@ double XbtRandom::uniform_real(double min, double max) do { numerator = mt19937_gen() - decltype(mt19937_gen)::min(); } while (numerator == divisor); - return min + (max - min) * numerator / divisor; + return min + (max - min) * static_cast(numerator) / divisor; } double XbtRandom::exponential(double lambda) @@ -107,15 +109,15 @@ double XbtRandom::normal(double mean, double sd) return z0 * sd + mean; } -static std::unique_ptr default_random(new XbtRandom); +static std::unique_ptr default_random = std::make_unique(); void set_implem_xbt() { - default_random.reset(new XbtRandom); + default_random = std::make_unique(); } void set_implem_std() { - default_random.reset(new StdRandom); + default_random = std::make_unique(); } void set_mersenne_seed(int seed) @@ -123,12 +125,12 @@ void set_mersenne_seed(int seed) default_random->set_seed(seed); } -bool read_mersenne_state(std::string filename) +bool read_mersenne_state(const std::string& filename) { return default_random->read_state(filename); } -bool write_mersenne_state(std::string filename) +bool write_mersenne_state(const std::string& filename) { return default_random->write_state(filename); } @@ -153,6 +155,4 @@ double normal(double mean, double sd) return default_random->normal(mean, sd); } -} // namespace random -} // namespace xbt -} // namespace simgrid +} // namespace simgrid::xbt::random