You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
172 lines
5.4 KiB
172 lines
5.4 KiB
/* |
|
* 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 device_new_allocator.h |
|
* \brief An allocator which allocates storage with \p device_new |
|
*/ |
|
|
|
#pragma once |
|
|
|
#include <thrust/detail/config.h> |
|
#include <thrust/device_ptr.h> |
|
#include <thrust/device_reference.h> |
|
#include <thrust/device_new.h> |
|
#include <thrust/device_delete.h> |
|
#include <limits> |
|
#include <stdexcept> |
|
|
|
namespace thrust |
|
{ |
|
|
|
/*! \addtogroup memory_management Memory Management |
|
* \addtogroup memory_management_classes Memory Management Classes |
|
* \ingroup memory_management |
|
* \{ |
|
*/ |
|
|
|
/*! \p device_new_allocator is a device memory allocator that employs the |
|
* \p device_new function for allocation. |
|
* |
|
* \see device_new |
|
* \see device_ptr |
|
* \see http://www.sgi.com/tech/stl/Allocators.html |
|
*/ |
|
template<typename T> |
|
class device_new_allocator |
|
{ |
|
public: |
|
/*! Type of element allocated, \c T. */ |
|
typedef T value_type; |
|
|
|
/*! Pointer to allocation, \c device_ptr<T>. */ |
|
typedef device_ptr<T> pointer; |
|
|
|
/*! \c const pointer to allocation, \c device_ptr<const T>. */ |
|
typedef device_ptr<const T> const_pointer; |
|
|
|
/*! Reference to allocated element, \c device_reference<T>. */ |
|
typedef device_reference<T> reference; |
|
|
|
/*! \c const reference to allocated element, \c device_reference<const T>. */ |
|
typedef device_reference<const T> const_reference; |
|
|
|
/*! Type of allocation size, \c std::size_t. */ |
|
typedef std::size_t size_type; |
|
|
|
/*! Type of allocation difference, \c pointer::difference_type. */ |
|
typedef typename pointer::difference_type difference_type; |
|
|
|
/*! The \p rebind metafunction provides the type of a \p device_new_allocator |
|
* instantiated with another type. |
|
* |
|
* \tparam U The other type to use for instantiation. |
|
*/ |
|
template<typename U> |
|
struct rebind |
|
{ |
|
/*! The typedef \p other gives the type of the rebound \p device_new_allocator. |
|
*/ |
|
typedef device_new_allocator<U> other; |
|
}; // end rebind |
|
|
|
/*! No-argument constructor has no effect. */ |
|
__host__ __device__ |
|
inline device_new_allocator() {} |
|
|
|
/*! No-argument destructor has no effect. */ |
|
__host__ __device__ |
|
inline ~device_new_allocator() {} |
|
|
|
/*! Copy constructor has no effect. */ |
|
__host__ __device__ |
|
inline device_new_allocator(device_new_allocator const&) {} |
|
|
|
/*! Constructor from other \p device_malloc_allocator has no effect. */ |
|
template<typename U> |
|
__host__ __device__ |
|
inline device_new_allocator(device_new_allocator<U> const&) {} |
|
|
|
/*! Returns the address of an allocated object. |
|
* \return <tt>&r</tt>. |
|
*/ |
|
__host__ __device__ |
|
inline pointer address(reference r) { return &r; } |
|
|
|
/*! Returns the address an allocated object. |
|
* \return <tt>&r</tt>. |
|
*/ |
|
__host__ __device__ |
|
inline const_pointer address(const_reference r) { return &r; } |
|
|
|
/*! Allocates storage for \p cnt objects. |
|
* \param cnt The number of objects to allocate. |
|
* \return A \p pointer to uninitialized storage for \p cnt objects. |
|
* \note Memory allocated by this function must be deallocated with \p deallocate. |
|
*/ |
|
__host__ |
|
inline pointer allocate(size_type cnt, |
|
const_pointer = const_pointer(static_cast<T*>(0))) |
|
{ |
|
if(cnt > this->max_size()) |
|
{ |
|
throw std::bad_alloc(); |
|
} // end if |
|
|
|
// use "::operator new" rather than keyword new |
|
return pointer(device_new<T>(cnt)); |
|
} // end allocate() |
|
|
|
/*! Deallocates storage for objects allocated with \p allocate. |
|
* \param p A \p pointer to the storage to deallocate. |
|
* \param cnt The size of the previous allocation. |
|
* \note Memory deallocated by this function must previously have been |
|
* allocated with \p allocate. |
|
*/ |
|
__host__ |
|
inline void deallocate(pointer p, size_type cnt) |
|
{ |
|
// use "::operator delete" rather than keyword delete |
|
device_delete(p); |
|
} // end deallocate() |
|
|
|
/*! Returns the largest value \c n for which <tt>allocate(n)</tt> might succeed. |
|
* \return The largest value \c n for which <tt>allocate(n)</tt> might succeed. |
|
*/ |
|
__host__ __device__ |
|
inline size_type max_size() const |
|
{ |
|
return std::numeric_limits<size_type>::max THRUST_PREVENT_MACRO_SUBSTITUTION () / sizeof(T); |
|
} // end max_size() |
|
|
|
/*! Compares against another \p device_malloc_allocator for equality. |
|
* \return \c true |
|
*/ |
|
__host__ __device__ |
|
inline bool operator==(device_new_allocator const&) { return true; } |
|
|
|
/*! Compares against another \p device_malloc_allocator for inequality. |
|
* \return \c false |
|
*/ |
|
__host__ __device__ |
|
inline bool operator!=(device_new_allocator const &a) {return !operator==(a); } |
|
}; // end device_new_allocator |
|
|
|
/*! \} |
|
*/ |
|
|
|
} // end thrust |
|
|
|
|