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

185 lines
6.2 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.
*/
#pragma once
#include <thrust/detail/config.h>
#include <thrust/iterator/iterator_adaptor.h>
#include <thrust/iterator/detail/iterator_traversal_tags.h>
#include <thrust/detail/type_traits/pointer_traits.h>
#include <thrust/detail/type_traits.h>
#include <thrust/detail/reference_forward_declaration.h>
namespace thrust
{
// declare pointer with default values of template parameters
template<typename Element, typename Tag, typename Reference = use_default, typename Derived = use_default> class pointer;
} // end thrust
// specialize std::iterator_traits to avoid problems with the name of
// pointer's constructor shadowing its nested pointer type
// do this before pointer is defined so the specialization is correctly
// used inside the definition
namespace std
{
template<typename Element, typename Tag, typename Reference, typename Derived>
struct iterator_traits<thrust::pointer<Element,Tag,Reference,Derived> >
{
private:
typedef thrust::pointer<Element,Tag,Reference,Derived> ptr;
public:
typedef typename ptr::iterator_category iterator_category;
typedef typename ptr::value_type value_type;
typedef typename ptr::difference_type difference_type;
// XXX implement this type (the result of operator->) later
typedef void pointer;
typedef typename ptr::reference reference;
}; // end iterator_traits
} // end std
namespace thrust
{
namespace detail
{
// this metafunction computes the type of iterator_adaptor thrust::pointer should inherit from
template<typename Element, typename Tag, typename Reference, typename Derived>
struct pointer_base
{
// void pointers should have no element type
// note that we remove_cv from the Element type to get the value_type
typedef typename thrust::detail::eval_if<
thrust::detail::is_void<typename thrust::detail::remove_const<Element>::type>::value,
thrust::detail::identity_<void>,
thrust::detail::remove_cv<Element>
>::type value_type;
// if no Derived type is given, just use pointer
typedef typename thrust::detail::eval_if<
thrust::detail::is_same<Derived,use_default>::value,
thrust::detail::identity_<pointer<Element,Tag,Reference,Derived> >,
thrust::detail::identity_<Derived>
>::type derived_type;
// void pointers should have no reference type
// if no Reference type is given, just use reference
typedef typename thrust::detail::eval_if<
thrust::detail::is_void<typename thrust::detail::remove_const<Element>::type>::value,
thrust::detail::identity_<void>,
thrust::detail::eval_if<
thrust::detail::is_same<Reference,use_default>::value,
thrust::detail::identity_<reference<Element,derived_type> >,
thrust::detail::identity_<Reference>
>
>::type reference_arg;
typedef thrust::iterator_adaptor<
derived_type, // pass along the type of our Derived class to iterator_adaptor
Element *, // we adapt a raw pointer
value_type, // the value type
Tag, // system tag
thrust::random_access_traversal_tag, // pointers have random access traversal
reference_arg, // pass along our Reference type
std::ptrdiff_t
> type;
}; // end pointer_base
} // end detail
// the base type for all of thrust's tagged pointers.
// for reasonable pointer-like semantics, derived types should reimplement the following:
// 1. no-argument constructor
// 2. constructor from OtherElement *
// 3. constructor from OtherPointer related by convertibility
// 4. assignment from OtherPointer related by convertibility
// These should just call the corresponding members of pointer.
template<typename Element, typename Tag, typename Reference, typename Derived>
class pointer
: public thrust::detail::pointer_base<Element,Tag,Reference,Derived>::type
{
private:
typedef typename thrust::detail::pointer_base<Element,Tag,Reference,Derived>::type super_t;
typedef typename thrust::detail::pointer_base<Element,Tag,Reference,Derived>::derived_type derived_type;
// friend iterator_core_access to give it access to dereference
friend class thrust::iterator_core_access;
__host__ __device__
typename super_t::reference dereference() const;
// don't provide access to this part of super_t's interface
using super_t::base;
using typename super_t::base_type;
public:
typedef typename super_t::base_type raw_pointer;
// constructors
__host__ __device__
pointer();
// OtherValue shall be convertible to Value
// XXX consider making the pointer implementation a template parameter which defaults to Element *
template<typename OtherElement>
__host__ __device__
explicit pointer(OtherElement *ptr);
// OtherPointer's element_type shall be convertible to Element
// OtherPointer's system shall be convertible to Tag
template<typename OtherPointer>
__host__ __device__
pointer(const OtherPointer &other,
typename thrust::detail::enable_if_pointer_is_convertible<
OtherPointer,
pointer<Element,Tag,Reference,Derived>
>::type * = 0);
// assignment
// OtherPointer's element_type shall be convertible to Element
// OtherPointer's system shall be convertible to Tag
template<typename OtherPointer>
__host__ __device__
typename thrust::detail::enable_if_pointer_is_convertible<
OtherPointer,
pointer,
derived_type &
>::type
operator=(const OtherPointer &other);
// observers
__host__ __device__
Element *get() const;
}; // end pointer
} // end thrust
#include <thrust/detail/pointer.inl>