Boost logo

Boost-Commit :

Subject: [Boost-commit] svn:boost r81229 - in trunk: boost/smart_ptr boost/smart_ptr/detail libs/smart_ptr/test
From: glenfe_at_[hidden]
Date: 2012-11-07 09:42:11


Author: glenfe
Date: 2012-11-07 09:42:10 EST (Wed, 07 Nov 2012)
New Revision: 81229
URL: http://svn.boost.org/trac/boost/changeset/81229

Log:
Add support for make_shared of array of arrays. Correctly destroy elements and construct elements for the variadic template constructor variants.
Added:
   trunk/boost/smart_ptr/detail/array_helper.hpp (contents, props changed)
   trunk/libs/smart_ptr/test/allocate_shared_arrays_test.cpp (contents, props changed)
   trunk/libs/smart_ptr/test/make_shared_arrays_test.cpp (contents, props changed)
Text files modified:
   trunk/boost/smart_ptr/allocate_shared_array.hpp | 2 +-
   trunk/boost/smart_ptr/detail/array_deleter.hpp | 12 +++++-------
   trunk/boost/smart_ptr/make_shared_array.hpp | 2 +-
   3 files changed, 7 insertions(+), 9 deletions(-)

Modified: trunk/boost/smart_ptr/allocate_shared_array.hpp
==============================================================================
--- trunk/boost/smart_ptr/allocate_shared_array.hpp (original)
+++ trunk/boost/smart_ptr/allocate_shared_array.hpp 2012-11-07 09:42:10 EST (Wed, 07 Nov 2012)
@@ -9,7 +9,7 @@
 #ifndef BOOST_SMART_PTR_ALLOCATE_SHARED_ARRAY_HPP
 #define BOOST_SMART_PTR_ALLOCATE_SHARED_ARRAY_HPP
 
-#include <boost/shared_ptr.hpp>
+#include <boost/smart_ptr/shared_ptr.hpp>
 #include <boost/smart_ptr/detail/allocate_array_helper.hpp>
 #include <boost/smart_ptr/detail/array_deleter.hpp>
 #include <boost/smart_ptr/detail/sp_if_array.hpp>

Modified: trunk/boost/smart_ptr/detail/array_deleter.hpp
==============================================================================
--- trunk/boost/smart_ptr/detail/array_deleter.hpp (original)
+++ trunk/boost/smart_ptr/detail/array_deleter.hpp 2012-11-07 09:42:10 EST (Wed, 07 Nov 2012)
@@ -10,6 +10,7 @@
 #define BOOST_SMART_PTR_DETAIL_ARRAY_DELETER_HPP
 
 #include <boost/config.hpp>
+#include <boost/smart_ptr/detail/array_helper.hpp>
 #include <cstddef>
 
 namespace boost {
@@ -25,23 +26,20 @@
             }
             void construct(T* memory, std::size_t count) {
                 for (object = memory; size < count; size++) {
- void* p1 = object + size;
- ::new(p1) T();
+ array_helper<T>::create(object[size]);
                 }
             }
 #if defined(BOOST_HAS_VARIADIC_TMPL) && defined(BOOST_HAS_RVALUE_REFS)
             template<typename... Args>
             void construct(T* memory, std::size_t count, Args&&... args) {
                 for (object = memory; size < count; size++) {
- void* p1 = object + size;
- ::new(p1) T(args...);
+ array_helper<T>::create(object[size], args...);
                 }
             }
 #endif
             void construct_noinit(T* memory, std::size_t count) {
                 for (object = memory; size < count; size++) {
- void* p1 = object + size;
- ::new(p1) T;
+ array_helper<T>::create_noinit(object[size]);
                 }
             }
             void operator()(T*) {
@@ -50,7 +48,7 @@
         private:
             void destroy() {
                 while (size > 0) {
- object[--size].~T();
+ array_helper<T>::destroy(object[--size]);
                 }
             }
             std::size_t size;

Added: trunk/boost/smart_ptr/detail/array_helper.hpp
==============================================================================
--- (empty file)
+++ trunk/boost/smart_ptr/detail/array_helper.hpp 2012-11-07 09:42:10 EST (Wed, 07 Nov 2012)
@@ -0,0 +1,64 @@
+/*
+ * Copyright (c) 2012 Glen Joseph Fernandes
+ * glenfe at live dot com
+ *
+ * Distributed under the Boost Software License,
+ * Version 1.0. (See accompanying file LICENSE_1_0.txt
+ * or copy at http://boost.org/LICENSE_1_0.txt)
+ */
+#ifndef BOOST_SMART_PTR_DETAIL_ARRAY_HELPER_HPP
+#define BOOST_SMART_PTR_DETAIL_ARRAY_HELPER_HPP
+
+namespace boost {
+ namespace detail {
+ template<typename T>
+ struct array_helper {
+ static void destroy(T& value) {
+ value.~T();
+ }
+#if defined(BOOST_HAS_VARIADIC_TMPL) && defined(BOOST_HAS_RVALUE_REFS)
+ template<typename... Args>
+ static void create(T& value, Args... args) {
+ void* p1 = &value;
+ ::new(p1) T(args...);
+ }
+#endif
+ };
+#if !defined(BOOST_NO_TEMPLATE_PARTIAL_SPECIALIZATION)
+ template<typename T, size_t N>
+ struct array_helper<T[N]> {
+ static void create(T value[N]) {
+ void* p1 = &value;
+ ::new(p1) T[N]();
+ }
+ static void create_noinit(T value[N]) {
+ void* p1 = &value;
+ ::new(p1) T[N];
+ }
+ static void destroy(T value[N]) {
+ array_helper<T>::destroy(value[N-1]);
+ array_helper<T[N-1]>::destroy(value);
+ }
+#if defined(BOOST_HAS_VARIADIC_TMPL) && defined(BOOST_HAS_RVALUE_REFS)
+ template<typename... Args>
+ static void create(T value[N], Args... args) {
+ array_helper<T[N-1]>::create(value, args);
+ array_helper<T>::create(value[N-1], args);
+ }
+#endif
+ };
+ template<typename T>
+ struct array_helper<T[0]> {
+ static void destroy(T[]) {
+ }
+#if defined(BOOST_HAS_VARIADIC_TMPL) && defined(BOOST_HAS_RVALUE_REFS)
+ template<typename... Args>
+ static void create(T[], Args...) {
+ }
+#endif
+ };
+#endif
+ }
+}
+
+#endif

Modified: trunk/boost/smart_ptr/make_shared_array.hpp
==============================================================================
--- trunk/boost/smart_ptr/make_shared_array.hpp (original)
+++ trunk/boost/smart_ptr/make_shared_array.hpp 2012-11-07 09:42:10 EST (Wed, 07 Nov 2012)
@@ -9,7 +9,7 @@
 #ifndef BOOST_SMART_PTR_MAKE_SHARED_ARRAY_HPP
 #define BOOST_SMART_PTR_MAKE_SHARED_ARRAY_HPP
 
-#include <boost/shared_ptr.hpp>
+#include <boost/smart_ptr/shared_ptr.hpp>
 #include <boost/smart_ptr/detail/array_deleter.hpp>
 #include <boost/smart_ptr/detail/make_array_helper.hpp>
 #include <boost/smart_ptr/detail/sp_if_array.hpp>

Added: trunk/libs/smart_ptr/test/allocate_shared_arrays_test.cpp
==============================================================================
--- (empty file)
+++ trunk/libs/smart_ptr/test/allocate_shared_arrays_test.cpp 2012-11-07 09:42:10 EST (Wed, 07 Nov 2012)
@@ -0,0 +1,75 @@
+/*
+ * Copyright (c) 2012 Glen Joseph Fernandes
+ * glenfe at live dot com
+ *
+ * Distributed under the Boost Software License,
+ * Version 1.0. (See accompanying file LICENSE_1_0.txt
+ * or copy at http://boost.org/LICENSE_1_0.txt)
+ */
+#include <boost/detail/lightweight_test.hpp>
+#include <boost/smart_ptr/allocate_shared_array.hpp>
+
+class type {
+public:
+ static unsigned int instances;
+ explicit type(int = 0, int = 0)
+ : member() {
+ instances++;
+ }
+ ~type() {
+ instances--;
+ }
+private:
+ type(const type&);
+ type& operator=(const type&);
+ double member;
+};
+
+unsigned int type::instances = 0;
+
+int main() {
+ {
+ boost::shared_ptr<int[][2][2]> a1 = boost::allocate_shared<int[][2][2]>(std::allocator<int>(), 2);
+ BOOST_TEST(a1.get() != 0);
+ BOOST_TEST(a1.use_count() == 1);
+ BOOST_TEST(a1[0][0][1] == 0);
+ BOOST_TEST(a1[0][1][0] == 0);
+ BOOST_TEST(a1[1][0][0] == 0);
+ }
+ {
+ boost::shared_ptr<const int[][2][2]> a1 = boost::allocate_shared<const int[][2][2]>(std::allocator<int>(), 2);
+ BOOST_TEST(a1.get() != 0);
+ BOOST_TEST(a1.use_count() == 1);
+ BOOST_TEST(a1[0][0][1] == 0);
+ BOOST_TEST(a1[0][1][0] == 0);
+ BOOST_TEST(a1[1][0][0] == 0);
+ }
+ BOOST_TEST(type::instances == 0);
+ {
+ boost::shared_ptr<type[][2][2]> a1 = boost::allocate_shared<type[][2][2]>(std::allocator<type>(), 2);
+ BOOST_TEST(a1.get() != 0);
+ BOOST_TEST(a1.use_count() == 1);
+ BOOST_TEST(type::instances == 8);
+ a1.reset();
+ BOOST_TEST(type::instances == 0);
+ }
+ BOOST_TEST(type::instances == 0);
+ {
+ boost::shared_ptr<const type[][2][2]> a1 = boost::allocate_shared<const type[][2][2]>(std::allocator<type>(), 2);
+ BOOST_TEST(a1.get() != 0);
+ BOOST_TEST(a1.use_count() == 1);
+ BOOST_TEST(type::instances == 8);
+ }
+#if defined(BOOST_HAS_VARIADIC_TMPL) && defined(BOOST_HAS_RVALUE_REFS)
+ BOOST_TEST(type::instances == 0);
+ {
+ boost::shared_ptr<type[][2][2]> a1 = boost::allocate_shared<type[][2][2]>(std::allocator<type>(), 2, 1, 5);
+ BOOST_TEST(a1.get() != 0);
+ BOOST_TEST(a1.use_count() == 1);
+ BOOST_TEST(type::instances == 8);
+ a1.reset();
+ BOOST_TEST(type::instances == 0);
+ }
+#endif
+ return boost::report_errors();
+}

Added: trunk/libs/smart_ptr/test/make_shared_arrays_test.cpp
==============================================================================
--- (empty file)
+++ trunk/libs/smart_ptr/test/make_shared_arrays_test.cpp 2012-11-07 09:42:10 EST (Wed, 07 Nov 2012)
@@ -0,0 +1,101 @@
+/*
+ * Copyright (c) 2012 Glen Joseph Fernandes
+ * glenfe at live dot com
+ *
+ * Distributed under the Boost Software License,
+ * Version 1.0. (See accompanying file LICENSE_1_0.txt
+ * or copy at http://boost.org/LICENSE_1_0.txt)
+ */
+#include <boost/detail/lightweight_test.hpp>
+#include <boost/smart_ptr/make_shared_array.hpp>
+
+class type {
+public:
+ static unsigned int instances;
+ explicit type(int = 0, int = 0)
+ : member() {
+ instances++;
+ }
+ ~type() {
+ instances--;
+ }
+private:
+ type(const type&);
+ type& operator=(const type&);
+ double member;
+};
+
+unsigned int type::instances = 0;
+
+int main() {
+ {
+ boost::shared_ptr<int[][2][2]> a1 = boost::make_shared<int[][2][2]>(2);
+ BOOST_TEST(a1.get() != 0);
+ BOOST_TEST(a1.use_count() == 1);
+ BOOST_TEST(a1[0][0][1] == 0);
+ BOOST_TEST(a1[0][1][0] == 0);
+ BOOST_TEST(a1[1][0][0] == 0);
+ }
+ {
+ boost::shared_ptr<const int[][2][2]> a1 = boost::make_shared<const int[][2][2]>(2);
+ BOOST_TEST(a1.get() != 0);
+ BOOST_TEST(a1.use_count() == 1);
+ BOOST_TEST(a1[0][0][1] == 0);
+ BOOST_TEST(a1[0][1][0] == 0);
+ BOOST_TEST(a1[1][0][0] == 0);
+ }
+ BOOST_TEST(type::instances == 0);
+ {
+ boost::shared_ptr<type[][2][2]> a1 = boost::make_shared<type[][2][2]>(2);
+ BOOST_TEST(a1.get() != 0);
+ BOOST_TEST(a1.use_count() == 1);
+ BOOST_TEST(type::instances == 8);
+ a1.reset();
+ BOOST_TEST(type::instances == 0);
+ }
+ BOOST_TEST(type::instances == 0);
+ {
+ boost::shared_ptr<const type[][2][2]> a1 = boost::make_shared<const type[][2][2]>(2);
+ BOOST_TEST(a1.get() != 0);
+ BOOST_TEST(a1.use_count() == 1);
+ BOOST_TEST(type::instances == 8);
+ }
+#if defined(BOOST_HAS_VARIADIC_TMPL) && defined(BOOST_HAS_RVALUE_REFS)
+ BOOST_TEST(type::instances == 0);
+ {
+ boost::shared_ptr<type[][2][2]> a1 = boost::make_shared<type[][2][2]>(2, 1, 5);
+ BOOST_TEST(a1.get() != 0);
+ BOOST_TEST(a1.use_count() == 1);
+ BOOST_TEST(type::instances == 8);
+ a1.reset();
+ BOOST_TEST(type::instances == 0);
+ }
+#endif
+ {
+ boost::shared_ptr<int[][2][2]> a1 = boost::make_shared_noinit<int[][2][2]>(2);
+ BOOST_TEST(a1.get() != 0);
+ BOOST_TEST(a1.use_count() == 1);
+ }
+ {
+ boost::shared_ptr<const int[][2][2]> a1 = boost::make_shared_noinit<const int[][2][2]>(2);
+ BOOST_TEST(a1.get() != 0);
+ BOOST_TEST(a1.use_count() == 1);
+ }
+ BOOST_TEST(type::instances == 0);
+ {
+ boost::shared_ptr<type[][2][2]> a1 = boost::make_shared_noinit<type[][2][2]>(2);
+ BOOST_TEST(a1.get() != 0);
+ BOOST_TEST(a1.use_count() == 1);
+ BOOST_TEST(type::instances == 8);
+ a1.reset();
+ BOOST_TEST(type::instances == 0);
+ }
+ BOOST_TEST(type::instances == 0);
+ {
+ boost::shared_ptr<const type[][2][2]> a1 = boost::make_shared_noinit<const type[][2][2]>(2);
+ BOOST_TEST(a1.get() != 0);
+ BOOST_TEST(a1.use_count() == 1);
+ BOOST_TEST(type::instances == 8);
+ }
+ return boost::report_errors();
+}


Boost-Commit list run by bdawes at acm.org, david.abrahams at rcn.com, gregod at cs.rpi.edu, cpdaniel at pacbell.net, john at johnmaddock.co.uk