Initial Commit

This commit is contained in:
Jordan Sherer
2018-02-08 21:28:33 -05:00
commit 678c1d3966
14352 changed files with 3176737 additions and 0 deletions
@@ -0,0 +1,280 @@
//---------------------------------------------------------------------------//
// Copyright (c) 2013 Kyle Lutz <kyle.r.lutz@gmail.com>
//
// 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
//
// See http://boostorg.github.com/compute for more information.
//---------------------------------------------------------------------------//
#ifndef BOOST_COMPUTE_ITERATOR_BUFFER_ITERATOR_HPP
#define BOOST_COMPUTE_ITERATOR_BUFFER_ITERATOR_HPP
#include <cstddef>
#include <iterator>
#include <boost/config.hpp>
#include <boost/type_traits.hpp>
#include <boost/static_assert.hpp>
#include <boost/utility/enable_if.hpp>
#include <boost/iterator/iterator_facade.hpp>
#include <boost/compute/buffer.hpp>
#include <boost/compute/detail/buffer_value.hpp>
#include <boost/compute/detail/is_buffer_iterator.hpp>
#include <boost/compute/detail/meta_kernel.hpp>
#include <boost/compute/detail/read_write_single_value.hpp>
#include <boost/compute/type_traits/is_device_iterator.hpp>
namespace boost {
namespace compute {
// forward declaration for buffer_iterator<T>
template<class T> class buffer_iterator;
namespace detail {
// helper class which defines the iterator_facade super-class
// type for buffer_iterator<T>
template<class T>
class buffer_iterator_base
{
public:
typedef ::boost::iterator_facade<
::boost::compute::buffer_iterator<T>,
T,
::std::random_access_iterator_tag,
::boost::compute::detail::buffer_value<T>
> type;
};
template<class T, class IndexExpr>
struct buffer_iterator_index_expr
{
typedef T result_type;
buffer_iterator_index_expr(const buffer &buffer,
size_t index,
const memory_object::address_space address_space,
const IndexExpr &expr)
: m_buffer(buffer),
m_index(index),
m_address_space(address_space),
m_expr(expr)
{
}
operator T() const
{
BOOST_STATIC_ASSERT_MSG(boost::is_integral<IndexExpr>::value,
"Index expression must be integral");
return buffer_value<T>(m_buffer, size_t(m_expr) * sizeof(T));
}
const buffer &m_buffer;
size_t m_index;
memory_object::address_space m_address_space;
IndexExpr m_expr;
};
template<class T, class IndexExpr>
inline meta_kernel& operator<<(meta_kernel &kernel,
const buffer_iterator_index_expr<T, IndexExpr> &expr)
{
if(expr.m_index == 0){
return kernel <<
kernel.get_buffer_identifier<T>(expr.m_buffer, expr.m_address_space) <<
'[' << expr.m_expr << ']';
}
else {
return kernel <<
kernel.get_buffer_identifier<T>(expr.m_buffer, expr.m_address_space) <<
'[' << uint_(expr.m_index) << "+(" << expr.m_expr << ")]";
}
}
} // end detail namespace
/// \class buffer_iterator
/// \brief An iterator for values in a buffer.
///
/// The buffer_iterator class iterates over values in a memory buffer on a
/// compute device. It is the most commonly used iterator in Boost.Compute
/// and is used by the \ref vector "vector<T>" and \ref array "array<T, N>"
/// container classes.
///
/// Buffer iterators store a reference to a memory buffer along with an index
/// into that memory buffer.
///
/// The buffer_iterator class allows for arbitrary OpenCL memory objects
/// (including those created outside of Boost.Compute) to be used with the
/// Boost.Compute algorithms (such as transform() and sort()). For example,
/// to reverse the contents of an OpenCL memory buffer containing a set of
/// integers:
///
/// \snippet test/test_buffer_iterator.cpp reverse_external_buffer
///
/// \see buffer, make_buffer_iterator()
template<class T>
class buffer_iterator : public detail::buffer_iterator_base<T>::type
{
public:
typedef typename detail::buffer_iterator_base<T>::type super_type;
typedef typename super_type::reference reference;
typedef typename super_type::difference_type difference_type;
buffer_iterator()
: m_index(0)
{
}
buffer_iterator(const buffer &buffer, size_t index)
: m_buffer(buffer.get(), false),
m_index(index)
{
}
buffer_iterator(const buffer_iterator<T> &other)
: m_buffer(other.m_buffer.get(), false),
m_index(other.m_index)
{
}
buffer_iterator<T>& operator=(const buffer_iterator<T> &other)
{
if(this != &other){
m_buffer.get() = other.m_buffer.get();
m_index = other.m_index;
}
return *this;
}
~buffer_iterator()
{
// set buffer to null so that its reference count will
// not be decremented when its destructor is called
m_buffer.get() = 0;
}
const buffer& get_buffer() const
{
return m_buffer;
}
size_t get_index() const
{
return m_index;
}
T read(command_queue &queue) const
{
BOOST_ASSERT(m_buffer.get());
BOOST_ASSERT(m_index < m_buffer.size() / sizeof(T));
return detail::read_single_value<T>(m_buffer, m_index, queue);
}
void write(const T &value, command_queue &queue)
{
BOOST_ASSERT(m_buffer.get());
BOOST_ASSERT(m_index < m_buffer.size() / sizeof(T));
detail::write_single_value<T>(value, m_buffer, m_index, queue);
}
/// \internal_
template<class Expr>
detail::buffer_iterator_index_expr<T, Expr>
operator[](const Expr &expr) const
{
BOOST_ASSERT(m_buffer.get());
return detail::buffer_iterator_index_expr<T, Expr>(
m_buffer, m_index, memory_object::global_memory, expr
);
}
private:
friend class ::boost::iterator_core_access;
/// \internal_
reference dereference() const
{
return detail::buffer_value<T>(m_buffer, m_index * sizeof(T));
}
/// \internal_
bool equal(const buffer_iterator<T> &other) const
{
return m_buffer.get() == other.m_buffer.get() &&
m_index == other.m_index;
}
/// \internal_
void increment()
{
m_index++;
}
/// \internal_
void decrement()
{
m_index--;
}
/// \internal_
void advance(difference_type n)
{
m_index = static_cast<size_t>(static_cast<difference_type>(m_index) + n);
}
/// \internal_
difference_type distance_to(const buffer_iterator<T> &other) const
{
return static_cast<difference_type>(other.m_index - m_index);
}
private:
const buffer m_buffer;
size_t m_index;
};
/// Creates a new \ref buffer_iterator for \p buffer at \p index.
///
/// \param buffer the \ref buffer object
/// \param index the index in the buffer
///
/// \return a \c buffer_iterator for \p buffer at \p index
template<class T>
inline buffer_iterator<T>
make_buffer_iterator(const buffer &buffer, size_t index = 0)
{
return buffer_iterator<T>(buffer, index);
}
/// \internal_ (is_device_iterator specialization for buffer_iterator)
template<class T>
struct is_device_iterator<buffer_iterator<T> > : boost::true_type {};
namespace detail {
// is_buffer_iterator specialization for buffer_iterator
template<class Iterator>
struct is_buffer_iterator<
Iterator,
typename boost::enable_if<
boost::is_same<
buffer_iterator<typename Iterator::value_type>,
typename boost::remove_const<Iterator>::type
>
>::type
> : public boost::true_type {};
} // end detail namespace
} // end compute namespace
} // end boost namespace
#endif // BOOST_COMPUTE_ITERATOR_BUFFER_ITERATOR_HPP
@@ -0,0 +1,209 @@
//---------------------------------------------------------------------------//
// Copyright (c) 2013 Kyle Lutz <kyle.r.lutz@gmail.com>
//
// 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
//
// See http://boostorg.github.com/compute for more information.
//---------------------------------------------------------------------------//
#ifndef BOOST_COMPUTE_ITERATOR_CONSTANT_BUFFER_ITERATOR_HPP
#define BOOST_COMPUTE_ITERATOR_CONSTANT_BUFFER_ITERATOR_HPP
#include <cstddef>
#include <iterator>
#include <boost/iterator/iterator_facade.hpp>
#include <boost/compute/buffer.hpp>
#include <boost/compute/iterator/buffer_iterator.hpp>
#include <boost/compute/type_traits/is_device_iterator.hpp>
namespace boost {
namespace compute {
// forward declaration for constant_buffer_iterator<T>
template<class T> class constant_buffer_iterator;
namespace detail {
// helper class which defines the iterator_facade super-class
// type for constant_buffer_iterator<T>
template<class T>
class constant_buffer_iterator_base
{
public:
typedef ::boost::iterator_facade<
::boost::compute::constant_buffer_iterator<T>,
T,
::std::random_access_iterator_tag,
::boost::compute::detail::buffer_value<T>
> type;
};
} // end detail namespace
/// \class constant_buffer_iterator
/// \brief An iterator for a buffer in the \c constant memory space.
///
/// The constant_buffer_iterator class provides an iterator for values in a
/// buffer in the \c constant memory space.
///
/// For iterating over values in the \c global memory space (the most common
/// case), use the buffer_iterator class.
///
/// \see buffer_iterator
template<class T>
class constant_buffer_iterator :
public detail::constant_buffer_iterator_base<T>::type
{
public:
typedef typename detail::constant_buffer_iterator_base<T>::type super_type;
typedef typename super_type::reference reference;
typedef typename super_type::difference_type difference_type;
constant_buffer_iterator()
: m_buffer(0),
m_index(0)
{
}
constant_buffer_iterator(const buffer &buffer, size_t index)
: m_buffer(&buffer),
m_index(index)
{
}
constant_buffer_iterator(const constant_buffer_iterator<T> &other)
: m_buffer(other.m_buffer),
m_index(other.m_index)
{
}
constant_buffer_iterator<T>& operator=(const constant_buffer_iterator<T> &other)
{
if(this != &other){
m_buffer = other.m_buffer;
m_index = other.m_index;
}
return *this;
}
~constant_buffer_iterator()
{
}
const buffer& get_buffer() const
{
return *m_buffer;
}
size_t get_index() const
{
return m_index;
}
T read(command_queue &queue) const
{
BOOST_ASSERT(m_buffer && m_buffer->get());
BOOST_ASSERT(m_index < m_buffer->size() / sizeof(T));
return detail::read_single_value<T>(m_buffer, m_index, queue);
}
void write(const T &value, command_queue &queue)
{
BOOST_ASSERT(m_buffer && m_buffer->get());
BOOST_ASSERT(m_index < m_buffer->size() / sizeof(T));
detail::write_single_value<T>(m_buffer, m_index, queue);
}
template<class Expr>
detail::buffer_iterator_index_expr<T, Expr>
operator[](const Expr &expr) const
{
BOOST_ASSERT(m_buffer);
BOOST_ASSERT(m_buffer->get());
return detail::buffer_iterator_index_expr<T, Expr>(
*m_buffer, m_index, memory_object::constant_memory, expr
);
}
private:
friend class ::boost::iterator_core_access;
reference dereference() const
{
return detail::buffer_value<T>(*m_buffer, m_index);
}
bool equal(const constant_buffer_iterator<T> &other) const
{
return m_buffer == other.m_buffer && m_index == other.m_index;
}
void increment()
{
m_index++;
}
void decrement()
{
m_index--;
}
void advance(difference_type n)
{
m_index = static_cast<size_t>(static_cast<difference_type>(m_index) + n);
}
difference_type distance_to(const constant_buffer_iterator<T> &other) const
{
return static_cast<difference_type>(other.m_index - m_index);
}
private:
const buffer *m_buffer;
size_t m_index;
};
/// Creates a new constant_buffer_iterator for \p buffer at \p index.
///
/// \param buffer the \ref buffer object
/// \param index the index in the buffer
///
/// \return a \c constant_buffer_iterator for \p buffer at \p index
template<class T>
inline constant_buffer_iterator<T>
make_constant_buffer_iterator(const buffer &buffer, size_t index = 0)
{
return constant_buffer_iterator<T>(buffer, index);
}
/// \internal_ (is_device_iterator specialization for constant_buffer_iterator)
template<class T>
struct is_device_iterator<constant_buffer_iterator<T> > : boost::true_type {};
namespace detail {
// is_buffer_iterator specialization for constant_buffer_iterator
template<class Iterator>
struct is_buffer_iterator<
Iterator,
typename boost::enable_if<
boost::is_same<
constant_buffer_iterator<typename Iterator::value_type>,
typename boost::remove_const<Iterator>::type
>
>::type
> : public boost::true_type {};
} // end detail namespace
} // end compute namespace
} // end boost namespace
#endif // BOOST_COMPUTE_ITERATOR_CONSTANT_BUFFER_ITERATOR_HPP
@@ -0,0 +1,171 @@
//---------------------------------------------------------------------------//
// Copyright (c) 2013 Kyle Lutz <kyle.r.lutz@gmail.com>
//
// 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
//
// See http://boostorg.github.com/compute for more information.
//---------------------------------------------------------------------------//
#ifndef BOOST_COMPUTE_ITERATOR_CONSTANT_ITERATOR_HPP
#define BOOST_COMPUTE_ITERATOR_CONSTANT_ITERATOR_HPP
#include <string>
#include <cstddef>
#include <iterator>
#include <boost/config.hpp>
#include <boost/iterator/iterator_facade.hpp>
#include <boost/compute/detail/meta_kernel.hpp>
#include <boost/compute/type_traits/is_device_iterator.hpp>
namespace boost {
namespace compute {
// forward declaration for constant_iterator<T>
template<class T> class constant_iterator;
namespace detail {
// helper class which defines the iterator_facade super-class
// type for constant_iterator<T>
template<class T>
class constant_iterator_base
{
public:
typedef ::boost::iterator_facade<
::boost::compute::constant_iterator<T>,
T,
::std::random_access_iterator_tag
> type;
};
} // end detail namespace
/// \class constant_iterator
/// \brief An iterator with a constant value.
///
/// The constant_iterator class provides an iterator which returns a constant
/// value when dereferenced.
///
/// For example, this could be used to implement the fill() algorithm in terms
/// of the copy() algorithm by copying from a range of constant iterators:
///
/// \snippet test/test_constant_iterator.cpp fill_with_copy
///
/// \see make_constant_iterator()
template<class T>
class constant_iterator : public detail::constant_iterator_base<T>::type
{
public:
typedef typename detail::constant_iterator_base<T>::type super_type;
typedef typename super_type::reference reference;
typedef typename super_type::difference_type difference_type;
constant_iterator(const T &value, size_t index = 0)
: m_value(value),
m_index(index)
{
}
constant_iterator(const constant_iterator<T> &other)
: m_value(other.m_value),
m_index(other.m_index)
{
}
constant_iterator<T>& operator=(const constant_iterator<T> &other)
{
if(this != &other){
m_value = other.m_value;
m_index = other.m_index;
}
return *this;
}
~constant_iterator()
{
}
size_t get_index() const
{
return m_index;
}
/// \internal_
template<class Expr>
detail::meta_kernel_literal<T> operator[](const Expr &expr) const
{
(void) expr;
return detail::meta_kernel::make_lit<T>(m_value);
}
private:
friend class ::boost::iterator_core_access;
/// \internal_
reference dereference() const
{
return m_value;
}
/// \internal_
bool equal(const constant_iterator<T> &other) const
{
return m_value == other.m_value && m_index == other.m_index;
}
/// \internal_
void increment()
{
m_index++;
}
/// \internal_
void decrement()
{
m_index--;
}
/// \internal_
void advance(difference_type n)
{
m_index = static_cast<size_t>(static_cast<difference_type>(m_index) + n);
}
/// \internal_
difference_type distance_to(const constant_iterator<T> &other) const
{
return static_cast<difference_type>(other.m_index - m_index);
}
private:
T m_value;
size_t m_index;
};
/// Returns a new constant_iterator with \p value at \p index.
///
/// \param value the constant value
/// \param index the iterators index
///
/// \return a \c constant_iterator with \p value
template<class T>
inline constant_iterator<T>
make_constant_iterator(const T &value, size_t index = 0)
{
return constant_iterator<T>(value, index);
}
/// \internal_ (is_device_iterator specialization for constant_iterator)
template<class T>
struct is_device_iterator<constant_iterator<T> > : boost::true_type {};
} // end compute namespace
} // end boost namespace
#endif // BOOST_COMPUTE_ITERATOR_CONSTANT_ITERATOR_HPP
@@ -0,0 +1,185 @@
//---------------------------------------------------------------------------//
// Copyright (c) 2013 Kyle Lutz <kyle.r.lutz@gmail.com>
//
// 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
//
// See http://boostorg.github.com/compute for more information.
//---------------------------------------------------------------------------//
#ifndef BOOST_COMPUTE_ITERATOR_COUNTING_ITERATOR_HPP
#define BOOST_COMPUTE_ITERATOR_COUNTING_ITERATOR_HPP
#include <string>
#include <cstddef>
#include <iterator>
#include <boost/config.hpp>
#include <boost/iterator/iterator_facade.hpp>
#include <boost/compute/detail/meta_kernel.hpp>
#include <boost/compute/type_traits/is_device_iterator.hpp>
namespace boost {
namespace compute {
// forward declaration for counting_iterator<T>
template<class T> class counting_iterator;
namespace detail {
// helper class which defines the iterator_facade super-class
// type for counting_iterator<T>
template<class T>
class counting_iterator_base
{
public:
typedef ::boost::iterator_facade<
::boost::compute::counting_iterator<T>,
T,
::std::random_access_iterator_tag
> type;
};
template<class T, class IndexExpr>
struct counting_iterator_index_expr
{
typedef T result_type;
counting_iterator_index_expr(const T &init, const IndexExpr &expr)
: m_init(init),
m_expr(expr)
{
}
const T &m_init;
IndexExpr m_expr;
};
template<class T, class IndexExpr>
inline meta_kernel& operator<<(meta_kernel &kernel,
const counting_iterator_index_expr<T, IndexExpr> &expr)
{
return kernel << '(' << expr.m_init << '+' << expr.m_expr << ')';
}
} // end detail namespace
/// \class counting_iterator
/// \brief The counting_iterator class implements a counting iterator.
///
/// A counting iterator returns an internal value (initialized with \p init)
/// which is incremented each time the iterator is incremented.
///
/// For example, this could be used to implement the iota() algorithm in terms
/// of the copy() algorithm by copying from a range of counting iterators:
///
/// \snippet test/test_counting_iterator.cpp iota_with_copy
///
/// \see make_counting_iterator()
template<class T>
class counting_iterator : public detail::counting_iterator_base<T>::type
{
public:
typedef typename detail::counting_iterator_base<T>::type super_type;
typedef typename super_type::reference reference;
typedef typename super_type::difference_type difference_type;
counting_iterator(const T &init)
: m_init(init)
{
}
counting_iterator(const counting_iterator<T> &other)
: m_init(other.m_init)
{
}
counting_iterator<T>& operator=(const counting_iterator<T> &other)
{
if(this != &other){
m_init = other.m_init;
}
return *this;
}
~counting_iterator()
{
}
size_t get_index() const
{
return 0;
}
template<class Expr>
detail::counting_iterator_index_expr<T, Expr>
operator[](const Expr &expr) const
{
return detail::counting_iterator_index_expr<T, Expr>(m_init, expr);
}
private:
friend class ::boost::iterator_core_access;
reference dereference() const
{
return m_init;
}
bool equal(const counting_iterator<T> &other) const
{
return m_init == other.m_init;
}
void increment()
{
m_init++;
}
void decrement()
{
m_init--;
}
void advance(difference_type n)
{
m_init += static_cast<T>(n);
}
difference_type distance_to(const counting_iterator<T> &other) const
{
return difference_type(other.m_init) - difference_type(m_init);
}
private:
T m_init;
};
/// Returns a new counting_iterator starting at \p init.
///
/// \param init the initial value
///
/// \return a counting_iterator with \p init.
///
/// For example, to create a counting iterator which returns unsigned integers
/// and increments from one:
/// \code
/// auto iter = make_counting_iterator<uint_>(1);
/// \endcode
template<class T>
inline counting_iterator<T> make_counting_iterator(const T &init)
{
return counting_iterator<T>(init);
}
/// \internal_ (is_device_iterator specialization for counting_iterator)
template<class T>
struct is_device_iterator<counting_iterator<T> > : boost::true_type {};
} // end compute namespace
} // end boost namespace
#endif // BOOST_COMPUTE_ITERATOR_COUNTING_ITERATOR_HPP
@@ -0,0 +1,52 @@
//---------------------------------------------------------------------------//
// Copyright (c) 2013 Kyle Lutz <kyle.r.lutz@gmail.com>
//
// 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
//
// See http://boostorg.github.com/compute for more information.
//---------------------------------------------------------------------------//
#ifndef BOOST_COMPUTE_ITERATOR_DETAIL_GET_BASE_ITERATOR_BUFFER_HPP
#define BOOST_COMPUTE_ITERATOR_DETAIL_GET_BASE_ITERATOR_BUFFER_HPP
namespace boost {
namespace compute {
namespace detail {
// returns the buffer for an iterator adaptor's base iterator if
// it exists, otherwise returns a null buffer object.
template<class Iterator>
inline const buffer&
get_base_iterator_buffer(const Iterator &iter,
typename boost::enable_if<
is_buffer_iterator<
typename Iterator::base_type
>
>::type* = 0)
{
return iter.base().get_buffer();
}
template<class Iterator>
inline const buffer&
get_base_iterator_buffer(const Iterator &iter,
typename boost::disable_if<
is_buffer_iterator<
typename Iterator::base_type
>
>::type* = 0)
{
(void) iter;
static buffer null_buffer;
return null_buffer;
}
} // end detail namespace
} // end compute namespace
} // end boost namespace
#endif // BOOST_COMPUTE_ITERATOR_DETAIL_GET_BASE_ITERATOR_BUFFER_HPP
@@ -0,0 +1,170 @@
//---------------------------------------------------------------------------//
// Copyright (c) 2013-2014 Kyle Lutz <kyle.r.lutz@gmail.com>
//
// 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
//
// See http://boostorg.github.com/compute for more information.
//---------------------------------------------------------------------------//
#ifndef BOOST_COMPUTE_ITERATOR_DISCARD_ITERATOR_HPP
#define BOOST_COMPUTE_ITERATOR_DISCARD_ITERATOR_HPP
#include <string>
#include <cstddef>
#include <iterator>
#include <boost/config.hpp>
#include <boost/iterator/iterator_facade.hpp>
#include <boost/compute/detail/meta_kernel.hpp>
#include <boost/compute/type_traits/is_device_iterator.hpp>
namespace boost {
namespace compute {
// forward declaration for discard_iterator
class discard_iterator;
namespace detail {
// helper class which defines the iterator_facade super-class
// type for discard_iterator
struct discard_iterator_base
{
typedef ::boost::iterator_facade<
::boost::compute::discard_iterator,
void,
::std::random_access_iterator_tag,
void *
> type;
};
template<class IndexExpr>
struct discard_iterator_index_expr
{
typedef void result_type;
discard_iterator_index_expr(const IndexExpr &expr)
: m_expr(expr)
{
}
IndexExpr m_expr;
};
template<class IndexExpr>
inline meta_kernel& operator<<(meta_kernel &kernel,
const discard_iterator_index_expr<IndexExpr> &expr)
{
(void) expr;
return kernel;
}
} // end detail namespace
/// \class discard_iterator
/// \brief An iterator which discards all values written to it.
///
/// \see make_discard_iterator(), constant_iterator
class discard_iterator : public detail::discard_iterator_base::type
{
public:
typedef detail::discard_iterator_base::type super_type;
typedef super_type::reference reference;
typedef super_type::difference_type difference_type;
discard_iterator(size_t index = 0)
: m_index(index)
{
}
discard_iterator(const discard_iterator &other)
: m_index(other.m_index)
{
}
discard_iterator& operator=(const discard_iterator &other)
{
if(this != &other){
m_index = other.m_index;
}
return *this;
}
~discard_iterator()
{
}
/// \internal_
template<class Expr>
detail::discard_iterator_index_expr<Expr>
operator[](const Expr &expr) const
{
return detail::discard_iterator_index_expr<Expr>(expr);
}
private:
friend class ::boost::iterator_core_access;
/// \internal_
reference dereference() const
{
return 0;
}
/// \internal_
bool equal(const discard_iterator &other) const
{
return m_index == other.m_index;
}
/// \internal_
void increment()
{
m_index++;
}
/// \internal_
void decrement()
{
m_index--;
}
/// \internal_
void advance(difference_type n)
{
m_index = static_cast<size_t>(static_cast<difference_type>(m_index) + n);
}
/// \internal_
difference_type distance_to(const discard_iterator &other) const
{
return static_cast<difference_type>(other.m_index - m_index);
}
private:
size_t m_index;
};
/// Returns a new discard_iterator with \p index.
///
/// \param index the index of the iterator
///
/// \return a \c discard_iterator at \p index
inline discard_iterator make_discard_iterator(size_t index = 0)
{
return discard_iterator(index);
}
/// internal_ (is_device_iterator specialization for discard_iterator)
template<>
struct is_device_iterator<discard_iterator> : boost::true_type {};
} // end compute namespace
} // end boost namespace
#endif // BOOST_COMPUTE_ITERATOR_DISCARD_ITERATOR_HPP
@@ -0,0 +1,186 @@
//---------------------------------------------------------------------------//
// Copyright (c) 2013 Kyle Lutz <kyle.r.lutz@gmail.com>
//
// 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
//
// See http://boostorg.github.com/compute for more information.
//---------------------------------------------------------------------------//
#ifndef BOOST_COMPUTE_ITERATOR_FUNCTION_INPUT_ITERATOR_HPP
#define BOOST_COMPUTE_ITERATOR_FUNCTION_INPUT_ITERATOR_HPP
#include <cstddef>
#include <iterator>
#include <boost/config.hpp>
#include <boost/iterator/iterator_facade.hpp>
#include <boost/compute/detail/meta_kernel.hpp>
#include <boost/compute/type_traits/is_device_iterator.hpp>
#include <boost/compute/type_traits/result_of.hpp>
namespace boost {
namespace compute {
// forward declaration for function_input_iterator<Function>
template<class Function> class function_input_iterator;
namespace detail {
// helper class which defines the iterator_facade super-class
// type for function_input_iterator<Function>
template<class Function>
class function_input_iterator_base
{
public:
typedef ::boost::iterator_facade<
::boost::compute::function_input_iterator<Function>,
typename ::boost::compute::result_of<Function()>::type,
::std::random_access_iterator_tag,
typename ::boost::compute::result_of<Function()>::type
> type;
};
template<class Function>
struct function_input_iterator_expr
{
typedef typename ::boost::compute::result_of<Function()>::type result_type;
function_input_iterator_expr(const Function &function)
: m_function(function)
{
}
Function m_function;
};
template<class Function>
inline meta_kernel& operator<<(meta_kernel &kernel,
const function_input_iterator_expr<Function> &expr)
{
return kernel << expr.m_function();
}
} // end detail namespace
/// \class function_input_iterator
/// \brief Iterator which returns the result of a function when dereferenced
///
/// For example:
///
/// \snippet test/test_function_input_iterator.cpp generate_42
///
/// \see make_function_input_iterator()
template<class Function>
class function_input_iterator :
public detail::function_input_iterator_base<Function>::type
{
public:
typedef typename detail::function_input_iterator_base<Function>::type super_type;
typedef typename super_type::reference reference;
typedef typename super_type::difference_type difference_type;
typedef Function function;
function_input_iterator(const Function &function, size_t index = 0)
: m_function(function),
m_index(index)
{
}
function_input_iterator(const function_input_iterator<Function> &other)
: m_function(other.m_function),
m_index(other.m_index)
{
}
function_input_iterator<Function>&
operator=(const function_input_iterator<Function> &other)
{
if(this != &other){
m_function = other.m_function;
m_index = other.m_index;
}
return *this;
}
~function_input_iterator()
{
}
size_t get_index() const
{
return m_index;
}
template<class Expr>
detail::function_input_iterator_expr<Function>
operator[](const Expr &expr) const
{
(void) expr;
return detail::function_input_iterator_expr<Function>(m_function);
}
private:
friend class ::boost::iterator_core_access;
reference dereference() const
{
return reference();
}
bool equal(const function_input_iterator<Function> &other) const
{
return m_function == other.m_function && m_index == other.m_index;
}
void increment()
{
m_index++;
}
void decrement()
{
m_index--;
}
void advance(difference_type n)
{
m_index = static_cast<size_t>(static_cast<difference_type>(m_index) + n);
}
difference_type
distance_to(const function_input_iterator<Function> &other) const
{
return static_cast<difference_type>(other.m_index - m_index);
}
private:
Function m_function;
size_t m_index;
};
/// Returns a function_input_iterator with \p function.
///
/// \param function function to execute when dereferenced
/// \param index index of the iterator
///
/// \return a \c function_input_iterator with \p function
template<class Function>
inline function_input_iterator<Function>
make_function_input_iterator(const Function &function, size_t index = 0)
{
return function_input_iterator<Function>(function, index);
}
/// \internal_ (is_device_iterator specialization for function_input_iterator)
template<class Function>
struct is_device_iterator<function_input_iterator<Function> > : boost::true_type {};
} // end compute namespace
} // end boost namespace
#endif // BOOST_COMPUTE_ITERATOR_FUNCTION_INPUT_ITERATOR_HPP
@@ -0,0 +1,192 @@
//---------------------------------------------------------------------------//
// Copyright (c) 2013 Kyle Lutz <kyle.r.lutz@gmail.com>
//
// 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
//
// See http://boostorg.github.com/compute for more information.
//---------------------------------------------------------------------------//
#ifndef BOOST_COMPUTE_ITERATOR_PERMUTATION_ITERATOR_HPP
#define BOOST_COMPUTE_ITERATOR_PERMUTATION_ITERATOR_HPP
#include <string>
#include <cstddef>
#include <iterator>
#include <boost/config.hpp>
#include <boost/iterator/iterator_adaptor.hpp>
#include <boost/compute/functional.hpp>
#include <boost/compute/detail/meta_kernel.hpp>
#include <boost/compute/detail/is_buffer_iterator.hpp>
#include <boost/compute/detail/read_write_single_value.hpp>
#include <boost/compute/iterator/detail/get_base_iterator_buffer.hpp>
#include <boost/compute/type_traits/is_device_iterator.hpp>
namespace boost {
namespace compute {
// forward declaration for transform_iterator
template<class ElementIterator, class IndexIterator>
class permutation_iterator;
namespace detail {
// helper class which defines the iterator_adaptor super-class
// type for permutation_iterator
template<class ElementIterator, class IndexIterator>
class permutation_iterator_base
{
public:
typedef ::boost::iterator_adaptor<
::boost::compute::permutation_iterator<ElementIterator, IndexIterator>,
ElementIterator
> type;
};
template<class ElementIterator, class IndexIterator, class IndexExpr>
struct permutation_iterator_access_expr
{
typedef typename std::iterator_traits<ElementIterator>::value_type result_type;
permutation_iterator_access_expr(const ElementIterator &e,
const IndexIterator &i,
const IndexExpr &expr)
: m_element_iter(e),
m_index_iter(i),
m_expr(expr)
{
}
ElementIterator m_element_iter;
IndexIterator m_index_iter;
IndexExpr m_expr;
};
template<class ElementIterator, class IndexIterator, class IndexExpr>
inline meta_kernel& operator<<(meta_kernel &kernel,
const permutation_iterator_access_expr<ElementIterator,
IndexIterator,
IndexExpr> &expr)
{
return kernel << expr.m_element_iter[expr.m_index_iter[expr.m_expr]];
}
} // end detail namespace
/// \class permutation_iterator
/// \brief The permutation_iterator class provides a permuation iterator
///
/// A permutation iterator iterates over a value range and an index range. When
/// dereferenced, it returns the value from the value range using the current
/// index from the index range.
///
/// For example, to reverse a range using the copy() algorithm and a permutation
/// sequence:
///
/// \snippet test/test_permutation_iterator.cpp reverse_range
///
/// \see make_permutation_iterator()
template<class ElementIterator, class IndexIterator>
class permutation_iterator
: public detail::permutation_iterator_base<ElementIterator,
IndexIterator>::type
{
public:
typedef typename
detail::permutation_iterator_base<ElementIterator,
IndexIterator>::type super_type;
typedef typename super_type::value_type value_type;
typedef typename super_type::reference reference;
typedef typename super_type::base_type base_type;
typedef typename super_type::difference_type difference_type;
typedef IndexIterator index_iterator;
permutation_iterator(ElementIterator e, IndexIterator i)
: super_type(e),
m_map(i)
{
}
permutation_iterator(const permutation_iterator<ElementIterator,
IndexIterator> &other)
: super_type(other),
m_map(other.m_map)
{
}
permutation_iterator<ElementIterator, IndexIterator>&
operator=(const permutation_iterator<ElementIterator,
IndexIterator> &other)
{
if(this != &other){
super_type::operator=(other);
m_map = other.m_map;
}
return *this;
}
~permutation_iterator()
{
}
size_t get_index() const
{
return super_type::base().get_index();
}
const buffer& get_buffer() const
{
return detail::get_base_iterator_buffer(*this);
}
template<class IndexExpr>
detail::permutation_iterator_access_expr<ElementIterator,
IndexIterator,
IndexExpr>
operator[](const IndexExpr &expr) const
{
return detail::permutation_iterator_access_expr<ElementIterator,
IndexIterator,
IndexExpr>(super_type::base(),
m_map,
expr);
}
private:
friend class ::boost::iterator_core_access;
reference dereference() const
{
return reference();
}
private:
IndexIterator m_map;
};
/// Returns a permutation_iterator for \p e using indices from \p i.
///
/// \param e the element range iterator
/// \param i the index range iterator
///
/// \return a \c permutation_iterator for \p e using \p i
template<class ElementIterator, class IndexIterator>
inline permutation_iterator<ElementIterator, IndexIterator>
make_permutation_iterator(ElementIterator e, IndexIterator i)
{
return permutation_iterator<ElementIterator, IndexIterator>(e, i);
}
/// \internal_ (is_device_iterator specialization for permutation_iterator)
template<class ElementIterator, class IndexIterator>
struct is_device_iterator<
permutation_iterator<ElementIterator, IndexIterator> > : boost::true_type {};
} // end compute namespace
} // end boost namespace
#endif // BOOST_COMPUTE_ITERATOR_PERMUTATION_ITERATOR_HPP
@@ -0,0 +1,227 @@
//---------------------------------------------------------------------------//
// Copyright (c) 2013 Kyle Lutz <kyle.r.lutz@gmail.com>
//
// 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
//
// See http://boostorg.github.com/compute for more information.
//---------------------------------------------------------------------------//
#ifndef BOOST_COMPUTE_ITERATOR_TRANSFORM_ITERATOR_HPP
#define BOOST_COMPUTE_ITERATOR_TRANSFORM_ITERATOR_HPP
#include <cstddef>
#include <iterator>
#include <boost/config.hpp>
#include <boost/iterator/iterator_adaptor.hpp>
#include <boost/compute/functional.hpp>
#include <boost/compute/detail/meta_kernel.hpp>
#include <boost/compute/detail/is_buffer_iterator.hpp>
#include <boost/compute/detail/read_write_single_value.hpp>
#include <boost/compute/iterator/detail/get_base_iterator_buffer.hpp>
#include <boost/compute/type_traits/is_device_iterator.hpp>
#include <boost/compute/type_traits/result_of.hpp>
namespace boost {
namespace compute {
// forward declaration for transform_iterator
template<class InputIterator, class UnaryFunction>
class transform_iterator;
namespace detail {
// meta-function returning the value_type for a transform_iterator
template<class InputIterator, class UnaryFunction>
struct make_transform_iterator_value_type
{
typedef typename std::iterator_traits<InputIterator>::value_type value_type;
typedef typename boost::compute::result_of<UnaryFunction(value_type)>::type type;
};
// helper class which defines the iterator_adaptor super-class
// type for transform_iterator
template<class InputIterator, class UnaryFunction>
class transform_iterator_base
{
public:
typedef ::boost::iterator_adaptor<
::boost::compute::transform_iterator<InputIterator, UnaryFunction>,
InputIterator,
typename make_transform_iterator_value_type<InputIterator, UnaryFunction>::type,
typename std::iterator_traits<InputIterator>::iterator_category,
typename make_transform_iterator_value_type<InputIterator, UnaryFunction>::type
> type;
};
template<class InputIterator, class UnaryFunction, class IndexExpr>
struct transform_iterator_index_expr
{
typedef typename
make_transform_iterator_value_type<
InputIterator,
UnaryFunction
>::type result_type;
transform_iterator_index_expr(const InputIterator &input_iter,
const UnaryFunction &transform_expr,
const IndexExpr &index_expr)
: m_input_iter(input_iter),
m_transform_expr(transform_expr),
m_index_expr(index_expr)
{
}
InputIterator m_input_iter;
UnaryFunction m_transform_expr;
IndexExpr m_index_expr;
};
template<class InputIterator, class UnaryFunction, class IndexExpr>
inline meta_kernel& operator<<(meta_kernel &kernel,
const transform_iterator_index_expr<InputIterator,
UnaryFunction,
IndexExpr> &expr)
{
return kernel << expr.m_transform_expr(expr.m_input_iter[expr.m_index_expr]);
}
} // end detail namespace
/// \class transform_iterator
/// \brief A transform iterator adaptor.
///
/// The transform_iterator adaptor applies a unary function to each element
/// produced from the underlying iterator when dereferenced.
///
/// For example, to copy from an input range to an output range while taking
/// the absolute value of each element:
///
/// \snippet test/test_transform_iterator.cpp copy_abs
///
/// \see buffer_iterator, make_transform_iterator()
template<class InputIterator, class UnaryFunction>
class transform_iterator :
public detail::transform_iterator_base<InputIterator, UnaryFunction>::type
{
public:
typedef typename
detail::transform_iterator_base<InputIterator,
UnaryFunction>::type super_type;
typedef typename super_type::value_type value_type;
typedef typename super_type::reference reference;
typedef typename super_type::base_type base_type;
typedef typename super_type::difference_type difference_type;
typedef UnaryFunction unary_function;
transform_iterator(InputIterator iterator, UnaryFunction transform)
: super_type(iterator),
m_transform(transform)
{
}
transform_iterator(const transform_iterator<InputIterator,
UnaryFunction> &other)
: super_type(other.base()),
m_transform(other.m_transform)
{
}
transform_iterator<InputIterator, UnaryFunction>&
operator=(const transform_iterator<InputIterator,
UnaryFunction> &other)
{
if(this != &other){
super_type::operator=(other);
m_transform = other.m_transform;
}
return *this;
}
~transform_iterator()
{
}
size_t get_index() const
{
return super_type::base().get_index();
}
const buffer& get_buffer() const
{
return detail::get_base_iterator_buffer(*this);
}
template<class IndexExpression>
detail::transform_iterator_index_expr<InputIterator, UnaryFunction, IndexExpression>
operator[](const IndexExpression &expr) const
{
return detail::transform_iterator_index_expr<InputIterator,
UnaryFunction,
IndexExpression>(super_type::base(),
m_transform,
expr);
}
private:
friend class ::boost::iterator_core_access;
reference dereference() const
{
const context &context = super_type::base().get_buffer().get_context();
command_queue queue(context, context.get_device());
detail::meta_kernel k("read");
size_t output_arg = k.add_arg<value_type *>(memory_object::global_memory, "output");
k << "*output = " << m_transform(super_type::base()[k.lit(0)]) << ";";
kernel kernel = k.compile(context);
buffer output_buffer(context, sizeof(value_type));
kernel.set_arg(output_arg, output_buffer);
queue.enqueue_task(kernel);
return detail::read_single_value<value_type>(output_buffer, queue);
}
private:
UnaryFunction m_transform;
};
/// Returns a transform_iterator for \p iterator with \p transform.
///
/// \param iterator the underlying iterator
/// \param transform the unary transform function
///
/// \return a \c transform_iterator for \p iterator with \p transform
///
/// For example, to create an iterator which returns the square-root of each
/// value in a \c vector<int>:
/// \code
/// auto sqrt_iterator = make_transform_iterator(vec.begin(), sqrt<int>());
/// \endcode
template<class InputIterator, class UnaryFunction>
inline transform_iterator<InputIterator, UnaryFunction>
make_transform_iterator(InputIterator iterator, UnaryFunction transform)
{
return transform_iterator<InputIterator,
UnaryFunction>(iterator, transform);
}
/// \internal_ (is_device_iterator specialization for transform_iterator)
template<class InputIterator, class UnaryFunction>
struct is_device_iterator<
transform_iterator<InputIterator, UnaryFunction> > : boost::true_type {};
} // end compute namespace
} // end boost namespace
#endif // BOOST_COMPUTE_ITERATOR_TRANSFORM_ITERATOR_HPP
@@ -0,0 +1,316 @@
//---------------------------------------------------------------------------//
// Copyright (c) 2013 Kyle Lutz <kyle.r.lutz@gmail.com>
//
// 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
//
// See http://boostorg.github.com/compute for more information.
//---------------------------------------------------------------------------//
#ifndef BOOST_COMPUTE_ITERATOR_ZIP_ITERATOR_HPP
#define BOOST_COMPUTE_ITERATOR_ZIP_ITERATOR_HPP
#include <cstddef>
#include <iterator>
#include <boost/config.hpp>
#include <boost/fusion/algorithm/iteration/for_each.hpp>
#include <boost/iterator/iterator_facade.hpp>
#include <boost/mpl/back_inserter.hpp>
#include <boost/mpl/transform.hpp>
#include <boost/mpl/vector.hpp>
#include <boost/preprocessor/repetition.hpp>
#include <boost/tuple/tuple.hpp>
#include <boost/tuple/tuple_comparison.hpp>
#include <boost/compute/config.hpp>
#include <boost/compute/functional.hpp>
#include <boost/compute/detail/meta_kernel.hpp>
#include <boost/compute/detail/mpl_vector_to_tuple.hpp>
#include <boost/compute/types/tuple.hpp>
#include <boost/compute/type_traits/is_device_iterator.hpp>
#include <boost/compute/type_traits/type_name.hpp>
namespace boost {
namespace compute {
// forward declaration for zip_iterator
template<class IteratorTuple>
class zip_iterator;
namespace detail {
namespace mpl = boost::mpl;
// meta-function returning the value_type for an iterator
template<class Iterator>
struct make_iterator_value_type
{
typedef typename std::iterator_traits<Iterator>::value_type type;
};
// meta-function returning the value_type for a zip_iterator
template<class IteratorTuple>
struct make_zip_iterator_value_type
{
typedef typename
detail::mpl_vector_to_tuple<
typename mpl::transform<
IteratorTuple,
make_iterator_value_type<mpl::_1>,
mpl::back_inserter<mpl::vector<> >
>::type
>::type type;
};
// helper class which defines the iterator_facade super-class
// type for zip_iterator
template<class IteratorTuple>
class zip_iterator_base
{
public:
typedef ::boost::iterator_facade<
::boost::compute::zip_iterator<IteratorTuple>,
typename make_zip_iterator_value_type<IteratorTuple>::type,
::std::random_access_iterator_tag,
typename make_zip_iterator_value_type<IteratorTuple>::type
> type;
};
template<class IteratorTuple, class IndexExpr>
struct zip_iterator_index_expr
{
typedef typename
make_zip_iterator_value_type<IteratorTuple>::type
result_type;
zip_iterator_index_expr(const IteratorTuple &iterators,
const IndexExpr &index_expr)
: m_iterators(iterators),
m_index_expr(index_expr)
{
}
IteratorTuple m_iterators;
IndexExpr m_index_expr;
};
/// \internal_
#define BOOST_COMPUTE_PRINT_ELEM(z, n, unused) \
BOOST_PP_EXPR_IF(n, << ", ") \
<< boost::get<n>(expr.m_iterators)[expr.m_index_expr]
/// \internal_
#define BOOST_COMPUTE_PRINT_ZIP_IDX(z, n, unused) \
template<BOOST_PP_ENUM_PARAMS(n, class Iterator), class IndexExpr> \
inline meta_kernel& operator<<( \
meta_kernel &kernel, \
const zip_iterator_index_expr< \
boost::tuple<BOOST_PP_ENUM_PARAMS(n, Iterator)>, \
IndexExpr \
> &expr) \
{ \
typedef typename \
boost::tuple<BOOST_PP_ENUM_PARAMS(n, Iterator)> \
tuple_type; \
typedef typename \
make_zip_iterator_value_type<tuple_type>::type \
value_type; \
kernel.inject_type<value_type>(); \
return kernel \
<< "(" << type_name<value_type>() << ")" \
<< "{ " \
BOOST_PP_REPEAT(n, BOOST_COMPUTE_PRINT_ELEM, ~) \
<< "}"; \
}
BOOST_PP_REPEAT_FROM_TO(1, BOOST_COMPUTE_MAX_ARITY, BOOST_COMPUTE_PRINT_ZIP_IDX, ~)
#undef BOOST_COMPUTE_PRINT_ZIP_IDX
#undef BOOST_COMPUTE_PRINT_ELEM
struct iterator_advancer
{
iterator_advancer(size_t n)
: m_distance(n)
{
}
template<class Iterator>
void operator()(Iterator &i) const
{
std::advance(i, m_distance);
}
size_t m_distance;
};
template<class Iterator>
void increment_iterator(Iterator &i)
{
i++;
}
template<class Iterator>
void decrement_iterator(Iterator &i)
{
i--;
}
} // end detail namespace
/// \class zip_iterator
/// \brief A zip iterator adaptor.
///
/// The zip_iterator class combines values from multiple input iterators. When
/// dereferenced it returns a tuple containing each value at the current
/// position in each input range.
///
/// \see make_zip_iterator()
template<class IteratorTuple>
class zip_iterator : public detail::zip_iterator_base<IteratorTuple>::type
{
public:
typedef typename
detail::zip_iterator_base<IteratorTuple>::type
super_type;
typedef typename super_type::value_type value_type;
typedef typename super_type::reference reference;
typedef typename super_type::difference_type difference_type;
typedef IteratorTuple iterator_tuple;
zip_iterator(IteratorTuple iterators)
: m_iterators(iterators)
{
}
zip_iterator(const zip_iterator<IteratorTuple> &other)
: m_iterators(other.m_iterators)
{
}
zip_iterator<IteratorTuple>&
operator=(const zip_iterator<IteratorTuple> &other)
{
if(this != &other){
super_type::operator=(other);
m_iterators = other.m_iterators;
}
return *this;
}
~zip_iterator()
{
}
const IteratorTuple& get_iterator_tuple() const
{
return m_iterators;
}
template<class IndexExpression>
detail::zip_iterator_index_expr<IteratorTuple, IndexExpression>
operator[](const IndexExpression &expr) const
{
return detail::zip_iterator_index_expr<IteratorTuple,
IndexExpression>(m_iterators,
expr);
}
private:
friend class ::boost::iterator_core_access;
reference dereference() const
{
return reference();
}
bool equal(const zip_iterator<IteratorTuple> &other) const
{
return m_iterators == other.m_iterators;
}
void increment()
{
boost::fusion::for_each(m_iterators, detail::increment_iterator);
}
void decrement()
{
boost::fusion::for_each(m_iterators, detail::decrement_iterator);
}
void advance(difference_type n)
{
boost::fusion::for_each(m_iterators, detail::iterator_advancer(n));
}
difference_type distance_to(const zip_iterator<IteratorTuple> &other) const
{
return std::distance(boost::get<0>(m_iterators),
boost::get<0>(other.m_iterators));
}
private:
IteratorTuple m_iterators;
};
/// Creates a zip_iterator for \p iterators.
///
/// \param iterators a tuple of input iterators to zip together
///
/// \return a \c zip_iterator for \p iterators
///
/// For example, to zip together iterators from three vectors (\c a, \c b, and
/// \p c):
/// \code
/// auto zipped = boost::compute::make_zip_iterator(
/// boost::make_tuple(a.begin(), b.begin(), c.begin())
/// );
/// \endcode
template<class IteratorTuple>
inline zip_iterator<IteratorTuple>
make_zip_iterator(IteratorTuple iterators)
{
return zip_iterator<IteratorTuple>(iterators);
}
/// \internal_ (is_device_iterator specialization for zip_iterator)
template<class IteratorTuple>
struct is_device_iterator<zip_iterator<IteratorTuple> > : boost::true_type {};
namespace detail {
// get<N>() specialization for zip_iterator
/// \internal_
#define BOOST_COMPUTE_ZIP_GET_N(z, n, unused) \
template<size_t N, class IteratorTuple, class IndexExpr, \
BOOST_PP_ENUM_PARAMS(n, class T)> \
inline meta_kernel& \
operator<<(meta_kernel &kernel, \
const invoked_get< \
N, \
zip_iterator_index_expr<IteratorTuple, IndexExpr>, \
boost::tuple<BOOST_PP_ENUM_PARAMS(n, T)> \
> &expr) \
{ \
typedef typename boost::tuple<BOOST_PP_ENUM_PARAMS(n, T)> Tuple; \
typedef typename boost::tuples::element<N, Tuple>::type T; \
BOOST_STATIC_ASSERT(N < size_t(boost::tuples::length<Tuple>::value)); \
kernel.inject_type<T>(); \
return kernel \
<< boost::get<N>(expr.m_arg.m_iterators)[expr.m_arg.m_index_expr]; \
}
BOOST_PP_REPEAT_FROM_TO(1, BOOST_COMPUTE_MAX_ARITY, BOOST_COMPUTE_ZIP_GET_N, ~)
#undef BOOST_COMPUTE_ZIP_GET_N
} // end detail namespace
} // end compute namespace
} // end boost namespace
#endif // BOOST_COMPUTE_ITERATOR_ZIP_ITERATOR_HPP