| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150 | // Copyright 2017 The Abseil Authors.//// Licensed under the Apache License, Version 2.0 (the "License");// you may not use this file except in compliance with the License.// You may obtain a copy of the License at////      https://www.apache.org/licenses/LICENSE-2.0//// Unless required by applicable law or agreed to in writing, software// distributed under the License is distributed on an "AS IS" BASIS,// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.// See the License for the specific language governing permissions and// limitations under the License.#ifndef ABSL_RANDOM_INTERNAL_NONSECURE_BASE_H_#define ABSL_RANDOM_INTERNAL_NONSECURE_BASE_H_#include <algorithm>#include <cstdint>#include <iostream>#include <iterator>#include <random>#include <string>#include <type_traits>#include <vector>#include "absl/base/macros.h"#include "absl/meta/type_traits.h"#include "absl/random/internal/pool_urbg.h"#include "absl/random/internal/salted_seed_seq.h"#include "absl/random/internal/seed_material.h"#include "absl/types/optional.h"#include "absl/types/span.h"namespace absl {ABSL_NAMESPACE_BEGINnamespace random_internal {// Each instance of NonsecureURBGBase<URBG> will be seeded by variates produced// by a thread-unique URBG-instance.template <typename URBG>class NonsecureURBGBase { public:  using result_type = typename URBG::result_type;  // Default constructor  NonsecureURBGBase() : urbg_(ConstructURBG()) {}  // Copy disallowed, move allowed.  NonsecureURBGBase(const NonsecureURBGBase&) = delete;  NonsecureURBGBase& operator=(const NonsecureURBGBase&) = delete;  NonsecureURBGBase(NonsecureURBGBase&&) = default;  NonsecureURBGBase& operator=(NonsecureURBGBase&&) = default;  // Constructor using a seed  template <class SSeq, typename = typename absl::enable_if_t<                            !std::is_same<SSeq, NonsecureURBGBase>::value>>  explicit NonsecureURBGBase(SSeq&& seq)      : urbg_(ConstructURBG(std::forward<SSeq>(seq))) {}  // Note: on MSVC, min() or max() can be interpreted as MIN() or MAX(), so we  // enclose min() or max() in parens as (min)() and (max)().  // Additionally, clang-format requires no space before this construction.  // NonsecureURBGBase::min()  static constexpr result_type(min)() { return (URBG::min)(); }  // NonsecureURBGBase::max()  static constexpr result_type(max)() { return (URBG::max)(); }  // NonsecureURBGBase::operator()()  result_type operator()() { return urbg_(); }  // NonsecureURBGBase::discard()  void discard(unsigned long long values) {  // NOLINT(runtime/int)    urbg_.discard(values);  }  bool operator==(const NonsecureURBGBase& other) const {    return urbg_ == other.urbg_;  }  bool operator!=(const NonsecureURBGBase& other) const {    return !(urbg_ == other.urbg_);  } private:  // Seeder is a custom seed sequence type where generate() fills the provided  // buffer via the RandenPool entropy source.  struct Seeder {    using result_type = uint32_t;    size_t size() { return 0; }    template <typename OutIterator>    void param(OutIterator) const {}    template <typename RandomAccessIterator>    void generate(RandomAccessIterator begin, RandomAccessIterator end) {      if (begin != end) {        // begin, end must be random access iterators assignable from uint32_t.        generate_impl(            std::integral_constant<bool, sizeof(*begin) == sizeof(uint32_t)>{},            begin, end);      }    }    // Commonly, generate is invoked with a pointer to a buffer which    // can be cast to a uint32_t.    template <typename RandomAccessIterator>    void generate_impl(std::integral_constant<bool, true>,                       RandomAccessIterator begin, RandomAccessIterator end) {      auto buffer = absl::MakeSpan(begin, end);      auto target = absl::MakeSpan(reinterpret_cast<uint32_t*>(buffer.data()),                                   buffer.size());      RandenPool<uint32_t>::Fill(target);    }    // The non-uint32_t case should be uncommon, and involves an extra copy,    // filling the uint32_t buffer and then mixing into the output.    template <typename RandomAccessIterator>    void generate_impl(std::integral_constant<bool, false>,                       RandomAccessIterator begin, RandomAccessIterator end) {      const size_t n = std::distance(begin, end);      absl::InlinedVector<uint32_t, 8> data(n, 0);      RandenPool<uint32_t>::Fill(absl::MakeSpan(data.begin(), data.end()));      std::copy(std::begin(data), std::end(data), begin);    }  };  static URBG ConstructURBG() {    Seeder seeder;    return URBG(seeder);  }  template <typename SSeq>  static URBG ConstructURBG(SSeq&& seq) {  // NOLINT(runtime/references)    auto salted_seq =        random_internal::MakeSaltedSeedSeq(std::forward<SSeq>(seq));    return URBG(salted_seq);  }  URBG urbg_;};}  // namespace random_internalABSL_NAMESPACE_END}  // namespace absl#endif  // ABSL_RANDOM_INTERNAL_NONSECURE_BASE_H_
 |