epoc32/include/stdapis/boost/detail/sp_counted_base_gcc_ia64.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_IA64_HPP_INCLUDED
williamr@2
     2
#define BOOST_DETAIL_SP_COUNTED_BASE_GCC_IA64_HPP_INCLUDED
williamr@2
     3
williamr@2
     4
//
williamr@2
     5
//  detail/sp_counted_base_gcc_ia64.hpp - g++ on IA64
williamr@2
     6
//
williamr@2
     7
//  Copyright (c) 2001, 2002, 2003 Peter Dimov and Multi Media Ltd.
williamr@2
     8
//  Copyright 2004-2006 Peter Dimov
williamr@2
     9
//  Copyright 2005 Ben Hutchings
williamr@2
    10
//
williamr@2
    11
//  Distributed under the Boost Software License, Version 1.0. (See
williamr@2
    12
//  accompanying file LICENSE_1_0.txt or copy at
williamr@2
    13
//  http://www.boost.org/LICENSE_1_0.txt)
williamr@2
    14
//
williamr@2
    15
//
williamr@2
    16
//  Lock-free algorithm by Alexander Terekhov
williamr@2
    17
//
williamr@2
    18
williamr@2
    19
#include <typeinfo>
williamr@2
    20
williamr@2
    21
namespace boost
williamr@2
    22
{
williamr@2
    23
williamr@2
    24
namespace detail
williamr@2
    25
{
williamr@2
    26
williamr@2
    27
inline void atomic_increment( int * pw )
williamr@2
    28
{
williamr@2
    29
    // ++*pw;
williamr@2
    30
williamr@2
    31
    int tmp;
williamr@2
    32
williamr@2
    33
    // No barrier is required here but fetchadd always has an acquire or
williamr@2
    34
    // release barrier associated with it.  We choose release as it should be
williamr@2
    35
    // cheaper.
williamr@2
    36
    __asm__ ("fetchadd4.rel %0=%1,1" :
williamr@2
    37
         "=r"(tmp), "=m"(*pw) :
williamr@2
    38
         "m"( *pw ));
williamr@2
    39
}
williamr@2
    40
williamr@2
    41
inline int atomic_decrement( int * pw )
williamr@2
    42
{
williamr@2
    43
    // return --*pw;
williamr@2
    44
williamr@2
    45
    int rv;
williamr@2
    46
williamr@2
    47
    __asm__ ("     fetchadd4.rel %0=%1,-1 ;; \n"
williamr@2
    48
             "     cmp.eq        p7,p0=1,%0 ;; \n"
williamr@2
    49
             "(p7) ld4.acq       %0=%1    " :
williamr@2
    50
             "=&r"(rv), "=m"(*pw) :
williamr@2
    51
             "m"( *pw ) :
williamr@2
    52
             "p7");
williamr@2
    53
williamr@2
    54
    return rv;
williamr@2
    55
}
williamr@2
    56
williamr@2
    57
inline int atomic_conditional_increment( int * pw )
williamr@2
    58
{
williamr@2
    59
    // if( *pw != 0 ) ++*pw;
williamr@2
    60
    // return *pw;
williamr@2
    61
williamr@2
    62
    int rv, tmp, tmp2;
williamr@2
    63
williamr@2
    64
    __asm__ ("0:   ld4          %0=%3           ;; \n"
williamr@2
    65
         "     cmp.eq       p7,p0=0,%0        ;; \n"
williamr@2
    66
         "(p7) br.cond.spnt 1f                \n"
williamr@2
    67
         "     mov          ar.ccv=%0         \n"
williamr@2
    68
         "     add          %1=1,%0           ;; \n"
williamr@2
    69
         "     cmpxchg4.acq %2=%3,%1,ar.ccv ;; \n"
williamr@2
    70
         "     cmp.ne       p7,p0=%0,%2       ;; \n"
williamr@2
    71
         "(p7) br.cond.spnt 0b                \n"
williamr@2
    72
         "     mov          %0=%1             ;; \n"
williamr@2
    73
         "1:" : 
williamr@2
    74
         "=&r"(rv), "=&r"(tmp), "=&r"(tmp2), "=m"(*pw) :
williamr@2
    75
         "m"( *pw ) :
williamr@2
    76
         "ar.ccv", "p7");
williamr@2
    77
williamr@2
    78
    return rv;
williamr@2
    79
}
williamr@2
    80
williamr@2
    81
class sp_counted_base
williamr@2
    82
{
williamr@2
    83
private:
williamr@2
    84
williamr@2
    85
    sp_counted_base( sp_counted_base const & );
williamr@2
    86
    sp_counted_base & operator= ( sp_counted_base const & );
williamr@2
    87
williamr@2
    88
    int use_count_;        // #shared
williamr@2
    89
    int weak_count_;       // #weak + (#shared != 0)
williamr@2
    90
williamr@2
    91
public:
williamr@2
    92
williamr@2
    93
    sp_counted_base(): use_count_( 1 ), weak_count_( 1 )
williamr@2
    94
    {
williamr@2
    95
    }
williamr@2
    96
williamr@2
    97
    virtual ~sp_counted_base() // nothrow
williamr@2
    98
    {
williamr@2
    99
    }
williamr@2
   100
williamr@2
   101
    // dispose() is called when use_count_ drops to zero, to release
williamr@2
   102
    // the resources managed by *this.
williamr@2
   103
williamr@2
   104
    virtual void dispose() = 0; // nothrow
williamr@2
   105
williamr@2
   106
    // destroy() is called when weak_count_ drops to zero.
williamr@2
   107
williamr@2
   108
    virtual void destroy() // nothrow
williamr@2
   109
    {
williamr@2
   110
        delete this;
williamr@2
   111
    }
williamr@2
   112
williamr@2
   113
    virtual void * get_deleter( std::type_info const & ti ) = 0;
williamr@2
   114
williamr@2
   115
    void add_ref_copy()
williamr@2
   116
    {
williamr@2
   117
        atomic_increment( &use_count_ );
williamr@2
   118
    }
williamr@2
   119
williamr@2
   120
    bool add_ref_lock() // true on success
williamr@2
   121
    {
williamr@2
   122
        return atomic_conditional_increment( &use_count_ ) != 0;
williamr@2
   123
    }
williamr@2
   124
williamr@2
   125
    void release() // nothrow
williamr@2
   126
    {
williamr@2
   127
        if( atomic_decrement( &use_count_ ) == 0 )
williamr@2
   128
        {
williamr@2
   129
            dispose();
williamr@2
   130
            weak_release();
williamr@2
   131
        }
williamr@2
   132
    }
williamr@2
   133
williamr@2
   134
    void weak_add_ref() // nothrow
williamr@2
   135
    {
williamr@2
   136
        atomic_increment( &weak_count_ );
williamr@2
   137
    }
williamr@2
   138
williamr@2
   139
    void weak_release() // nothrow
williamr@2
   140
    {
williamr@2
   141
        if( atomic_decrement( &weak_count_ ) == 0 )
williamr@2
   142
        {
williamr@2
   143
            destroy();
williamr@2
   144
        }
williamr@2
   145
    }
williamr@2
   146
williamr@2
   147
    long use_count() const // nothrow
williamr@2
   148
    {
williamr@2
   149
        return static_cast<int const volatile &>( use_count_ ); // TODO use ld.acq here
williamr@2
   150
    }
williamr@2
   151
};
williamr@2
   152
williamr@2
   153
} // namespace detail
williamr@2
   154
williamr@2
   155
} // namespace boost
williamr@2
   156
williamr@2
   157
#endif  // #ifndef BOOST_DETAIL_SP_COUNTED_BASE_GCC_IA64_HPP_INCLUDED