/*=============================================================================
  Copyright (c) 2011-2016 Bolero MURAKAMI
  https://github.com/bolero-MURAKAMI/Sprout

  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)
=============================================================================*/
#ifndef SPROUT_RANDOM_SEED_ARRAY_HPP
#define SPROUT_RANDOM_SEED_ARRAY_HPP

#include <cstdint>
#include <iterator>
#include <initializer_list>
#include <sprout/config.hpp>
#include <sprout/workaround/std/cstddef.hpp>
#include <sprout/array/array.hpp>
#include <sprout/array/make_array.hpp>
#include <sprout/index_tuple/metafunction.hpp>
#include <sprout/container/traits.hpp>
#include <sprout/container/functions.hpp>
#include <sprout/container/indexes.hpp>
#include <sprout/iterator/type_traits/is_iterator_of.hpp>
#include <sprout/type_traits/enabler_if.hpp>
#include <sprout/algorithm/fixed/results.hpp>
#include <sprout/algorithm/fixed/copy.hpp>
#include <sprout/algorithm/fixed/transform.hpp>
#include <sprout/algorithm/cxx14/copy.hpp>
#include <sprout/algorithm/cxx14/fill.hpp>
#include <sprout/range/algorithm/fixed/copy.hpp>
#include <sprout/container/functions.hpp>
#include <sprout/pit/pit.hpp>
#include <sprout/detail/container_complate.hpp>
#include HDR_ALGORITHM_MIN_MAX_SSCRISK_CEL_OR_SPROUT

namespace sprout {
	namespace random {
		namespace detail {
			template<typename UIntType>
			class seed_array_init {
			public:
				template<typename IntType>
				SPROUT_CONSTEXPR UIntType operator()(IntType const& x) const {
					return static_cast<UIntType>(x % (std::uint_least64_t(2) << 32));
				}
			};

			template<std::size_t N>
			class seed_array_impl {
			public:
				typedef std::uint_least32_t result_type;
			protected:
				typedef sprout::array<result_type, N> array_type;
			protected:
				array_type v;
			public:
				SPROUT_CONSTEXPR seed_array_impl() SPROUT_NOEXCEPT
					: v{{}}
				{}
				template<typename InputIterator>
				SPROUT_CONSTEXPR seed_array_impl(InputIterator begin, InputIterator end)
					: v(sprout::fixed::transform<array_type>(begin, end, sprout::random::detail::seed_array_init<result_type>()))
				{}
			};
		}	// namespace detail
		//
		// seed_array
		//
		template<std::size_t N>
		class seed_array
			: public sprout::random::detail::seed_array_impl<N>
		{
		private:
			typedef sprout::random::detail::seed_array_impl<N> impl_type;
		public:
			typedef typename impl_type::result_type result_type;
			typedef result_type value_type;
			typedef std::size_t size_type;
		private:
			using impl_type::v;
		private:
			static SPROUT_CONSTEXPR result_type tp(result_type x) {
				return x ^ (x >> 27);
			}
			static SPROUT_CONSTEXPR size_type tval(size_type n) {
				return (n >= 623) ? 11
					: (n >= 68) ? 7
					: (n >= 39) ? 5
					: (n >= 7) ? 3
					: (n - 1) / 2
					;
			}
			template<typename Result, sprout::index_t... Indexes>
			static SPROUT_CONSTEXPR typename sprout::fixed::results::algorithm<Result>::type set2_impl(
				Result const& result,
				size_type kpn, size_type kqn, size_type kn,
				typename sprout::container_traits<Result>::value_type r3,
				typename sprout::container_traits<Result>::value_type r4,
				sprout::index_tuple<Indexes...>
				)
			{
				return sprout::remake<Result>(
					result,
					sprout::size(result),
					(Indexes == kn ? r4
						: (result[Indexes]
							^ (Indexes == kpn ? r3 : typename sprout::container_traits<Result>::value_type())
							^ (Indexes == kqn ? r4 : typename sprout::container_traits<Result>::value_type())
							)
						)...
					);
			}
			template<typename Result>
			static SPROUT_CONSTEXPR typename sprout::fixed::results::algorithm<Result>::type set2(
				Result const& result,
				size_type kpn, size_type kqn, size_type kn,
				typename sprout::container_traits<Result>::value_type r3,
				typename sprout::container_traits<Result>::value_type r4_3
				)
			{
				return set2_impl(result, kpn, kqn, kn, r3, r3 - r4_3, sprout::container_indexes<Result>::make());
			}
			template<typename Result, sprout::index_t... Indexes>
			static SPROUT_CONSTEXPR typename sprout::fixed::results::algorithm<Result>::type set1_impl(
				Result const& result,
				size_type kpn, size_type kqn, size_type kn,
				typename sprout::container_traits<Result>::value_type r1,
				typename sprout::container_traits<Result>::value_type r2,
				sprout::index_tuple<Indexes...>
				)
			{
				return sprout::remake<Result>(
					result,
					sprout::size(result),
					(Indexes == kn ? r2
						: (result[Indexes]
							+ (Indexes == kpn ? r1 : typename sprout::container_traits<Result>::value_type())
							+ (Indexes == kqn ? r2 : typename sprout::container_traits<Result>::value_type())
							)
						)...
					);
			}
			template<typename Result>
			static SPROUT_CONSTEXPR typename sprout::fixed::results::algorithm<Result>::type set1(
				Result const& result,
				size_type kpn, size_type kqn, size_type kn,
				typename sprout::container_traits<Result>::value_type r1,
				typename sprout::container_traits<Result>::value_type r2_1
				)
			{
				return set1_impl(result, kpn, kqn, kn, r1, r1 + r2_1, sprout::container_indexes<Result>::make());
			}
			template<typename Result>
			SPROUT_CONSTEXPR typename sprout::fixed::results::algorithm<Result>::type gen2(
				Result const& result,
				size_type n, size_type s, size_type m, size_type t, size_type p, size_type q,
				size_type k
				) const
			{
				return k < m + n ? gen2(
						set2(
							result, (k + p) % n, (k + q) % n, k % n,
							1566083941 * tp(result[k % n] + result[(k + p) % n] + result[(k + n - 1) % n]),
							k % n
							),
						n, s, m, t, p, q,
						k + 1
						)
					: sprout::deep_copy(result)
					;
			}
			template<typename Result>
			SPROUT_CONSTEXPR typename sprout::fixed::results::algorithm<Result>::type gen1_2(
				Result const& result,
				size_type n, size_type s, size_type m, size_type t, size_type p, size_type q,
				size_type k
				) const
			{
				return k < m ? gen1_2(
						set1(
							result, (k + p) % n, (k + q) % n, k % n,
							1664525 * tp(result[k % n] ^ result[(k + p) % n] ^ result[(k - 1) % n]),
							k % n
							),
						n, s, m, t, p, q,
						k + 1
						)
					: sprout::deep_copy(result)
					;
			}
			template<typename Result>
			SPROUT_CONSTEXPR typename sprout::fixed::results::algorithm<Result>::type gen1_1(
				Result const& result,
				size_type n, size_type s, size_type m, size_type t, size_type p, size_type q,
				size_type k
				) const
			{
				return k <= s ? gen1_1(
						set1(
							result, (k + p) % n, (k + q) % n, k % n,
							1664525 * tp(result[k % n] ^ result[(k + p) % n] ^ result[(k - 1) % n]),
							k % n + v[k - 1]
							),
						n, s, m, t, p, q,
						k + 1
						)
					: sprout::deep_copy(result)
					;
			}
			template<typename Result>
			SPROUT_CONSTEXPR typename sprout::fixed::results::algorithm<Result>::type gen1_0(
				Result const& result,
				size_type n, size_type s, size_type p, size_type q
				) const
			{
				return set1(
					result, p, q, size_type(0),
					1664525 * tp(result[0] ^ result[p] ^ result[n - 1]),
					s
					);
			}
			template<typename Result>
			SPROUT_CONSTEXPR typename sprout::fixed::results::algorithm<Result>::type gen_impl(
				Result const& result,
				size_type n, size_type s, size_type m, size_type t, size_type p, size_type q
				) const
			{
				return gen2(
					gen1_2(
						gen1_1(
							gen1_0(
								result,
								n, s, p, q
								),
							n, s, m, t, p, q,
							1
							),
						n, s, m, t, p, q,
						s + 1
						),
					n, s, m, t, p, q,
					m
					);
			}
			template<typename Result, typename TempResult>
			SPROUT_CONSTEXPR typename sprout::fixed::results::algorithm<Result>::type gen(
				Result const& result, TempResult const& temp_result,
				size_type n, size_type s, size_type m, size_type t
				) const
			{
				return sprout::range::fixed::copy(
					gen_impl(temp_result, n, s, m, t, (n - t) / 2, (n - t) / 2 + t),
					result
					);
			}
		public:
			SPROUT_CONSTEXPR seed_array() SPROUT_NOEXCEPT
				: impl_type()
			{}
			seed_array(seed_array const&) = default;
			template<typename T>
			SPROUT_INITIALIZER_LIST_CONSTEXPR seed_array(std::initializer_list<T> il)
				: impl_type(il.begin(), il.end())
			{}
			template<typename InputIterator>
			SPROUT_CONSTEXPR seed_array(InputIterator begin, InputIterator end)
				: impl_type(begin, end)
			{}
			template<typename InputRange>
			explicit SPROUT_CONSTEXPR seed_array(InputRange const& rng)
				: impl_type(sprout::begin(rng), sprout::end(rng))
			{}
			template<typename RandomAccessIterator>
			SPROUT_CXX14_CONSTEXPR void generate(RandomAccessIterator begin, RandomAccessIterator end) const {
				typedef typename std::iterator_traits<RandomAccessIterator>::value_type value_type;
				if (begin == end) {
					return;
				}
				sprout::fill(begin, end, static_cast<value_type>(0x8b8b8b8bu));
				size_type n = end - begin;
				size_type s = size();
				size_type m = NS_SSCRISK_CEL_OR_SPROUT::max(s + 1, n);
				size_type t = tval(n);
				size_type p = (n - t) / 2;
				size_type q = p + t;
				for (size_type k = 0; k < m; ++k) {
					size_type kn = k % n;
					size_type kpn = (k + p) % n;
					size_type kqn = (k + q) % n;
					size_type k1n = (k + n - 1) % n;
					value_type r1 = 1664525 * tp(begin[kn] ^ begin[kpn] ^ begin[k1n]);
					value_type r2 = r1 + (
						k == 0 ? s
						: k <= s ? (kn + v[k - 1])
						: kn
						);
					begin[kpn] += r1;
					begin[kqn] += r2;
					begin[kn] = r2;
				}
				for (size_type k = m; k < m + n; ++k) {
					size_type kn = k % n;
					size_type kpn = (k + p) % n;
					size_type kqn = (k + q) % n;
					size_type k1n = (k + n - 1) % n;
					value_type r3 = 1566083941 * tp(begin[kn] + begin[kpn] + begin[k1n]);
					value_type r4 = r3 - kn;
					begin[kpn] ^= r3;
					begin[kqn] ^= r4;
					begin[kn] = r4;
				}
			}
			template<typename Result>
			SPROUT_CONSTEXPR typename sprout::fixed::results::algorithm<Result>::type generate(Result const& result) const {
				typedef typename sprout::container_traits<Result>::value_type value_type;
				typedef sprout::array<value_type, sprout::container_traits<Result>::static_size> temp_type;
				return !sprout::empty(result) ? gen(
						result, sprout::pit<temp_type>(static_cast<value_type>(0x8b8b8b8bu)),
						sprout::size(result), size(), sprout::max<size_type>(size() + 1, sprout::size(result)),
						tval(sprout::size(result))
						)
					: sprout::detail::container_complate(result)
					;
			}
			template<typename Result>
			SPROUT_CONSTEXPR typename sprout::fixed::results::algorithm<Result>::type generate() const {
				return generate(sprout::pit<Result>());
			}
			SPROUT_CONSTEXPR size_type size() const SPROUT_NOEXCEPT {
				return v.size();
			}
			template<
				typename OutputIterator,
				typename sprout::enabler_if<sprout::is_iterator_outputable<OutputIterator>::value>::type = sprout::enabler
			>
			SPROUT_CXX14_CONSTEXPR void param(OutputIterator dest) const {
				sprout::copy(v.begin(), v.end(), dest);
			}
			SPROUT_CONSTEXPR sprout::array<result_type, N> const& param() const {
				return v;
			}
			template<
				typename Result,
				typename sprout::enabler_if<!sprout::is_iterator_outputable<Result>::value>::type = sprout::enabler
			>
			SPROUT_CONSTEXPR typename sprout::fixed::results::algorithm<Result>::type param(Result const& result) const {
				return sprout::fixed::copy(v.begin(), v.end(), result);
			}
			template<typename Result>
			SPROUT_CONSTEXPR typename sprout::fixed::results::algorithm<Result>::type param() const {
				return sprout::fixed::copy<Result>(v.begin(), v.end());
			}
		};

		//
		// make_seed_array
		//
		template<typename... Args>
		inline SPROUT_CONSTEXPR sprout::random::seed_array<sizeof...(Args)>
		make_seed_array(Args&&... args) {
			return sprout::random::seed_array<sizeof...(Args)>(
				sprout::make_common_array(SPROUT_FORWARD(Args, args)...)
				);
		}
	}	// namespace random

	using sprout::random::seed_array;
	using sprout::random::make_seed_array;
}	// namespace sprout

#endif	// #ifndef SPROUT_RANDOM_SEED_ARRAY_HPP