epoc32/include/stdapis/boost/detail/sp_counted_base_gcc_ppc.hpp
author William Roberts <williamr@symbian.org>
Tue, 16 Mar 2010 16:12:26 +0000
branchSymbian2
changeset 2 2fe1408b6811
permissions -rw-r--r--
Final list of Symbian^2 public API header files
williamr@2
     1
#ifndef BOOST_DETAIL_SP_COUNTED_BASE_GCC_PPC_HPP_INCLUDED
williamr@2
     2
#define BOOST_DETAIL_SP_COUNTED_BASE_GCC_PPC_HPP_INCLUDED
williamr@2
     3
williamr@2
     4
// MS compatible compilers support #pragma once
williamr@2
     5
williamr@2
     6
#if defined(_MSC_VER) && (_MSC_VER >= 1020)
williamr@2
     7
# pragma once
williamr@2
     8
#endif
williamr@2
     9
williamr@2
    10
//
williamr@2
    11
//  detail/sp_counted_base_gcc_ppc.hpp - g++ on PowerPC
williamr@2
    12
//
williamr@2
    13
//  Copyright (c) 2001, 2002, 2003 Peter Dimov and Multi Media Ltd.
williamr@2
    14
//  Copyright 2004-2005 Peter Dimov
williamr@2
    15
//
williamr@2
    16
//  Distributed under the Boost Software License, Version 1.0. (See
williamr@2
    17
//  accompanying file LICENSE_1_0.txt or copy at
williamr@2
    18
//  http://www.boost.org/LICENSE_1_0.txt)
williamr@2
    19
//
williamr@2
    20
//
williamr@2
    21
//  Lock-free algorithm by Alexander Terekhov
williamr@2
    22
//
williamr@2
    23
//  Thanks to Ben Hitchings for the #weak + (#shared != 0)
williamr@2
    24
//  formulation
williamr@2
    25
//
williamr@2
    26
williamr@2
    27
#include <typeinfo>
williamr@2
    28
williamr@2
    29
namespace boost
williamr@2
    30
{
williamr@2
    31
williamr@2
    32
namespace detail
williamr@2
    33
{
williamr@2
    34
williamr@2
    35
inline void atomic_increment( int * pw )
williamr@2
    36
{
williamr@2
    37
    // ++*pw;
williamr@2
    38
williamr@2
    39
    int tmp;
williamr@2
    40
williamr@2
    41
    __asm__
williamr@2
    42
    (
williamr@2
    43
        "0:\n\t"
williamr@2
    44
        "lwarx %1, 0, %2\n\t"
williamr@2
    45
        "addi %1, %1, 1\n\t"
williamr@2
    46
        "stwcx. %1, 0, %2\n\t"
williamr@2
    47
        "bne- 0b":
williamr@2
    48
williamr@2
    49
        "=m"( *pw ), "=&b"( tmp ):
williamr@2
    50
        "r"( pw ), "m"( *pw ):
williamr@2
    51
        "cc"
williamr@2
    52
    );
williamr@2
    53
}
williamr@2
    54
williamr@2
    55
inline int atomic_decrement( int * pw )
williamr@2
    56
{
williamr@2
    57
    // return --*pw;
williamr@2
    58
williamr@2
    59
    int rv;
williamr@2
    60
williamr@2
    61
    __asm__ __volatile__
williamr@2
    62
    (
williamr@2
    63
        "sync\n\t"
williamr@2
    64
        "0:\n\t"
williamr@2
    65
        "lwarx %1, 0, %2\n\t"
williamr@2
    66
        "addi %1, %1, -1\n\t"
williamr@2
    67
        "stwcx. %1, 0, %2\n\t"
williamr@2
    68
        "bne- 0b\n\t"
williamr@2
    69
        "isync":
williamr@2
    70
williamr@2
    71
        "=m"( *pw ), "=&b"( rv ):
williamr@2
    72
        "r"( pw ), "m"( *pw ):
williamr@2
    73
        "memory", "cc"
williamr@2
    74
    );
williamr@2
    75
williamr@2
    76
    return rv;
williamr@2
    77
}
williamr@2
    78
williamr@2
    79
inline int atomic_conditional_increment( int * pw )
williamr@2
    80
{
williamr@2
    81
    // if( *pw != 0 ) ++*pw;
williamr@2
    82
    // return *pw;
williamr@2
    83
williamr@2
    84
    int rv;
williamr@2
    85
williamr@2
    86
    __asm__
williamr@2
    87
    (
williamr@2
    88
        "0:\n\t"
williamr@2
    89
        "lwarx %1, 0, %2\n\t"
williamr@2
    90
        "cmpwi %1, 0\n\t"
williamr@2
    91
        "beq 1f\n\t"
williamr@2
    92
        "addi %1, %1, 1\n\t"
williamr@2
    93
        "1:\n\t"
williamr@2
    94
        "stwcx. %1, 0, %2\n\t"
williamr@2
    95
        "bne- 0b":
williamr@2
    96
williamr@2
    97
        "=m"( *pw ), "=&b"( rv ):
williamr@2
    98
        "r"( pw ), "m"( *pw ):
williamr@2
    99
        "cc"
williamr@2
   100
    );
williamr@2
   101
williamr@2
   102
    return rv;
williamr@2
   103
}
williamr@2
   104
williamr@2
   105
class sp_counted_base
williamr@2
   106
{
williamr@2
   107
private:
williamr@2
   108
williamr@2
   109
    sp_counted_base( sp_counted_base const & );
williamr@2
   110
    sp_counted_base & operator= ( sp_counted_base const & );
williamr@2
   111
williamr@2
   112
    int use_count_;        // #shared
williamr@2
   113
    int weak_count_;       // #weak + (#shared != 0)
williamr@2
   114
williamr@2
   115
public:
williamr@2
   116
williamr@2
   117
    sp_counted_base(): use_count_( 1 ), weak_count_( 1 )
williamr@2
   118
    {
williamr@2
   119
    }
williamr@2
   120
williamr@2
   121
    virtual ~sp_counted_base() // nothrow
williamr@2
   122
    {
williamr@2
   123
    }
williamr@2
   124
williamr@2
   125
    // dispose() is called when use_count_ drops to zero, to release
williamr@2
   126
    // the resources managed by *this.
williamr@2
   127
williamr@2
   128
    virtual void dispose() = 0; // nothrow
williamr@2
   129
williamr@2
   130
    // destroy() is called when weak_count_ drops to zero.
williamr@2
   131
williamr@2
   132
    virtual void destroy() // nothrow
williamr@2
   133
    {
williamr@2
   134
        delete this;
williamr@2
   135
    }
williamr@2
   136
williamr@2
   137
    virtual void * get_deleter( std::type_info const & ti ) = 0;
williamr@2
   138
williamr@2
   139
    void add_ref_copy()
williamr@2
   140
    {
williamr@2
   141
        atomic_increment( &use_count_ );
williamr@2
   142
    }
williamr@2
   143
williamr@2
   144
    bool add_ref_lock() // true on success
williamr@2
   145
    {
williamr@2
   146
        return atomic_conditional_increment( &use_count_ ) != 0;
williamr@2
   147
    }
williamr@2
   148
williamr@2
   149
    void release() // nothrow
williamr@2
   150
    {
williamr@2
   151
        if( atomic_decrement( &use_count_ ) == 0 )
williamr@2
   152
        {
williamr@2
   153
            dispose();
williamr@2
   154
            weak_release();
williamr@2
   155
        }
williamr@2
   156
    }
williamr@2
   157
williamr@2
   158
    void weak_add_ref() // nothrow
williamr@2
   159
    {
williamr@2
   160
        atomic_increment( &weak_count_ );
williamr@2
   161
    }
williamr@2
   162
williamr@2
   163
    void weak_release() // nothrow
williamr@2
   164
    {
williamr@2
   165
        if( atomic_decrement( &weak_count_ ) == 0 )
williamr@2
   166
        {
williamr@2
   167
            destroy();
williamr@2
   168
        }
williamr@2
   169
    }
williamr@2
   170
williamr@2
   171
    long use_count() const // nothrow
williamr@2
   172
    {
williamr@2
   173
        return static_cast<int const volatile &>( use_count_ );
williamr@2
   174
    }
williamr@2
   175
};
williamr@2
   176
williamr@2
   177
} // namespace detail
williamr@2
   178
williamr@2
   179
} // namespace boost
williamr@2
   180
williamr@2
   181
#endif  // #ifndef BOOST_DETAIL_SP_COUNTED_BASE_GCC_PPC_HPP_INCLUDED