419 lines
10 KiB
C
Raw Normal View History

2014-03-18 22:17:40 +01:00
/*
* Copyright 2008-2012 NVIDIA Corporation
*
* 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
*
* http://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.
*/
#pragma once
#include <thrust/iterator/iterator_traits.h>
#include <thrust/iterator/iterator_facade.h>
#include <thrust/iterator/iterator_categories.h>
#include <thrust/iterator/detail/minimum_category.h>
#include <thrust/iterator/detail/minimum_system.h>
#include <thrust/tuple.h>
#include <thrust/detail/tuple_meta_transform.h>
#include <thrust/detail/tuple_transform.h>
#include <thrust/detail/type_traits.h>
#include <thrust/iterator/detail/tuple_of_iterator_references.h>
namespace thrust
{
// forward declare zip_iterator for zip_iterator_base
template<typename IteratorTuple> class zip_iterator;
namespace detail
{
// Functors to be used with tuple algorithms
//
template<typename DiffType>
class advance_iterator
{
public:
inline __host__ __device__
advance_iterator(DiffType step) : m_step(step) {}
template<typename Iterator>
inline __host__ __device__
void operator()(Iterator& it) const
{ it += m_step; }
private:
DiffType m_step;
}; // end advance_iterator
struct increment_iterator
{
template<typename Iterator>
inline __host__ __device__
void operator()(Iterator& it)
{ ++it; }
}; // end increment_iterator
struct decrement_iterator
{
template<typename Iterator>
inline __host__ __device__
void operator()(Iterator& it)
{ --it; }
}; // end decrement_iterator
struct dereference_iterator
{
template<typename Iterator>
struct apply
{
typedef typename
iterator_traits<Iterator>::reference
type;
}; // end apply
// XXX silence warnings of the form "calling a __host__ function from a __host__ __device__ function is not allowed
__thrust_hd_warning_disable__
template<typename Iterator>
__host__ __device__
typename apply<Iterator>::type operator()(Iterator const& it)
{
return *it;
}
}; // end dereference_iterator
// The namespace tuple_impl_specific provides two meta-
// algorithms and two algorithms for tuples.
namespace tuple_impl_specific
{
// define apply1 for tuple_meta_transform_impl
template<typename UnaryMetaFunctionClass, class Arg>
struct apply1
: UnaryMetaFunctionClass::template apply<Arg>
{
}; // end apply1
// define apply2 for tuple_meta_accumulate_impl
template<typename UnaryMetaFunctionClass, class Arg1, class Arg2>
struct apply2
: UnaryMetaFunctionClass::template apply<Arg1,Arg2>
{
}; // end apply2
// Meta-accumulate algorithm for tuples. Note: The template
// parameter StartType corresponds to the initial value in
// ordinary accumulation.
//
template<class Tuple, class BinaryMetaFun, class StartType>
struct tuple_meta_accumulate;
template<
typename Tuple
, class BinaryMetaFun
, typename StartType
>
struct tuple_meta_accumulate_impl
{
typedef typename apply2<
BinaryMetaFun
, typename Tuple::head_type
, typename tuple_meta_accumulate<
typename Tuple::tail_type
, BinaryMetaFun
, StartType
>::type
>::type type;
};
template<
typename Tuple
, class BinaryMetaFun
, typename StartType
>
struct tuple_meta_accumulate
: thrust::detail::eval_if<
thrust::detail::is_same<Tuple, thrust::null_type>::value
, thrust::detail::identity_<StartType>
, tuple_meta_accumulate_impl<
Tuple
, BinaryMetaFun
, StartType
>
> // end eval_if
{
}; // end tuple_meta_accumulate
// transform algorithm for tuples. The template parameter Fun
// must be a unary functor which is also a unary metafunction
// class that computes its return type based on its argument
// type. For example:
//
// struct to_ptr
// {
// template <class Arg>
// struct apply
// {
// typedef Arg* type;
// }
//
// template <class Arg>
// Arg* operator()(Arg x);
// };
// for_each algorithm for tuples.
//
template<typename Fun, typename System>
inline __host__ __device__
Fun tuple_for_each(thrust::null_type, Fun f, System *)
{
return f;
} // end tuple_for_each()
template<typename Tuple, typename Fun, typename System>
inline __host__ __device__
Fun tuple_for_each(Tuple& t, Fun f, System *dispatch_tag)
{
f( t.get_head() );
return tuple_for_each(t.get_tail(), f, dispatch_tag);
} // end tuple_for_each()
template<typename Tuple, typename Fun>
inline __host__ __device__
Fun tuple_for_each(Tuple& t, Fun f, thrust::host_system_tag *dispatch_tag)
{
// XXX this path is required in order to accomodate pure host iterators
// (such as std::vector::iterator) in a zip_iterator
#ifndef __CUDA_ARCH__
f( t.get_head() );
return tuple_for_each(t.get_tail(), f, dispatch_tag);
#else
// this code will never be called
return f;
#endif
} // end tuple_for_each()
// Equality of tuples. NOTE: "==" for tuples currently (7/2003)
// has problems under some compilers, so I just do my own.
// No point in bringing in a bunch of #ifdefs here. This is
// going to go away with the next tuple implementation anyway.
//
__host__ __device__
inline bool tuple_equal(thrust::null_type, thrust::null_type)
{ return true; }
template<typename Tuple1, typename Tuple2>
__host__ __device__
bool tuple_equal(Tuple1 const& t1, Tuple2 const& t2)
{
return t1.get_head() == t2.get_head() &&
tuple_equal(t1.get_tail(), t2.get_tail());
} // end tuple_equal()
} // end end tuple_impl_specific
// Metafunction to obtain the type of the tuple whose element types
// are the value_types of an iterator tupel.
//
template<typename IteratorTuple>
struct tuple_of_value_types
: tuple_meta_transform<
IteratorTuple,
iterator_value
>
{
}; // end tuple_of_value_types
struct minimum_category_lambda
{
template<typename T1, typename T2>
struct apply : minimum_category<T1,T2>
{};
};
// Metafunction to obtain the minimal traversal tag in a tuple
// of iterators.
//
template<typename IteratorTuple>
struct minimum_traversal_category_in_iterator_tuple
{
typedef typename tuple_meta_transform<
IteratorTuple
, thrust::iterator_traversal
>::type tuple_of_traversal_tags;
typedef typename tuple_impl_specific::tuple_meta_accumulate<
tuple_of_traversal_tags
, minimum_category_lambda
, thrust::random_access_traversal_tag
>::type type;
};
struct minimum_system_lambda
{
template<typename T1, typename T2>
struct apply : minimum_system<T1,T2>
{};
};
// Metafunction to obtain the minimal system tag in a tuple
// of iterators.
template<typename IteratorTuple>
struct minimum_system_in_iterator_tuple
{
typedef typename thrust::detail::tuple_meta_transform<
IteratorTuple,
thrust::iterator_system
>::type tuple_of_system_tags;
typedef typename tuple_impl_specific::tuple_meta_accumulate<
tuple_of_system_tags,
minimum_system_lambda,
thrust::any_system_tag
>::type type;
};
namespace zip_iterator_base_ns
{
template<int i, typename Tuple>
struct tuple_elements_helper
: eval_if<
(i < tuple_size<Tuple>::value),
tuple_element<i,Tuple>,
identity_<thrust::null_type>
>
{};
template<typename Tuple>
struct tuple_elements
{
typedef typename tuple_elements_helper<0,Tuple>::type T0;
typedef typename tuple_elements_helper<1,Tuple>::type T1;
typedef typename tuple_elements_helper<2,Tuple>::type T2;
typedef typename tuple_elements_helper<3,Tuple>::type T3;
typedef typename tuple_elements_helper<4,Tuple>::type T4;
typedef typename tuple_elements_helper<5,Tuple>::type T5;
typedef typename tuple_elements_helper<6,Tuple>::type T6;
typedef typename tuple_elements_helper<7,Tuple>::type T7;
typedef typename tuple_elements_helper<8,Tuple>::type T8;
typedef typename tuple_elements_helper<9,Tuple>::type T9;
};
template<typename IteratorTuple>
struct tuple_of_iterator_references
{
// get a thrust::tuple of the iterators' references
typedef typename tuple_meta_transform<
IteratorTuple,
iterator_reference
>::type tuple_of_references;
// get at the individual tuple element types by name
typedef tuple_elements<tuple_of_references> elements;
// map thrust::tuple<T...> to tuple_of_iterator_references<T...>
typedef thrust::detail::tuple_of_iterator_references<
typename elements::T0,
typename elements::T1,
typename elements::T2,
typename elements::T3,
typename elements::T4,
typename elements::T5,
typename elements::T6,
typename elements::T7,
typename elements::T8,
typename elements::T9
> type;
};
} // end zip_iterator_base_ns
///////////////////////////////////////////////////////////////////
//
// Class zip_iterator_base
//
// Builds and exposes the iterator facade type from which the zip
// iterator will be derived.
//
template<typename IteratorTuple>
struct zip_iterator_base
{
//private:
// reference type is the type of the tuple obtained from the
// iterators' reference types.
typedef typename zip_iterator_base_ns::tuple_of_iterator_references<IteratorTuple>::type reference;
// Boost's Value type is the same as reference type.
//typedef reference value_type;
typedef typename tuple_of_value_types<IteratorTuple>::type value_type;
// Difference type is the first iterator's difference type
typedef typename thrust::iterator_traits<
typename thrust::tuple_element<0, IteratorTuple>::type
>::difference_type difference_type;
// Iterator system is the minimum system tag in the
// iterator tuple
typedef typename
minimum_system_in_iterator_tuple<IteratorTuple>::type system;
// Traversal category is the minimum traversal category in the
// iterator tuple
typedef typename
minimum_traversal_category_in_iterator_tuple<IteratorTuple>::type traversal_category;
public:
// The iterator facade type from which the zip iterator will
// be derived.
typedef thrust::iterator_facade<
zip_iterator<IteratorTuple>,
value_type,
system,
traversal_category,
reference,
difference_type
> type;
}; // end zip_iterator_base
} // end detail
} // end thrust