1 #ifndef BOOST_SMART_PTR_DETAIL_SP_COUNTED_BASE_PT_HPP_INCLUDED
2 #define BOOST_SMART_PTR_DETAIL_SP_COUNTED_BASE_PT_HPP_INCLUDED
4 // MS compatible compilers support #pragma once
6 #if defined(_MSC_VER) && (_MSC_VER >= 1020)
11 // detail/sp_counted_base_pt.hpp
13 // Copyright (c) 2001, 2002, 2003 Peter Dimov and Multi Media Ltd.
14 // Copyright 2004-2005 Peter Dimov
16 // Distributed under the Boost Software License, Version 1.0. (See
17 // accompanying file LICENSE_1_0.txt or copy at
18 // http://www.boost.org/LICENSE_1_0.txt)
21 #include <boost/smart_ptr/detail/sp_typeinfo_.hpp>
22 #include <boost/assert.hpp>
23 #include <boost/config.hpp>
24 #include <boost/cstdint.hpp>
27 #if defined(BOOST_SP_REPORT_IMPLEMENTATION)
29 #include <boost/config/pragma_message.hpp>
30 BOOST_PRAGMA_MESSAGE("Using pthread_mutex sp_counted_base")
40 class BOOST_SYMBOL_VISIBLE sp_counted_base
44 sp_counted_base( sp_counted_base const & );
45 sp_counted_base & operator= ( sp_counted_base const & );
47 boost::int_least32_t use_count_; // #shared
48 boost::int_least32_t weak_count_; // #weak + (#shared != 0)
50 mutable pthread_mutex_t m_;
54 sp_counted_base(): use_count_( 1 ), weak_count_( 1 )
56 // HPUX 10.20 / DCE has a nonstandard pthread_mutex_init
58 #if defined(__hpux) && defined(_DECTHREADS_)
59 BOOST_VERIFY( pthread_mutex_init( &m_, pthread_mutexattr_default ) == 0 );
61 BOOST_VERIFY( pthread_mutex_init( &m_, 0 ) == 0 );
65 virtual ~sp_counted_base() // nothrow
67 BOOST_VERIFY( pthread_mutex_destroy( &m_ ) == 0 );
70 // dispose() is called when use_count_ drops to zero, to release
71 // the resources managed by *this.
73 virtual void dispose() = 0; // nothrow
75 // destroy() is called when weak_count_ drops to zero.
77 virtual void destroy() // nothrow
82 virtual void * get_deleter( sp_typeinfo_ const & ti ) = 0;
83 virtual void * get_local_deleter( sp_typeinfo_ const & ti ) = 0;
84 virtual void * get_untyped_deleter() = 0;
88 BOOST_VERIFY( pthread_mutex_lock( &m_ ) == 0 );
90 BOOST_VERIFY( pthread_mutex_unlock( &m_ ) == 0 );
93 bool add_ref_lock() // true on success
95 BOOST_VERIFY( pthread_mutex_lock( &m_ ) == 0 );
96 bool r = use_count_ == 0? false: ( ++use_count_, true );
97 BOOST_VERIFY( pthread_mutex_unlock( &m_ ) == 0 );
101 void release() // nothrow
103 BOOST_VERIFY( pthread_mutex_lock( &m_ ) == 0 );
104 boost::int_least32_t new_use_count = --use_count_;
105 BOOST_VERIFY( pthread_mutex_unlock( &m_ ) == 0 );
107 if( new_use_count == 0 )
114 void weak_add_ref() // nothrow
116 BOOST_VERIFY( pthread_mutex_lock( &m_ ) == 0 );
118 BOOST_VERIFY( pthread_mutex_unlock( &m_ ) == 0 );
121 void weak_release() // nothrow
123 BOOST_VERIFY( pthread_mutex_lock( &m_ ) == 0 );
124 boost::int_least32_t new_weak_count = --weak_count_;
125 BOOST_VERIFY( pthread_mutex_unlock( &m_ ) == 0 );
127 if( new_weak_count == 0 )
133 long use_count() const // nothrow
135 BOOST_VERIFY( pthread_mutex_lock( &m_ ) == 0 );
136 boost::int_least32_t r = use_count_;
137 BOOST_VERIFY( pthread_mutex_unlock( &m_ ) == 0 );
143 } // namespace detail
147 #endif // #ifndef BOOST_SMART_PTR_DETAIL_SP_COUNTED_BASE_PT_HPP_INCLUDED