williamr@2: #ifndef BOOST_DETAIL_SP_COUNTED_BASE_CW_PPC_HPP_INCLUDED williamr@2: #define BOOST_DETAIL_SP_COUNTED_BASE_CW_PPC_HPP_INCLUDED williamr@2: williamr@2: // MS compatible compilers support #pragma once williamr@2: williamr@2: #if defined(_MSC_VER) && (_MSC_VER >= 1020) williamr@2: # pragma once williamr@2: #endif williamr@2: williamr@2: // williamr@2: // detail/sp_counted_base_cw_ppc.hpp - CodeWarrior on PowerPC williamr@2: // williamr@2: // Copyright (c) 2001, 2002, 2003 Peter Dimov and Multi Media Ltd. williamr@2: // Copyright 2004-2005 Peter Dimov williamr@2: // williamr@2: // Distributed under the Boost Software License, Version 1.0. (See williamr@2: // accompanying file LICENSE_1_0.txt or copy at williamr@2: // http://www.boost.org/LICENSE_1_0.txt) williamr@2: // williamr@2: // williamr@2: // Lock-free algorithm by Alexander Terekhov williamr@2: // williamr@2: // Thanks to Ben Hitchings for the #weak + (#shared != 0) williamr@2: // formulation williamr@2: // williamr@2: williamr@2: #include williamr@2: williamr@2: namespace boost williamr@2: { williamr@2: williamr@2: namespace detail williamr@2: { williamr@2: williamr@2: inline void atomic_increment( register long * pw ) williamr@2: { williamr@2: register int a; williamr@2: williamr@2: asm williamr@2: { williamr@2: loop: williamr@2: williamr@2: lwarx a, 0, pw williamr@2: addi a, a, 1 williamr@2: stwcx. a, 0, pw williamr@2: bne- loop williamr@2: } williamr@2: } williamr@2: williamr@2: inline long atomic_decrement( register long * pw ) williamr@2: { williamr@2: register int a; williamr@2: williamr@2: asm williamr@2: { williamr@2: sync williamr@2: williamr@2: loop: williamr@2: williamr@2: lwarx a, 0, pw williamr@2: addi a, a, -1 williamr@2: stwcx. a, 0, pw williamr@2: bne- loop williamr@2: williamr@2: isync williamr@2: } williamr@2: williamr@2: return a; williamr@2: } williamr@2: williamr@2: inline long atomic_conditional_increment( register long * pw ) williamr@2: { williamr@2: register int a; williamr@2: williamr@2: asm williamr@2: { williamr@2: loop: williamr@2: williamr@2: lwarx a, 0, pw williamr@2: cmpwi a, 0 williamr@2: beq store williamr@2: williamr@2: addi a, a, 1 williamr@2: williamr@2: store: williamr@2: williamr@2: stwcx. a, 0, pw williamr@2: bne- loop williamr@2: } williamr@2: williamr@2: return a; williamr@2: } williamr@2: williamr@2: class sp_counted_base williamr@2: { williamr@2: private: williamr@2: williamr@2: sp_counted_base( sp_counted_base const & ); williamr@2: sp_counted_base & operator= ( sp_counted_base const & ); williamr@2: williamr@2: long use_count_; // #shared williamr@2: long weak_count_; // #weak + (#shared != 0) williamr@2: williamr@2: public: williamr@2: williamr@2: sp_counted_base(): use_count_( 1 ), weak_count_( 1 ) williamr@2: { williamr@2: } williamr@2: williamr@2: virtual ~sp_counted_base() // nothrow williamr@2: { williamr@2: } williamr@2: williamr@2: // dispose() is called when use_count_ drops to zero, to release williamr@2: // the resources managed by *this. williamr@2: williamr@2: virtual void dispose() = 0; // nothrow williamr@2: williamr@2: // destroy() is called when weak_count_ drops to zero. williamr@2: williamr@2: virtual void destroy() // nothrow williamr@2: { williamr@2: delete this; williamr@2: } williamr@2: williamr@2: virtual void * get_deleter( std::type_info const & ti ) = 0; williamr@2: williamr@2: void add_ref_copy() williamr@2: { williamr@2: atomic_increment( &use_count_ ); williamr@2: } williamr@2: williamr@2: bool add_ref_lock() // true on success williamr@2: { williamr@2: return atomic_conditional_increment( &use_count_ ) != 0; williamr@2: } williamr@2: williamr@2: void release() // nothrow williamr@2: { williamr@2: if( atomic_decrement( &use_count_ ) == 0 ) williamr@2: { williamr@2: dispose(); williamr@2: weak_release(); williamr@2: } williamr@2: } williamr@2: williamr@2: void weak_add_ref() // nothrow williamr@2: { williamr@2: atomic_increment( &weak_count_ ); williamr@2: } williamr@2: williamr@2: void weak_release() // nothrow williamr@2: { williamr@2: if( atomic_decrement( &weak_count_ ) == 0 ) williamr@2: { williamr@2: destroy(); williamr@2: } williamr@2: } williamr@2: williamr@2: long use_count() const // nothrow williamr@2: { williamr@2: return static_cast( use_count_ ); williamr@2: } williamr@2: }; williamr@2: williamr@2: } // namespace detail williamr@2: williamr@2: } // namespace boost williamr@2: williamr@2: #endif // #ifndef BOOST_DETAIL_SP_COUNTED_BASE_CW_PPC_HPP_INCLUDED