Jolt Physics
A multi core friendly Game Physics Engine
Loading...
Searching...
No Matches
Reference.h
Go to the documentation of this file.
1// Jolt Physics Library (https://github.com/jrouwe/JoltPhysics)
2// SPDX-FileCopyrightText: 2021 Jorrit Rouwe
3// SPDX-License-Identifier: MIT
4
5#pragma once
6
7#include <Jolt/Core/Atomics.h>
8
10
11// Forward declares
12template <class T> class Ref;
13template <class T> class RefConst;
14
33template <class T>
35{
36public:
38 inline RefTarget() = default;
39 inline RefTarget(const RefTarget &) { /* Do not copy refcount */ }
40 inline ~RefTarget() { JPH_IF_ENABLE_ASSERTS(uint32 value = mRefCount.load(memory_order_relaxed);) JPH_ASSERT(value == 0 || value == cEmbedded); }
41
45 inline void SetEmbedded() const { JPH_IF_ENABLE_ASSERTS(uint32 old = ) mRefCount.fetch_add(cEmbedded, memory_order_relaxed); JPH_ASSERT(old < cEmbedded); }
46
48 inline RefTarget & operator = (const RefTarget &) { /* Don't copy refcount */ return *this; }
49
51 uint32 GetRefCount() const { return mRefCount.load(memory_order_relaxed); }
52
54 inline void AddRef() const
55 {
56 // Adding a reference can use relaxed memory ordering
57 mRefCount.fetch_add(1, memory_order_relaxed);
58 }
59
60 inline void Release() const
61 {
62 #ifndef JPH_TSAN_ENABLED
63 // Releasing a reference must use release semantics...
64 if (mRefCount.fetch_sub(1, memory_order_release) == 1)
65 {
66 // ... so that we can use acquire to ensure that we see any updates from other threads that released a ref before deleting the object
67 atomic_thread_fence(memory_order_acquire);
68 delete static_cast<const T *>(this);
69 }
70 #else
71 // But under TSAN, we cannot use atomic_thread_fence, so we use an acq_rel operation unconditionally instead
72 if (mRefCount.fetch_sub(1, memory_order_acq_rel) == 1)
73 delete static_cast<const T *>(this);
74 #endif
75 }
76
78 static int sInternalGetRefCountOffset() { return offsetof(T, mRefCount); }
79
80protected:
81 static constexpr uint32 cEmbedded = 0x0ebedded;
82
83 mutable atomic<uint32> mRefCount = 0;
84};
85
88{
89public:
91 virtual ~RefTargetVirtual() = default;
92
94 virtual void AddRef() = 0;
95
97 virtual void Release() = 0;
98};
99
105template <class T>
106class Ref
107{
108public:
110 inline Ref() : mPtr(nullptr) { }
111 inline Ref(T *inRHS) : mPtr(inRHS) { AddRef(); }
112 inline Ref(const Ref<T> &inRHS) : mPtr(inRHS.mPtr) { AddRef(); }
113 inline Ref(Ref<T> &&inRHS) noexcept : mPtr(inRHS.mPtr) { inRHS.mPtr = nullptr; }
114 inline ~Ref() { Release(); }
115
117 inline Ref<T> & operator = (T *inRHS) { if (mPtr != inRHS) { Release(); mPtr = inRHS; AddRef(); } return *this; }
118 inline Ref<T> & operator = (const Ref<T> &inRHS) { if (mPtr != inRHS.mPtr) { Release(); mPtr = inRHS.mPtr; AddRef(); } return *this; }
119 inline Ref<T> & operator = (Ref<T> &&inRHS) noexcept { if (mPtr != inRHS.mPtr) { Release(); mPtr = inRHS.mPtr; inRHS.mPtr = nullptr; } return *this; }
120
122 inline operator T *() const { return mPtr; }
123
125 inline T * operator -> () const { return mPtr; }
126 inline T & operator * () const { return *mPtr; }
127
129 inline bool operator == (const T * inRHS) const { return mPtr == inRHS; }
130 inline bool operator == (const Ref<T> &inRHS) const { return mPtr == inRHS.mPtr; }
131 inline bool operator != (const T * inRHS) const { return mPtr != inRHS; }
132 inline bool operator != (const Ref<T> &inRHS) const { return mPtr != inRHS.mPtr; }
133
135 inline T * GetPtr() const { return mPtr; }
136
138 void ** InternalGetPointer() { return reinterpret_cast<void **>(&mPtr); }
139
140private:
141 template <class T2> friend class RefConst;
142
144 inline void AddRef() { if (mPtr != nullptr) mPtr->AddRef(); }
145 inline void Release() { if (mPtr != nullptr) mPtr->Release(); }
146
147 T * mPtr;
148};
149
155template <class T>
157{
158public:
160 inline RefConst() : mPtr(nullptr) { }
161 inline RefConst(const T * inRHS) : mPtr(inRHS) { AddRef(); }
162 inline RefConst(const RefConst<T> &inRHS) : mPtr(inRHS.mPtr) { AddRef(); }
163 inline RefConst(RefConst<T> &&inRHS) noexcept : mPtr(inRHS.mPtr) { inRHS.mPtr = nullptr; }
164 inline RefConst(const Ref<T> &inRHS) : mPtr(inRHS.mPtr) { AddRef(); }
165 inline RefConst(Ref<T> &&inRHS) noexcept : mPtr(inRHS.mPtr) { inRHS.mPtr = nullptr; }
166 inline ~RefConst() { Release(); }
167
169 inline RefConst<T> & operator = (const T * inRHS) { if (mPtr != inRHS) { Release(); mPtr = inRHS; AddRef(); } return *this; }
170 inline RefConst<T> & operator = (const RefConst<T> &inRHS) { if (mPtr != inRHS.mPtr) { Release(); mPtr = inRHS.mPtr; AddRef(); } return *this; }
171 inline RefConst<T> & operator = (RefConst<T> &&inRHS) noexcept { if (mPtr != inRHS.mPtr) { Release(); mPtr = inRHS.mPtr; inRHS.mPtr = nullptr; } return *this; }
172 inline RefConst<T> & operator = (const Ref<T> &inRHS) { if (mPtr != inRHS.mPtr) { Release(); mPtr = inRHS.mPtr; AddRef(); } return *this; }
173 inline RefConst<T> & operator = (Ref<T> &&inRHS) noexcept { if (mPtr != inRHS.mPtr) { Release(); mPtr = inRHS.mPtr; inRHS.mPtr = nullptr; } return *this; }
174
176 inline operator const T * () const { return mPtr; }
177
179 inline const T * operator -> () const { return mPtr; }
180 inline const T & operator * () const { return *mPtr; }
181
183 inline bool operator == (const T * inRHS) const { return mPtr == inRHS; }
184 inline bool operator == (const RefConst<T> &inRHS) const { return mPtr == inRHS.mPtr; }
185 inline bool operator == (const Ref<T> &inRHS) const { return mPtr == inRHS.mPtr; }
186 inline bool operator != (const T * inRHS) const { return mPtr != inRHS; }
187 inline bool operator != (const RefConst<T> &inRHS) const { return mPtr != inRHS.mPtr; }
188 inline bool operator != (const Ref<T> &inRHS) const { return mPtr != inRHS.mPtr; }
189
191 inline const T * GetPtr() const { return mPtr; }
192
194 void ** InternalGetPointer() { return const_cast<void **>(reinterpret_cast<const void **>(&mPtr)); }
195
196private:
198 inline void AddRef() { if (mPtr != nullptr) mPtr->AddRef(); }
199 inline void Release() { if (mPtr != nullptr) mPtr->Release(); }
200
201 const T * mPtr;
202};
203
205
206JPH_SUPPRESS_WARNING_PUSH
207JPH_CLANG_SUPPRESS_WARNING("-Wc++98-compat")
208
209namespace std
210{
212 template <class T>
213 struct hash<JPH::Ref<T>>
214 {
215 size_t operator () (const JPH::Ref<T> &inRHS) const
216 {
217 return hash<T *> { }(inRHS.GetPtr());
218 }
219 };
220
222 template <class T>
223 struct hash<JPH::RefConst<T>>
224 {
225 size_t operator () (const JPH::RefConst<T> &inRHS) const
226 {
227 return hash<const T *> { }(inRHS.GetPtr());
228 }
229 };
230}
231
232JPH_SUPPRESS_WARNING_POP
#define JPH_EXPORT
Definition: Core.h:236
#define JPH_NAMESPACE_END
Definition: Core.h:378
#define JPH_CLANG_SUPPRESS_WARNING(w)
Definition: Core.h:263
std::uint32_t uint32
Definition: Core.h:455
#define JPH_NAMESPACE_BEGIN
Definition: Core.h:372
#define JPH_IF_ENABLE_ASSERTS(...)
Definition: IssueReporting.h:35
#define JPH_ASSERT(...)
Definition: IssueReporting.h:33
Definition: Reference.h:157
const T * GetPtr() const
Get pointer.
Definition: Reference.h:191
bool operator!=(const T *inRHS) const
Definition: Reference.h:186
RefConst()
Constructor.
Definition: Reference.h:160
~RefConst()
Definition: Reference.h:166
RefConst(const T *inRHS)
Definition: Reference.h:161
const T & operator*() const
Definition: Reference.h:180
RefConst< T > & operator=(const T *inRHS)
Assignment operators.
Definition: Reference.h:169
RefConst(const Ref< T > &inRHS)
Definition: Reference.h:164
void ** InternalGetPointer()
INTERNAL HELPER FUNCTION USED BY SERIALIZATION.
Definition: Reference.h:194
const T * operator->() const
Access like a normal pointer.
Definition: Reference.h:179
RefConst(RefConst< T > &&inRHS) noexcept
Definition: Reference.h:163
RefConst(const RefConst< T > &inRHS)
Definition: Reference.h:162
bool operator==(const T *inRHS) const
Comparison.
Definition: Reference.h:183
RefConst(Ref< T > &&inRHS) noexcept
Definition: Reference.h:165
Definition: Reference.h:107
Ref(const Ref< T > &inRHS)
Definition: Reference.h:112
T * GetPtr() const
Get pointer.
Definition: Reference.h:135
bool operator!=(const T *inRHS) const
Definition: Reference.h:131
Ref()
Constructor.
Definition: Reference.h:110
~Ref()
Definition: Reference.h:114
T * operator->() const
Access like a normal pointer.
Definition: Reference.h:125
void ** InternalGetPointer()
INTERNAL HELPER FUNCTION USED BY SERIALIZATION.
Definition: Reference.h:138
Ref< T > & operator=(T *inRHS)
Assignment operators.
Definition: Reference.h:117
Ref(Ref< T > &&inRHS) noexcept
Definition: Reference.h:113
Ref(T *inRHS)
Definition: Reference.h:111
bool operator==(const T *inRHS) const
Comparison.
Definition: Reference.h:129
T & operator*() const
Definition: Reference.h:126
Definition: Reference.h:35
void Release() const
Definition: Reference.h:60
atomic< uint32 > mRefCount
Current reference count.
Definition: Reference.h:83
RefTarget & operator=(const RefTarget &)
Assignment operator.
Definition: Reference.h:48
void SetEmbedded() const
Definition: Reference.h:45
RefTarget()=default
Constructor.
RefTarget(const RefTarget &)
Definition: Reference.h:39
static int sInternalGetRefCountOffset()
INTERNAL HELPER FUNCTION USED BY SERIALIZATION.
Definition: Reference.h:78
~RefTarget()
assert no one is referencing us
Definition: Reference.h:40
uint32 GetRefCount() const
Get current refcount of this object.
Definition: Reference.h:51
void AddRef() const
Add or release a reference to this object.
Definition: Reference.h:54
static constexpr uint32 cEmbedded
A large value that gets added to the refcount to mark the object as embedded.
Definition: Reference.h:81
Pure virtual version of RefTarget.
Definition: Reference.h:88
virtual ~RefTargetVirtual()=default
Virtual destructor.
virtual void Release()=0
Virtual release reference.
virtual void AddRef()=0
Virtual add reference.
Definition: Array.h:575