Chris@16
|
1 /* boost random/seed_seq.hpp header file
|
Chris@16
|
2 *
|
Chris@16
|
3 * Copyright Steven Watanabe 2010
|
Chris@16
|
4 * Distributed under the Boost Software License, Version 1.0. (See
|
Chris@16
|
5 * accompanying file LICENSE_1_0.txt or copy at
|
Chris@16
|
6 * http://www.boost.org/LICENSE_1_0.txt)
|
Chris@16
|
7 *
|
Chris@16
|
8 * See http://www.boost.org for most recent version including documentation.
|
Chris@16
|
9 *
|
Chris@101
|
10 * $Id$
|
Chris@16
|
11 *
|
Chris@16
|
12 */
|
Chris@16
|
13
|
Chris@16
|
14 #ifndef BOOST_RANDOM_SEED_SEQ_HPP
|
Chris@16
|
15 #define BOOST_RANDOM_SEED_SEQ_HPP
|
Chris@16
|
16
|
Chris@16
|
17 #include <boost/config.hpp>
|
Chris@16
|
18 #include <boost/cstdint.hpp>
|
Chris@16
|
19 #include <boost/range/begin.hpp>
|
Chris@16
|
20 #include <boost/range/end.hpp>
|
Chris@16
|
21 #include <cstddef>
|
Chris@16
|
22 #include <vector>
|
Chris@16
|
23 #include <algorithm>
|
Chris@16
|
24 #include <iterator>
|
Chris@16
|
25
|
Chris@16
|
26 #ifndef BOOST_NO_CXX11_HDR_INITIALIZER_LIST
|
Chris@16
|
27 #include <initializer_list>
|
Chris@16
|
28 #endif
|
Chris@16
|
29
|
Chris@16
|
30 namespace boost {
|
Chris@16
|
31 namespace random {
|
Chris@16
|
32
|
Chris@16
|
33 /**
|
Chris@16
|
34 * The class @c seed_seq stores a sequence of 32-bit words
|
Chris@16
|
35 * for seeding a \pseudo_random_number_generator. These
|
Chris@16
|
36 * words will be combined to fill the entire state of the
|
Chris@16
|
37 * generator.
|
Chris@16
|
38 */
|
Chris@16
|
39 class seed_seq {
|
Chris@16
|
40 public:
|
Chris@16
|
41 typedef boost::uint_least32_t result_type;
|
Chris@16
|
42
|
Chris@16
|
43 /** Initializes a seed_seq to hold an empty sequence. */
|
Chris@16
|
44 seed_seq() {}
|
Chris@16
|
45 #ifndef BOOST_NO_CXX11_HDR_INITIALIZER_LIST
|
Chris@16
|
46 /** Initializes the sequence from an initializer_list. */
|
Chris@16
|
47 template<class T>
|
Chris@16
|
48 seed_seq(const std::initializer_list<T>& il) : v(il.begin(), il.end()) {}
|
Chris@16
|
49 #endif
|
Chris@16
|
50 /** Initializes the sequence from an iterator range. */
|
Chris@16
|
51 template<class Iter>
|
Chris@16
|
52 seed_seq(Iter first, Iter last) : v(first, last) {}
|
Chris@16
|
53 /** Initializes the sequence from Boost.Range range. */
|
Chris@16
|
54 template<class Range>
|
Chris@16
|
55 explicit seed_seq(const Range& range)
|
Chris@16
|
56 : v(boost::begin(range), boost::end(range)) {}
|
Chris@16
|
57
|
Chris@16
|
58 /**
|
Chris@16
|
59 * Fills a range with 32-bit values based on the stored sequence.
|
Chris@16
|
60 *
|
Chris@16
|
61 * Requires: Iter must be a Random Access Iterator whose value type
|
Chris@16
|
62 * is an unsigned integral type at least 32 bits wide.
|
Chris@16
|
63 */
|
Chris@16
|
64 template<class Iter>
|
Chris@16
|
65 void generate(Iter first, Iter last) const
|
Chris@16
|
66 {
|
Chris@16
|
67 typedef typename std::iterator_traits<Iter>::value_type value_type;
|
Chris@16
|
68 std::fill(first, last, static_cast<value_type>(0x8b8b8b8bu));
|
Chris@16
|
69 std::size_t s = v.size();
|
Chris@16
|
70 std::size_t n = last - first;
|
Chris@16
|
71 std::size_t t =
|
Chris@16
|
72 (n >= 623) ? 11 :
|
Chris@16
|
73 (n >= 68) ? 7 :
|
Chris@16
|
74 (n >= 39) ? 5 :
|
Chris@16
|
75 (n >= 7) ? 3 :
|
Chris@16
|
76 (n - 1)/2;
|
Chris@16
|
77 std::size_t p = (n - t) / 2;
|
Chris@16
|
78 std::size_t q = p + t;
|
Chris@16
|
79 std::size_t m = (std::max)(s+1, n);
|
Chris@16
|
80 value_type mask = 0xffffffffu;
|
Chris@16
|
81 for(std::size_t k = 0; k < m; ++k) {
|
Chris@16
|
82 value_type r1 =
|
Chris@16
|
83 *(first + k%n) ^ *(first + (k+p)%n) ^ *(first + (k+n-1)%n);
|
Chris@16
|
84 r1 = r1 ^ (r1 >> 27);
|
Chris@16
|
85 r1 = (r1 * 1664525u) & mask;
|
Chris@16
|
86 value_type r2 = r1 +
|
Chris@16
|
87 ((k == 0) ? s :
|
Chris@16
|
88 (k <= s) ? k % n + v[k - 1] :
|
Chris@16
|
89 (k % n));
|
Chris@16
|
90 *(first + (k+p)%n) = (*(first + (k+p)%n) + r1) & mask;
|
Chris@16
|
91 *(first + (k+q)%n) = (*(first + (k+q)%n) + r2) & mask;
|
Chris@16
|
92 *(first + k%n) = r2;
|
Chris@16
|
93 }
|
Chris@16
|
94 for(std::size_t k = m; k < m + n; ++k) {
|
Chris@16
|
95 value_type r3 =
|
Chris@16
|
96 (*(first + k%n) + *(first + (k+p)%n) + *(first + (k+n-1)%n))
|
Chris@16
|
97 & mask;
|
Chris@16
|
98 r3 = r3 ^ (r3 >> 27);
|
Chris@16
|
99 r3 = (r3 * 1566083941u) & mask;
|
Chris@16
|
100 value_type r4 = r3 - k%m;
|
Chris@16
|
101 *(first + (k+p)%n) ^= r3;
|
Chris@16
|
102 *(first + (k+q)%n) ^= r4;
|
Chris@16
|
103 *(first + k%n) = r4;
|
Chris@16
|
104 }
|
Chris@16
|
105 }
|
Chris@16
|
106 /** Returns the size of the sequence. */
|
Chris@16
|
107 std::size_t size() const { return v.size(); }
|
Chris@16
|
108 /** Writes the stored sequence to iter. */
|
Chris@16
|
109 template<class Iter>
|
Chris@16
|
110 void param(Iter out) { std::copy(v.begin(), v.end(), out); }
|
Chris@16
|
111 private:
|
Chris@16
|
112 std::vector<result_type> v;
|
Chris@16
|
113 };
|
Chris@16
|
114
|
Chris@16
|
115 }
|
Chris@16
|
116 }
|
Chris@16
|
117
|
Chris@16
|
118 #endif
|