| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  | #ifndef BOOST_DETAIL_SHARED_ARRAY_NMT_HPP_INCLUDED
 | 
					
						
							|  |  |  | #define BOOST_DETAIL_SHARED_ARRAY_NMT_HPP_INCLUDED
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //  detail/shared_array_nmt.hpp - shared_array.hpp without member templates
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //  (C) Copyright Greg Colvin and Beman Dawes 1998, 1999.
 | 
					
						
							|  |  |  | //  Copyright (c) 2001, 2002 Peter Dimov
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2004-08-19 15:19:17 +00:00
										 |  |  | //  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)
 | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  | //
 | 
					
						
							| 
									
										
										
										
											2002-02-04 11:15:40 +00:00
										 |  |  | //  See http://www.boost.org/libs/smart_ptr/shared_array.htm for documentation.
 | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  | //
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <boost/assert.hpp>
 | 
					
						
							|  |  |  | #include <boost/checked_delete.hpp>
 | 
					
						
							| 
									
										
										
										
											2002-08-14 12:27:22 +00:00
										 |  |  | #include <boost/throw_exception.hpp>
 | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  | #include <boost/detail/atomic_count.hpp>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-08-14 12:27:22 +00:00
										 |  |  | #include <cstddef>          // for std::ptrdiff_t
 | 
					
						
							|  |  |  | #include <algorithm>        // for std::swap
 | 
					
						
							|  |  |  | #include <functional>       // for std::less
 | 
					
						
							|  |  |  | #include <new>              // for std::bad_alloc
 | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace boost | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template<class T> class shared_array | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     typedef detail::atomic_count count_type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     typedef T element_type; | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |     explicit shared_array(T * p = 0): px(p) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2002-08-14 12:27:22 +00:00
										 |  |  | #ifndef BOOST_NO_EXCEPTIONS
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  |         try  // prevent leak if new throws
 | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             pn = new count_type(1); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         catch(...) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2002-08-14 12:27:22 +00:00
										 |  |  |             boost::checked_array_delete(p); | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  |             throw; | 
					
						
							| 
									
										
										
										
											2002-08-14 12:27:22 +00:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         pn = new count_type(1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if(pn == 0) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             boost::checked_array_delete(p); | 
					
						
							|  |  |  |             boost::throw_exception(std::bad_alloc()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2002-08-14 12:27:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  |     ~shared_array() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if(--*pn == 0) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2002-08-14 12:27:22 +00:00
										 |  |  |             boost::checked_array_delete(px); | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  |             delete pn; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     shared_array(shared_array const & r) : px(r.px)  // never throws
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         pn = r.pn; | 
					
						
							|  |  |  |         ++*pn; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2002-08-14 12:27:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  |     shared_array & operator=(shared_array const & r) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         shared_array(r).swap(*this); | 
					
						
							|  |  |  |         return *this; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2002-08-14 12:27:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  |     void reset(T * p = 0) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2002-01-24 19:16:12 +00:00
										 |  |  |         BOOST_ASSERT(p == 0 || p != px); | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  |         shared_array(p).swap(*this); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2002-08-14 12:27:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  |     T * get() const  // never throws
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return px; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     T & operator[](std::ptrdiff_t i) const  // never throws
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         BOOST_ASSERT(px != 0); | 
					
						
							|  |  |  |         BOOST_ASSERT(i >= 0); | 
					
						
							|  |  |  |         return px[i]; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2002-08-14 12:27:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  |     long use_count() const  // never throws
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return *pn; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool unique() const  // never throws
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return *pn == 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2002-08-14 12:27:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  |     void swap(shared_array<T> & other)  // never throws
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         std::swap(px, other.px); | 
					
						
							|  |  |  |         std::swap(pn, other.pn); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2002-08-14 12:27:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2002-08-14 12:27:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-01-22 13:38:52 +00:00
										 |  |  |     T * px;            // contained pointer
 | 
					
						
							|  |  |  |     count_type * pn;   // ptr to reference counter
 | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  | };  // shared_array
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template<class T, class U> inline bool operator==(shared_array<T> const & a, shared_array<U> const & b) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return a.get() == b.get(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template<class T, class U> inline bool operator!=(shared_array<T> const & a, shared_array<U> const & b) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return a.get() != b.get(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template<class T> inline bool operator<(shared_array<T> const & a, shared_array<T> const & b) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return std::less<T*>()(a.get(), b.get()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template<class T> void swap(shared_array<T> & a, shared_array<T> & b) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     a.swap(b); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace boost
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif  // #ifndef BOOST_DETAIL_SHARED_ARRAY_NMT_HPP_INCLUDED
 |