|  | /* | 
|  | * 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 = skia_private::STArray<kInitialTrackedResourcesCount, T>; | 
|  | TrackedResourceArray<sk_sp<const GrManagedResource>> fTrackedResources; | 
|  | TrackedResourceArray<gr_rp<const GrRecycledResource>> fTrackedRecycledResources; | 
|  | skia_private::STArray<16, sk_sp<const GrBuffer>> fTrackedGpuBuffers; | 
|  | skia_private::STArray<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. | 
|  | skia_private::STArray<1, VkBufferMemoryBarrier> fBufferBarriers; | 
|  | skia_private::STArray<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, | 
|  | skia_private::TArray<GrVkSemaphore::Resource*>& signalSemaphores, | 
|  | skia_private::TArray<GrVkSemaphore::Resource*>& waitSemaphores); | 
|  |  | 
|  | void forceSync(GrVkGpu* gpu); | 
|  |  | 
|  | bool finished(GrVkGpu* gpu); | 
|  |  | 
|  | void addFinishedProc(sk_sp<skgpu::RefCntedCallback> finishedProc); | 
|  |  | 
|  | void callFinishedProcs() { | 
|  | fFinishedProcs.clear(); | 
|  | } | 
|  |  | 
|  | 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; | 
|  |  | 
|  | skia_private::TArray<std::unique_ptr<GrVkSecondaryCommandBuffer>, true> fSecondaryCommandBuffers; | 
|  | VkFence                                                     fSubmitFence; | 
|  | skia_private::TArray<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 |