119 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			119 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
|   | /* boost random/seed_seq.hpp header file | ||
|  |  * | ||
|  |  * Copyright Steven Watanabe 2010 | ||
|  |  * Distributed under the Boost Software License, Version 1.0. (See | ||
|  |  * accompanying file LICENSE_1_0.txt or copy at | ||
|  |  * http://www.boost.org/LICENSE_1_0.txt) | ||
|  |  * | ||
|  |  * See http://www.boost.org for most recent version including documentation. | ||
|  |  * | ||
|  |  * $Id$ | ||
|  |  * | ||
|  |  */ | ||
|  | 
 | ||
|  | #ifndef BOOST_RANDOM_SEED_SEQ_HPP | ||
|  | #define BOOST_RANDOM_SEED_SEQ_HPP | ||
|  | 
 | ||
|  | #include <boost/config.hpp> | ||
|  | #include <boost/cstdint.hpp> | ||
|  | #include <boost/range/begin.hpp> | ||
|  | #include <boost/range/end.hpp> | ||
|  | #include <cstddef> | ||
|  | #include <vector> | ||
|  | #include <algorithm> | ||
|  | #include <iterator> | ||
|  | 
 | ||
|  | #ifndef BOOST_NO_CXX11_HDR_INITIALIZER_LIST | ||
|  | #include <initializer_list> | ||
|  | #endif | ||
|  | 
 | ||
|  | namespace boost { | ||
|  | namespace random { | ||
|  | 
 | ||
|  | /** | ||
|  |  * The class @c seed_seq stores a sequence of 32-bit words | ||
|  |  * for seeding a \pseudo_random_number_generator.  These | ||
|  |  * words will be combined to fill the entire state of the | ||
|  |  * generator. | ||
|  |  */ | ||
|  | class seed_seq { | ||
|  | public: | ||
|  |     typedef boost::uint_least32_t result_type; | ||
|  | 
 | ||
|  |     /** Initializes a seed_seq to hold an empty sequence. */ | ||
|  |     seed_seq() {} | ||
|  | #ifndef BOOST_NO_CXX11_HDR_INITIALIZER_LIST | ||
|  |     /** Initializes the sequence from an initializer_list. */ | ||
|  |     template<class T> | ||
|  |     seed_seq(const std::initializer_list<T>& il) : v(il.begin(), il.end()) {} | ||
|  | #endif | ||
|  |     /** Initializes the sequence from an iterator range. */ | ||
|  |     template<class Iter> | ||
|  |     seed_seq(Iter first, Iter last) : v(first, last) {} | ||
|  |     /** Initializes the sequence from Boost.Range range. */ | ||
|  |     template<class Range> | ||
|  |     explicit seed_seq(const Range& range) | ||
|  |       : v(boost::begin(range), boost::end(range)) {} | ||
|  | 
 | ||
|  |     /** | ||
|  |      * Fills a range with 32-bit values based on the stored sequence. | ||
|  |      * | ||
|  |      * Requires: Iter must be a Random Access Iterator whose value type | ||
|  |      * is an unsigned integral type at least 32 bits wide. | ||
|  |      */ | ||
|  |     template<class Iter> | ||
|  |     void generate(Iter first, Iter last) const | ||
|  |     { | ||
|  |         typedef typename std::iterator_traits<Iter>::value_type value_type; | ||
|  |         std::fill(first, last, static_cast<value_type>(0x8b8b8b8bu)); | ||
|  |         std::size_t s = v.size(); | ||
|  |         std::size_t n = last - first; | ||
|  |         std::size_t t = | ||
|  |             (n >= 623) ? 11 : | ||
|  |             (n >=  68) ?  7 : | ||
|  |             (n >=  39) ?  5 : | ||
|  |             (n >=   7) ?  3 : | ||
|  |             (n - 1)/2; | ||
|  |         std::size_t p = (n - t) / 2; | ||
|  |         std::size_t q = p + t; | ||
|  |         std::size_t m = (std::max)(s+1, n); | ||
|  |         value_type mask = 0xffffffffu; | ||
|  |         for(std::size_t k = 0; k < m; ++k) { | ||
|  |             value_type r1 = static_cast<value_type> | ||
|  |                 (*(first + k%n) ^ *(first + (k+p)%n) ^ *(first + (k+n-1)%n)); | ||
|  |             r1 = r1 ^ (r1 >> 27); | ||
|  |             r1 = (r1 * 1664525u) & mask; | ||
|  |             value_type r2 = static_cast<value_type>(r1 + | ||
|  |                 ((k == 0) ? s : | ||
|  |                  (k <= s) ? k % n + v[k - 1] : | ||
|  |                  (k % n))); | ||
|  |             *(first + (k+p)%n) = (*(first + (k+p)%n) + r1) & mask; | ||
|  |             *(first + (k+q)%n) = (*(first + (k+q)%n) + r2) & mask; | ||
|  |             *(first + k%n) = r2; | ||
|  |         } | ||
|  |         for(std::size_t k = m; k < m + n; ++k) { | ||
|  |             value_type r3 = static_cast<value_type> | ||
|  |                 ((*(first + k%n) + *(first + (k+p)%n) + *(first + (k+n-1)%n)) | ||
|  |                 & mask); | ||
|  |             r3 = r3 ^ (r3 >> 27); | ||
|  |             r3 = (r3 * 1566083941u) & mask; | ||
|  |             value_type r4 = static_cast<value_type>(r3 - k%m); | ||
|  |             *(first + (k+p)%n) ^= r3; | ||
|  |             *(first + (k+q)%n) ^= r4; | ||
|  |             *(first + k%n) = r4; | ||
|  |         } | ||
|  |     } | ||
|  |     /** Returns the size of the sequence. */ | ||
|  |     std::size_t size() const { return v.size(); } | ||
|  |     /** Writes the stored sequence to iter. */ | ||
|  |     template<class Iter> | ||
|  |     void param(Iter out) { std::copy(v.begin(), v.end(), out); } | ||
|  | private: | ||
|  |     std::vector<result_type> v; | ||
|  | }; | ||
|  | 
 | ||
|  | } | ||
|  | } | ||
|  | 
 | ||
|  | #endif |