226 lines
8.5 KiB
C++
226 lines
8.5 KiB
C++
// Copyright (c) 2012-2020 Wojciech Figat. All rights reserved.
|
|
|
|
#pragma once
|
|
|
|
#include "Engine/Graphics/GPUContext.h"
|
|
|
|
#if GRAPHICS_API_VULKAN
|
|
|
|
#include "GPUDeviceVulkan.h"
|
|
#include "Types.h"
|
|
|
|
class QueueVulkan;
|
|
class CmdBufferManagerVulkan;
|
|
class ResourceOwnerVulkan;
|
|
class GPUTextureViewVulkan;
|
|
class GPUBufferVulkan;
|
|
class GPUPipelineStateVulkan;
|
|
class ComputePipelineStateVulkan;
|
|
class GPUConstantBufferVulkan;
|
|
class DescriptorPoolVulkan;
|
|
class DescriptorSetLayoutVulkan;
|
|
|
|
/// <summary>
|
|
/// Enables using batched pipeline barriers to improve performance.
|
|
/// </summary>
|
|
#define VK_ENABLE_BARRIERS_BATCHING 1
|
|
|
|
/// <summary>
|
|
/// Enables pipeline barriers debugging.
|
|
/// </summary>
|
|
#define VK_ENABLE_BARRIERS_DEBUG (BUILD_DEBUG && 0)
|
|
|
|
/// <summary>
|
|
/// Size of the pipeline barriers buffer size (will be auto-flushed on overflow).
|
|
/// </summary>
|
|
#define VK_BARRIER_BUFFER_SIZE 16
|
|
|
|
#if VK_ENABLE_BARRIERS_BATCHING
|
|
|
|
/// <summary>
|
|
/// The Vulkan pipeline resources layout barrier batching structure.
|
|
/// </summary>
|
|
struct PipelineBarrierVulkan
|
|
{
|
|
public:
|
|
|
|
VkPipelineStageFlags SourceStage = 0;
|
|
VkPipelineStageFlags DestStage = 0;
|
|
Array<VkImageMemoryBarrier, FixedAllocation<VK_BARRIER_BUFFER_SIZE>> ImageBarriers;
|
|
Array<VkBufferMemoryBarrier, FixedAllocation<VK_BARRIER_BUFFER_SIZE>> BufferBarriers;
|
|
#if VK_ENABLE_BARRIERS_DEBUG
|
|
Array<GPUTextureViewVulkan*, FixedAllocation<VK_BARRIER_BUFFER_SIZE>> ImageBarriersDebug;
|
|
#endif
|
|
|
|
public:
|
|
|
|
inline void Reset()
|
|
{
|
|
SourceStage = 0;
|
|
DestStage = 0;
|
|
ImageBarriers.Clear();
|
|
BufferBarriers.Clear();
|
|
#if VK_ENABLE_BARRIERS_DEBUG
|
|
ImageBarriersDebug.Clear();
|
|
#endif
|
|
}
|
|
|
|
void AddImageBarrier(VkImage image, const VkImageSubresourceRange& range, VkImageLayout srcLayout, VkImageLayout dstLayout, GPUTextureViewVulkan* handle);
|
|
void AddBufferBarrier(VkBuffer buffer, VkDeviceSize offset, VkDeviceSize size, VkAccessFlags srcAccess, VkAccessFlags dstAccess);
|
|
|
|
inline bool IsFull() const
|
|
{
|
|
return ImageBarriers.Count() == VK_BARRIER_BUFFER_SIZE || BufferBarriers.Count() == VK_BARRIER_BUFFER_SIZE;
|
|
}
|
|
|
|
inline bool HasBarrier() const
|
|
{
|
|
return ImageBarriers.Count() + BufferBarriers.Count() != 0;
|
|
}
|
|
|
|
void Execute(CmdBufferVulkan* cmdBuffer);
|
|
};
|
|
|
|
#endif
|
|
|
|
/// <summary>
|
|
/// GPU Context for Vulkan backend.
|
|
/// </summary>
|
|
class GPUContextVulkan : public GPUContext
|
|
{
|
|
private:
|
|
|
|
GPUDeviceVulkan* _device;
|
|
QueueVulkan* _queue;
|
|
CmdBufferManagerVulkan* _cmdBufferManager;
|
|
#if VK_ENABLE_BARRIERS_BATCHING
|
|
PipelineBarrierVulkan _barriers;
|
|
#endif
|
|
|
|
int32 _psDirtyFlag : 1;
|
|
int32 _rtDirtyFlag : 1;
|
|
int32 _cbDirtyFlag : 1;
|
|
int32 _srDirtyFlag : 1;
|
|
int32 _uaDirtyFlag : 1;
|
|
|
|
int32 _rtCount;
|
|
|
|
RenderPassVulkan* _renderPass;
|
|
GPUPipelineStateVulkan* _currentState;
|
|
GPUTextureViewVulkan* _rtDepth;
|
|
GPUTextureViewVulkan* _rtHandles[GPU_MAX_RT_BINDED];
|
|
DescriptorOwnerResourceVulkan* _cbHandles[GPU_MAX_CB_BINDED];
|
|
DescriptorOwnerResourceVulkan* _srHandles[GPU_MAX_SR_BINDED];
|
|
DescriptorOwnerResourceVulkan* _uaHandles[GPU_MAX_UA_BINDED];
|
|
DescriptorOwnerResourceVulkan** _handles[(int32)SpirvShaderResourceBindingType::MAX];
|
|
|
|
#if VULKAN_USE_DESCRIPTOR_POOL_MANAGER
|
|
typedef Array<DescriptorPoolVulkan*> DescriptorPoolArray;
|
|
Dictionary<uint32, DescriptorPoolArray> _descriptorPools;
|
|
#else
|
|
Array<DescriptorPoolVulkan*> _descriptorPools;
|
|
#endif
|
|
|
|
public:
|
|
|
|
/// <summary>
|
|
/// Initializes a new instance of the <see cref="GPUContextVulkan"/> class.
|
|
/// </summary>
|
|
/// <param name="device">The graphics device.</param>
|
|
/// <param name="queue">The commands submission device.</param>
|
|
GPUContextVulkan(GPUDeviceVulkan* device, QueueVulkan* queue);
|
|
|
|
/// <summary>
|
|
/// Finalizes an instance of the <see cref="GPUContextVulkan"/> class.
|
|
/// </summary>
|
|
~GPUContextVulkan();
|
|
|
|
public:
|
|
|
|
QueueVulkan* GetQueue() const
|
|
{
|
|
return _queue;
|
|
}
|
|
|
|
CmdBufferManagerVulkan* GetCmdBufferManager() const
|
|
{
|
|
return _cmdBufferManager;
|
|
}
|
|
|
|
void AddImageBarrier(VkImage image, VkImageLayout srcLayout, VkImageLayout dstLayout, VkImageSubresourceRange& subresourceRange, GPUTextureViewVulkan* handle);
|
|
void AddImageBarrier(GPUTextureViewVulkan* handle, VkImageLayout dstLayout);
|
|
void AddImageBarrier(GPUTextureVulkan* texture, int32 mipSlice, int32 arraySlice, VkImageLayout dstLayout);
|
|
void AddImageBarrier(GPUTextureVulkan* texture, VkImageLayout dstLayout);
|
|
void AddBufferBarrier(GPUBufferVulkan* buffer, VkAccessFlags dstAccess);
|
|
|
|
void FlushBarriers();
|
|
|
|
// outSets must have been previously pre-allocated
|
|
DescriptorPoolVulkan* AllocateDescriptorSets(const VkDescriptorSetAllocateInfo& descriptorSetAllocateInfo, const DescriptorSetLayoutVulkan& layout, VkDescriptorSet* outSets);
|
|
|
|
void BeginRenderPass();
|
|
|
|
void EndRenderPass();
|
|
|
|
private:
|
|
|
|
void UpdateDescriptorSets(const struct SpirvShaderDescriptorInfo& descriptorInfo, class DescriptorSetWriterVulkan& dsWriter, bool& needsWrite);
|
|
void UpdateDescriptorSets(GPUPipelineStateVulkan* pipelineState);
|
|
void UpdateDescriptorSets(ComputePipelineStateVulkan* pipelineState);
|
|
void BindPipeline();
|
|
void OnDrawCall();
|
|
|
|
public:
|
|
|
|
// [GPUContext]
|
|
void FrameBegin() override;
|
|
void FrameEnd() override;
|
|
#if GPU_ALLOW_PROFILE_EVENTS
|
|
void EventBegin(const Char* name) override;
|
|
void EventEnd() override;
|
|
#endif
|
|
void* GetNativePtr() const override;
|
|
bool IsDepthBufferBinded() override;
|
|
void Clear(GPUTextureView* rt, const Color& color) override;
|
|
void ClearDepth(GPUTextureView* depthBuffer, float depthValue) override;
|
|
void ClearUA(GPUBuffer* buf, const Vector4& value) override;
|
|
void ResetRenderTarget() override;
|
|
void SetRenderTarget(GPUTextureView* rt) override;
|
|
void SetRenderTarget(GPUTextureView* depthBuffer, GPUTextureView* rt) override;
|
|
void SetRenderTarget(GPUTextureView* depthBuffer, const Span<GPUTextureView*>& rts) override;
|
|
void SetRenderTarget(GPUTextureView* rt, GPUBuffer* uaOutput) override;
|
|
void ResetSR() override;
|
|
void ResetUA() override;
|
|
void ResetCB() override;
|
|
void BindCB(int32 slot, GPUConstantBuffer* cb) override;
|
|
void BindSR(int32 slot, GPUResourceView* view) override;
|
|
void BindUA(int32 slot, GPUResourceView* view) override;
|
|
void BindVB(const Span<GPUBuffer*>& vertexBuffers, const uint32* vertexBuffersOffsets = nullptr) override;
|
|
void BindIB(GPUBuffer* indexBuffer) override;
|
|
void UpdateCB(GPUConstantBuffer* cb, const void* data) override;
|
|
void Dispatch(GPUShaderProgramCS* shader, uint32 threadGroupCountX, uint32 threadGroupCountY, uint32 threadGroupCountZ) override;
|
|
void DispatchIndirect(GPUShaderProgramCS* shader, GPUBuffer* bufferForArgs, uint32 offsetForArgs) override;
|
|
void ResolveMultisample(GPUTexture* sourceMultisampleTexture, GPUTexture* destTexture, int32 sourceSubResource, int32 destSubResource, PixelFormat format) override;
|
|
void DrawInstanced(uint32 verticesCount, uint32 instanceCount, int32 startInstance, int32 startVertex) override;
|
|
void DrawIndexedInstanced(uint32 indicesCount, uint32 instanceCount, int32 startInstance, int32 startVertex, int32 startIndex) override;
|
|
void DrawInstancedIndirect(GPUBuffer* bufferForArgs, uint32 offsetForArgs) override;
|
|
void DrawIndexedInstancedIndirect(GPUBuffer* bufferForArgs, uint32 offsetForArgs) override;
|
|
void SetViewport(const Viewport& viewport) override;
|
|
void SetScissor(const Rectangle& scissorRect) override;
|
|
GPUPipelineState* GetState() const override;
|
|
void SetState(GPUPipelineState* state) override;
|
|
void ClearState() override;
|
|
void FlushState() override;
|
|
void Flush() override;
|
|
void UpdateBuffer(GPUBuffer* buffer, const void* data, uint32 size, uint32 offset) override;
|
|
void CopyBuffer(GPUBuffer* dstBuffer, GPUBuffer* srcBuffer, uint32 size, uint32 dstOffset, uint32 srcOffset) override;
|
|
void UpdateTexture(GPUTexture* texture, int32 arrayIndex, int32 mipIndex, const void* data, uint32 rowPitch, uint32 slicePitch) override;
|
|
void CopyTexture(GPUTexture* dstResource, uint32 dstSubresource, uint32 dstX, uint32 dstY, uint32 dstZ, GPUTexture* srcResource, uint32 srcSubresource) override;
|
|
void ResetCounter(GPUBuffer* buffer) override;
|
|
void CopyCounter(GPUBuffer* dstBuffer, uint32 dstOffset, GPUBuffer* srcBuffer) override;
|
|
void CopyResource(GPUResource* dstResource, GPUResource* srcResource) override;
|
|
void CopySubresource(GPUResource* dstResource, uint32 dstSubresource, GPUResource* srcResource, uint32 srcSubresource) override;
|
|
};
|
|
|
|
#endif
|