Diligent Engine  v.2.4.g
ResourceReleaseQueue.hpp
Go to the documentation of this file.
1 /*
2  * Copyright 2019-2021 Diligent Graphics LLC
3  * Copyright 2015-2019 Egor Yusov
4  *
5  * Licensed under the Apache License, Version 2.0 (the "License");
6  * you may not use this file except in compliance with the License.
7  * You may obtain a copy of the License at
8  *
9  * http://www.apache.org/licenses/LICENSE-2.0
10  *
11  * Unless required by applicable law or agreed to in writing, software
12  * distributed under the License is distributed on an "AS IS" BASIS,
13  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14  * See the License for the specific language governing permissions and
15  * limitations under the License.
16  *
17  * In no event and under no legal theory, whether in tort (including negligence),
18  * contract, or otherwise, unless required by applicable law (such as deliberate
19  * and grossly negligent acts) or agreed to in writing, shall any Contributor be
20  * liable for any damages, including any direct, indirect, special, incidental,
21  * or consequential damages of any character arising as a result of this License or
22  * out of the use or inability to use the software (including but not limited to damages
23  * for loss of goodwill, work stoppage, computer failure or malfunction, or any and
24  * all other commercial damages or losses), even if such Contributor has been advised
25  * of the possibility of such damages.
26  */
27 
28 // Helper class that handles free memory block management to accommodate variable-size allocation requests
29 // See http://diligentgraphics.com/diligent-engine/architecture/d3d12/variable-size-memory-allocations-manager/
30 
31 #pragma once
32 
35 
36 #include <mutex>
37 #include <deque>
38 
39 #include "../../../Primitives/interface/MemoryAllocator.h"
40 #include "../../../Common/interface/STDAllocator.hpp"
41 #include "../../../Platforms/interface/Atomics.hpp"
42 #include "../../../Platforms/Basic/interface/DebugUtilities.hpp"
43 
44 namespace Diligent
45 {
46 
49 {
50 public:
51  // ___________________________ ___________________________
52  // |DynamicStaleResourceWrapper| |DynamicStaleResourceWrapper|
53  // | | | |
54  // | m_pStaleResource | | m_pStaleResource |
55  // |__________|________________| |__________|________________|
56  // | |
57  // | |
58  // | |
59  // __________V_______________________________________ __________V___________________________________
60  // |SpecificSharedStaleResource<VulkanBufferWrapper> | |SpecificStaleResource<VulkanMemoryAllocation> |
61  // | | | |
62  // | VulkanBufferWrapper m_SpecificResource; | | VulkanMemoryAllocation m_SpecificResource; |
63  // | AtomicLong m_RefCounter | |______________________________________________|
64  // |__________________________________________________|
65  //
66 
67  template <typename ResourceType, typename = typename std::enable_if<std::is_object<ResourceType>::value>::type>
68  static DynamicStaleResourceWrapper Create(ResourceType&& Resource, Atomics::Long NumReferences)
69  {
70  VERIFY_EXPR(NumReferences >= 1);
71 
72  class SpecificStaleResource final : public StaleResourceBase
73  {
74  public:
75  SpecificStaleResource(ResourceType&& SpecificResource) :
76  m_SpecificResource(std::move(SpecificResource))
77  {}
78 
79  // clang-format off
80  SpecificStaleResource (const SpecificStaleResource&) = delete;
81  SpecificStaleResource (SpecificStaleResource&&) = delete;
82  SpecificStaleResource& operator = (const SpecificStaleResource&) = delete;
83  SpecificStaleResource& operator = (SpecificStaleResource&&) = delete;
84  // clang-format on
85 
86  virtual void Release() override final
87  {
88  delete this;
89  }
90 
91  private:
92  ResourceType m_SpecificResource;
93  };
94 
95  class SpecificSharedStaleResource final : public StaleResourceBase
96  {
97  public:
98  SpecificSharedStaleResource(ResourceType&& SpecificResource, Atomics::Long NumReferences) :
99  m_SpecificResource(std::move(SpecificResource))
100  {
101  m_RefCounter = NumReferences;
102  }
103 
104  // clang-format off
105  SpecificSharedStaleResource (const SpecificSharedStaleResource&) = delete;
106  SpecificSharedStaleResource (SpecificSharedStaleResource&&) = delete;
107  SpecificSharedStaleResource& operator = (const SpecificSharedStaleResource&) = delete;
108  SpecificSharedStaleResource& operator = (SpecificSharedStaleResource&&) = delete;
109  // clang-format on
110 
111  virtual void Release() override final
112  {
113  if (Atomics::AtomicDecrement(m_RefCounter) == 0)
114  {
115  delete this;
116  }
117  }
118 
119  private:
120  ResourceType m_SpecificResource;
121  Atomics::AtomicLong m_RefCounter;
122  };
123 
125  NumReferences == 1 ?
126  static_cast<StaleResourceBase*>(new SpecificStaleResource{std::move(Resource)}) :
127  static_cast<StaleResourceBase*>(new SpecificSharedStaleResource{std::move(Resource), NumReferences})};
128  }
129 
131  m_pStaleResource(std::move(rhs.m_pStaleResource))
132  {
133  rhs.m_pStaleResource = nullptr;
134  }
135 
137  m_pStaleResource{rhs.m_pStaleResource}
138  {
139  }
140 
141  // clang-format off
144  // clang-format on
145 
147  {
148  m_pStaleResource = nullptr;
149  }
150 
152  {
153  if (m_pStaleResource != nullptr)
154  m_pStaleResource->Release();
155  }
156 
157 private:
158  class StaleResourceBase
159  {
160  public:
161  virtual ~StaleResourceBase() = 0;
162  virtual void Release() = 0;
163  };
164 
165  DynamicStaleResourceWrapper(StaleResourceBase* pStaleResource) :
166  m_pStaleResource(pStaleResource)
167  {}
168 
169  StaleResourceBase* m_pStaleResource;
170 };
171 
172 inline DynamicStaleResourceWrapper::StaleResourceBase::~StaleResourceBase()
173 {
174 }
175 
177 template <typename ResourceType>
179 {
180 public:
181  static StaticStaleResourceWrapper Create(ResourceType&& Resource, Atomics::Long NumReferences)
182  {
183  VERIFY(NumReferences == 1, "Number of references must be 1 for StaticStaleResourceWrapper");
184  return StaticStaleResourceWrapper{std::move(Resource)};
185  }
186 
188  m_StaleResource(std::move(rhs.m_StaleResource))
189  {}
190 
192  {
193  m_StaleResource = std::move(rhs.m_StaleResource);
194  return *this;
195  }
196 
197  // clang-format off
200  // clang-format on
201 
202 private:
203  StaticStaleResourceWrapper(ResourceType&& StaleResource) :
204  m_StaleResource{std::move(StaleResource)}
205  {}
206 
207  ResourceType m_StaleResource;
208 };
209 
211 
220 template <typename ResourceWrapperType>
222 {
223 public:
224  // clang-format off
226  m_ReleaseQueue (STD_ALLOCATOR_RAW_MEM(ReleaseQueueElemType, Allocator, "Allocator for deque<ReleaseQueueElemType>")),
227  m_StaleResources(STD_ALLOCATOR_RAW_MEM(ReleaseQueueElemType, Allocator, "Allocator for deque<ReleaseQueueElemType>"))
228  {}
229  // clang-format on
230 
232  {
233  DEV_CHECK_ERR(m_StaleResources.empty(), "Not all stale objects were destroyed");
234  DEV_CHECK_ERR(m_ReleaseQueue.empty(), "Release queue is not empty");
235  }
236 
240  template <typename ResourceType, typename = typename std::enable_if<std::is_object<ResourceType>::value>::type>
241  static ResourceWrapperType CreateWrapper(ResourceType&& Resource, Atomics::Long NumReferences)
242  {
243  return ResourceWrapperType::Create(std::move(Resource), NumReferences);
244  }
245 
249  template <typename ResourceType, typename = typename std::enable_if<std::is_object<ResourceType>::value>::type>
250  void SafeReleaseResource(ResourceType&& Resource, Uint64 NextCommandListNumber)
251  {
252  SafeReleaseResource(CreateWrapper(std::move(Resource), 1), NextCommandListNumber);
253  }
254 
258  void SafeReleaseResource(ResourceWrapperType&& Wrapper, Uint64 NextCommandListNumber)
259  {
260  std::lock_guard<std::mutex> LockGuard(m_StaleObjectsMutex);
261  m_StaleResources.emplace_back(NextCommandListNumber, std::move(Wrapper));
262  }
263 
267  void SafeReleaseResource(const ResourceWrapperType& Wrapper, Uint64 NextCommandListNumber)
268  {
269  std::lock_guard<std::mutex> LockGuard(m_StaleObjectsMutex);
270  m_StaleResources.emplace_back(NextCommandListNumber, Wrapper);
271  }
272 
276  template <typename ResourceType, typename = typename std::enable_if<std::is_object<ResourceType>::value>::type>
277  void DiscardResource(ResourceType&& Resource, Uint64 FenceValue)
278  {
279  DiscardResource(CreateWrapper(std::move(Resource), 1), FenceValue);
280  }
281 
285  void DiscardResource(ResourceWrapperType&& Wrapper, Uint64 FenceValue)
286  {
287  std::lock_guard<std::mutex> ReleaseQueueLock(m_ReleaseQueueMutex);
288  m_ReleaseQueue.emplace_back(FenceValue, std::move(Wrapper));
289  }
290 
294  void DiscardResource(const ResourceWrapperType& Wrapper, Uint64 FenceValue)
295  {
296  std::lock_guard<std::mutex> ReleaseQueueLock(m_ReleaseQueueMutex);
297  m_ReleaseQueue.emplace_back(FenceValue, Wrapper);
298  }
299 
303  template <typename ResourceType, typename IteratorType>
304  void DiscardResources(Uint64 FenceValue, IteratorType Iterator)
305  {
306  std::lock_guard<std::mutex> ReleaseQueueLock(m_ReleaseQueueMutex);
307  ResourceType Resource;
308  while (Iterator(Resource))
309  {
310  m_ReleaseQueue.emplace_back(FenceValue, CreateWrapper(std::move(Resource), 1));
311  }
312  }
313 
321  void DiscardStaleResources(Uint64 SubmittedCmdBuffNumber, Uint64 FenceValue)
322  {
323  // Only discard these stale objects that were released before CmdBuffNumber
324  // was executed
325  std::lock_guard<std::mutex> StaleObjectsLock(m_StaleObjectsMutex);
326  std::lock_guard<std::mutex> ReleaseQueueLock(m_ReleaseQueueMutex);
327  while (!m_StaleResources.empty())
328  {
329  auto& FirstStaleObj = m_StaleResources.front();
330  if (FirstStaleObj.first <= SubmittedCmdBuffNumber)
331  {
332  m_ReleaseQueue.emplace_back(FenceValue, std::move(FirstStaleObj.second));
333  m_StaleResources.pop_front();
334  }
335  else
336  break;
337  }
338  }
339 
340 
344  void Purge(Uint64 CompletedFenceValue)
345  {
346  std::lock_guard<std::mutex> LockGuard(m_ReleaseQueueMutex);
347 
348  // Release all objects whose associated fence value is at most CompletedFenceValue
349  // See http://diligentgraphics.com/diligent-engine/architecture/d3d12/managing-resource-lifetimes/
350  while (!m_ReleaseQueue.empty())
351  {
352  auto& FirstObj = m_ReleaseQueue.front();
353  if (FirstObj.first <= CompletedFenceValue)
354  m_ReleaseQueue.pop_front();
355  else
356  break;
357  }
358  }
359 
361  size_t GetStaleResourceCount() const
362  {
363  return m_StaleResources.size();
364  }
365 
368  {
369  return m_ReleaseQueue.size();
370  }
371 
372 private:
373  std::mutex m_ReleaseQueueMutex;
374  using ReleaseQueueElemType = std::pair<Uint64, ResourceWrapperType>;
375  std::deque<ReleaseQueueElemType, STDAllocatorRawMem<ReleaseQueueElemType>> m_ReleaseQueue;
376 
377  std::mutex m_StaleObjectsMutex;
378  std::deque<ReleaseQueueElemType, STDAllocatorRawMem<ReleaseQueueElemType>> m_StaleResources;
379 };
380 
381 } // namespace Diligent
Diligent::ResourceReleaseQueue::GetPendingReleaseResourceCount
size_t GetPendingReleaseResourceCount() const
Returns the number of resources pending release.
Definition: ResourceReleaseQueue.hpp:367
Diligent::DynamicStaleResourceWrapper
Helper class that wraps stale resources of different types.
Definition: ResourceReleaseQueue.hpp:48
BasicAtomics::Long
long Long
Definition: BasicAtomics.hpp:34
Diligent::StaticStaleResourceWrapper
Helper class that wraps stale resources of the same type.
Definition: ResourceReleaseQueue.hpp:178
Diligent::Uint64
uint64_t Uint64
64-bit unsigned integer
Definition: BasicTypes.h:50
Diligent::ResourceReleaseQueue::SafeReleaseResource
void SafeReleaseResource(const ResourceWrapperType &Wrapper, Uint64 NextCommandListNumber)
Moves a copy of the resource wrapper to the stale resources queue.
Definition: ResourceReleaseQueue.hpp:267
Diligent::StaticStaleResourceWrapper::StaticStaleResourceWrapper
StaticStaleResourceWrapper(StaticStaleResourceWrapper &&rhs) noexcept
Definition: ResourceReleaseQueue.hpp:187
DEV_CHECK_ERR
#define DEV_CHECK_ERR(...)
Definition: DebugUtilities.hpp:90
Diligent::ResourceReleaseQueue
Facilitates safe resource destruction in D3D12 and Vulkan.
Definition: ResourceReleaseQueue.hpp:221
Diligent::ResourceReleaseQueue::DiscardResources
void DiscardResources(Uint64 FenceValue, IteratorType Iterator)
Adds multiple resources directly to the release queue.
Definition: ResourceReleaseQueue.hpp:304
Diligent::StaticStaleResourceWrapper::operator=
StaticStaleResourceWrapper & operator=(StaticStaleResourceWrapper &&rhs) noexcept
Definition: ResourceReleaseQueue.hpp:191
Diligent::StaticStaleResourceWrapper::Create
static StaticStaleResourceWrapper Create(ResourceType &&Resource, Atomics::Long NumReferences)
Definition: ResourceReleaseQueue.hpp:181
Diligent::ResourceReleaseQueue::SafeReleaseResource
void SafeReleaseResource(ResourceType &&Resource, Uint64 NextCommandListNumber)
Moves a resource to the stale resources queue.
Definition: ResourceReleaseQueue.hpp:250
Diligent::ResourceReleaseQueue::DiscardResource
void DiscardResource(ResourceWrapperType &&Wrapper, Uint64 FenceValue)
Adds a resource wrapper directly to the release queue.
Definition: ResourceReleaseQueue.hpp:285
Diligent::ResourceReleaseQueue::DiscardResource
void DiscardResource(ResourceType &&Resource, Uint64 FenceValue)
Adds a resource directly to the release queue.
Definition: ResourceReleaseQueue.hpp:277
Diligent::ResourceReleaseQueue::DiscardStaleResources
void DiscardStaleResources(Uint64 SubmittedCmdBuffNumber, Uint64 FenceValue)
Moves stale objects to the release queue.
Definition: ResourceReleaseQueue.hpp:321
Diligent::ResourceReleaseQueue::ResourceReleaseQueue
ResourceReleaseQueue(IMemoryAllocator &Allocator)
Definition: ResourceReleaseQueue.hpp:225
Diligent::ResourceReleaseQueue::Purge
void Purge(Uint64 CompletedFenceValue)
Removes all objects from the release queue whose fence value is less than or equal to CompletedFenceV...
Definition: ResourceReleaseQueue.hpp:344
Diligent::DynamicStaleResourceWrapper::operator=
DynamicStaleResourceWrapper & operator=(const DynamicStaleResourceWrapper &)=delete
Diligent::DynamicStaleResourceWrapper::DynamicStaleResourceWrapper
DynamicStaleResourceWrapper(DynamicStaleResourceWrapper &&rhs) noexcept
Definition: ResourceReleaseQueue.hpp:130
Diligent::ResourceReleaseQueue::SafeReleaseResource
void SafeReleaseResource(ResourceWrapperType &&Wrapper, Uint64 NextCommandListNumber)
Moves a resource wrapper to the stale resources queue.
Definition: ResourceReleaseQueue.hpp:258
BasicAtomics::AtomicLong
std::atomic< Long > AtomicLong
Definition: BasicAtomics.hpp:35
Diligent::ResourceReleaseQueue::DiscardResource
void DiscardResource(const ResourceWrapperType &Wrapper, Uint64 FenceValue)
Adds a copy of the resource wrapper directly to the release queue.
Definition: ResourceReleaseQueue.hpp:294
BasicAtomics::AtomicDecrement
static Type AtomicDecrement(std::atomic< Type > &Val)
Definition: BasicAtomics.hpp:48
Diligent::DynamicStaleResourceWrapper::GiveUpOwnership
void GiveUpOwnership()
Definition: ResourceReleaseQueue.hpp:146
Diligent::ResourceReleaseQueue::~ResourceReleaseQueue
~ResourceReleaseQueue()
Definition: ResourceReleaseQueue.hpp:231
Diligent::IMemoryAllocator
Base interface for a raw memory allocator.
Definition: MemoryAllocator.h:41
STD_ALLOCATOR_RAW_MEM
#define STD_ALLOCATOR_RAW_MEM(Type, Allocator, Description)
Definition: STDAllocator.hpp:179
Diligent::ResourceReleaseQueue::GetStaleResourceCount
size_t GetStaleResourceCount() const
Returns the number of stale resources.
Definition: ResourceReleaseQueue.hpp:361
Diligent::DynamicStaleResourceWrapper::DynamicStaleResourceWrapper
DynamicStaleResourceWrapper(const DynamicStaleResourceWrapper &rhs) noexcept
Definition: ResourceReleaseQueue.hpp:136
VERIFY_EXPR
#define VERIFY_EXPR(...)
Definition: DebugUtilities.hpp:79
Diligent::ResourceReleaseQueue::CreateWrapper
static ResourceWrapperType CreateWrapper(ResourceType &&Resource, Atomics::Long NumReferences)
Creates a resource wrapper for the specific resource type.
Definition: ResourceReleaseQueue.hpp:241
VERIFY
#define VERIFY(...)
Definition: DebugUtilities.hpp:76
Diligent::DynamicStaleResourceWrapper::~DynamicStaleResourceWrapper
~DynamicStaleResourceWrapper()
Definition: ResourceReleaseQueue.hpp:151
Diligent::DynamicStaleResourceWrapper::Create
static DynamicStaleResourceWrapper Create(ResourceType &&Resource, Atomics::Long NumReferences)
Definition: ResourceReleaseQueue.hpp:68
Diligent
The library uses Direct3D-style math:
Definition: AdvancedMath.hpp:37