Path: blob/main/contrib/llvm-project/libcxx/include/__random/seed_seq.h
35233 views
//===----------------------------------------------------------------------===//1//2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.3// See https://llvm.org/LICENSE.txt for license information.4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception5//6//===----------------------------------------------------------------------===//78#ifndef _LIBCPP___RANDOM_SEED_SEQ_H9#define _LIBCPP___RANDOM_SEED_SEQ_H1011#include <__algorithm/copy.h>12#include <__algorithm/fill.h>13#include <__algorithm/max.h>14#include <__config>15#include <__iterator/iterator_traits.h>16#include <__type_traits/is_unsigned.h>17#include <cstdint>18#include <initializer_list>19#include <vector>2021#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)22# pragma GCC system_header23#endif2425_LIBCPP_PUSH_MACROS26#include <__undef_macros>2728_LIBCPP_BEGIN_NAMESPACE_STD2930class _LIBCPP_TEMPLATE_VIS seed_seq {31public:32// types33typedef uint32_t result_type;3435// constructors36_LIBCPP_HIDE_FROM_ABI seed_seq() _NOEXCEPT {}37#ifndef _LIBCPP_CXX03_LANG38template <class _Tp, __enable_if_t<is_integral<_Tp>::value, int> = 0>39_LIBCPP_HIDE_FROM_ABI seed_seq(initializer_list<_Tp> __il) {40__init(__il.begin(), __il.end());41}42#endif // _LIBCPP_CXX03_LANG4344template <class _InputIterator>45_LIBCPP_HIDE_FROM_ABI seed_seq(_InputIterator __first, _InputIterator __last) {46static_assert(is_integral<typename iterator_traits<_InputIterator>::value_type>::value,47"Mandates: iterator_traits<InputIterator>::value_type is an integer type");48__init(__first, __last);49}5051// generating functions52template <class _RandomAccessIterator>53_LIBCPP_HIDE_FROM_ABI void generate(_RandomAccessIterator __first, _RandomAccessIterator __last);5455// property functions56_LIBCPP_HIDE_FROM_ABI size_t size() const _NOEXCEPT { return __v_.size(); }57template <class _OutputIterator>58_LIBCPP_HIDE_FROM_ABI void param(_OutputIterator __dest) const {59std::copy(__v_.begin(), __v_.end(), __dest);60}6162seed_seq(const seed_seq&) = delete;63void operator=(const seed_seq&) = delete;6465_LIBCPP_HIDE_FROM_ABI static result_type _Tp(result_type __x) { return __x ^ (__x >> 27); }6667private:68template <class _InputIterator>69_LIBCPP_HIDE_FROM_ABI void __init(_InputIterator __first, _InputIterator __last);7071vector<result_type> __v_;72};7374template <class _InputIterator>75void seed_seq::__init(_InputIterator __first, _InputIterator __last) {76for (_InputIterator __s = __first; __s != __last; ++__s)77__v_.push_back(*__s & 0xFFFFFFFF);78}7980template <class _RandomAccessIterator>81void seed_seq::generate(_RandomAccessIterator __first, _RandomAccessIterator __last) {82using _ValueType = typename iterator_traits<_RandomAccessIterator>::value_type;83static_assert(is_unsigned<_ValueType>::value && sizeof(_ValueType) >= sizeof(uint32_t),84"[rand.util.seedseq]/7 requires the value_type of the iterator to be an unsigned "85"integer capable of accommodating 32-bit quantities.");8687if (__first != __last) {88std::fill(__first, __last, 0x8b8b8b8b);89const size_t __n = static_cast<size_t>(__last - __first);90const size_t __s = __v_.size();91const size_t __t = (__n >= 623) ? 11 : (__n >= 68) ? 7 : (__n >= 39) ? 5 : (__n >= 7) ? 3 : (__n - 1) / 2;92const size_t __p = (__n - __t) / 2;93const size_t __q = __p + __t;94const size_t __m = std::max(__s + 1, __n);95// __k = 0;96{97result_type __r = 1664525 * _Tp(__first[0] ^ __first[__p] ^ __first[__n - 1]);98__first[__p] += __r;99__r += __s;100__first[__q] += __r;101__first[0] = __r;102}103// Initialize indexing terms used with if statements as an optimization to104// avoid calculating modulo n on every loop iteration for each term.105size_t __kmodn = 0; // __k % __n106size_t __k1modn = __n - 1; // (__k - 1) % __n107size_t __kpmodn = __p % __n; // (__k + __p) % __n108size_t __kqmodn = __q % __n; // (__k + __q) % __n109110for (size_t __k = 1; __k <= __s; ++__k) {111if (++__kmodn == __n)112__kmodn = 0;113if (++__k1modn == __n)114__k1modn = 0;115if (++__kpmodn == __n)116__kpmodn = 0;117if (++__kqmodn == __n)118__kqmodn = 0;119120result_type __r = 1664525 * _Tp(__first[__kmodn] ^ __first[__kpmodn] ^ __first[__k1modn]);121__first[__kpmodn] += __r;122__r += __kmodn + __v_[__k - 1];123__first[__kqmodn] += __r;124__first[__kmodn] = __r;125}126for (size_t __k = __s + 1; __k < __m; ++__k) {127if (++__kmodn == __n)128__kmodn = 0;129if (++__k1modn == __n)130__k1modn = 0;131if (++__kpmodn == __n)132__kpmodn = 0;133if (++__kqmodn == __n)134__kqmodn = 0;135136result_type __r = 1664525 * _Tp(__first[__kmodn] ^ __first[__kpmodn] ^ __first[__k1modn]);137__first[__kpmodn] += __r;138__r += __kmodn;139__first[__kqmodn] += __r;140__first[__kmodn] = __r;141}142for (size_t __k = __m; __k < __m + __n; ++__k) {143if (++__kmodn == __n)144__kmodn = 0;145if (++__k1modn == __n)146__k1modn = 0;147if (++__kpmodn == __n)148__kpmodn = 0;149if (++__kqmodn == __n)150__kqmodn = 0;151152result_type __r = 1566083941 * _Tp(__first[__kmodn] + __first[__kpmodn] + __first[__k1modn]);153__first[__kpmodn] ^= __r;154__r -= __kmodn;155__first[__kqmodn] ^= __r;156__first[__kmodn] = __r;157}158}159}160161_LIBCPP_END_NAMESPACE_STD162163_LIBCPP_POP_MACROS164165#endif // _LIBCPP___RANDOM_SEED_SEQ_H166167168