mirror of
				https://github.com/encounter/dawn-cmake.git
				synced 2025-10-24 18:50:29 +00:00 
			
		
		
		
	Use CommandAllocatorManager to create and track lifetimes of ID3D12CommandAllocators
This commit is contained in:
		
							parent
							
								
									2157002570
								
							
						
					
					
						commit
						78f1619446
					
				| @ -18,6 +18,7 @@ | |||||||
| #include "GLFW/glfw3.h" | #include "GLFW/glfw3.h" | ||||||
| #include "GLFW/glfw3native.h" | #include "GLFW/glfw3native.h" | ||||||
| 
 | 
 | ||||||
|  | #include <initializer_list> | ||||||
| #include <assert.h> | #include <assert.h> | ||||||
| #include <wrl.h> | #include <wrl.h> | ||||||
| #include <d3d12.h> | #include <d3d12.h> | ||||||
| @ -32,6 +33,11 @@ namespace d3d12 { | |||||||
|     void Init(ComPtr<ID3D12Device> d3d12Device, nxtProcTable* procs, nxtDevice* device); |     void Init(ComPtr<ID3D12Device> d3d12Device, nxtProcTable* procs, nxtDevice* device); | ||||||
|     ComPtr<ID3D12CommandQueue> GetCommandQueue(nxtDevice device); |     ComPtr<ID3D12CommandQueue> GetCommandQueue(nxtDevice device); | ||||||
|     void SetNextRenderTargetDescriptor(nxtDevice device, D3D12_CPU_DESCRIPTOR_HANDLE renderTargetDescriptor); |     void SetNextRenderTargetDescriptor(nxtDevice device, D3D12_CPU_DESCRIPTOR_HANDLE renderTargetDescriptor); | ||||||
|  |     uint64_t GetSerial(const nxtDevice device); | ||||||
|  |     void NextSerial(nxtDevice device); | ||||||
|  |     void ExecuteCommandLists(nxtDevice device, std::initializer_list<ID3D12CommandList*> commandLists); | ||||||
|  |     void WaitForSerial(nxtDevice device, uint64_t serial); | ||||||
|  |     ComPtr<ID3D12CommandAllocator> ReserveCommandAllocator(nxtDevice device); | ||||||
| } | } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -102,15 +108,13 @@ class D3D12Binding : public BackendBinding { | |||||||
| 
 | 
 | ||||||
|             rtvDescriptorSize = d3d12Device->GetDescriptorHandleIncrementSize(D3D12_DESCRIPTOR_HEAP_TYPE_RTV); |             rtvDescriptorSize = d3d12Device->GetDescriptorHandleIncrementSize(D3D12_DESCRIPTOR_HEAP_TYPE_RTV); | ||||||
| 
 | 
 | ||||||
|             // Create a RTV and command allocators for each frame.
 |             // Create a RTV for each frame.
 | ||||||
|             { |             { | ||||||
|                 D3D12_CPU_DESCRIPTOR_HANDLE renderTargetViewHandle = renderTargetViewHeap->GetCPUDescriptorHandleForHeapStart(); |                 D3D12_CPU_DESCRIPTOR_HANDLE renderTargetViewHandle = renderTargetViewHeap->GetCPUDescriptorHandleForHeapStart(); | ||||||
|                 for (uint32_t n = 0; n < kFrameCount; ++n) { |                 for (uint32_t n = 0; n < kFrameCount; ++n) { | ||||||
|                     ASSERT_SUCCESS(swapChain->GetBuffer(n, IID_PPV_ARGS(&renderTargetResources[n]))); |                     ASSERT_SUCCESS(swapChain->GetBuffer(n, IID_PPV_ARGS(&renderTargetResources[n]))); | ||||||
|                     d3d12Device->CreateRenderTargetView(renderTargetResources[n].Get(), nullptr, renderTargetViewHandle); |                     d3d12Device->CreateRenderTargetView(renderTargetResources[n].Get(), nullptr, renderTargetViewHandle); | ||||||
|                     renderTargetViewHandle.ptr += rtvDescriptorSize; |                     renderTargetViewHandle.ptr += rtvDescriptorSize; | ||||||
| 
 |  | ||||||
|                     ASSERT_SUCCESS(d3d12Device->CreateCommandAllocator(D3D12_COMMAND_LIST_TYPE_DIRECT, IID_PPV_ARGS(&commandAllocators[n]))); |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
| @ -118,23 +122,19 @@ class D3D12Binding : public BackendBinding { | |||||||
|             previousRenderTargetIndex = renderTargetIndex = swapChain->GetCurrentBackBufferIndex(); |             previousRenderTargetIndex = renderTargetIndex = swapChain->GetCurrentBackBufferIndex(); | ||||||
|             previousRenderTargetIndex = renderTargetIndex == 0 ? 1 : 0; |             previousRenderTargetIndex = renderTargetIndex == 0 ? 1 : 0; | ||||||
| 
 | 
 | ||||||
|             // Initialize the current frame, the last completed frame, and all last frame each render target was used to 0
 |             // Initial the serial for all render targets
 | ||||||
|             // so that it looks like we completed all previous frames
 |             const uint64_t initialSerial = backend::d3d12::GetSerial(backendDevice); | ||||||
|             currentFrameNumber = 0; |  | ||||||
|             for (uint32_t n = 0; n < kFrameCount; ++n) { |             for (uint32_t n = 0; n < kFrameCount; ++n) { | ||||||
|                 lastFrameRenderTargetWasUsed[n] = 0; |                 lastSerialRenderTargetWasUsed[n] = initialSerial; | ||||||
|             } |             } | ||||||
|             ASSERT_SUCCESS(d3d12Device->CreateFence(0, D3D12_FENCE_FLAG_NONE, IID_PPV_ARGS(&fence))); |  | ||||||
| 
 |  | ||||||
|             fenceEvent = CreateEvent(nullptr, FALSE, FALSE, nullptr); |  | ||||||
|             ASSERT(fenceEvent != nullptr); |  | ||||||
| 
 | 
 | ||||||
|             // Transition the first frame to be a render target
 |             // Transition the first frame to be a render target
 | ||||||
|             { |             { | ||||||
|  |                 ComPtr<ID3D12CommandAllocator> commandAllocator = backend::d3d12::ReserveCommandAllocator(backendDevice); | ||||||
|                 ASSERT_SUCCESS(d3d12Device->CreateCommandList( |                 ASSERT_SUCCESS(d3d12Device->CreateCommandList( | ||||||
|                     0, |                     0, | ||||||
|                     D3D12_COMMAND_LIST_TYPE_DIRECT, |                     D3D12_COMMAND_LIST_TYPE_DIRECT, | ||||||
|                     commandAllocators[previousRenderTargetIndex].Get(), |                     commandAllocator.Get(), | ||||||
|                     nullptr, |                     nullptr, | ||||||
|                     IID_PPV_ARGS(&commandList) |                     IID_PPV_ARGS(&commandList) | ||||||
|                 )); |                 )); | ||||||
| @ -149,7 +149,9 @@ class D3D12Binding : public BackendBinding { | |||||||
|                 commandList->ResourceBarrier(1, &resourceBarrier); |                 commandList->ResourceBarrier(1, &resourceBarrier); | ||||||
|                 ASSERT_SUCCESS(commandList->Close()); |                 ASSERT_SUCCESS(commandList->Close()); | ||||||
|                 ID3D12CommandList* commandLists[] = { commandList.Get() }; |                 ID3D12CommandList* commandLists[] = { commandList.Get() }; | ||||||
|                 commandQueue->ExecuteCommandLists(_countof(commandLists), commandLists); |                 backend::d3d12::ExecuteCommandLists(backendDevice, { commandList.Get() }); | ||||||
|  | 
 | ||||||
|  |                 backend::d3d12::NextSerial(backendDevice); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             D3D12_CPU_DESCRIPTOR_HANDLE renderTargetViewHandle = renderTargetViewHeap->GetCPUDescriptorHandleForHeapStart(); |             D3D12_CPU_DESCRIPTOR_HANDLE renderTargetViewHandle = renderTargetViewHeap->GetCPUDescriptorHandleForHeapStart(); | ||||||
| @ -160,7 +162,8 @@ class D3D12Binding : public BackendBinding { | |||||||
|         void SwapBuffers() override { |         void SwapBuffers() override { | ||||||
|             // Transition current frame's render target for presenting
 |             // Transition current frame's render target for presenting
 | ||||||
|             { |             { | ||||||
|                 ASSERT_SUCCESS(commandList->Reset(commandAllocators[renderTargetIndex].Get(), nullptr)); |                 ComPtr<ID3D12CommandAllocator> commandAllocator = backend::d3d12::ReserveCommandAllocator(backendDevice); | ||||||
|  |                 ASSERT_SUCCESS(commandList->Reset(commandAllocator.Get(), nullptr)); | ||||||
|                 D3D12_RESOURCE_BARRIER resourceBarrier; |                 D3D12_RESOURCE_BARRIER resourceBarrier; | ||||||
|                 resourceBarrier.Transition.pResource = renderTargetResources[renderTargetIndex].Get(); |                 resourceBarrier.Transition.pResource = renderTargetResources[renderTargetIndex].Get(); | ||||||
|                 resourceBarrier.Transition.StateBefore = D3D12_RESOURCE_STATE_RENDER_TARGET; |                 resourceBarrier.Transition.StateBefore = D3D12_RESOURCE_STATE_RENDER_TARGET; | ||||||
| @ -171,14 +174,15 @@ class D3D12Binding : public BackendBinding { | |||||||
|                 commandList->ResourceBarrier(1, &resourceBarrier); |                 commandList->ResourceBarrier(1, &resourceBarrier); | ||||||
|                 ASSERT_SUCCESS(commandList->Close()); |                 ASSERT_SUCCESS(commandList->Close()); | ||||||
|                 ID3D12CommandList* commandLists[] = { commandList.Get() }; |                 ID3D12CommandList* commandLists[] = { commandList.Get() }; | ||||||
|                 commandQueue->ExecuteCommandLists(_countof(commandLists), commandLists); |                 backend::d3d12::ExecuteCommandLists(backendDevice, { commandList.Get() }); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             ASSERT_SUCCESS(swapChain->Present(1, 0)); |             ASSERT_SUCCESS(swapChain->Present(1, 0)); | ||||||
| 
 | 
 | ||||||
|             // Transition last frame's render target back to being a render target
 |             // Transition last frame's render target back to being a render target
 | ||||||
|             { |             { | ||||||
|                 ASSERT_SUCCESS(commandList->Reset(commandAllocators[renderTargetIndex].Get(), nullptr)); |                 ComPtr<ID3D12CommandAllocator> commandAllocator = backend::d3d12::ReserveCommandAllocator(backendDevice); | ||||||
|  |                 ASSERT_SUCCESS(commandList->Reset(commandAllocator.Get(), nullptr)); | ||||||
|                 D3D12_RESOURCE_BARRIER resourceBarrier; |                 D3D12_RESOURCE_BARRIER resourceBarrier; | ||||||
|                 resourceBarrier.Transition.pResource = renderTargetResources[previousRenderTargetIndex].Get(); |                 resourceBarrier.Transition.pResource = renderTargetResources[previousRenderTargetIndex].Get(); | ||||||
|                 resourceBarrier.Transition.StateBefore = D3D12_RESOURCE_STATE_PRESENT; |                 resourceBarrier.Transition.StateBefore = D3D12_RESOURCE_STATE_PRESENT; | ||||||
| @ -189,29 +193,20 @@ class D3D12Binding : public BackendBinding { | |||||||
|                 commandList->ResourceBarrier(1, &resourceBarrier); |                 commandList->ResourceBarrier(1, &resourceBarrier); | ||||||
|                 ASSERT_SUCCESS(commandList->Close()); |                 ASSERT_SUCCESS(commandList->Close()); | ||||||
|                 ID3D12CommandList* commandLists[] = { commandList.Get() }; |                 ID3D12CommandList* commandLists[] = { commandList.Get() }; | ||||||
|                 commandQueue->ExecuteCommandLists(_countof(commandLists), commandLists); |                 backend::d3d12::ExecuteCommandLists(backendDevice, { commandList.Get() }); | ||||||
|             } |             } | ||||||
|             ASSERT_SUCCESS(commandQueue->Signal(fence.Get(), currentFrameNumber)); |  | ||||||
| 
 | 
 | ||||||
|             // Advance to the next frame
 |             backend::d3d12::NextSerial(backendDevice); | ||||||
|             currentFrameNumber++; |  | ||||||
| 
 | 
 | ||||||
|             previousRenderTargetIndex = renderTargetIndex; |             previousRenderTargetIndex = renderTargetIndex; | ||||||
|             renderTargetIndex = swapChain->GetCurrentBackBufferIndex(); |             renderTargetIndex = swapChain->GetCurrentBackBufferIndex(); | ||||||
| 
 | 
 | ||||||
|             // If the next render target is not ready to be rendered yet, wait until it is ready.
 |             // If the next render target is not ready to be rendered yet, wait until it is ready.
 | ||||||
|             // If the last completed frame is less than the last requested frame for this render target,
 |             // If the last completed serial is less than the last requested serial for this render target,
 | ||||||
|             // then the commands previously executed on this render target have not yet completed
 |             // then the commands previously executed on this render target have not yet completed
 | ||||||
|             const uint64_t lastFrameCompletedOnGPU = fence->GetCompletedValue(); |             backend::d3d12::WaitForSerial(backendDevice, lastSerialRenderTargetWasUsed[renderTargetIndex]); | ||||||
|             if (lastFrameCompletedOnGPU < lastFrameRenderTargetWasUsed[renderTargetIndex]) { |  | ||||||
|                 ASSERT_SUCCESS(fence->SetEventOnCompletion(lastFrameRenderTargetWasUsed[renderTargetIndex], fenceEvent)); |  | ||||||
|                 WaitForSingleObjectEx(fenceEvent, INFINITE, FALSE); |  | ||||||
|             } |  | ||||||
| 
 | 
 | ||||||
|             lastFrameRenderTargetWasUsed[renderTargetIndex] = currentFrameNumber; |             lastSerialRenderTargetWasUsed[renderTargetIndex] = backend::d3d12::GetSerial(backendDevice); | ||||||
| 
 |  | ||||||
|             // The block above checked that the commands in this allocator are done executing
 |  | ||||||
|             ASSERT_SUCCESS(commandAllocators[renderTargetIndex]->Reset()); |  | ||||||
| 
 | 
 | ||||||
|             // Tell the backend to render to the current render target
 |             // Tell the backend to render to the current render target
 | ||||||
|             D3D12_CPU_DESCRIPTOR_HANDLE renderTargetViewHandle = renderTargetViewHeap->GetCPUDescriptorHandleForHeapStart(); |             D3D12_CPU_DESCRIPTOR_HANDLE renderTargetViewHandle = renderTargetViewHeap->GetCPUDescriptorHandleForHeapStart(); | ||||||
| @ -237,12 +232,8 @@ class D3D12Binding : public BackendBinding { | |||||||
|         // Frame synchronization. Updated every frame
 |         // Frame synchronization. Updated every frame
 | ||||||
|         uint32_t renderTargetIndex; |         uint32_t renderTargetIndex; | ||||||
|         uint32_t previousRenderTargetIndex; |         uint32_t previousRenderTargetIndex; | ||||||
|         uint64_t currentFrameNumber; |         uint64_t lastSerialRenderTargetWasUsed[kFrameCount]; | ||||||
|         uint64_t lastFrameRenderTargetWasUsed[kFrameCount]; |  | ||||||
|         ComPtr<ID3D12CommandAllocator> commandAllocators[kFrameCount]; |  | ||||||
|         ComPtr<ID3D12GraphicsCommandList> commandList; |         ComPtr<ID3D12GraphicsCommandList> commandList; | ||||||
|         ComPtr<ID3D12Fence> fence; |  | ||||||
|         HANDLE fenceEvent; |  | ||||||
| 
 | 
 | ||||||
|         static void ASSERT_SUCCESS(HRESULT hr) { |         static void ASSERT_SUCCESS(HRESULT hr) { | ||||||
|             assert(SUCCEEDED(hr)); |             assert(SUCCEEDED(hr)); | ||||||
|  | |||||||
| @ -219,6 +219,8 @@ if (WIN32) | |||||||
|     list(APPEND BACKEND_SOURCES |     list(APPEND BACKEND_SOURCES | ||||||
|         ${D3D12_DIR}/BufferD3D12.cpp |         ${D3D12_DIR}/BufferD3D12.cpp | ||||||
|         ${D3D12_DIR}/BufferD3D12.h |         ${D3D12_DIR}/BufferD3D12.h | ||||||
|  |         ${D3D12_DIR}/CommandAllocatorManager.cpp | ||||||
|  |         ${D3D12_DIR}/CommandAllocatorManager.h | ||||||
|         ${D3D12_DIR}/CommandBufferD3D12.cpp |         ${D3D12_DIR}/CommandBufferD3D12.cpp | ||||||
|         ${D3D12_DIR}/CommandBufferD3D12.h |         ${D3D12_DIR}/CommandBufferD3D12.h | ||||||
|         ${D3D12_DIR}/D3D12Backend.cpp |         ${D3D12_DIR}/D3D12Backend.cpp | ||||||
|  | |||||||
| @ -78,6 +78,8 @@ namespace backend { | |||||||
|             void Clear(); |             void Clear(); | ||||||
|             void ClearUpTo(Serial serial); |             void ClearUpTo(Serial serial); | ||||||
| 
 | 
 | ||||||
|  |             Serial FirstSerial() const; | ||||||
|  | 
 | ||||||
|         private: |         private: | ||||||
|             // Returns the first StorageIterator that a serial bigger than serial.
 |             // Returns the first StorageIterator that a serial bigger than serial.
 | ||||||
|             StorageIterator FindUpTo(Serial serial) const; |             StorageIterator FindUpTo(Serial serial) const; | ||||||
| @ -143,6 +145,12 @@ namespace backend { | |||||||
|         storage.erase(storage.begin(), FindUpTo(serial)); |         storage.erase(storage.begin(), FindUpTo(serial)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     template<typename T> | ||||||
|  |     Serial SerialQueue<T>::FirstSerial() const { | ||||||
|  |         ASSERT(!Empty()); | ||||||
|  |         return storage.front().first; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     template<typename T> |     template<typename T> | ||||||
|     typename SerialQueue<T>::StorageIterator SerialQueue<T>::FindUpTo(Serial serial) const { |     typename SerialQueue<T>::StorageIterator SerialQueue<T>::FindUpTo(Serial serial) const { | ||||||
|         auto it = storage.begin(); |         auto it = storage.begin(); | ||||||
|  | |||||||
							
								
								
									
										66
									
								
								src/backend/d3d12/CommandAllocatorManager.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								src/backend/d3d12/CommandAllocatorManager.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,66 @@ | |||||||
|  | // Copyright 2017 The NXT Authors
 | ||||||
|  | //
 | ||||||
|  | // Licensed under the Apache License, Version 2.0 (the "License");
 | ||||||
|  | // you may not use this file except in compliance with the License.
 | ||||||
|  | // You may obtain a copy of the License at
 | ||||||
|  | //
 | ||||||
|  | //     http://www.apache.org/licenses/LICENSE-2.0
 | ||||||
|  | //
 | ||||||
|  | // Unless required by applicable law or agreed to in writing, software
 | ||||||
|  | // distributed under the License is distributed on an "AS IS" BASIS,
 | ||||||
|  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | ||||||
|  | // See the License for the specific language governing permissions and
 | ||||||
|  | // limitations under the License.
 | ||||||
|  | 
 | ||||||
|  | #include "CommandAllocatorManager.h" | ||||||
|  | 
 | ||||||
|  | #include "D3D12Backend.h" | ||||||
|  | 
 | ||||||
|  | #include "common/BitSetIterator.h" | ||||||
|  | 
 | ||||||
|  | namespace backend { | ||||||
|  | namespace d3d12 { | ||||||
|  | 
 | ||||||
|  |     CommandAllocatorManager::CommandAllocatorManager(Device* device) : device(device), allocatorCount(0) { | ||||||
|  |         freeAllocators.set(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ComPtr<ID3D12CommandAllocator> CommandAllocatorManager::ReserveCommandAllocator() { | ||||||
|  |         // If there are no free allocators, get the oldest serial in flight and wait on it
 | ||||||
|  |         if (freeAllocators.none()) { | ||||||
|  |             const uint64_t firstSerial = inFlightCommandAllocators.FirstSerial(); | ||||||
|  |             device->WaitForSerial(firstSerial); | ||||||
|  |             ResetCompletedAllocators(firstSerial); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         ASSERT(freeAllocators.any()); | ||||||
|  | 
 | ||||||
|  |         // Get the index of the first free allocator from the bitset
 | ||||||
|  |         unsigned int firstFreeIndex = *(IterateBitSet(freeAllocators).begin()); | ||||||
|  | 
 | ||||||
|  |         if (firstFreeIndex >= allocatorCount) { | ||||||
|  |             ASSERT(firstFreeIndex == allocatorCount); | ||||||
|  |             allocatorCount++; | ||||||
|  |             ASSERT_SUCCESS(device->GetD3D12Device()->CreateCommandAllocator(D3D12_COMMAND_LIST_TYPE_DIRECT, IID_PPV_ARGS(&commandAllocators[firstFreeIndex]))); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // Mark the command allocator as used
 | ||||||
|  |         freeAllocators.reset(firstFreeIndex); | ||||||
|  | 
 | ||||||
|  |         // Enqueue the command allocator. It will be scheduled for reset after the next ExecuteCommandLists
 | ||||||
|  |         inFlightCommandAllocators.Enqueue({commandAllocators[firstFreeIndex], firstFreeIndex}, device->GetSerial()); | ||||||
|  | 
 | ||||||
|  |         return commandAllocators[firstFreeIndex]; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void CommandAllocatorManager::ResetCompletedAllocators(uint64_t lastCompletedSerial) { | ||||||
|  |         // Reset all command allocators that are no longer in flight
 | ||||||
|  |         for (auto it : inFlightCommandAllocators.IterateUpTo(lastCompletedSerial)) { | ||||||
|  |             ASSERT_SUCCESS(it.commandAllocator->Reset()); | ||||||
|  |             freeAllocators.set(it.index); | ||||||
|  |         } | ||||||
|  |         inFlightCommandAllocators.ClearUpTo(lastCompletedSerial); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | } | ||||||
							
								
								
									
										58
									
								
								src/backend/d3d12/CommandAllocatorManager.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								src/backend/d3d12/CommandAllocatorManager.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,58 @@ | |||||||
|  | // Copyright 2017 The NXT Authors
 | ||||||
|  | //
 | ||||||
|  | // Licensed under the Apache License, Version 2.0 (the "License");
 | ||||||
|  | // you may not use this file except in compliance with the License.
 | ||||||
|  | // You may obtain a copy of the License at
 | ||||||
|  | //
 | ||||||
|  | //     http://www.apache.org/licenses/LICENSE-2.0
 | ||||||
|  | //
 | ||||||
|  | // Unless required by applicable law or agreed to in writing, software
 | ||||||
|  | // distributed under the License is distributed on an "AS IS" BASIS,
 | ||||||
|  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | ||||||
|  | // See the License for the specific language governing permissions and
 | ||||||
|  | // limitations under the License.
 | ||||||
|  | 
 | ||||||
|  | #ifndef BACKEND_D3D12_COMMANDALLOCATORMANAGER_H_ | ||||||
|  | #define BACKEND_D3D12_COMMANDALLOCATORMANAGER_H_ | ||||||
|  | 
 | ||||||
|  | #include "d3d12_platform.h" | ||||||
|  | 
 | ||||||
|  | #include "common/SerialQueue.h" | ||||||
|  | 
 | ||||||
|  | #include <bitset> | ||||||
|  | 
 | ||||||
|  | namespace backend { | ||||||
|  | namespace d3d12 { | ||||||
|  | 
 | ||||||
|  |     class Device; | ||||||
|  | 
 | ||||||
|  |     class CommandAllocatorManager { | ||||||
|  |         public: | ||||||
|  |             CommandAllocatorManager(Device* device); | ||||||
|  | 
 | ||||||
|  |             // A CommandAllocator that is reserved must be used before the next Device::Tick where the next serial has completed on the GPU
 | ||||||
|  |             // at this time, the CommandAllocator will be reset
 | ||||||
|  |             ComPtr<ID3D12CommandAllocator> ReserveCommandAllocator(); | ||||||
|  |             void ResetCompletedAllocators(uint64_t lastCompletedSerial); | ||||||
|  | 
 | ||||||
|  |         private: | ||||||
|  |             Device* device; | ||||||
|  | 
 | ||||||
|  |             // This must be at least 2 because the Device and Queue use separate command allocators
 | ||||||
|  |             static constexpr unsigned int kMaxCommandAllocators = 32; | ||||||
|  |             unsigned int allocatorCount; | ||||||
|  | 
 | ||||||
|  |             struct IndexedCommandAllocator { | ||||||
|  |                 ComPtr<ID3D12CommandAllocator> commandAllocator; | ||||||
|  |                 unsigned int index; | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |             ComPtr<ID3D12CommandAllocator> commandAllocators[kMaxCommandAllocators]; | ||||||
|  |             std::bitset<kMaxCommandAllocators> freeAllocators; | ||||||
|  |             SerialQueue<IndexedCommandAllocator> inFlightCommandAllocators; | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #endif // BACKEND_D3D12_COMMANDALLOCATORMANAGER_H_
 | ||||||
| @ -44,26 +44,56 @@ namespace d3d12 { | |||||||
|         backendDevice->SetNextRenderTargetDescriptor(renderTargetDescriptor); |         backendDevice->SetNextRenderTargetDescriptor(renderTargetDescriptor); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     uint64_t GetSerial(const nxtDevice device) { | ||||||
|  |         const Device* backendDevice = reinterpret_cast<const Device*>(device); | ||||||
|  |         return backendDevice->GetSerial(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void NextSerial(nxtDevice device) { | ||||||
|  |         Device* backendDevice = reinterpret_cast<Device*>(device); | ||||||
|  |         backendDevice->NextSerial(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void ExecuteCommandLists(nxtDevice device, std::initializer_list<ID3D12CommandList*> commandLists) { | ||||||
|  |         Device* backendDevice = reinterpret_cast<Device*>(device); | ||||||
|  |         backendDevice->ExecuteCommandLists(commandLists); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void WaitForSerial(nxtDevice device, uint64_t serial) { | ||||||
|  |         Device* backendDevice = reinterpret_cast<Device*>(device); | ||||||
|  |         backendDevice->WaitForSerial(serial); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ComPtr<ID3D12CommandAllocator> ReserveCommandAllocator(nxtDevice device) { | ||||||
|  |         Device* backendDevice = reinterpret_cast<Device*>(device); | ||||||
|  |         return backendDevice->GetCommandAllocatorManager()->ReserveCommandAllocator(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     void ASSERT_SUCCESS(HRESULT hr) { |     void ASSERT_SUCCESS(HRESULT hr) { | ||||||
|         ASSERT(SUCCEEDED(hr)); |         ASSERT(SUCCEEDED(hr)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     Device::Device(ComPtr<ID3D12Device> d3d12Device) : d3d12Device(d3d12Device), resourceUploader(this) { |     Device::Device(ComPtr<ID3D12Device> d3d12Device) | ||||||
|  |         : d3d12Device(d3d12Device), | ||||||
|  |           commandAllocatorManager(this), | ||||||
|  |           resourceUploader(this), | ||||||
|  |           pendingCommands{ commandAllocatorManager.ReserveCommandAllocator() } { | ||||||
|  | 
 | ||||||
|         D3D12_COMMAND_QUEUE_DESC queueDesc = {}; |         D3D12_COMMAND_QUEUE_DESC queueDesc = {}; | ||||||
|         queueDesc.Flags = D3D12_COMMAND_QUEUE_FLAG_NONE; |         queueDesc.Flags = D3D12_COMMAND_QUEUE_FLAG_NONE; | ||||||
|         queueDesc.Type = D3D12_COMMAND_LIST_TYPE_DIRECT; |         queueDesc.Type = D3D12_COMMAND_LIST_TYPE_DIRECT; | ||||||
|         ASSERT_SUCCESS(d3d12Device->CreateCommandQueue(&queueDesc, IID_PPV_ARGS(&commandQueue))); |         ASSERT_SUCCESS(d3d12Device->CreateCommandQueue(&queueDesc, IID_PPV_ARGS(&commandQueue))); | ||||||
| 
 | 
 | ||||||
|         ASSERT_SUCCESS(d3d12Device->CreateCommandAllocator(D3D12_COMMAND_LIST_TYPE_DIRECT, IID_PPV_ARGS(&pendingCommandAllocator))); |  | ||||||
|         ASSERT_SUCCESS(d3d12Device->CreateCommandList( |         ASSERT_SUCCESS(d3d12Device->CreateCommandList( | ||||||
|             0, |             0, | ||||||
|             D3D12_COMMAND_LIST_TYPE_DIRECT, |             D3D12_COMMAND_LIST_TYPE_DIRECT, | ||||||
|             pendingCommandAllocator.Get(), |             pendingCommands.commandAllocator.Get(), | ||||||
|             nullptr, |             nullptr, | ||||||
|             IID_PPV_ARGS(&pendingCommandList) |             IID_PPV_ARGS(&pendingCommands.commandList) | ||||||
|         )); |         )); | ||||||
|  |         pendingCommands.open = true; | ||||||
| 
 | 
 | ||||||
|         ASSERT_SUCCESS(d3d12Device->CreateFence(serial++, D3D12_FENCE_FLAG_NONE, IID_PPV_ARGS(&fence))); |         ASSERT_SUCCESS(d3d12Device->CreateFence(serial, D3D12_FENCE_FLAG_NONE, IID_PPV_ARGS(&fence))); | ||||||
|         fenceEvent = CreateEvent(nullptr, FALSE, FALSE, nullptr); |         fenceEvent = CreateEvent(nullptr, FALSE, FALSE, nullptr); | ||||||
|         ASSERT(fenceEvent != nullptr); |         ASSERT(fenceEvent != nullptr); | ||||||
|     } |     } | ||||||
| @ -79,12 +109,14 @@ namespace d3d12 { | |||||||
|         return commandQueue; |         return commandQueue; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     ComPtr<ID3D12CommandAllocator> Device::GetPendingCommandAllocator() { |  | ||||||
|         return pendingCommandAllocator; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     ComPtr<ID3D12GraphicsCommandList> Device::GetPendingCommandList() { |     ComPtr<ID3D12GraphicsCommandList> Device::GetPendingCommandList() { | ||||||
|         return pendingCommandList; |         // Callers of GetPendingCommandList do so to record commands. Only reserve a command allocator when it is needed so we don't submit empty command lists
 | ||||||
|  |         if (!pendingCommands.open) { | ||||||
|  |             pendingCommands.commandAllocator = commandAllocatorManager.ReserveCommandAllocator(); | ||||||
|  |             ASSERT_SUCCESS(pendingCommands.commandList->Reset(pendingCommands.commandAllocator.Get(), nullptr)); | ||||||
|  |             pendingCommands.open = true; | ||||||
|  |         } | ||||||
|  |         return pendingCommands.commandList; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     D3D12_CPU_DESCRIPTOR_HANDLE Device::GetCurrentRenderTargetDescriptor() { |     D3D12_CPU_DESCRIPTOR_HANDLE Device::GetCurrentRenderTargetDescriptor() { | ||||||
| @ -95,44 +127,50 @@ namespace d3d12 { | |||||||
|         return &resourceUploader; |         return &resourceUploader; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     CommandAllocatorManager* Device::GetCommandAllocatorManager() { | ||||||
|  |         return &commandAllocatorManager; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     void Device::SetNextRenderTargetDescriptor(D3D12_CPU_DESCRIPTOR_HANDLE renderTargetDescriptor) { |     void Device::SetNextRenderTargetDescriptor(D3D12_CPU_DESCRIPTOR_HANDLE renderTargetDescriptor) { | ||||||
|         this->renderTargetDescriptor = renderTargetDescriptor; |         this->renderTargetDescriptor = renderTargetDescriptor; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void Device::TickImpl() { |     void Device::TickImpl() { | ||||||
|         // Execute any pending commands
 |         // Perform cleanup operations to free unused objects
 | ||||||
|         ASSERT_SUCCESS(pendingCommandList->Close()); |  | ||||||
|         ID3D12CommandList* commandLists[] = { pendingCommandList.Get() }; |  | ||||||
|         commandQueue->ExecuteCommandLists(_countof(commandLists), commandLists); |  | ||||||
| 
 |  | ||||||
|         IncrementSerial(); |  | ||||||
| 
 |  | ||||||
|         // Signal when the pending commands have finished
 |  | ||||||
|         ASSERT_SUCCESS(commandQueue->Signal(fence.Get(), GetSerial())); |  | ||||||
| 
 |  | ||||||
|         // Handle objects awaiting GPU execution
 |  | ||||||
|         const uint64_t lastCompletedSerial = fence->GetCompletedValue(); |         const uint64_t lastCompletedSerial = fence->GetCompletedValue(); | ||||||
|         resourceUploader.FreeCompletedResources(lastCompletedSerial); |         resourceUploader.FreeCompletedResources(lastCompletedSerial); | ||||||
| 
 |         commandAllocatorManager.ResetCompletedAllocators(lastCompletedSerial); | ||||||
|         // TODO(enga@google.com): This will stall on the submit because
 |  | ||||||
|         // the commands must finish exeuting before the ID3D12CommandAllocator is reset.
 |  | ||||||
|         // This should be fixed / optimized by using multiple command allocators.
 |  | ||||||
|         const uint64_t currentFence = GetSerial(); |  | ||||||
|         if (lastCompletedSerial < currentFence) { |  | ||||||
|             ASSERT_SUCCESS(fence->SetEventOnCompletion(currentFence, fenceEvent)); |  | ||||||
|             WaitForSingleObject(fenceEvent, INFINITE); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         ASSERT_SUCCESS(pendingCommandAllocator->Reset()); |  | ||||||
|         ASSERT_SUCCESS(pendingCommandList->Reset(pendingCommandAllocator.Get(), NULL)); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     uint64_t Device::GetSerial() const { |     uint64_t Device::GetSerial() const { | ||||||
|         return serial; |         return serial; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void Device::IncrementSerial() { |     void Device::NextSerial() { | ||||||
|         serial++; |         ASSERT_SUCCESS(commandQueue->Signal(fence.Get(), serial++)); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void Device::WaitForSerial(uint64_t serial) { | ||||||
|  |         const uint64_t lastCompletedSerial = fence->GetCompletedValue(); | ||||||
|  |         if (lastCompletedSerial < serial) { | ||||||
|  |             ASSERT_SUCCESS(fence->SetEventOnCompletion(serial, fenceEvent)); | ||||||
|  |             WaitForSingleObject(fenceEvent, INFINITE); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void Device::ExecuteCommandLists(std::initializer_list<ID3D12CommandList*> commandLists) { | ||||||
|  |         // If there are pending commands, prepend them to ExecuteCommandLists
 | ||||||
|  |         if (pendingCommands.open) { | ||||||
|  |             std::vector<ID3D12CommandList*> lists(commandLists.size() + 1); | ||||||
|  |             pendingCommands.commandList->Close(); | ||||||
|  |             pendingCommands.open = false; | ||||||
|  |             lists[0] = pendingCommands.commandList.Get(); | ||||||
|  |             std::copy(commandLists.begin(), commandLists.end(), lists.begin() + 1); | ||||||
|  |             commandQueue->ExecuteCommandLists(commandLists.size() + 1, lists.data()); | ||||||
|  |         } else { | ||||||
|  |             std::vector<ID3D12CommandList*> lists(commandLists); | ||||||
|  |             commandQueue->ExecuteCommandLists(commandLists.size(), lists.data()); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     BindGroupBase* Device::CreateBindGroup(BindGroupBuilder* builder) { |     BindGroupBase* Device::CreateBindGroup(BindGroupBuilder* builder) { | ||||||
|  | |||||||
| @ -33,6 +33,7 @@ | |||||||
| 
 | 
 | ||||||
| #include "d3d12_platform.h" | #include "d3d12_platform.h" | ||||||
| #include "ResourceUploader.h" | #include "ResourceUploader.h" | ||||||
|  | #include "CommandAllocatorManager.h" | ||||||
| 
 | 
 | ||||||
| namespace backend { | namespace backend { | ||||||
| namespace d3d12 { | namespace d3d12 { | ||||||
| @ -109,32 +110,42 @@ namespace d3d12 { | |||||||
| 
 | 
 | ||||||
|             ComPtr<ID3D12Device> GetD3D12Device(); |             ComPtr<ID3D12Device> GetD3D12Device(); | ||||||
|             ComPtr<ID3D12CommandQueue> GetCommandQueue(); |             ComPtr<ID3D12CommandQueue> GetCommandQueue(); | ||||||
|             ComPtr<ID3D12CommandAllocator> GetPendingCommandAllocator(); |             CommandAllocatorManager* GetCommandAllocatorManager(); | ||||||
|             ComPtr<ID3D12GraphicsCommandList> GetPendingCommandList(); |  | ||||||
|             D3D12_CPU_DESCRIPTOR_HANDLE GetCurrentRenderTargetDescriptor(); |  | ||||||
|             ResourceUploader* GetResourceUploader(); |             ResourceUploader* GetResourceUploader(); | ||||||
| 
 | 
 | ||||||
|  |             ComPtr<ID3D12GraphicsCommandList> GetPendingCommandList(); | ||||||
|  |              | ||||||
|  |             D3D12_CPU_DESCRIPTOR_HANDLE GetCurrentRenderTargetDescriptor(); | ||||||
|             void SetNextRenderTargetDescriptor(D3D12_CPU_DESCRIPTOR_HANDLE renderTargetDescriptor); |             void SetNextRenderTargetDescriptor(D3D12_CPU_DESCRIPTOR_HANDLE renderTargetDescriptor); | ||||||
| 
 | 
 | ||||||
|             uint64_t GetSerial() const; |             uint64_t GetSerial() const; | ||||||
|             void IncrementSerial(); |             void NextSerial(); | ||||||
|  |             void WaitForSerial(uint64_t serial); | ||||||
|  | 
 | ||||||
|  |             void ExecuteCommandLists(std::initializer_list<ID3D12CommandList*> commandLists); | ||||||
| 
 | 
 | ||||||
|             // NXT API
 |             // NXT API
 | ||||||
|             void Reference(); |             void Reference(); | ||||||
|             void Release(); |             void Release(); | ||||||
| 
 | 
 | ||||||
|         private: |         private: | ||||||
|             ComPtr<ID3D12Device> d3d12Device; |  | ||||||
|             ComPtr<ID3D12CommandQueue> commandQueue; |  | ||||||
|             ComPtr<ID3D12CommandAllocator> pendingCommandAllocator; |  | ||||||
|             ComPtr<ID3D12GraphicsCommandList> pendingCommandList; |  | ||||||
|             D3D12_CPU_DESCRIPTOR_HANDLE renderTargetDescriptor; |  | ||||||
| 
 |  | ||||||
|             ResourceUploader resourceUploader; |  | ||||||
| 
 |  | ||||||
|             uint64_t serial = 0; |             uint64_t serial = 0; | ||||||
|             ComPtr<ID3D12Fence> fence; |             ComPtr<ID3D12Fence> fence; | ||||||
|             HANDLE fenceEvent; |             HANDLE fenceEvent; | ||||||
|  | 
 | ||||||
|  |             ComPtr<ID3D12Device> d3d12Device; | ||||||
|  |             ComPtr<ID3D12CommandQueue> commandQueue; | ||||||
|  | 
 | ||||||
|  |             CommandAllocatorManager commandAllocatorManager; | ||||||
|  |             ResourceUploader resourceUploader; | ||||||
|  |              | ||||||
|  |             struct PendingCommandList { | ||||||
|  |                 ComPtr<ID3D12CommandAllocator> commandAllocator; | ||||||
|  |                 ComPtr<ID3D12GraphicsCommandList> commandList; | ||||||
|  |                 bool open = false; | ||||||
|  |             } pendingCommands; | ||||||
|  | 
 | ||||||
|  |             D3D12_CPU_DESCRIPTOR_HANDLE renderTargetDescriptor; | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -23,45 +23,32 @@ namespace d3d12 { | |||||||
|     Queue::Queue(Device* device, QueueBuilder* builder) |     Queue::Queue(Device* device, QueueBuilder* builder) | ||||||
|         : QueueBase(builder), device(device) { |         : QueueBase(builder), device(device) { | ||||||
| 
 | 
 | ||||||
|         ASSERT_SUCCESS(device->GetD3D12Device()->CreateCommandAllocator(D3D12_COMMAND_LIST_TYPE_DIRECT, IID_PPV_ARGS(&commandAllocator))); |         // TODO(enga@google.com): We don't need this allocator, but it's needed for command list initialization. Is there a better way to do this?
 | ||||||
|  |         // Is CommandList creation expensive or can it be done every Queue::Submit?
 | ||||||
|  |         ComPtr<ID3D12CommandAllocator> temporaryCommandAllocator; | ||||||
|  |         ASSERT_SUCCESS(device->GetD3D12Device()->CreateCommandAllocator(D3D12_COMMAND_LIST_TYPE_DIRECT, IID_PPV_ARGS(&temporaryCommandAllocator))); | ||||||
|         ASSERT_SUCCESS(device->GetD3D12Device()->CreateCommandList( |         ASSERT_SUCCESS(device->GetD3D12Device()->CreateCommandList( | ||||||
|             0, |             0, | ||||||
|             D3D12_COMMAND_LIST_TYPE_DIRECT, |             D3D12_COMMAND_LIST_TYPE_DIRECT, | ||||||
|             commandAllocator.Get(), |             temporaryCommandAllocator.Get(), | ||||||
|             nullptr, |             nullptr, | ||||||
|             IID_PPV_ARGS(&commandList) |             IID_PPV_ARGS(&commandList) | ||||||
|         )); |         )); | ||||||
|         ASSERT_SUCCESS(commandList->Close()); |         ASSERT_SUCCESS(commandList->Close()); | ||||||
| 
 |  | ||||||
|         ASSERT_SUCCESS(device->GetD3D12Device()->CreateFence(fenceValue, D3D12_FENCE_FLAG_NONE, IID_PPV_ARGS(&fence))); |  | ||||||
|         fenceEvent = CreateEvent(nullptr, FALSE, FALSE, nullptr); |  | ||||||
|         ASSERT(fenceEvent != nullptr); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void Queue::Submit(uint32_t numCommands, CommandBuffer* const * commands) { |     void Queue::Submit(uint32_t numCommands, CommandBuffer* const * commands) { | ||||||
|         device->Tick(); |         device->Tick(); | ||||||
| 
 | 
 | ||||||
|         // TODO(enga@google.com): This will stall on the previous submit because
 |         ASSERT_SUCCESS(commandList->Reset(device->GetCommandAllocatorManager()->ReserveCommandAllocator().Get(), nullptr)); | ||||||
|         // the commands must finish exeuting before the ID3D12CommandAllocator is reset.
 |  | ||||||
|         // This should be fixed / optimized by using multiple command allocators.
 |  | ||||||
|         const uint64_t currentFence = fenceValue++; |  | ||||||
|         ASSERT_SUCCESS(device->GetCommandQueue()->Signal(fence.Get(), fenceValue)); |  | ||||||
| 
 |  | ||||||
|         if (fence->GetCompletedValue() < currentFence) { |  | ||||||
|             ASSERT_SUCCESS(fence->SetEventOnCompletion(currentFence, fenceEvent)); |  | ||||||
|             WaitForSingleObject(fenceEvent, INFINITE); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         ASSERT_SUCCESS(commandAllocator->Reset()); |  | ||||||
|         ASSERT_SUCCESS(commandList->Reset(commandAllocator.Get(), NULL)); |  | ||||||
| 
 |  | ||||||
|         for (uint32_t i = 0; i < numCommands; ++i) { |         for (uint32_t i = 0; i < numCommands; ++i) { | ||||||
|             commands[i]->FillCommands(commandList); |             commands[i]->FillCommands(commandList); | ||||||
|         } |         } | ||||||
|         ASSERT_SUCCESS(commandList->Close()); |         ASSERT_SUCCESS(commandList->Close()); | ||||||
| 
 | 
 | ||||||
|         ID3D12CommandList* commandLists[] = { commandList.Get() }; |         device->ExecuteCommandLists({ commandList.Get() }); | ||||||
|         device->GetCommandQueue()->ExecuteCommandLists(_countof(commandLists), commandLists); | 
 | ||||||
|  |         device->NextSerial(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
|  | |||||||
| @ -35,11 +35,7 @@ namespace d3d12 { | |||||||
|         private: |         private: | ||||||
|             Device* device; |             Device* device; | ||||||
| 
 | 
 | ||||||
|             ComPtr<ID3D12CommandAllocator> commandAllocator; |  | ||||||
|             ComPtr<ID3D12GraphicsCommandList> commandList; |             ComPtr<ID3D12GraphicsCommandList> commandList; | ||||||
|             ComPtr<ID3D12Fence> fence; |  | ||||||
|             uint64_t fenceValue = 0; |  | ||||||
|             HANDLE fenceEvent; |  | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
|  | |||||||
| @ -70,7 +70,7 @@ namespace d3d12 { | |||||||
|         uploadResource->Unmap(0, &writeRange); |         uploadResource->Unmap(0, &writeRange); | ||||||
|         device->GetPendingCommandList()->CopyBufferRegion(resource.Get(), start, uploadResource.Get(), 0, count); |         device->GetPendingCommandList()->CopyBufferRegion(resource.Get(), start, uploadResource.Get(), 0, count); | ||||||
| 
 | 
 | ||||||
|         uploadingResources.Enqueue(std::move(uploadResource), device->GetSerial() + 1); |         uploadingResources.Enqueue(std::move(uploadResource), device->GetSerial()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void ResourceUploader::FreeCompletedResources(const uint64_t lastCompletedSerial) { |     void ResourceUploader::FreeCompletedResources(const uint64_t lastCompletedSerial) { | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user