blob: c2746de78ce786b6f19d2ef0cceda51cc03069aa [file] [log] [blame]
/*
* Copyright 2015 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifndef GrVkCommandBuffer_DEFINED
#define GrVkCommandBuffer_DEFINED
#include "include/gpu/vk/GrVkTypes.h"
#include "src/gpu/ganesh/GrManagedResource.h"
#include "src/gpu/ganesh/GrRefCnt.h"
#include "src/gpu/ganesh/vk/GrVkGpu.h"
#include "src/gpu/ganesh/vk/GrVkSemaphore.h"
#include "src/gpu/ganesh/vk/GrVkUtil.h"
class GrVkFramebuffer;
class GrVkImage;
class GrVkPipeline;
class GrVkPipelineState;
class GrVkRenderPass;
class GrVkRenderTarget;
class GrVkCommandBuffer {
public:
virtual ~GrVkCommandBuffer() {}
void invalidateState();
////////////////////////////////////////////////////////////////////////////
// CommandBuffer commands
////////////////////////////////////////////////////////////////////////////
enum BarrierType {
kBufferMemory_BarrierType,
kImageMemory_BarrierType
};
void pipelineBarrier(const GrVkGpu* gpu,
const GrManagedResource* resource,
VkPipelineStageFlags srcStageMask,
VkPipelineStageFlags dstStageMask,
bool byRegion,
BarrierType barrierType,
void* barrier);
void bindInputBuffer(GrVkGpu* gpu, uint32_t binding, sk_sp<const GrBuffer> buffer);
void bindIndexBuffer(GrVkGpu* gpu, sk_sp<const GrBuffer> buffer);
void bindPipeline(const GrVkGpu* gpu, sk_sp<const GrVkPipeline> pipeline);
void bindDescriptorSets(const GrVkGpu* gpu,
VkPipelineLayout layout,
uint32_t firstSet,
uint32_t setCount,
const VkDescriptorSet* descriptorSets,
uint32_t dynamicOffsetCount,
const uint32_t* dynamicOffsets);
void pushConstants(const GrVkGpu* gpu, VkPipelineLayout layout,
VkShaderStageFlags stageFlags, uint32_t offset, uint32_t size,
const void* values);
void setViewport(const GrVkGpu* gpu,
uint32_t firstViewport,
uint32_t viewportCount,
const VkViewport* viewports);
void setScissor(const GrVkGpu* gpu,
uint32_t firstScissor,
uint32_t scissorCount,
const VkRect2D* scissors);
void setBlendConstants(const GrVkGpu* gpu, const float blendConstants[4]);
// Commands that only work inside of a render pass
void clearAttachments(const GrVkGpu* gpu,
int numAttachments,
const VkClearAttachment* attachments,
int numRects,
const VkClearRect* clearRects);
void drawIndexed(const GrVkGpu* gpu,
uint32_t indexCount,
uint32_t instanceCount,
uint32_t firstIndex,
int32_t vertexOffset,
uint32_t firstInstance);
void draw(const GrVkGpu* gpu,
uint32_t vertexCount,
uint32_t instanceCount,
uint32_t firstVertex,
uint32_t firstInstance);
void drawIndirect(const GrVkGpu* gpu,
sk_sp<const GrBuffer> indirectBuffer,
VkDeviceSize offset,
uint32_t drawCount,
uint32_t stride);
void drawIndexedIndirect(const GrVkGpu* gpu,
sk_sp<const GrBuffer> indirectBuffer,
VkDeviceSize offset,
uint32_t drawCount,
uint32_t stride);
// Add ref-counted resource that will be tracked and released when this command buffer finishes
// execution
void addResource(sk_sp<const GrManagedResource> resource) {
SkASSERT(resource);
fTrackedResources.push_back(std::move(resource));
}
void addResource(const GrManagedResource* resource) {
this->addResource(sk_ref_sp(resource));
}
// Add ref-counted resource that will be tracked and released when this command buffer finishes
// execution. When it is released, it will signal that the resource can be recycled for reuse.
void addRecycledResource(gr_rp<const GrRecycledResource> resource) {
SkASSERT(resource);
fTrackedRecycledResources.push_back(std::move(resource));
}
void addRecycledResource(const GrRecycledResource* resource) {
this->addRecycledResource(gr_ref_rp<const GrRecycledResource>(resource));
}
void addGrBuffer(sk_sp<const GrBuffer> buffer) {
fTrackedGpuBuffers.push_back(std::move(buffer));
}
void addGrSurface(sk_sp<const GrSurface> surface) {
fTrackedGpuSurfaces.push_back(std::move(surface));
}
void releaseResources();
void freeGPUData(const GrGpu* gpu, VkCommandPool pool) const;
bool hasWork() const { return fHasWork; }
protected:
GrVkCommandBuffer(VkCommandBuffer cmdBuffer, bool isWrapped = false)
: fIsActive(isWrapped) // All wrapped command buffers start as active
, fCmdBuffer(cmdBuffer)
, fIsWrapped(isWrapped) {
this->invalidateState();
}
bool isWrapped() const { return fIsWrapped; }
void addingWork(const GrVkGpu* gpu);
void submitPipelineBarriers(const GrVkGpu* gpu, bool forSelfDependency = false);
private:
static constexpr int kInitialTrackedResourcesCount = 32;
protected:
template <typename T> using TrackedResourceArray = SkSTArray<kInitialTrackedResourcesCount, T>;
TrackedResourceArray<sk_sp<const GrManagedResource>> fTrackedResources;
TrackedResourceArray<gr_rp<const GrRecycledResource>> fTrackedRecycledResources;
SkSTArray<16, sk_sp<const GrBuffer>> fTrackedGpuBuffers;
SkSTArray<16, gr_cb<const GrSurface>> fTrackedGpuSurfaces;
// Tracks whether we are in the middle of a command buffer begin/end calls and thus can add
// new commands to the buffer;
bool fIsActive;
bool fHasWork = false;
// Stores a pointer to the current active render pass (i.e. begin has been called but not
// end). A nullptr means there is no active render pass. The GrVKCommandBuffer does not own
// the render pass.
const GrVkRenderPass* fActiveRenderPass = nullptr;
VkCommandBuffer fCmdBuffer;
virtual void onReleaseResources() {}
virtual void onFreeGPUData(const GrVkGpu* gpu) const = 0;
static constexpr uint32_t kMaxInputBuffers = 2;
VkBuffer fBoundInputBuffers[kMaxInputBuffers];
VkBuffer fBoundIndexBuffer;
// Cached values used for dynamic state updates
VkViewport fCachedViewport;
VkRect2D fCachedScissor;
float fCachedBlendConstant[4];
// Tracking of memory barriers so that we can submit them all in a batch together.
SkSTArray<1, VkBufferMemoryBarrier> fBufferBarriers;
SkSTArray<2, VkImageMemoryBarrier> fImageBarriers;
bool fBarriersByRegion = false;
VkPipelineStageFlags fSrcStageMask = 0;
VkPipelineStageFlags fDstStageMask = 0;
bool fIsWrapped;
};
class GrVkSecondaryCommandBuffer;
class GrVkPrimaryCommandBuffer : public GrVkCommandBuffer {
public:
~GrVkPrimaryCommandBuffer() override;
static GrVkPrimaryCommandBuffer* Create(GrVkGpu* gpu, VkCommandPool cmdPool);
void begin(GrVkGpu* gpu);
void end(GrVkGpu* gpu, bool abandoningBuffer = false);
// Begins render pass on this command buffer. The framebuffer from GrVkRenderTarget will be used
// in the render pass.
bool beginRenderPass(GrVkGpu* gpu,
const GrVkRenderPass*,
sk_sp<const GrVkFramebuffer>,
const VkClearValue clearValues[],
const GrSurface* target,
const SkIRect& bounds,
bool forSecondaryCB);
void endRenderPass(const GrVkGpu* gpu);
void nexSubpass(GrVkGpu* gpu, bool forSecondaryCB);
// Submits the SecondaryCommandBuffer into this command buffer. It is required that we are
// currently inside a render pass that is compatible with the one used to create the
// SecondaryCommandBuffer.
void executeCommands(const GrVkGpu* gpu,
std::unique_ptr<GrVkSecondaryCommandBuffer> secondaryBuffer);
// Commands that only work outside of a render pass
void clearColorImage(const GrVkGpu* gpu,
GrVkImage* image,
const VkClearColorValue* color,
uint32_t subRangeCount,
const VkImageSubresourceRange* subRanges);
void clearDepthStencilImage(const GrVkGpu* gpu,
GrVkImage* image,
const VkClearDepthStencilValue* color,
uint32_t subRangeCount,
const VkImageSubresourceRange* subRanges);
void copyImage(const GrVkGpu* gpu,
GrVkImage* srcImage,
VkImageLayout srcLayout,
GrVkImage* dstImage,
VkImageLayout dstLayout,
uint32_t copyRegionCount,
const VkImageCopy* copyRegions);
void blitImage(const GrVkGpu* gpu,
const GrManagedResource* srcResource,
VkImage srcImage,
VkImageLayout srcLayout,
const GrManagedResource* dstResource,
VkImage dstImage,
VkImageLayout dstLayout,
uint32_t blitRegionCount,
const VkImageBlit* blitRegions,
VkFilter filter);
void blitImage(const GrVkGpu* gpu,
const GrVkImage& srcImage,
const GrVkImage& dstImage,
uint32_t blitRegionCount,
const VkImageBlit* blitRegions,
VkFilter filter);
void copyImageToBuffer(const GrVkGpu* gpu,
GrVkImage* srcImage,
VkImageLayout srcLayout,
sk_sp<GrGpuBuffer> dstBuffer,
uint32_t copyRegionCount,
const VkBufferImageCopy* copyRegions);
// All uses of copyBufferToImage are done with buffers from our staging manager. The staging
// manager will handle making sure the command buffer refs the buffer. Thus we just pass in the
// raw VkBuffer here and don't worry about refs.
void copyBufferToImage(const GrVkGpu* gpu,
VkBuffer srcBuffer,
GrVkImage* dstImage,
VkImageLayout dstLayout,
uint32_t copyRegionCount,
const VkBufferImageCopy* copyRegions);
void fillBuffer(GrVkGpu* gpu,
sk_sp<GrGpuBuffer>,
VkDeviceSize offset,
VkDeviceSize size,
uint32_t data);
void copyBuffer(GrVkGpu* gpu,
sk_sp<GrGpuBuffer> srcBuffer,
sk_sp<GrGpuBuffer> dstBuffer,
uint32_t regionCount,
const VkBufferCopy* regions);
void updateBuffer(GrVkGpu* gpu,
sk_sp<GrVkBuffer> dstBuffer,
VkDeviceSize dstOffset,
VkDeviceSize dataSize,
const void* data);
void resolveImage(GrVkGpu* gpu,
const GrVkImage& srcImage,
const GrVkImage& dstImage,
uint32_t regionCount,
const VkImageResolve* regions);
bool submitToQueue(GrVkGpu* gpu, VkQueue queue,
SkTArray<GrVkSemaphore::Resource*>& signalSemaphores,
SkTArray<GrVkSemaphore::Resource*>& waitSemaphores);
void forceSync(GrVkGpu* gpu);
bool finished(GrVkGpu* gpu);
void addFinishedProc(sk_sp<skgpu::RefCntedCallback> finishedProc);
void callFinishedProcs() {
fFinishedProcs.reset();
}
void recycleSecondaryCommandBuffers(GrVkCommandPool* cmdPool);
private:
explicit GrVkPrimaryCommandBuffer(VkCommandBuffer cmdBuffer)
: INHERITED(cmdBuffer)
, fSubmitFence(VK_NULL_HANDLE) {}
void onFreeGPUData(const GrVkGpu* gpu) const override;
void onReleaseResources() override;
SkTArray<std::unique_ptr<GrVkSecondaryCommandBuffer>, true> fSecondaryCommandBuffers;
VkFence fSubmitFence;
SkTArray<sk_sp<skgpu::RefCntedCallback>> fFinishedProcs;
using INHERITED = GrVkCommandBuffer;
};
class GrVkSecondaryCommandBuffer : public GrVkCommandBuffer {
public:
static GrVkSecondaryCommandBuffer* Create(GrVkGpu* gpu, GrVkCommandPool* cmdPool);
// Used for wrapping an external secondary command buffer.
static GrVkSecondaryCommandBuffer* Create(VkCommandBuffer externalSecondaryCB,
const GrVkRenderPass* externalRenderPass);
void begin(GrVkGpu* gpu, const GrVkFramebuffer* framebuffer,
const GrVkRenderPass* compatibleRenderPass);
void end(GrVkGpu* gpu);
void recycle(GrVkCommandPool* cmdPool);
VkCommandBuffer vkCommandBuffer() { return fCmdBuffer; }
private:
explicit GrVkSecondaryCommandBuffer(VkCommandBuffer cmdBuffer,
const GrVkRenderPass* externalRenderPass)
: INHERITED(cmdBuffer, SkToBool(externalRenderPass)) {
fActiveRenderPass = externalRenderPass;
}
void onFreeGPUData(const GrVkGpu* gpu) const override {}
// Used for accessing fIsActive (on GrVkCommandBuffer)
friend class GrVkPrimaryCommandBuffer;
using INHERITED = GrVkCommandBuffer;
};
#endif