266 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
			
		
		
	
	
			266 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| /*=============================================================================
 | |
|     Copyright (c) 2011 Eric Niebler
 | |
| 
 | |
|     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)
 | |
| ==============================================================================*/
 | |
| #if !defined(BOOST_FUSION_SEGMENTED_ITERATOR_NEXT_IMPL_HPP_INCLUDED)
 | |
| #define BOOST_FUSION_SEGMENTED_ITERATOR_NEXT_IMPL_HPP_INCLUDED
 | |
| 
 | |
| #include <boost/fusion/support/config.hpp>
 | |
| #include <boost/type_traits/add_const.hpp>
 | |
| #include <boost/type_traits/remove_reference.hpp>
 | |
| #include <boost/fusion/iterator/equal_to.hpp>
 | |
| #include <boost/fusion/container/list/cons_fwd.hpp>
 | |
| #include <boost/fusion/iterator/next.hpp>
 | |
| #include <boost/fusion/iterator/deref.hpp>
 | |
| 
 | |
| namespace boost { namespace fusion
 | |
| {
 | |
|     template <typename First, typename Second>
 | |
|     struct iterator_range;
 | |
| 
 | |
|     template <typename Context>
 | |
|     struct segmented_iterator;
 | |
| 
 | |
|     namespace detail
 | |
|     {
 | |
|         template <typename Sequence, typename Stack>
 | |
|         struct segmented_begin_impl;
 | |
| 
 | |
|         //bool is_invalid(stack)
 | |
|         //{
 | |
|         //  return empty(car(stack));
 | |
|         //}
 | |
| 
 | |
|         template <typename Stack>
 | |
|         struct is_invalid
 | |
|           : result_of::equal_to<
 | |
|                 typename Stack::car_type::begin_type,
 | |
|                 typename Stack::car_type::end_type
 | |
|             >
 | |
|         {};
 | |
| 
 | |
|         ////Advance the first iterator in the seq at the
 | |
|         ////top of a stack of iterator ranges. Return the
 | |
|         ////new stack.
 | |
|         //auto pop_front_car(stack)
 | |
|         //{
 | |
|         //  return cons(iterator_range(next(begin(car(stack))), end(car(stack))), cdr(stack));
 | |
|         //}
 | |
| 
 | |
|         template <typename Stack>
 | |
|         struct pop_front_car
 | |
|         {
 | |
|             typedef 
 | |
|                 iterator_range<
 | |
|                     typename result_of::next<
 | |
|                         typename Stack::car_type::begin_type
 | |
|                     >::type
 | |
|                   , typename Stack::car_type::end_type
 | |
|                 >
 | |
|             car_type;
 | |
|             
 | |
|             typedef
 | |
|                 cons<car_type, typename Stack::cdr_type>
 | |
|             type;
 | |
| 
 | |
|             BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
 | |
|             static type call(Stack const & stack)
 | |
|             {
 | |
|                 return type(
 | |
|                     car_type(fusion::next(stack.car.first), stack.car.last),
 | |
|                     stack.cdr);
 | |
|             }
 | |
|         };
 | |
| 
 | |
|         template <
 | |
|             typename Stack,
 | |
|             typename Next   = typename pop_front_car<Stack>::type,
 | |
|             bool IsInvalid  = is_invalid<Next>::value,
 | |
|             int StackSize   = Stack::size::value>
 | |
|         struct segmented_next_impl_recurse;
 | |
| 
 | |
|         // Handle the case where the top of the stack has no usable 
 | |
|         //auto segmented_next_impl_recurse3(stack)
 | |
|         //{
 | |
|         //  if (size(stack) == 1)
 | |
|         //    return cons(iterator_range(end(car(stack)), end(car(stack))), nil_);
 | |
|         //  else
 | |
|         //    return segmented_next_impl_recurse(stack.cdr);
 | |
|         //}
 | |
| 
 | |
|         template <
 | |
|             typename Stack,
 | |
|             int StackSize = Stack::size::value>
 | |
|         struct segmented_next_impl_recurse3
 | |
|         {
 | |
|             typedef segmented_next_impl_recurse<typename Stack::cdr_type> impl;
 | |
|             typedef typename impl::type type;
 | |
| 
 | |
|             BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
 | |
|             static type call(Stack const & stack)
 | |
|             {
 | |
|                 return impl::call(stack.cdr);
 | |
|             }
 | |
|         };
 | |
| 
 | |
|         template <typename Stack>
 | |
|         struct segmented_next_impl_recurse3<Stack, 1>
 | |
|         {
 | |
|             typedef typename Stack::car_type::end_type end_type;
 | |
|             typedef iterator_range<end_type, end_type> range_type;
 | |
|             typedef cons<range_type> type;
 | |
| 
 | |
|             BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
 | |
|             static type call(Stack const & stack)
 | |
|             {
 | |
|                 return type(range_type(stack.car.last, stack.car.last));
 | |
|             }
 | |
|         };
 | |
| 
 | |
|         //auto segmented_next_impl_recurse2(stack)
 | |
|         //{
 | |
|         //  auto res = segmented_begin_impl(front(car(stack)), stack);
 | |
|         //  if (is_invalid(res))
 | |
|         //    return segmented_next_impl_recurse3(stack);
 | |
|         //  else
 | |
|         //    return res;
 | |
|         //}
 | |
| 
 | |
|         template <
 | |
|             typename Stack,
 | |
|             typename Sequence  =
 | |
|                 typename remove_reference<
 | |
|                     typename add_const<
 | |
|                         typename result_of::deref<
 | |
|                             typename Stack::car_type::begin_type
 | |
|                         >::type
 | |
|                     >::type
 | |
|                 >::type,
 | |
|             typename Result =
 | |
|                 typename segmented_begin_impl<Sequence, Stack>::type,
 | |
|             bool IsInvalid  =
 | |
|                 is_invalid<Result>::value>
 | |
|         struct segmented_next_impl_recurse2
 | |
|         {
 | |
|             typedef segmented_next_impl_recurse3<Stack> impl;
 | |
|             typedef typename impl::type type;
 | |
| 
 | |
|             BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
 | |
|             static type call(Stack const & stack)
 | |
|             {
 | |
|                 return impl::call(stack);
 | |
|             }
 | |
|         };
 | |
| 
 | |
|         template <typename Stack, typename Sequence, typename Result>
 | |
|         struct segmented_next_impl_recurse2<Stack, Sequence, Result, false>
 | |
|         {
 | |
|             typedef Result type;
 | |
| 
 | |
|             BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
 | |
|             static type call(Stack const & stack)
 | |
|             {
 | |
|                 return segmented_begin_impl<Sequence, Stack>::call(*stack.car.first, stack);
 | |
|             }
 | |
|         };
 | |
| 
 | |
|         //auto segmented_next_impl_recurse(stack)
 | |
|         //{
 | |
|         //  auto next = pop_front_car(stack);
 | |
|         //  if (is_invalid(next))
 | |
|         //    if (1 == size(stack))
 | |
|         //      return next;
 | |
|         //    else
 | |
|         //      return segmented_next_impl_recurse(cdr(stack));
 | |
|         //  else
 | |
|         //    return segmented_next_impl_recurse2(next)
 | |
|         //}
 | |
| 
 | |
|         template <typename Stack, typename Next, bool IsInvalid, int StackSize>
 | |
|         struct segmented_next_impl_recurse
 | |
|         {
 | |
|             typedef
 | |
|                 typename segmented_next_impl_recurse<typename Stack::cdr_type>::type
 | |
|             type;
 | |
| 
 | |
|             BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
 | |
|             static type call(Stack const& stack)
 | |
|             {
 | |
|                 return segmented_next_impl_recurse<typename Stack::cdr_type>::call(stack.cdr);
 | |
|             }
 | |
|         };
 | |
| 
 | |
|         template <typename Stack, typename Next>
 | |
|         struct segmented_next_impl_recurse<Stack, Next, true, 1>
 | |
|         {
 | |
|             typedef Next type;
 | |
| 
 | |
|             BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
 | |
|             static type call(Stack const & stack)
 | |
|             {
 | |
|                 return pop_front_car<Stack>::call(stack);
 | |
|             }
 | |
|         };
 | |
| 
 | |
|         template <typename Stack, typename Next, int StackSize>
 | |
|         struct segmented_next_impl_recurse<Stack, Next, false, StackSize>
 | |
|         {
 | |
|             typedef segmented_next_impl_recurse2<Next> impl;
 | |
|             typedef typename impl::type type;
 | |
| 
 | |
|             BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
 | |
|             static type call(Stack const & stack)
 | |
|             {
 | |
|                 return impl::call(pop_front_car<Stack>::call(stack));
 | |
|             }
 | |
|         };
 | |
| 
 | |
|         //auto segmented_next_impl(stack)
 | |
|         //{
 | |
|         //  // car(stack) is a seq of values, not a seq of segments
 | |
|         //  auto next = pop_front_car(stack);
 | |
|         //  if (is_invalid(next))
 | |
|         //    return segmented_next_impl_recurse(cdr(next));
 | |
|         //  else
 | |
|         //    return next;
 | |
|         //}
 | |
| 
 | |
|         template <
 | |
|             typename Stack,
 | |
|             typename Next   = typename pop_front_car<Stack>::type,
 | |
|             bool IsInvalid  = is_invalid<Next>::value>
 | |
|         struct segmented_next_impl_aux
 | |
|         {
 | |
|             typedef segmented_next_impl_recurse<typename Stack::cdr_type> impl;
 | |
|             typedef typename impl::type type;
 | |
| 
 | |
|             BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
 | |
|             static type call(Stack const & stack)
 | |
|             {
 | |
|                 return impl::call(stack.cdr);
 | |
|             }
 | |
|         };
 | |
| 
 | |
|         template <typename Stack, typename Next>
 | |
|         struct segmented_next_impl_aux<Stack, Next, false>
 | |
|         {
 | |
|             typedef Next type;
 | |
| 
 | |
|             BOOST_CONSTEXPR BOOST_FUSION_GPU_ENABLED
 | |
|             static type call(Stack const & stack)
 | |
|             {
 | |
|                 return pop_front_car<Stack>::call(stack);
 | |
|             }
 | |
|         };
 | |
| 
 | |
|         template <typename Stack>
 | |
|         struct segmented_next_impl
 | |
|           : segmented_next_impl_aux<Stack>
 | |
|         {};
 | |
|     }
 | |
| }}
 | |
| 
 | |
| #endif
 | 
