annotate DEPENDENCIES/generic/include/boost/smart_ptr/detail/sp_counted_base_snc_ps3.hpp @ 41:ed9f24243b59

OS/X fix
author Chris Cannam
date Thu, 07 Aug 2014 09:18:32 +0100
parents 2665513ce2d3
children
rev   line source
Chris@16 1 #ifndef BOOST_SMART_PTR_DETAIL_SP_COUNTED_BASE_SNC_PS3_HPP_INCLUDED
Chris@16 2 #define BOOST_SMART_PTR_DETAIL_SP_COUNTED_BASE_SNC_PS3_HPP_INCLUDED
Chris@16 3
Chris@16 4 // MS compatible compilers support #pragma once
Chris@16 5 #if defined(_MSC_VER) && (_MSC_VER >= 1020)
Chris@16 6 # pragma once
Chris@16 7 #endif
Chris@16 8
Chris@16 9 // detail/sp_counted_base_gcc_sparc.hpp - g++ on Sparc V8+
Chris@16 10 //
Chris@16 11 // Copyright (c) 2006 Piotr Wyderski
Chris@16 12 // Copyright (c) 2006 Tomas Puverle
Chris@16 13 // Copyright (c) 2006 Peter Dimov
Chris@16 14 // Copyright (c) 2011 Emil Dotchevski
Chris@16 15 //
Chris@16 16 // Distributed under the Boost Software License, Version 1.0.
Chris@16 17 // See accompanying file LICENSE_1_0.txt or copy at
Chris@16 18 // http://www.boost.org/LICENSE_1_0.txt
Chris@16 19 //
Chris@16 20 // Thanks to Michael van der Westhuizen
Chris@16 21
Chris@16 22 #include <boost/detail/sp_typeinfo.hpp>
Chris@16 23 #include <inttypes.h> // uint32_t
Chris@16 24
Chris@16 25 namespace boost
Chris@16 26 {
Chris@16 27
Chris@16 28 namespace detail
Chris@16 29 {
Chris@16 30
Chris@16 31 inline uint32_t compare_and_swap( uint32_t * dest_, uint32_t compare_, uint32_t swap_ )
Chris@16 32 {
Chris@16 33 return __builtin_cellAtomicCompareAndSwap32(dest_,compare_,swap_);
Chris@16 34 }
Chris@16 35
Chris@16 36 inline uint32_t atomic_fetch_and_add( uint32_t * pw, uint32_t dv )
Chris@16 37 {
Chris@16 38 // long r = *pw;
Chris@16 39 // *pw += dv;
Chris@16 40 // return r;
Chris@16 41
Chris@16 42 for( ;; )
Chris@16 43 {
Chris@16 44 uint32_t r = *pw;
Chris@16 45
Chris@16 46 if( __builtin_expect((compare_and_swap(pw, r, r + dv) == r), 1) )
Chris@16 47 {
Chris@16 48 return r;
Chris@16 49 }
Chris@16 50 }
Chris@16 51 }
Chris@16 52
Chris@16 53 inline void atomic_increment( uint32_t * pw )
Chris@16 54 {
Chris@16 55 (void) __builtin_cellAtomicIncr32( pw );
Chris@16 56 }
Chris@16 57
Chris@16 58 inline uint32_t atomic_decrement( uint32_t * pw )
Chris@16 59 {
Chris@16 60 return __builtin_cellAtomicDecr32( pw );
Chris@16 61 }
Chris@16 62
Chris@16 63 inline uint32_t atomic_conditional_increment( uint32_t * pw )
Chris@16 64 {
Chris@16 65 // long r = *pw;
Chris@16 66 // if( r != 0 ) ++*pw;
Chris@16 67 // return r;
Chris@16 68
Chris@16 69 for( ;; )
Chris@16 70 {
Chris@16 71 uint32_t r = *pw;
Chris@16 72
Chris@16 73 if( r == 0 )
Chris@16 74 {
Chris@16 75 return r;
Chris@16 76 }
Chris@16 77
Chris@16 78 if( __builtin_expect( ( compare_and_swap( pw, r, r + 1 ) == r ), 1 ) )
Chris@16 79 {
Chris@16 80 return r;
Chris@16 81 }
Chris@16 82 }
Chris@16 83 }
Chris@16 84
Chris@16 85 class sp_counted_base
Chris@16 86 {
Chris@16 87 private:
Chris@16 88
Chris@16 89 sp_counted_base( sp_counted_base const & );
Chris@16 90 sp_counted_base & operator= ( sp_counted_base const & );
Chris@16 91
Chris@16 92 uint32_t use_count_; // #shared
Chris@16 93 uint32_t weak_count_; // #weak + (#shared != 0)
Chris@16 94
Chris@16 95 public:
Chris@16 96
Chris@16 97 sp_counted_base(): use_count_( 1 ), weak_count_( 1 )
Chris@16 98 {
Chris@16 99 }
Chris@16 100
Chris@16 101 virtual ~sp_counted_base() // nothrow
Chris@16 102 {
Chris@16 103 }
Chris@16 104
Chris@16 105 // dispose() is called when use_count_ drops to zero, to release
Chris@16 106 // the resources managed by *this.
Chris@16 107
Chris@16 108 virtual void dispose() = 0; // nothrow
Chris@16 109
Chris@16 110 // destroy() is called when weak_count_ drops to zero.
Chris@16 111
Chris@16 112 virtual void destroy() // nothrow
Chris@16 113 {
Chris@16 114 delete this;
Chris@16 115 }
Chris@16 116
Chris@16 117 virtual void * get_deleter( sp_typeinfo const & ti ) = 0;
Chris@16 118 virtual void * get_untyped_deleter() = 0;
Chris@16 119
Chris@16 120 void add_ref_copy()
Chris@16 121 {
Chris@16 122 atomic_increment( &use_count_ );
Chris@16 123 }
Chris@16 124
Chris@16 125 bool add_ref_lock() // true on success
Chris@16 126 {
Chris@16 127 return atomic_conditional_increment( &use_count_ ) != 0;
Chris@16 128 }
Chris@16 129
Chris@16 130 void release() // nothrow
Chris@16 131 {
Chris@16 132 if( atomic_decrement( &use_count_ ) == 1 )
Chris@16 133 {
Chris@16 134 dispose();
Chris@16 135 weak_release();
Chris@16 136 }
Chris@16 137 }
Chris@16 138
Chris@16 139 void weak_add_ref() // nothrow
Chris@16 140 {
Chris@16 141 atomic_increment( &weak_count_ );
Chris@16 142 }
Chris@16 143
Chris@16 144 void weak_release() // nothrow
Chris@16 145 {
Chris@16 146 if( atomic_decrement( &weak_count_ ) == 1 )
Chris@16 147 {
Chris@16 148 destroy();
Chris@16 149 }
Chris@16 150 }
Chris@16 151
Chris@16 152 long use_count() const // nothrow
Chris@16 153 {
Chris@16 154 return const_cast< uint32_t const volatile & >( use_count_ );
Chris@16 155 }
Chris@16 156 };
Chris@16 157
Chris@16 158 } // namespace detail
Chris@16 159
Chris@16 160 } // namespace boost
Chris@16 161
Chris@16 162 #endif // #ifndef BOOST_SMART_PTR_DETAIL_SP_COUNTED_BASE_SNC_PS3_HPP_INCLUDED