blob: 33571521188452aedae5d5560bd67a7fa9fa1df9 [file] [log] [blame]
Arthur O'Dwyer344cef62021-11-19 14:58:51 -05001//===----------------------------------------------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
9#ifndef _LIBCPP___RANDOM_DISCARD_BLOCK_ENGINE_H
10#define _LIBCPP___RANDOM_DISCARD_BLOCK_ENGINE_H
11
12#include <__config>
13#include <__random/is_seed_sequence.h>
14#include <__utility/move.h>
15#include <climits>
16#include <iosfwd>
17#include <type_traits>
18
19#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
20#pragma GCC system_header
21#endif
22
23_LIBCPP_PUSH_MACROS
24#include <__undef_macros>
25
26_LIBCPP_BEGIN_NAMESPACE_STD
27
28template<class _Engine, size_t __p, size_t __r>
29class _LIBCPP_TEMPLATE_VIS discard_block_engine
30{
31 _Engine __e_;
32 int __n_;
33
34 static_assert( 0 < __r, "discard_block_engine invalid parameters");
35 static_assert(__r <= __p, "discard_block_engine invalid parameters");
36 static_assert(__r <= INT_MAX, "discard_block_engine invalid parameters");
37public:
38 // types
39 typedef typename _Engine::result_type result_type;
40
41 // engine characteristics
42 static _LIBCPP_CONSTEXPR const size_t block_size = __p;
43 static _LIBCPP_CONSTEXPR const size_t used_block = __r;
44
45#ifdef _LIBCPP_CXX03_LANG
46 static const result_type _Min = _Engine::_Min;
47 static const result_type _Max = _Engine::_Max;
48#else
49 static _LIBCPP_CONSTEXPR const result_type _Min = _Engine::min();
50 static _LIBCPP_CONSTEXPR const result_type _Max = _Engine::max();
51#endif
52
53 _LIBCPP_INLINE_VISIBILITY
54 static _LIBCPP_CONSTEXPR result_type min() { return _Engine::min(); }
55 _LIBCPP_INLINE_VISIBILITY
56 static _LIBCPP_CONSTEXPR result_type max() { return _Engine::max(); }
57
58 // constructors and seeding functions
59 _LIBCPP_INLINE_VISIBILITY
60 discard_block_engine() : __n_(0) {}
61 _LIBCPP_INLINE_VISIBILITY
62 explicit discard_block_engine(const _Engine& __e)
63 : __e_(__e), __n_(0) {}
64#ifndef _LIBCPP_CXX03_LANG
65 _LIBCPP_INLINE_VISIBILITY
66 explicit discard_block_engine(_Engine&& __e)
67 : __e_(_VSTD::move(__e)), __n_(0) {}
68#endif // _LIBCPP_CXX03_LANG
69 _LIBCPP_INLINE_VISIBILITY
70 explicit discard_block_engine(result_type __sd) : __e_(__sd), __n_(0) {}
71 template<class _Sseq>
72 _LIBCPP_INLINE_VISIBILITY
73 explicit discard_block_engine(_Sseq& __q,
74 typename enable_if<__is_seed_sequence<_Sseq, discard_block_engine>::value &&
75 !is_convertible<_Sseq, _Engine>::value>::type* = 0)
76 : __e_(__q), __n_(0) {}
77 _LIBCPP_INLINE_VISIBILITY
78 void seed() {__e_.seed(); __n_ = 0;}
79 _LIBCPP_INLINE_VISIBILITY
80 void seed(result_type __sd) {__e_.seed(__sd); __n_ = 0;}
81 template<class _Sseq>
82 _LIBCPP_INLINE_VISIBILITY
83 typename enable_if
84 <
85 __is_seed_sequence<_Sseq, discard_block_engine>::value,
86 void
87 >::type
88 seed(_Sseq& __q) {__e_.seed(__q); __n_ = 0;}
89
90 // generating functions
91 result_type operator()();
92 _LIBCPP_INLINE_VISIBILITY
93 void discard(unsigned long long __z) {for (; __z; --__z) operator()();}
94
95 // property functions
96 _LIBCPP_INLINE_VISIBILITY
97 const _Engine& base() const _NOEXCEPT {return __e_;}
98
99 template<class _Eng, size_t _Pp, size_t _Rp>
100 friend
101 bool
102 operator==(
103 const discard_block_engine<_Eng, _Pp, _Rp>& __x,
104 const discard_block_engine<_Eng, _Pp, _Rp>& __y);
105
106 template<class _Eng, size_t _Pp, size_t _Rp>
107 friend
108 bool
109 operator!=(
110 const discard_block_engine<_Eng, _Pp, _Rp>& __x,
111 const discard_block_engine<_Eng, _Pp, _Rp>& __y);
112
113 template <class _CharT, class _Traits,
114 class _Eng, size_t _Pp, size_t _Rp>
115 friend
116 basic_ostream<_CharT, _Traits>&
117 operator<<(basic_ostream<_CharT, _Traits>& __os,
118 const discard_block_engine<_Eng, _Pp, _Rp>& __x);
119
120 template <class _CharT, class _Traits,
121 class _Eng, size_t _Pp, size_t _Rp>
122 friend
123 basic_istream<_CharT, _Traits>&
124 operator>>(basic_istream<_CharT, _Traits>& __is,
125 discard_block_engine<_Eng, _Pp, _Rp>& __x);
126};
127
128template<class _Engine, size_t __p, size_t __r>
129 _LIBCPP_CONSTEXPR const size_t discard_block_engine<_Engine, __p, __r>::block_size;
130
131template<class _Engine, size_t __p, size_t __r>
132 _LIBCPP_CONSTEXPR const size_t discard_block_engine<_Engine, __p, __r>::used_block;
133
134template<class _Engine, size_t __p, size_t __r>
135typename discard_block_engine<_Engine, __p, __r>::result_type
136discard_block_engine<_Engine, __p, __r>::operator()()
137{
138 if (__n_ >= static_cast<int>(__r))
139 {
140 __e_.discard(__p - __r);
141 __n_ = 0;
142 }
143 ++__n_;
144 return __e_();
145}
146
147template<class _Eng, size_t _Pp, size_t _Rp>
148inline _LIBCPP_INLINE_VISIBILITY
149bool
150operator==(const discard_block_engine<_Eng, _Pp, _Rp>& __x,
151 const discard_block_engine<_Eng, _Pp, _Rp>& __y)
152{
153 return __x.__n_ == __y.__n_ && __x.__e_ == __y.__e_;
154}
155
156template<class _Eng, size_t _Pp, size_t _Rp>
157inline _LIBCPP_INLINE_VISIBILITY
158bool
159operator!=(const discard_block_engine<_Eng, _Pp, _Rp>& __x,
160 const discard_block_engine<_Eng, _Pp, _Rp>& __y)
161{
162 return !(__x == __y);
163}
164
165template <class _CharT, class _Traits,
166 class _Eng, size_t _Pp, size_t _Rp>
167basic_ostream<_CharT, _Traits>&
168operator<<(basic_ostream<_CharT, _Traits>& __os,
169 const discard_block_engine<_Eng, _Pp, _Rp>& __x)
170{
171 __save_flags<_CharT, _Traits> __lx(__os);
172 typedef basic_ostream<_CharT, _Traits> _Ostream;
173 __os.flags(_Ostream::dec | _Ostream::left);
174 _CharT __sp = __os.widen(' ');
175 __os.fill(__sp);
176 return __os << __x.__e_ << __sp << __x.__n_;
177}
178
179template <class _CharT, class _Traits,
180 class _Eng, size_t _Pp, size_t _Rp>
181basic_istream<_CharT, _Traits>&
182operator>>(basic_istream<_CharT, _Traits>& __is,
183 discard_block_engine<_Eng, _Pp, _Rp>& __x)
184{
185 __save_flags<_CharT, _Traits> __lx(__is);
186 typedef basic_istream<_CharT, _Traits> _Istream;
187 __is.flags(_Istream::dec | _Istream::skipws);
188 _Eng __e;
189 int __n;
190 __is >> __e >> __n;
191 if (!__is.fail())
192 {
193 __x.__e_ = __e;
194 __x.__n_ = __n;
195 }
196 return __is;
197}
198
199_LIBCPP_END_NAMESPACE_STD
200
201_LIBCPP_POP_MACROS
202
203#endif // _LIBCPP___RANDOM_DISCARD_BLOCK_ENGINE_H