epoc32/include/stdapis/boost/detail/sp_counted_base_pt.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_PT_HPP_INCLUDED
williamr@2
     2
#define BOOST_DETAIL_SP_COUNTED_BASE_PT_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_pt.hpp
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
#include <typeinfo>
williamr@2
    22
#include <pthread.h>
williamr@2
    23
williamr@2
    24
namespace boost
williamr@2
    25
{
williamr@2
    26
williamr@2
    27
namespace detail
williamr@2
    28
{
williamr@2
    29
williamr@2
    30
class sp_counted_base
williamr@2
    31
{
williamr@2
    32
private:
williamr@2
    33
williamr@2
    34
    sp_counted_base( sp_counted_base const & );
williamr@2
    35
    sp_counted_base & operator= ( sp_counted_base const & );
williamr@2
    36
williamr@2
    37
    long use_count_;        // #shared
williamr@2
    38
    long weak_count_;       // #weak + (#shared != 0)
williamr@2
    39
williamr@2
    40
    mutable pthread_mutex_t m_;
williamr@2
    41
williamr@2
    42
public:
williamr@2
    43
williamr@2
    44
    sp_counted_base(): use_count_( 1 ), weak_count_( 1 )
williamr@2
    45
    {
williamr@2
    46
// HPUX 10.20 / DCE has a nonstandard pthread_mutex_init
williamr@2
    47
williamr@2
    48
#if defined(__hpux) && defined(_DECTHREADS_)
williamr@2
    49
        pthread_mutex_init( &m_, pthread_mutexattr_default );
williamr@2
    50
#else
williamr@2
    51
        pthread_mutex_init( &m_, 0 );
williamr@2
    52
#endif
williamr@2
    53
    }
williamr@2
    54
williamr@2
    55
    virtual ~sp_counted_base() // nothrow
williamr@2
    56
    {
williamr@2
    57
        pthread_mutex_destroy( &m_ );
williamr@2
    58
    }
williamr@2
    59
williamr@2
    60
    // dispose() is called when use_count_ drops to zero, to release
williamr@2
    61
    // the resources managed by *this.
williamr@2
    62
williamr@2
    63
    virtual void dispose() = 0; // nothrow
williamr@2
    64
williamr@2
    65
    // destroy() is called when weak_count_ drops to zero.
williamr@2
    66
williamr@2
    67
    virtual void destroy() // nothrow
williamr@2
    68
    {
williamr@2
    69
        delete this;
williamr@2
    70
    }
williamr@2
    71
williamr@2
    72
    virtual void * get_deleter( std::type_info const & ti ) = 0;
williamr@2
    73
williamr@2
    74
    void add_ref_copy()
williamr@2
    75
    {
williamr@2
    76
        pthread_mutex_lock( &m_ );
williamr@2
    77
        ++use_count_;
williamr@2
    78
        pthread_mutex_unlock( &m_ );
williamr@2
    79
    }
williamr@2
    80
williamr@2
    81
    bool add_ref_lock() // true on success
williamr@2
    82
    {
williamr@2
    83
        pthread_mutex_lock( &m_ );
williamr@2
    84
        bool r = use_count_ == 0? false: ( ++use_count_, true );
williamr@2
    85
        pthread_mutex_unlock( &m_ );
williamr@2
    86
        return r;
williamr@2
    87
    }
williamr@2
    88
williamr@2
    89
    void release() // nothrow
williamr@2
    90
    {
williamr@2
    91
        pthread_mutex_lock( &m_ );
williamr@2
    92
        long new_use_count = --use_count_;
williamr@2
    93
        pthread_mutex_unlock( &m_ );
williamr@2
    94
williamr@2
    95
        if( new_use_count == 0 )
williamr@2
    96
        {
williamr@2
    97
            dispose();
williamr@2
    98
            weak_release();
williamr@2
    99
        }
williamr@2
   100
    }
williamr@2
   101
williamr@2
   102
    void weak_add_ref() // nothrow
williamr@2
   103
    {
williamr@2
   104
        pthread_mutex_lock( &m_ );
williamr@2
   105
        ++weak_count_;
williamr@2
   106
        pthread_mutex_unlock( &m_ );
williamr@2
   107
    }
williamr@2
   108
williamr@2
   109
    void weak_release() // nothrow
williamr@2
   110
    {
williamr@2
   111
        pthread_mutex_lock( &m_ );
williamr@2
   112
        long new_weak_count = --weak_count_;
williamr@2
   113
        pthread_mutex_unlock( &m_ );
williamr@2
   114
williamr@2
   115
        if( new_weak_count == 0 )
williamr@2
   116
        {
williamr@2
   117
            destroy();
williamr@2
   118
        }
williamr@2
   119
    }
williamr@2
   120
williamr@2
   121
    long use_count() const // nothrow
williamr@2
   122
    {
williamr@2
   123
        pthread_mutex_lock( &m_ );
williamr@2
   124
        long r = use_count_;
williamr@2
   125
        pthread_mutex_unlock( &m_ );
williamr@2
   126
williamr@2
   127
        return r;
williamr@2
   128
    }
williamr@2
   129
};
williamr@2
   130
williamr@2
   131
} // namespace detail
williamr@2
   132
williamr@2
   133
} // namespace boost
williamr@2
   134
williamr@2
   135
#endif  // #ifndef BOOST_DETAIL_SP_COUNTED_BASE_PT_HPP_INCLUDED