#ifndef BOOST_DETAIL_SHARED_COUNT_HPP_INCLUDED #define BOOST_DETAIL_SHARED_COUNT_HPP_INCLUDED #if _MSC_VER >= 1020 #pragma once #endif // // detail/shared_count.hpp // // Copyright (c) 2001, 2002 Peter Dimov and Multi Media Ltd. // // Permission to copy, use, modify, sell and distribute this software // is granted provided this copyright notice appears in all copies. // This software is provided "as is" without express or implied // warranty, and with no claim as to its suitability for any purpose. // #include #ifndef BOOST_NO_AUTO_PTR # include #endif #include #include namespace boost { namespace detail { class counted_base { public: typedef atomic_count count_type; // pre: initial_use_count <= initial_weak_count explicit counted_base(long initial_use_count, long initial_weak_count): use_count_(initial_use_count), weak_count_(initial_weak_count), self_deleter_(&self_delete) { } virtual ~counted_base() // nothrow { } // dispose() is called when use_count_ drops to zero, to release // the resources managed by *this. // // counted_base doesn't manage any resources except itself, and // the default implementation is a no-op. // // dispose() is not pure virtual since weak_ptr instantiates a // counted_base in its default constructor. virtual void dispose() // nothrow { } void add_ref() // nothrow { ++use_count_; ++weak_count_; } void release() // nothrow { if(--use_count_ == 0) { dispose(); } if(--weak_count_ == 0) { // not a direct 'delete this', because the inlined // release() may use a different heap manager self_deleter_(this); } } void weak_add_ref() // nothrow { ++weak_count_; } void weak_release() // nothrow { if(--weak_count_ == 0) { self_deleter_(this); } } long use_count() const // nothrow { return use_count_; } private: counted_base(counted_base const &); counted_base & operator= (counted_base const &); static void self_delete(counted_base * p) { delete p; } // inv: use_count_ <= weak_count_ count_type use_count_; count_type weak_count_; void (*self_deleter_) (counted_base *); }; template class counted_base_impl: public counted_base { private: P ptr; // copy constructor must not throw D del; // copy constructor must not throw counted_base_impl(counted_base_impl const &); counted_base_impl & operator= (counted_base_impl const &); public: // pre: initial_use_count <= initial_weak_count, d(p) must not throw counted_base_impl(P p, D d, long initial_use_count, long initial_weak_count): counted_base(initial_use_count, initial_weak_count), ptr(p), del(d) { } virtual void dispose() // nothrow { del(ptr); } }; class shared_count { private: counted_base * pi_; friend class weak_count; public: template shared_count(P p, D d): pi_(0) { try { pi_ = new counted_base_impl(p, d, 1, 1); } catch(...) { d(p); // delete p throw; } } #ifndef BOOST_NO_AUTO_PTR // auto_ptr is special cased to provide the strong guarantee template explicit shared_count(std::auto_ptr & r): pi_(new counted_base_impl< Y *, checked_deleter >(r.get(), checked_deleter(), 1, 1)) { r.release(); } #endif ~shared_count() // nothrow { pi_->release(); } shared_count(shared_count const & r): pi_(r.pi_) // nothrow { pi_->add_ref(); } shared_count & operator= (shared_count const & r) // nothrow { counted_base * tmp = r.pi_; tmp->add_ref(); pi_->release(); pi_ = tmp; return *this; } void swap(shared_count & r) // nothrow { counted_base * tmp = r.pi_; r.pi_ = pi_; pi_ = tmp; } long use_count() const // nothrow { return pi_->use_count(); } bool unique() const // nothrow { return pi_->use_count() == 1; } }; class weak_count { private: counted_base * pi_; public: weak_count(): pi_(new counted_base(0, 1)) // can throw { } weak_count(shared_count const & r): pi_(r.pi_) // nothrow { pi_->weak_add_ref(); } weak_count(weak_count const & r): pi_(r.pi_) // nothrow { pi_->weak_add_ref(); } ~weak_count() // nothrow { pi_->weak_release(); } weak_count & operator= (shared_count const & r) // nothrow { counted_base * tmp = r.pi_; tmp->weak_add_ref(); pi_->weak_release(); pi_ = tmp; return *this; } weak_count & operator= (weak_count const & r) // nothrow { counted_base * tmp = r.pi_; tmp->weak_add_ref(); pi_->weak_release(); pi_ = tmp; return *this; } void swap(weak_count & r) // nothrow { counted_base * tmp = r.pi_; r.pi_ = pi_; pi_ = tmp; } long use_count() const // nothrow { return pi_->use_count(); } }; } // namespace detail } // namespace boost #endif // #ifndef BOOST_DETAIL_SHARED_COUNT_HPP_INCLUDED