2014-03-18 22:17:40 +01:00

244 lines
7.9 KiB
C++

/*
* 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.
*/
/*! \file thrust/iterator/counting_iterator.h
* \brief An iterator which returns an increasing incrementable value
* when dereferenced
*/
/*
* Copyright David Abrahams 2003.
*
* Distributed under the Boost Software License, Version 1.0.
* (See accompanying NOTICE file for the complete license)
*
* For more information, see http://www.boost.org
*/
#pragma once
#include <thrust/detail/config.h>
#include <thrust/iterator/iterator_adaptor.h>
#include <thrust/iterator/iterator_facade.h>
#include <thrust/iterator/iterator_categories.h>
// #include the details first
#include <thrust/iterator/detail/counting_iterator.inl>
namespace thrust
{
/*! \addtogroup iterators
* \{
*/
/*! \addtogroup fancyiterator Fancy Iterators
* \ingroup iterators
* \{
*/
/*! \p counting_iterator is an iterator which represents a pointer into a range
* of sequentially changing values. This iterator is useful for creating a range
* filled with a sequence without explicitly storing it in memory. Using
* \p counting_iterator saves memory capacity and bandwidth.
*
* The following code snippet demonstrates how to create a \p counting_iterator whose
* \c value_type is \c int and which sequentially increments by \c 1.
*
* \code
* #include <thrust/iterator/counting_iterator.h>
* ...
* // create iterators
* thrust::counting_iterator<int> first(10);
* thrust::counting_iterator<int> last = first + 3;
*
* first[0] // returns 10
* first[1] // returns 11
* first[100] // returns 110
*
* // sum of [first, last)
* thrust::reduce(first, last); // returns 33 (i.e. 10 + 11 + 12)
*
* // initialize vector to [0,1,2,..]
* thrust::counting_iterator<int> iter(0);
* thrust::device_vector<int> vec(500);
* thrust::copy(iter, iter + vec.size(), vec.begin());
* \endcode
*
* This next example demonstrates how to use a \p counting_iterator with the
* \p thrust::copy_if function to compute the indices of the non-zero elements
* of a \p device_vector. In this example, we use the \p make_counting_iterator
* function to avoid specifying the type of the \p counting_iterator.
*
* \code
* #include <thrust/iterator/counting_iterator.h>
* #include <thrust/copy.h>
* #include <thrust/functional.h>
* #include <thrust/device_vector.h>
*
* int main(void)
* {
* // this example computes indices for all the nonzero values in a sequence
*
* // sequence of zero and nonzero values
* thrust::device_vector<int> stencil(8);
* stencil[0] = 0;
* stencil[1] = 1;
* stencil[2] = 1;
* stencil[3] = 0;
* stencil[4] = 0;
* stencil[5] = 1;
* stencil[6] = 0;
* stencil[7] = 1;
*
* // storage for the nonzero indices
* thrust::device_vector<int> indices(8);
*
* // compute indices of nonzero elements
* typedef thrust::device_vector<int>::iterator IndexIterator;
*
* // use make_counting_iterator to define the sequence [0, 8)
* IndexIterator indices_end = thrust::copy_if(thrust::make_counting_iterator(0),
* thrust::make_counting_iterator(8),
* stencil.begin(),
* indices.begin(),
* thrust::identity<int>());
* // indices now contains [1,2,5,7]
*
* return 0;
* }
* \endcode
*
* \see make_counting_iterator
*/
template<typename Incrementable,
typename System = use_default,
typename Traversal = use_default,
typename Difference = use_default>
class counting_iterator
: public detail::counting_iterator_base<Incrementable, System, Traversal, Difference>::type
{
/*! \cond
*/
typedef typename detail::counting_iterator_base<Incrementable, System, Traversal, Difference>::type super_t;
friend class thrust::iterator_core_access;
public:
typedef typename super_t::reference reference;
typedef typename super_t::difference_type difference_type;
/*! \endcond
*/
/*! Null constructor initializes this \p counting_iterator's \c Incrementable
* counter using its null constructor.
*/
__host__ __device__
counting_iterator(void){};
/*! Copy constructor copies the value of another \p counting_iterator into a
* new \p counting_iterator.
*
* \p rhs The \p counting_iterator to copy.
*/
__host__ __device__
counting_iterator(counting_iterator const &rhs):super_t(rhs.base()){}
/*! Copy constructor copies the value of another counting_iterator
* with related System type.
*
* \param rhs The \p counting_iterator to copy.
*/
template<typename OtherSystem>
__host__ __device__
counting_iterator(counting_iterator<Incrementable, OtherSystem, Traversal, Difference> const &rhs,
typename thrust::detail::enable_if_convertible<
typename thrust::iterator_system<counting_iterator<Incrementable,OtherSystem,Traversal,Difference> >::type,
typename thrust::iterator_system<super_t>::type
>::type * = 0)
: super_t(rhs.base()){}
/*! This \c explicit constructor copies the value of an \c Incrementable
* into a new \p counting_iterator's \c Incrementable counter.
*
* \param x The initial value of the new \p counting_iterator's \c Incrementable
* counter.
*/
__host__ __device__
explicit counting_iterator(Incrementable x):super_t(x){}
/*! \cond
*/
private:
__host__ __device__
reference dereference(void) const
{
return this->base_reference();
}
// note that we implement equal specially for floating point counting_iterator
template <typename OtherIncrementable, typename OtherSystem, typename OtherTraversal, typename OtherDifference>
__host__ __device__
bool equal(counting_iterator<OtherIncrementable, OtherSystem, OtherTraversal, OtherDifference> const& y) const
{
typedef thrust::detail::counting_iterator_equal<difference_type,Incrementable,OtherIncrementable> e;
return e::equal(this->base(), y.base());
}
template <class OtherIncrementable>
__host__ __device__
difference_type
distance_to(counting_iterator<OtherIncrementable, System, Traversal, Difference> const& y) const
{
typedef typename
thrust::detail::eval_if<
thrust::detail::is_numeric<Incrementable>::value,
thrust::detail::identity_<thrust::detail::number_distance<difference_type, Incrementable, OtherIncrementable> >,
thrust::detail::identity_<thrust::detail::iterator_distance<difference_type, Incrementable, OtherIncrementable> >
>::type d;
return d::distance(this->base(), y.base());
}
/*! \endcond
*/
}; // end counting_iterator
/*! \p make_counting_iterator creates a \p counting_iterator
* using an initial value for its \c Incrementable counter.
*
* \param x The initial value of the new \p counting_iterator's counter.
* \return A new \p counting_iterator whose counter has been initialized to \p x.
*/
template <typename Incrementable>
inline __host__ __device__
counting_iterator<Incrementable> make_counting_iterator(Incrementable x)
{
return counting_iterator<Incrementable>(x);
}
/*! \} // end fancyiterators
*/
/*! \} // end iterators
*/
} // end thrust