// Copyright (c) 1997-2001 // Utrecht University (The Netherlands), // ETH Zurich (Switzerland), // INRIA Sophia-Antipolis (France), // Max-Planck-Institute Saarbruecken (Germany), // and Tel-Aviv University (Israel). All rights reserved. // // This file is part of CGAL (www.cgal.org) // // $URL: https://github.com/CGAL/cgal/blob/v5.1/Random_numbers/include/CGAL/Random.h $ // $Id: Random.h 0779373 2020-03-26T13:31:46+01:00 Sébastien Loriot // SPDX-License-Identifier: LGPL-3.0-or-later OR LicenseRef-Commercial // // // Author(s) : Sven Schoenherr , Sylvain Pion, Andreas Fabri #ifndef CGAL_RANDOM_H #define CGAL_RANDOM_H #include #include #include #include #if defined(BOOST_MSVC) # pragma warning(push) # pragma warning(disable:4244) #endif #include #if defined(BOOST_MSVC) # pragma warning(pop) #endif #include #include #include #include #include namespace CGAL { namespace internal { struct Random_print_seed{}; } class Random { public: // types struct State { std::string rng; unsigned int random_value, val, seed; State() {} State(std::string rng, unsigned int random_value, unsigned int val, unsigned int seed) : rng(rng), random_value(random_value), val(val), seed(seed) {} }; // creation CGAL_EXPORT Random( ); CGAL_EXPORT Random( internal::Random_print_seed ); CGAL_EXPORT Random( unsigned int seed ); // seed CGAL_EXPORT unsigned int get_seed ( ) const; // operations bool get_bool( ) { return( static_cast< bool>( rng() & 1)); } template IntType uniform_smallint(IntType lower, IntType upper) { // uniform_smallint has a closed interval, CGAL a halfopen typedef boost::rand48::result_type result_type; boost::uniform_smallint dist(static_cast(lower), static_cast(upper-1)); boost::variate_generator > generator(rng,dist); return static_cast(generator()); } template IntType uniform_smallint(IntType lower) { return uniform_smallint(lower,9); } template IntType uniform_smallint() { return uniform_smallint(0,9); } template IntType uniform_int(IntType lower, IntType upper) { // uniform_int has a closed interval, CGAL a halfopen boost::uniform_int dist(lower,upper); boost::variate_generator > generator(rng,dist); return generator(); } template IntType uniform_int(IntType lower) { return uniform_int(lower,9); } template IntType uniform_int() { return uniform_int(0,9); } template IntType operator () (IntType upper) { return uniform_int(0, upper-1); } int get_int(int lower, int upper) { return uniform_int(lower,upper-1); } template RealType uniform_real( RealType lower, RealType upper) { // uniform_real as well as CGAL have a halfopen interval boost::uniform_real dist(lower,upper); boost::variate_generator > generator(rng,dist); return generator(); } template RealType uniform_real( RealType lower) { return uniform_real(lower, 1.0); } template RealType uniform_real() { return uniform_real(0.0, 1.0); } template RealType uniform_01() { // uniform_01 as well as CGAL have a halfopen interval boost::uniform_01 dist; boost::variate_generator > generator(rng,dist); return generator(); } double get_double( double lower = 0.0, double upper = 1.0) { return uniform_real(lower, upper); } // state CGAL_EXPORT void save_state( State& state) const; CGAL_EXPORT void restore_state( const State& state); // Computes a random int value smaller than 2^b. // It's supposed to be fast, useful for randomized algorithms. // The distribution is not perfectly flat, but this is a sacrifice against // efficiency. template int get_bits() { CGAL_assertion(0>= 1; // Shifting by b would be slightly better, but is slower. return ret; } bool operator==(Random rd) const { return (rng == rd.rng) && (random_value == rd.random_value) && (val == rd.val) && (seed == rd.seed); } private: // data members unsigned int random_value; // Current 15 bits random value. unsigned int val; // random_value shifted by used bits. unsigned int seed; boost::rand48 rng; }; inline Random& get_default_random() { #if (defined( CGAL_TEST_SUITE ) || defined( CGAL_PRINT_SEED )) && !defined(CGAL_HEADER_ONLY) internal::Random_print_seed rps; CGAL_STATIC_THREAD_LOCAL_VARIABLE(Random, default_random, rps); #else CGAL_STATIC_THREAD_LOCAL_VARIABLE_0(Random, default_random); #endif return default_random; } #ifndef CGAL_NO_DEPRECATED_CODE namespace { CGAL_DEPRECATED_UNUSED CGAL::Random& default_random = get_default_random(); } #endif // CGAL_NO_DEPRECATED_CODE } //namespace CGAL #ifdef CGAL_HEADER_ONLY #include #endif // CGAL_HEADER_ONLY #endif // CGAL_RANDOM_H