HDK
 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Groups Pages
refBase.h
Go to the documentation of this file.
1 //
2 // Copyright 2016 Pixar
3 //
4 // Licensed under the Apache License, Version 2.0 (the "Apache License")
5 // with the following modification; you may not use this file except in
6 // compliance with the Apache License and the following modification to it:
7 // Section 6. Trademarks. is deleted and replaced with:
8 //
9 // 6. Trademarks. This License does not grant permission to use the trade
10 // names, trademarks, service marks, or product names of the Licensor
11 // and its affiliates, except as required to comply with Section 4(c) of
12 // the License and to reproduce the content of the NOTICE file.
13 //
14 // You may obtain a copy of the Apache License at
15 //
16 // http://www.apache.org/licenses/LICENSE-2.0
17 //
18 // Unless required by applicable law or agreed to in writing, software
19 // distributed under the Apache License with the above modification is
20 // distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
21 // KIND, either express or implied. See the Apache License for the specific
22 // language governing permissions and limitations under the Apache License.
23 //
24 #ifndef PXR_BASE_TF_REF_BASE_H
25 #define PXR_BASE_TF_REF_BASE_H
26 
27 /// \file tf/refBase.h
28 /// \ingroup group_tf_Memory
29 
30 #include "pxr/pxr.h"
31 
32 #include "pxr/base/tf/api.h"
33 
34 #include <atomic>
35 #include <cmath>
36 
38 
39 template <class T> class TfRefPtr;
40 template <class T> class TfWeakPtr;
41 
42 /// \class TfRefBase
43 /// \ingroup group_tf_Memory
44 ///
45 /// Enable a concrete base class for use with \c TfRefPtr.
46 ///
47 /// You should be familiar with the \c TfRefPtr type before reading further.
48 ///
49 /// A class (but not an interface class) is enabled for reference
50 /// counting via the \c TfRefPtr type by publicly deriving from \c
51 /// TfRefBase.
52 ///
53 /// For example,
54 /// \code
55 /// #include "pxr/base/tf/refPtr.h"
56 ///
57 /// class Simple : public TfRefBase {
58 /// public:
59 /// TfRefPtr<Simple> New() {
60 /// return TfCreateRefPtr(new Simple);
61 /// }
62 /// private:
63 /// Simple();
64 /// };
65 /// \endcode
66 ///
67 /// The class \c Simple can now only be manipulated in terms of
68 /// a \c TfRefPtr<Simple>.
69 ///
70 /// To disable the cost of the "unique changed" system, derive
71 /// from TfSimpleRefBase instead.
72 ///
73 class TfRefBase {
74 public:
75 
76  typedef void (*UniqueChangedFuncPtr)(TfRefBase const *, bool);
78  void (*lock)();
80  void (*unlock)();
81  };
82 
83  // This mimics the old TfRefCount's default ctor behavior, which set
84  // _refCount to 1.
85  TfRefBase() : _refCount(1) {}
86 
87  // This mimics the old TfRefCount's copy ctor behavior, which set _refCount
88  // to 1 on copy construction.
89  TfRefBase(TfRefBase const &) : _refCount(1) {}
90 
91  // This mimics the old TfRefCount's copy assignment behavior, which took no
92  // action.
94  return *this;
95  }
96 
97  /// Return the current reference count of this object.
98  size_t GetCurrentCount() const {
99  // Return the absolute value since the sign encodes whether or not this
100  // TfRefBase invokes the UniqueChangedListener.
101  return std::abs(_refCount.load(std::memory_order_relaxed));
102  }
103 
104  /// Return true if only one \c TfRefPtr points to this object.
105  bool IsUnique() const {
106  return GetCurrentCount() == 1;
107  }
108 
109  void SetShouldInvokeUniqueChangedListener(bool shouldCall) {
110  int curValue = _refCount.load(std::memory_order_relaxed);
111  while ((curValue > 0 && shouldCall) ||
112  (curValue < 0 && !shouldCall)) {
113  if (_refCount.compare_exchange_weak(curValue, -curValue)) {
114  return;
115  }
116  }
117  }
118 
119  TF_API static void SetUniqueChangedListener(UniqueChangedListener listener);
120 
121 protected:
122  /*
123  * Prohibit deletion through a TfRefBase pointer.
124  */
125  TF_API virtual ~TfRefBase();
126 
127 private:
128  // For TfRefPtr's use.
129  std::atomic_int &_GetRefCount() const {
130  return _refCount;
131  }
132 
133  // Note! Counts can be both positive or negative. Negative counts indicate
134  // that we must invoke the _uniqueChangedListener if the count goes 1 -> 2
135  // or 2 -> 1 (which is really -1 -> -2 or -2 -> -1).
136  mutable std::atomic_int _refCount;
137 
138  static UniqueChangedListener _uniqueChangedListener;
139  template <typename T> friend class TfRefPtr;
141  friend struct Tf_RefPtr_Counter;
142 
143  template <typename T> friend TfRefPtr<T>
145 };
146 
147 /// \class TfSimpleRefBase
148 /// \ingroup group_tf_Memory
149 ///
150 /// Enable a concrete base class for use with \c TfRefPtr that inhibits the
151 /// "unique changed" facility of TfRefPtr.
152 ///
153 /// Derive from this class if you don't plan on wrapping your
154 /// reference-counted object via hboost::python.
155 ///
156 class TfSimpleRefBase : public TfRefBase {
157 public:
158  TF_API virtual ~TfSimpleRefBase();
159 };
160 
162 
163 #endif // PXR_BASE_TF_REF_BASE_H
bool IsUnique() const
Return true if only one TfRefPtr points to this object.
Definition: refBase.h:105
#define TF_API
Definition: api.h:40
TfRefBase & operator=(TfRefBase const &)
Definition: refBase.h:93
void
Definition: png.h:1083
friend TfRefPtr< T > TfCreateRefPtrFromProtectedWeakPtr(TfWeakPtr< T > const &)
Definition: weakPtr.h:277
virtual TF_API ~TfRefBase()
void(* UniqueChangedFuncPtr)(TfRefBase const *, bool)
Definition: refBase.h:76
TfRefBase(TfRefBase const &)
Definition: refBase.h:89
atomic< int > atomic_int
Definition: atomic.h:25
virtual TF_API ~TfSimpleRefBase()
UniqueChangedFuncPtr func
Definition: refBase.h:79
static TF_API void SetUniqueChangedListener(UniqueChangedListener listener)
size_t GetCurrentCount() const
Return the current reference count of this object.
Definition: refBase.h:98
PXR_NAMESPACE_CLOSE_SCOPE PXR_NAMESPACE_OPEN_SCOPE
Definition: path.h:1432
#define PXR_NAMESPACE_CLOSE_SCOPE
Definition: pxr.h:91
IMATH_INTERNAL_NAMESPACE_HEADER_ENTER IMATH_HOSTDEVICE constexpr T abs(T a) IMATH_NOEXCEPT
Definition: ImathFun.h:26
void SetShouldInvokeUniqueChangedListener(bool shouldCall)
Definition: refBase.h:109
TfRefBase()
Definition: refBase.h:85