2012-11-06 14:17:32 +00:00
|
|
|
/*
|
2014-01-23 20:40:46 -08:00
|
|
|
* Copyright (c) 2012-2014 Glen Joseph Fernandes
|
2012-11-06 14:17:32 +00:00
|
|
|
* glenfe at live dot com
|
|
|
|
*
|
2012-12-03 05:56:17 +00:00
|
|
|
* Distributed under the Boost Software License,
|
|
|
|
* Version 1.0. (See accompanying file LICENSE_1_0.txt
|
2012-11-06 14:17:32 +00:00
|
|
|
* or copy at http://boost.org/LICENSE_1_0.txt)
|
|
|
|
*/
|
|
|
|
#ifndef BOOST_SMART_PTR_DETAIL_ARRAY_DELETER_HPP
|
|
|
|
#define BOOST_SMART_PTR_DETAIL_ARRAY_DELETER_HPP
|
|
|
|
|
2014-02-04 00:29:51 -08:00
|
|
|
#include <boost/config.hpp>
|
2014-01-28 02:27:49 -08:00
|
|
|
#include <boost/smart_ptr/detail/array_traits.hpp>
|
2014-02-04 08:43:36 -08:00
|
|
|
#include <boost/smart_ptr/detail/as_pair.hpp>
|
2014-02-04 00:29:51 -08:00
|
|
|
#include <boost/type_traits/has_trivial_constructor.hpp>
|
|
|
|
#include <boost/type_traits/has_trivial_destructor.hpp>
|
|
|
|
#if !defined(BOOST_NO_CXX11_ALLOCATOR)
|
|
|
|
#include <memory>
|
|
|
|
#endif
|
2012-11-06 14:17:32 +00:00
|
|
|
|
|
|
|
namespace boost {
|
|
|
|
namespace detail {
|
|
|
|
template<typename T>
|
2014-02-04 00:29:51 -08:00
|
|
|
struct ms_deleter_base;
|
2014-01-23 20:40:46 -08:00
|
|
|
|
2012-11-28 07:32:30 +00:00
|
|
|
template<typename T>
|
2014-02-04 00:29:51 -08:00
|
|
|
struct ms_deleter_base<T[]> {
|
|
|
|
ms_deleter_base(std::size_t size_)
|
2014-02-02 22:00:08 -08:00
|
|
|
: size(size_) {
|
2012-11-10 01:33:29 +00:00
|
|
|
}
|
2014-01-23 20:40:46 -08:00
|
|
|
|
2012-11-06 14:17:32 +00:00
|
|
|
std::size_t size;
|
|
|
|
};
|
2014-01-23 20:40:46 -08:00
|
|
|
|
2012-11-28 06:07:45 +00:00
|
|
|
template<typename T, std::size_t N>
|
2014-02-04 00:29:51 -08:00
|
|
|
struct ms_deleter_base<T[N]> {
|
2014-02-02 22:00:08 -08:00
|
|
|
enum {
|
|
|
|
size = N
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2014-02-04 00:29:51 -08:00
|
|
|
template<typename T, typename A>
|
|
|
|
class as_deleter
|
|
|
|
: ms_deleter_base<T> {
|
|
|
|
using ms_deleter_base<T>::size;
|
2014-02-02 22:00:08 -08:00
|
|
|
|
2012-11-28 06:07:45 +00:00
|
|
|
public:
|
2014-02-02 22:00:08 -08:00
|
|
|
typedef typename array_inner<T>::type type;
|
2014-02-04 00:29:51 -08:00
|
|
|
|
2014-02-04 08:43:36 -08:00
|
|
|
as_deleter(const A& allocator)
|
|
|
|
: pair(allocator, 0) {
|
2012-11-28 06:07:45 +00:00
|
|
|
}
|
2014-01-23 20:40:46 -08:00
|
|
|
|
2014-02-04 08:43:36 -08:00
|
|
|
as_deleter(const A& allocator, std::size_t size_)
|
2014-02-04 00:29:51 -08:00
|
|
|
: ms_deleter_base<T>(size_),
|
2014-02-04 08:43:36 -08:00
|
|
|
pair(allocator, 0) {
|
2014-02-04 00:29:51 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void init(type* memory) {
|
|
|
|
value_init(memory);
|
2014-02-04 08:43:36 -08:00
|
|
|
pair.data = memory;
|
2014-02-04 00:29:51 -08:00
|
|
|
}
|
2014-02-02 22:00:08 -08:00
|
|
|
|
2014-02-04 00:29:51 -08:00
|
|
|
template<std::size_t N>
|
|
|
|
void init(type* memory, const type* value) {
|
|
|
|
value_init<N>(memory, value);
|
2014-02-04 08:43:36 -08:00
|
|
|
pair.data = memory;
|
2014-02-04 00:29:51 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void operator()(const void*) {
|
2014-02-04 08:43:36 -08:00
|
|
|
if (pair.data) {
|
|
|
|
destroy(pair.data, size);
|
2012-12-07 01:53:35 +00:00
|
|
|
}
|
2012-11-28 06:07:45 +00:00
|
|
|
}
|
2014-01-23 20:40:46 -08:00
|
|
|
|
2014-02-04 00:29:51 -08:00
|
|
|
private:
|
2014-02-04 08:43:36 -08:00
|
|
|
|
2014-02-04 00:29:51 -08:00
|
|
|
#if !defined(BOOST_NO_CXX11_ALLOCATOR)
|
|
|
|
typedef typename std::allocator_traits<A>::
|
2014-02-04 08:43:36 -08:00
|
|
|
template rebind_alloc<type> TA;
|
2014-02-04 00:29:51 -08:00
|
|
|
typedef typename std::allocator_traits<A>::
|
2014-02-04 08:43:36 -08:00
|
|
|
template rebind_traits<type> TT;
|
|
|
|
#else
|
|
|
|
typedef typename A::
|
|
|
|
template rebind<type>::other TA;
|
2014-02-04 00:29:51 -08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
void destroy(type*, std::size_t, boost::true_type) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void destroy(type* memory, std::size_t n, boost::false_type) {
|
|
|
|
for (std::size_t i = n; i > 0;) {
|
|
|
|
#if !defined(BOOST_NO_CXX11_ALLOCATOR)
|
2014-02-04 08:43:36 -08:00
|
|
|
TT::destroy(pair, &memory[--i]);
|
2014-02-04 00:29:51 -08:00
|
|
|
#else
|
|
|
|
memory[--i].~type();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void destroy(type* memory, std::size_t n) {
|
|
|
|
boost::has_trivial_destructor<type> tag;
|
|
|
|
destroy(memory, n, tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
void value_init(type* memory, boost::true_type) {
|
|
|
|
for (std::size_t i = 0; i < size; i++) {
|
|
|
|
#if !defined(BOOST_NO_CXX11_ALLOCATOR)
|
2014-02-04 08:43:36 -08:00
|
|
|
TT::construct(pair, memory + i);
|
2014-02-04 00:29:51 -08:00
|
|
|
#else
|
|
|
|
void* p1 = memory + i;
|
|
|
|
::new(p1) type();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void value_init(type* memory, boost::false_type) {
|
|
|
|
#if !defined(BOOST_NO_EXCEPTIONS)
|
|
|
|
std::size_t i = 0;
|
|
|
|
try {
|
|
|
|
for (; i < size; i++) {
|
|
|
|
#if !defined(BOOST_NO_CXX11_ALLOCATOR)
|
2014-02-04 08:43:36 -08:00
|
|
|
TT::construct(pair, memory + i);
|
2014-02-04 00:29:51 -08:00
|
|
|
#else
|
|
|
|
void* p1 = memory + i;
|
|
|
|
::new(p1) type();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
} catch (...) {
|
|
|
|
destroy(memory, i);
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
for (std::size_t i = 0; i < size; i++) {
|
|
|
|
#if !defined(BOOST_NO_CXX11_ALLOCATOR)
|
2014-02-04 08:43:36 -08:00
|
|
|
TT::construct(pair, memory + i);
|
2014-02-04 00:29:51 -08:00
|
|
|
#else
|
|
|
|
void* p1 = memory + i;
|
|
|
|
::new(p1) type();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void value_init(type* memory) {
|
|
|
|
boost::has_trivial_default_constructor<type> tag;
|
|
|
|
value_init(memory, tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<std::size_t N>
|
|
|
|
void value_init(type* memory, const type* list) {
|
|
|
|
#if !defined(BOOST_NO_EXCEPTIONS)
|
|
|
|
std::size_t i = 0;
|
|
|
|
try {
|
|
|
|
for (; i < size; i++) {
|
|
|
|
#if !defined(BOOST_NO_CXX11_ALLOCATOR)
|
2014-02-04 08:43:36 -08:00
|
|
|
TT::construct(pair, memory + i, list[i % N]);
|
2014-02-04 00:29:51 -08:00
|
|
|
#else
|
|
|
|
void* p1 = memory + i;
|
|
|
|
::new(p1) type(list[i % N]);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
} catch (...) {
|
|
|
|
destroy(memory, i);
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
for (std::size_t i = 0; i < size; i++) {
|
|
|
|
#if !defined(BOOST_NO_CXX11_ALLOCATOR)
|
|
|
|
TT::construct(allocator, memory + i, list[i % N]);
|
|
|
|
#else
|
|
|
|
void* p1 = memory + i;
|
|
|
|
::new(p1) type(list[i % N]);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-02-04 08:43:36 -08:00
|
|
|
as_pair<TA, type*> pair;
|
2014-02-04 00:29:51 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
class ms_deleter
|
|
|
|
: ms_deleter_base<T> {
|
|
|
|
using ms_deleter_base<T>::size;
|
|
|
|
|
|
|
|
public:
|
|
|
|
typedef typename array_inner<T>::type type;
|
|
|
|
|
|
|
|
ms_deleter()
|
|
|
|
: object(0) {
|
|
|
|
}
|
|
|
|
|
|
|
|
ms_deleter(std::size_t size_)
|
|
|
|
: ms_deleter_base<T>(size_),
|
|
|
|
object(0) {
|
|
|
|
}
|
|
|
|
|
2014-02-02 22:00:08 -08:00
|
|
|
void init(type* memory) {
|
2014-02-04 00:29:51 -08:00
|
|
|
value_init(memory);
|
2012-12-07 01:53:35 +00:00
|
|
|
object = memory;
|
2012-11-28 06:07:45 +00:00
|
|
|
}
|
2014-01-23 20:40:46 -08:00
|
|
|
|
2014-02-02 22:00:08 -08:00
|
|
|
template<std::size_t N>
|
|
|
|
void init(type* memory, const type* value) {
|
2014-02-04 00:29:51 -08:00
|
|
|
value_init<N>(memory, value);
|
2012-12-07 01:53:35 +00:00
|
|
|
object = memory;
|
2012-11-28 06:07:45 +00:00
|
|
|
}
|
2014-01-23 20:40:46 -08:00
|
|
|
|
2014-02-02 22:00:08 -08:00
|
|
|
void noinit(type* memory) {
|
2014-02-04 00:29:51 -08:00
|
|
|
default_init(memory);
|
2012-12-07 01:53:35 +00:00
|
|
|
object = memory;
|
2012-11-28 06:07:45 +00:00
|
|
|
}
|
2014-01-23 20:40:46 -08:00
|
|
|
|
2012-11-28 06:07:45 +00:00
|
|
|
void operator()(const void*) {
|
|
|
|
if (object) {
|
2014-02-04 00:29:51 -08:00
|
|
|
destroy(object, size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
void destroy(type*, std::size_t, boost::true_type) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void destroy(type* memory, std::size_t n, boost::false_type) {
|
|
|
|
for (std::size_t i = n; i > 0;) {
|
|
|
|
memory[--i].~type();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void destroy(type* memory, std::size_t n) {
|
|
|
|
boost::has_trivial_destructor<type> tag;
|
|
|
|
destroy(memory, n, tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
void value_init(type* memory, boost::true_type) {
|
|
|
|
for (std::size_t i = 0; i < size; i++) {
|
|
|
|
void* p1 = memory + i;
|
|
|
|
::new(p1) type();
|
2012-11-28 06:07:45 +00:00
|
|
|
}
|
|
|
|
}
|
2014-01-23 20:40:46 -08:00
|
|
|
|
2014-02-04 00:29:51 -08:00
|
|
|
void value_init(type* memory, boost::false_type) {
|
|
|
|
#if !defined(BOOST_NO_EXCEPTIONS)
|
|
|
|
std::size_t i = 0;
|
|
|
|
try {
|
|
|
|
for (; i < size; i++) {
|
|
|
|
void* p1 = memory + i;
|
|
|
|
::new(p1) type();
|
|
|
|
}
|
|
|
|
} catch (...) {
|
|
|
|
destroy(memory, i);
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
for (std::size_t i = 0; i < size; i++) {
|
|
|
|
void* p1 = memory + i;
|
|
|
|
::new(p1) type();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void value_init(type* memory) {
|
|
|
|
boost::has_trivial_default_constructor<type> tag;
|
|
|
|
value_init(memory, tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
void default_init(type*, boost::true_type) {
|
|
|
|
}
|
|
|
|
|
|
|
|
void default_init(type* memory, boost::false_type) {
|
|
|
|
#if !defined(BOOST_NO_EXCEPTIONS)
|
|
|
|
std::size_t i = 0;
|
|
|
|
try {
|
|
|
|
for (; i < size; i++) {
|
|
|
|
void* p1 = memory + i;
|
|
|
|
::new(p1) type;
|
|
|
|
}
|
|
|
|
} catch (...) {
|
|
|
|
destroy(memory, i);
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
for (std::size_t i = 0; i < size; i++) {
|
|
|
|
void* p1 = memory + i;
|
|
|
|
::new(p1) type;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void default_init(type* memory) {
|
|
|
|
boost::has_trivial_default_constructor<type> tag;
|
|
|
|
default_init(memory, tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<std::size_t N>
|
|
|
|
void value_init(type* memory, const type* list) {
|
|
|
|
#if !defined(BOOST_NO_EXCEPTIONS)
|
|
|
|
std::size_t i = 0;
|
|
|
|
try {
|
|
|
|
for (; i < size; i++) {
|
|
|
|
void* p1 = memory + i;
|
|
|
|
::new(p1) type(list[i % N]);
|
|
|
|
}
|
|
|
|
} catch (...) {
|
|
|
|
destroy(memory, i);
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
for (std::size_t i = 0; i < size; i++) {
|
|
|
|
void* p1 = memory + i;
|
|
|
|
::new(p1) type(list[i % N]);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2012-12-07 01:53:35 +00:00
|
|
|
private:
|
2014-02-02 22:00:08 -08:00
|
|
|
type* object;
|
2014-02-04 00:29:51 -08:00
|
|
|
};
|
2012-11-06 14:17:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|