epoc32/include/stdapis/boost/detail/sp_counted_base_gcc_x86.hpp
author William Roberts <williamr@symbian.org>
Wed, 31 Mar 2010 12:33:34 +0100
branchSymbian3
changeset 4 837f303aceeb
permissions -rw-r--r--
Current Symbian^3 public API header files (from PDK 3.0.h)
This is the epoc32/include tree with the "platform" subtrees removed, and
all but a selected few mbg and rsg files removed.
williamr@2
     1
#ifndef BOOST_DETAIL_SP_COUNTED_BASE_GCC_X86_HPP_INCLUDED
williamr@2
     2
#define BOOST_DETAIL_SP_COUNTED_BASE_GCC_X86_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_x86.hpp - g++ on 486+ or AMD64
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 int atomic_exchange_and_add( int * pw, int dv )
williamr@2
    36
{
williamr@2
    37
    // int r = *pw;
williamr@2
    38
    // *pw += dv;
williamr@2
    39
    // return r;
williamr@2
    40
williamr@2
    41
    int r;
williamr@2
    42
williamr@2
    43
    __asm__ __volatile__
williamr@2
    44
    (
williamr@2
    45
        "lock\n\t"
williamr@2
    46
        "xadd %1, %0":
williamr@2
    47
        "=m"( *pw ), "=r"( r ): // outputs (%0, %1)
williamr@2
    48
        "m"( *pw ), "1"( dv ): // inputs (%2, %3 == %1)
williamr@2
    49
        "memory", "cc" // clobbers
williamr@2
    50
    );
williamr@2
    51
williamr@2
    52
    return r;
williamr@2
    53
}
williamr@2
    54
williamr@2
    55
inline void atomic_increment( int * pw )
williamr@2
    56
{
williamr@2
    57
    //atomic_exchange_and_add( pw, 1 );
williamr@2
    58
williamr@2
    59
    __asm__
williamr@2
    60
    (
williamr@2
    61
        "lock\n\t"
williamr@2
    62
        "incl %0":
williamr@2
    63
        "=m"( *pw ): // output (%0)
williamr@2
    64
        "m"( *pw ): // input (%1)
williamr@2
    65
        "cc" // clobbers
williamr@2
    66
    );
williamr@2
    67
}
williamr@2
    68
williamr@2
    69
inline int atomic_conditional_increment( int * pw )
williamr@2
    70
{
williamr@2
    71
    // int rv = *pw;
williamr@2
    72
    // if( rv != 0 ) ++*pw;
williamr@2
    73
    // return rv;
williamr@2
    74
williamr@2
    75
    int rv, tmp;
williamr@2
    76
williamr@2
    77
    __asm__
williamr@2
    78
    (
williamr@2
    79
        "movl %0, %%eax\n\t"
williamr@2
    80
        "0:\n\t"
williamr@2
    81
        "test %%eax, %%eax\n\t"
williamr@2
    82
        "je 1f\n\t"
williamr@2
    83
        "movl %%eax, %2\n\t"
williamr@2
    84
        "incl %2\n\t"
williamr@2
    85
        "lock\n\t"
williamr@2
    86
        "cmpxchgl %2, %0\n\t"
williamr@2
    87
        "jne 0b\n\t"
williamr@2
    88
        "1:":
williamr@2
    89
        "=m"( *pw ), "=&a"( rv ), "=&r"( tmp ): // outputs (%0, %1, %2)
williamr@2
    90
        "m"( *pw ): // input (%3)
williamr@2
    91
        "cc" // clobbers
williamr@2
    92
    );
williamr@2
    93
williamr@2
    94
    return rv;
williamr@2
    95
}
williamr@2
    96
williamr@2
    97
class sp_counted_base
williamr@2
    98
{
williamr@2
    99
private:
williamr@2
   100
williamr@2
   101
    sp_counted_base( sp_counted_base const & );
williamr@2
   102
    sp_counted_base & operator= ( sp_counted_base const & );
williamr@2
   103
williamr@2
   104
    int use_count_;        // #shared
williamr@2
   105
    int weak_count_;       // #weak + (#shared != 0)
williamr@2
   106
williamr@2
   107
public:
williamr@2
   108
williamr@2
   109
    sp_counted_base(): use_count_( 1 ), weak_count_( 1 )
williamr@2
   110
    {
williamr@2
   111
    }
williamr@2
   112
williamr@2
   113
    virtual ~sp_counted_base() // nothrow
williamr@2
   114
    {
williamr@2
   115
    }
williamr@2
   116
williamr@2
   117
    // dispose() is called when use_count_ drops to zero, to release
williamr@2
   118
    // the resources managed by *this.
williamr@2
   119
williamr@2
   120
    virtual void dispose() = 0; // nothrow
williamr@2
   121
williamr@2
   122
    // destroy() is called when weak_count_ drops to zero.
williamr@2
   123
williamr@2
   124
    virtual void destroy() // nothrow
williamr@2
   125
    {
williamr@2
   126
        delete this;
williamr@2
   127
    }
williamr@2
   128
williamr@2
   129
    virtual void * get_deleter( std::type_info const & ti ) = 0;
williamr@2
   130
williamr@2
   131
    void add_ref_copy()
williamr@2
   132
    {
williamr@2
   133
        atomic_increment( &use_count_ );
williamr@2
   134
    }
williamr@2
   135
williamr@2
   136
    bool add_ref_lock() // true on success
williamr@2
   137
    {
williamr@2
   138
        return atomic_conditional_increment( &use_count_ ) != 0;
williamr@2
   139
    }
williamr@2
   140
williamr@2
   141
    void release() // nothrow
williamr@2
   142
    {
williamr@2
   143
        if( atomic_exchange_and_add( &use_count_, -1 ) == 1 )
williamr@2
   144
        {
williamr@2
   145
            dispose();
williamr@2
   146
            weak_release();
williamr@2
   147
        }
williamr@2
   148
    }
williamr@2
   149
williamr@2
   150
    void weak_add_ref() // nothrow
williamr@2
   151
    {
williamr@2
   152
        atomic_increment( &weak_count_ );
williamr@2
   153
    }
williamr@2
   154
williamr@2
   155
    void weak_release() // nothrow
williamr@2
   156
    {
williamr@2
   157
        if( atomic_exchange_and_add( &weak_count_, -1 ) == 1 )
williamr@2
   158
        {
williamr@2
   159
            destroy();
williamr@2
   160
        }
williamr@2
   161
    }
williamr@2
   162
williamr@2
   163
    long use_count() const // nothrow
williamr@2
   164
    {
williamr@2
   165
        return static_cast<int const volatile &>( use_count_ );
williamr@2
   166
    }
williamr@2
   167
};
williamr@2
   168
williamr@2
   169
} // namespace detail
williamr@2
   170
williamr@2
   171
} // namespace boost
williamr@2
   172
williamr@2
   173
#endif  // #ifndef BOOST_DETAIL_SP_COUNTED_BASE_GCC_X86_HPP_INCLUDED