399 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			399 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
//===----------------------------------------------------------------------===//
 | 
						|
//
 | 
						|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
 | 
						|
// See https://llvm.org/LICENSE.txt for license information.
 | 
						|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
#ifndef _LIBCPP___RANDOM_LINEAR_CONGRUENTIAL_ENGINE_H
 | 
						|
#define _LIBCPP___RANDOM_LINEAR_CONGRUENTIAL_ENGINE_H
 | 
						|
 | 
						|
#include <__config>
 | 
						|
#include <__random/is_seed_sequence.h>
 | 
						|
#include <cstdint>
 | 
						|
#include <iosfwd>
 | 
						|
#include <type_traits>
 | 
						|
 | 
						|
#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
 | 
						|
#  pragma GCC system_header
 | 
						|
#endif
 | 
						|
 | 
						|
_LIBCPP_PUSH_MACROS
 | 
						|
#include <__undef_macros>
 | 
						|
 | 
						|
_LIBCPP_BEGIN_NAMESPACE_STD
 | 
						|
 | 
						|
template <unsigned long long __a, unsigned long long __c,
 | 
						|
          unsigned long long __m, unsigned long long _Mp,
 | 
						|
          bool _MightOverflow = (__a != 0 && __m != 0 && __m-1 > (_Mp-__c)/__a),
 | 
						|
          bool _OverflowOK = ((__m | (__m-1)) > __m), // m = 2^n
 | 
						|
          bool _SchrageOK = (__a != 0 && __m != 0 && __m % __a <= __m / __a)> // r <= q
 | 
						|
struct __lce_alg_picker
 | 
						|
{
 | 
						|
    static_assert(__a != 0 || __m != 0 || !_MightOverflow || _OverflowOK || _SchrageOK,
 | 
						|
                  "The current values of a, c, and m cannot generate a number "
 | 
						|
                  "within bounds of linear_congruential_engine.");
 | 
						|
 | 
						|
    static _LIBCPP_CONSTEXPR const bool __use_schrage = _MightOverflow &&
 | 
						|
                                                        !_OverflowOK &&
 | 
						|
                                                        _SchrageOK;
 | 
						|
};
 | 
						|
 | 
						|
template <unsigned long long __a, unsigned long long __c,
 | 
						|
          unsigned long long __m, unsigned long long _Mp,
 | 
						|
          bool _UseSchrage = __lce_alg_picker<__a, __c, __m, _Mp>::__use_schrage>
 | 
						|
struct __lce_ta;
 | 
						|
 | 
						|
// 64
 | 
						|
 | 
						|
template <unsigned long long __a, unsigned long long __c, unsigned long long __m>
 | 
						|
struct __lce_ta<__a, __c, __m, (unsigned long long)(~0), true>
 | 
						|
{
 | 
						|
    typedef unsigned long long result_type;
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    static result_type next(result_type __x)
 | 
						|
    {
 | 
						|
        // Schrage's algorithm
 | 
						|
        const result_type __q = __m / __a;
 | 
						|
        const result_type __r = __m % __a;
 | 
						|
        const result_type __t0 = __a * (__x % __q);
 | 
						|
        const result_type __t1 = __r * (__x / __q);
 | 
						|
        __x = __t0 + (__t0 < __t1) * __m - __t1;
 | 
						|
        __x += __c - (__x >= __m - __c) * __m;
 | 
						|
        return __x;
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
template <unsigned long long __a, unsigned long long __m>
 | 
						|
struct __lce_ta<__a, 0, __m, (unsigned long long)(~0), true>
 | 
						|
{
 | 
						|
    typedef unsigned long long result_type;
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    static result_type next(result_type __x)
 | 
						|
    {
 | 
						|
        // Schrage's algorithm
 | 
						|
        const result_type __q = __m / __a;
 | 
						|
        const result_type __r = __m % __a;
 | 
						|
        const result_type __t0 = __a * (__x % __q);
 | 
						|
        const result_type __t1 = __r * (__x / __q);
 | 
						|
        __x = __t0 + (__t0 < __t1) * __m - __t1;
 | 
						|
        return __x;
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
template <unsigned long long __a, unsigned long long __c, unsigned long long __m>
 | 
						|
struct __lce_ta<__a, __c, __m, (unsigned long long)(~0), false>
 | 
						|
{
 | 
						|
    typedef unsigned long long result_type;
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    static result_type next(result_type __x)
 | 
						|
    {
 | 
						|
        return (__a * __x + __c) % __m;
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
template <unsigned long long __a, unsigned long long __c>
 | 
						|
struct __lce_ta<__a, __c, 0, (unsigned long long)(~0), false>
 | 
						|
{
 | 
						|
    typedef unsigned long long result_type;
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    static result_type next(result_type __x)
 | 
						|
    {
 | 
						|
        return __a * __x + __c;
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
// 32
 | 
						|
 | 
						|
template <unsigned long long _Ap, unsigned long long _Cp, unsigned long long _Mp>
 | 
						|
struct __lce_ta<_Ap, _Cp, _Mp, unsigned(~0), true>
 | 
						|
{
 | 
						|
    typedef unsigned result_type;
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    static result_type next(result_type __x)
 | 
						|
    {
 | 
						|
        const result_type __a = static_cast<result_type>(_Ap);
 | 
						|
        const result_type __c = static_cast<result_type>(_Cp);
 | 
						|
        const result_type __m = static_cast<result_type>(_Mp);
 | 
						|
        // Schrage's algorithm
 | 
						|
        const result_type __q = __m / __a;
 | 
						|
        const result_type __r = __m % __a;
 | 
						|
        const result_type __t0 = __a * (__x % __q);
 | 
						|
        const result_type __t1 = __r * (__x / __q);
 | 
						|
        __x = __t0 + (__t0 < __t1) * __m - __t1;
 | 
						|
        __x += __c - (__x >= __m - __c) * __m;
 | 
						|
        return __x;
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
template <unsigned long long _Ap, unsigned long long _Mp>
 | 
						|
struct __lce_ta<_Ap, 0, _Mp, unsigned(~0), true>
 | 
						|
{
 | 
						|
    typedef unsigned result_type;
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    static result_type next(result_type __x)
 | 
						|
    {
 | 
						|
        const result_type __a = static_cast<result_type>(_Ap);
 | 
						|
        const result_type __m = static_cast<result_type>(_Mp);
 | 
						|
        // Schrage's algorithm
 | 
						|
        const result_type __q = __m / __a;
 | 
						|
        const result_type __r = __m % __a;
 | 
						|
        const result_type __t0 = __a * (__x % __q);
 | 
						|
        const result_type __t1 = __r * (__x / __q);
 | 
						|
        __x = __t0 + (__t0 < __t1) * __m - __t1;
 | 
						|
        return __x;
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
template <unsigned long long _Ap, unsigned long long _Cp, unsigned long long _Mp>
 | 
						|
struct __lce_ta<_Ap, _Cp, _Mp, unsigned(~0), false>
 | 
						|
{
 | 
						|
    typedef unsigned result_type;
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    static result_type next(result_type __x)
 | 
						|
    {
 | 
						|
        const result_type __a = static_cast<result_type>(_Ap);
 | 
						|
        const result_type __c = static_cast<result_type>(_Cp);
 | 
						|
        const result_type __m = static_cast<result_type>(_Mp);
 | 
						|
        return (__a * __x + __c) % __m;
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
template <unsigned long long _Ap, unsigned long long _Cp>
 | 
						|
struct __lce_ta<_Ap, _Cp, 0, unsigned(~0), false>
 | 
						|
{
 | 
						|
    typedef unsigned result_type;
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    static result_type next(result_type __x)
 | 
						|
    {
 | 
						|
        const result_type __a = static_cast<result_type>(_Ap);
 | 
						|
        const result_type __c = static_cast<result_type>(_Cp);
 | 
						|
        return __a * __x + __c;
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
// 16
 | 
						|
 | 
						|
template <unsigned long long __a, unsigned long long __c, unsigned long long __m, bool __b>
 | 
						|
struct __lce_ta<__a, __c, __m, (unsigned short)(~0), __b>
 | 
						|
{
 | 
						|
    typedef unsigned short result_type;
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    static result_type next(result_type __x)
 | 
						|
    {
 | 
						|
        return static_cast<result_type>(__lce_ta<__a, __c, __m, unsigned(~0)>::next(__x));
 | 
						|
    }
 | 
						|
};
 | 
						|
 | 
						|
template <class _UIntType, _UIntType __a, _UIntType __c, _UIntType __m>
 | 
						|
class _LIBCPP_TEMPLATE_VIS linear_congruential_engine;
 | 
						|
 | 
						|
template <class _CharT, class _Traits,
 | 
						|
          class _Up, _Up _Ap, _Up _Cp, _Up _Np>
 | 
						|
_LIBCPP_INLINE_VISIBILITY
 | 
						|
basic_ostream<_CharT, _Traits>&
 | 
						|
operator<<(basic_ostream<_CharT, _Traits>& __os,
 | 
						|
           const linear_congruential_engine<_Up, _Ap, _Cp, _Np>&);
 | 
						|
 | 
						|
template <class _CharT, class _Traits,
 | 
						|
          class _Up, _Up _Ap, _Up _Cp, _Up _Np>
 | 
						|
basic_istream<_CharT, _Traits>&
 | 
						|
operator>>(basic_istream<_CharT, _Traits>& __is,
 | 
						|
           linear_congruential_engine<_Up, _Ap, _Cp, _Np>& __x);
 | 
						|
 | 
						|
template <class _UIntType, _UIntType __a, _UIntType __c, _UIntType __m>
 | 
						|
class _LIBCPP_TEMPLATE_VIS linear_congruential_engine
 | 
						|
{
 | 
						|
public:
 | 
						|
    // types
 | 
						|
    typedef _UIntType result_type;
 | 
						|
 | 
						|
private:
 | 
						|
    result_type __x_;
 | 
						|
 | 
						|
    static _LIBCPP_CONSTEXPR const result_type _Mp = result_type(~0);
 | 
						|
 | 
						|
    static_assert(__m == 0 || __a < __m, "linear_congruential_engine invalid parameters");
 | 
						|
    static_assert(__m == 0 || __c < __m, "linear_congruential_engine invalid parameters");
 | 
						|
    static_assert(is_unsigned<_UIntType>::value, "_UIntType must be unsigned type");
 | 
						|
public:
 | 
						|
    static _LIBCPP_CONSTEXPR const result_type _Min = __c == 0u ? 1u: 0u;
 | 
						|
    static _LIBCPP_CONSTEXPR const result_type _Max = __m - 1u;
 | 
						|
    static_assert(_Min < _Max,           "linear_congruential_engine invalid parameters");
 | 
						|
 | 
						|
    // engine characteristics
 | 
						|
    static _LIBCPP_CONSTEXPR const result_type multiplier = __a;
 | 
						|
    static _LIBCPP_CONSTEXPR const result_type increment = __c;
 | 
						|
    static _LIBCPP_CONSTEXPR const result_type modulus = __m;
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    static _LIBCPP_CONSTEXPR result_type min() {return _Min;}
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    static _LIBCPP_CONSTEXPR result_type max() {return _Max;}
 | 
						|
    static _LIBCPP_CONSTEXPR const result_type default_seed = 1u;
 | 
						|
 | 
						|
    // constructors and seeding functions
 | 
						|
#ifndef _LIBCPP_CXX03_LANG
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    linear_congruential_engine() : linear_congruential_engine(default_seed) {}
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    explicit linear_congruential_engine(result_type __s) { seed(__s); }
 | 
						|
#else
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    explicit linear_congruential_engine(result_type __s = default_seed) {
 | 
						|
      seed(__s);
 | 
						|
    }
 | 
						|
#endif
 | 
						|
    template<class _Sseq>
 | 
						|
        _LIBCPP_INLINE_VISIBILITY
 | 
						|
        explicit linear_congruential_engine(_Sseq& __q,
 | 
						|
        typename enable_if<__is_seed_sequence<_Sseq, linear_congruential_engine>::value>::type* = 0)
 | 
						|
        {seed(__q);}
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    void seed(result_type __s = default_seed)
 | 
						|
        {seed(integral_constant<bool, __m == 0>(),
 | 
						|
              integral_constant<bool, __c == 0>(), __s);}
 | 
						|
    template<class _Sseq>
 | 
						|
        _LIBCPP_INLINE_VISIBILITY
 | 
						|
        typename enable_if
 | 
						|
        <
 | 
						|
            __is_seed_sequence<_Sseq, linear_congruential_engine>::value,
 | 
						|
            void
 | 
						|
        >::type
 | 
						|
        seed(_Sseq& __q)
 | 
						|
            {__seed(__q, integral_constant<unsigned,
 | 
						|
                1 + (__m == 0 ? (sizeof(result_type) * __CHAR_BIT__ - 1)/32
 | 
						|
                             :  (__m > 0x100000000ull))>());}
 | 
						|
 | 
						|
    // generating functions
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    result_type operator()()
 | 
						|
        {return __x_ = static_cast<result_type>(__lce_ta<__a, __c, __m, _Mp>::next(__x_));}
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    void discard(unsigned long long __z) {for (; __z; --__z) operator()();}
 | 
						|
 | 
						|
    friend _LIBCPP_INLINE_VISIBILITY
 | 
						|
    bool operator==(const linear_congruential_engine& __x,
 | 
						|
                    const linear_congruential_engine& __y)
 | 
						|
        {return __x.__x_ == __y.__x_;}
 | 
						|
    friend _LIBCPP_INLINE_VISIBILITY
 | 
						|
    bool operator!=(const linear_congruential_engine& __x,
 | 
						|
                    const linear_congruential_engine& __y)
 | 
						|
        {return !(__x == __y);}
 | 
						|
 | 
						|
private:
 | 
						|
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    void seed(true_type, true_type, result_type __s) {__x_ = __s == 0 ? 1 : __s;}
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    void seed(true_type, false_type, result_type __s) {__x_ = __s;}
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    void seed(false_type, true_type, result_type __s) {__x_ = __s % __m == 0 ?
 | 
						|
                                                                 1 : __s % __m;}
 | 
						|
    _LIBCPP_INLINE_VISIBILITY
 | 
						|
    void seed(false_type, false_type, result_type __s) {__x_ = __s % __m;}
 | 
						|
 | 
						|
    template<class _Sseq>
 | 
						|
        void __seed(_Sseq& __q, integral_constant<unsigned, 1>);
 | 
						|
    template<class _Sseq>
 | 
						|
        void __seed(_Sseq& __q, integral_constant<unsigned, 2>);
 | 
						|
 | 
						|
    template <class _CharT, class _Traits,
 | 
						|
              class _Up, _Up _Ap, _Up _Cp, _Up _Np>
 | 
						|
    friend
 | 
						|
    basic_ostream<_CharT, _Traits>&
 | 
						|
    operator<<(basic_ostream<_CharT, _Traits>& __os,
 | 
						|
               const linear_congruential_engine<_Up, _Ap, _Cp, _Np>&);
 | 
						|
 | 
						|
    template <class _CharT, class _Traits,
 | 
						|
              class _Up, _Up _Ap, _Up _Cp, _Up _Np>
 | 
						|
    friend
 | 
						|
    basic_istream<_CharT, _Traits>&
 | 
						|
    operator>>(basic_istream<_CharT, _Traits>& __is,
 | 
						|
               linear_congruential_engine<_Up, _Ap, _Cp, _Np>& __x);
 | 
						|
};
 | 
						|
 | 
						|
template <class _UIntType, _UIntType __a, _UIntType __c, _UIntType __m>
 | 
						|
    _LIBCPP_CONSTEXPR const typename linear_congruential_engine<_UIntType, __a, __c, __m>::result_type
 | 
						|
    linear_congruential_engine<_UIntType, __a, __c, __m>::multiplier;
 | 
						|
 | 
						|
template <class _UIntType, _UIntType __a, _UIntType __c, _UIntType __m>
 | 
						|
    _LIBCPP_CONSTEXPR const typename linear_congruential_engine<_UIntType, __a, __c, __m>::result_type
 | 
						|
    linear_congruential_engine<_UIntType, __a, __c, __m>::increment;
 | 
						|
 | 
						|
template <class _UIntType, _UIntType __a, _UIntType __c, _UIntType __m>
 | 
						|
    _LIBCPP_CONSTEXPR const typename linear_congruential_engine<_UIntType, __a, __c, __m>::result_type
 | 
						|
    linear_congruential_engine<_UIntType, __a, __c, __m>::modulus;
 | 
						|
 | 
						|
template <class _UIntType, _UIntType __a, _UIntType __c, _UIntType __m>
 | 
						|
    _LIBCPP_CONSTEXPR const typename linear_congruential_engine<_UIntType, __a, __c, __m>::result_type
 | 
						|
    linear_congruential_engine<_UIntType, __a, __c, __m>::default_seed;
 | 
						|
 | 
						|
template <class _UIntType, _UIntType __a, _UIntType __c, _UIntType __m>
 | 
						|
template<class _Sseq>
 | 
						|
void
 | 
						|
linear_congruential_engine<_UIntType, __a, __c, __m>::__seed(_Sseq& __q,
 | 
						|
                                                 integral_constant<unsigned, 1>)
 | 
						|
{
 | 
						|
    const unsigned __k = 1;
 | 
						|
    uint32_t __ar[__k+3];
 | 
						|
    __q.generate(__ar, __ar + __k + 3);
 | 
						|
    result_type __s = static_cast<result_type>(__ar[3] % __m);
 | 
						|
    __x_ = __c == 0 && __s == 0 ? result_type(1) : __s;
 | 
						|
}
 | 
						|
 | 
						|
template <class _UIntType, _UIntType __a, _UIntType __c, _UIntType __m>
 | 
						|
template<class _Sseq>
 | 
						|
void
 | 
						|
linear_congruential_engine<_UIntType, __a, __c, __m>::__seed(_Sseq& __q,
 | 
						|
                                                 integral_constant<unsigned, 2>)
 | 
						|
{
 | 
						|
    const unsigned __k = 2;
 | 
						|
    uint32_t __ar[__k+3];
 | 
						|
    __q.generate(__ar, __ar + __k + 3);
 | 
						|
    result_type __s = static_cast<result_type>((__ar[3] +
 | 
						|
                                              ((uint64_t)__ar[4] << 32)) % __m);
 | 
						|
    __x_ = __c == 0 && __s == 0 ? result_type(1) : __s;
 | 
						|
}
 | 
						|
 | 
						|
template <class _CharT, class _Traits,
 | 
						|
          class _UIntType, _UIntType __a, _UIntType __c, _UIntType __m>
 | 
						|
inline _LIBCPP_INLINE_VISIBILITY
 | 
						|
basic_ostream<_CharT, _Traits>&
 | 
						|
operator<<(basic_ostream<_CharT, _Traits>& __os,
 | 
						|
           const linear_congruential_engine<_UIntType, __a, __c, __m>& __x)
 | 
						|
{
 | 
						|
    __save_flags<_CharT, _Traits> __lx(__os);
 | 
						|
    typedef basic_ostream<_CharT, _Traits> _Ostream;
 | 
						|
    __os.flags(_Ostream::dec | _Ostream::left);
 | 
						|
    __os.fill(__os.widen(' '));
 | 
						|
    return __os << __x.__x_;
 | 
						|
}
 | 
						|
 | 
						|
template <class _CharT, class _Traits,
 | 
						|
          class _UIntType, _UIntType __a, _UIntType __c, _UIntType __m>
 | 
						|
basic_istream<_CharT, _Traits>&
 | 
						|
operator>>(basic_istream<_CharT, _Traits>& __is,
 | 
						|
           linear_congruential_engine<_UIntType, __a, __c, __m>& __x)
 | 
						|
{
 | 
						|
    __save_flags<_CharT, _Traits> __lx(__is);
 | 
						|
    typedef basic_istream<_CharT, _Traits> _Istream;
 | 
						|
    __is.flags(_Istream::dec | _Istream::skipws);
 | 
						|
    _UIntType __t;
 | 
						|
    __is >> __t;
 | 
						|
    if (!__is.fail())
 | 
						|
        __x.__x_ = __t;
 | 
						|
    return __is;
 | 
						|
}
 | 
						|
 | 
						|
typedef linear_congruential_engine<uint_fast32_t, 16807, 0, 2147483647>
 | 
						|
                                                                   minstd_rand0;
 | 
						|
typedef linear_congruential_engine<uint_fast32_t, 48271, 0, 2147483647>
 | 
						|
                                                                    minstd_rand;
 | 
						|
 | 
						|
_LIBCPP_END_NAMESPACE_STD
 | 
						|
 | 
						|
_LIBCPP_POP_MACROS
 | 
						|
 | 
						|
#endif // _LIBCPP___RANDOM_LINEAR_CONGRUENTIAL_ENGINE_H
 |