mirror of
https://github.com/baldurk/renderdoc.git
synced 2026-05-27 20:31:02 +00:00
2809 lines
102 KiB
C++
2809 lines
102 KiB
C++
/******************************************************************************
|
|
* The MIT License (MIT)
|
|
*
|
|
* Copyright (c) 2015-2017 Baldur Karlsson
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
* THE SOFTWARE.
|
|
******************************************************************************/
|
|
|
|
#include "../vk_core.h"
|
|
|
|
std::vector<VkImageMemoryBarrier> WrappedVulkan::GetImplicitRenderPassBarriers(uint32_t subpass)
|
|
{
|
|
ResourceId rp, fb;
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
rp = m_RenderState.renderPass;
|
|
fb = m_RenderState.framebuffer;
|
|
}
|
|
else
|
|
{
|
|
rp = m_BakedCmdBufferInfo[m_LastCmdBufferID].state.renderPass;
|
|
fb = m_BakedCmdBufferInfo[m_LastCmdBufferID].state.framebuffer;
|
|
}
|
|
|
|
std::vector<VkImageMemoryBarrier> ret;
|
|
|
|
VulkanCreationInfo::Framebuffer fbinfo = m_CreationInfo.m_Framebuffer[fb];
|
|
VulkanCreationInfo::RenderPass rpinfo = m_CreationInfo.m_RenderPass[rp];
|
|
|
|
std::vector<VkAttachmentReference> atts;
|
|
|
|
// a bit of dancing to get a subpass index. Because we don't increment
|
|
// the subpass counter on EndRenderPass the value is the same for the last
|
|
// NextSubpass. Instead we pass in the subpass index of ~0U for End
|
|
if(subpass == ~0U)
|
|
{
|
|
// we transition all attachments to finalLayout from whichever they
|
|
// were in previously
|
|
atts.resize(rpinfo.attachments.size());
|
|
for(size_t i = 0; i < rpinfo.attachments.size(); i++)
|
|
{
|
|
atts[i].attachment = (uint32_t)i;
|
|
atts[i].layout = rpinfo.attachments[i].finalLayout;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if(IsActiveReplaying(m_State))
|
|
subpass = m_RenderState.subpass;
|
|
else
|
|
subpass = m_BakedCmdBufferInfo[m_LastCmdBufferID].state.subpass;
|
|
|
|
// transition the attachments in this subpass
|
|
for(size_t i = 0; i < rpinfo.subpasses[subpass].colorAttachments.size(); i++)
|
|
{
|
|
uint32_t attIdx = rpinfo.subpasses[subpass].colorAttachments[i];
|
|
|
|
if(attIdx == VK_ATTACHMENT_UNUSED)
|
|
continue;
|
|
|
|
atts.push_back(VkAttachmentReference());
|
|
atts.back().attachment = attIdx;
|
|
atts.back().layout = rpinfo.subpasses[subpass].colorLayouts[i];
|
|
}
|
|
|
|
for(size_t i = 0; i < rpinfo.subpasses[subpass].inputAttachments.size(); i++)
|
|
{
|
|
uint32_t attIdx = rpinfo.subpasses[subpass].inputAttachments[i];
|
|
|
|
if(attIdx == VK_ATTACHMENT_UNUSED)
|
|
continue;
|
|
|
|
atts.push_back(VkAttachmentReference());
|
|
atts.back().attachment = attIdx;
|
|
atts.back().layout = rpinfo.subpasses[subpass].inputLayouts[i];
|
|
}
|
|
|
|
int32_t ds = rpinfo.subpasses[subpass].depthstencilAttachment;
|
|
|
|
if(ds != -1)
|
|
{
|
|
atts.push_back(VkAttachmentReference());
|
|
atts.back().attachment = (uint32_t)rpinfo.subpasses[subpass].depthstencilAttachment;
|
|
atts.back().layout = rpinfo.subpasses[subpass].depthstencilLayout;
|
|
}
|
|
}
|
|
|
|
for(size_t i = 0; i < atts.size(); i++)
|
|
{
|
|
uint32_t idx = atts[i].attachment;
|
|
|
|
VkImageMemoryBarrier barrier = {VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER};
|
|
|
|
ResourceId view = fbinfo.attachments[idx].view;
|
|
|
|
barrier.subresourceRange = m_CreationInfo.m_ImageView[view].range;
|
|
barrier.image = Unwrap(
|
|
GetResourceManager()->GetCurrentHandle<VkImage>(m_CreationInfo.m_ImageView[view].image));
|
|
|
|
barrier.newLayout = atts[i].layout;
|
|
|
|
// search back from this subpass to see which layout it was in before. If it's
|
|
// not been used in a previous subpass, then default to initialLayout
|
|
barrier.oldLayout = rpinfo.attachments[idx].initialLayout;
|
|
|
|
if(subpass == ~0U)
|
|
subpass = (uint32_t)rpinfo.subpasses.size();
|
|
|
|
// subpass is at this point a 1-indexed value essentially, as it's the index
|
|
// of the subpass we just finished (or 0 if we're in BeginRenderPass in which
|
|
// case the loop just skips completely and we use initialLayout, which is
|
|
// correct).
|
|
|
|
for(uint32_t s = subpass; s > 0; s--)
|
|
{
|
|
bool found = false;
|
|
|
|
for(size_t a = 0; !found && a < rpinfo.subpasses[s - 1].colorAttachments.size(); a++)
|
|
{
|
|
if(rpinfo.subpasses[s - 1].colorAttachments[a] == idx)
|
|
{
|
|
barrier.oldLayout = rpinfo.subpasses[s - 1].colorLayouts[a];
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(found)
|
|
break;
|
|
|
|
for(size_t a = 0; !found && a < rpinfo.subpasses[s - 1].inputAttachments.size(); a++)
|
|
{
|
|
if(rpinfo.subpasses[s - 1].inputAttachments[a] == idx)
|
|
{
|
|
barrier.oldLayout = rpinfo.subpasses[s - 1].inputLayouts[a];
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(found)
|
|
break;
|
|
|
|
if((uint32_t)rpinfo.subpasses[s - 1].depthstencilAttachment == idx)
|
|
{
|
|
barrier.oldLayout = rpinfo.subpasses[s - 1].depthstencilLayout;
|
|
break;
|
|
}
|
|
}
|
|
|
|
ReplacePresentableImageLayout(barrier.oldLayout);
|
|
ReplacePresentableImageLayout(barrier.newLayout);
|
|
|
|
ret.push_back(barrier);
|
|
}
|
|
|
|
// erase any do-nothing barriers
|
|
for(auto it = ret.begin(); it != ret.end();)
|
|
{
|
|
if(it->oldLayout == it->newLayout)
|
|
it = ret.erase(it);
|
|
else
|
|
++it;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
string WrappedVulkan::MakeRenderPassOpString(bool store)
|
|
{
|
|
string opDesc = "";
|
|
|
|
const VulkanCreationInfo::RenderPass &info =
|
|
m_CreationInfo.m_RenderPass[m_BakedCmdBufferInfo[m_LastCmdBufferID].state.renderPass];
|
|
const VulkanCreationInfo::Framebuffer &fbinfo =
|
|
m_CreationInfo.m_Framebuffer[m_BakedCmdBufferInfo[m_LastCmdBufferID].state.framebuffer];
|
|
|
|
const vector<VkAttachmentDescription> &atts = info.attachments;
|
|
|
|
if(atts.empty())
|
|
{
|
|
opDesc = "-";
|
|
}
|
|
else
|
|
{
|
|
bool colsame = true;
|
|
|
|
uint32_t subpass = m_BakedCmdBufferInfo[m_LastCmdBufferID].state.subpass;
|
|
|
|
// find which attachment is the depth-stencil one
|
|
int32_t dsAttach = info.subpasses[subpass].depthstencilAttachment;
|
|
bool hasStencil = false;
|
|
bool depthonly = false;
|
|
|
|
// if there is a depth-stencil attachment, see if it has a stencil
|
|
// component and if the subpass is depth only (no other attachments)
|
|
if(dsAttach >= 0)
|
|
{
|
|
hasStencil = !IsDepthOnlyFormat(fbinfo.attachments[dsAttach].format);
|
|
depthonly = info.subpasses[subpass].colorAttachments.size() == 0;
|
|
}
|
|
|
|
const std::vector<uint32_t> &cols = info.subpasses[subpass].colorAttachments;
|
|
|
|
// we check all non-UNUSED attachments to see if they're all the same.
|
|
// To begin with we point to an invalid attachment index
|
|
uint32_t col0 = VK_ATTACHMENT_UNUSED;
|
|
|
|
// look through all other color attachments to see if they're identical
|
|
for(size_t i = 0; i < cols.size(); i++)
|
|
{
|
|
const uint32_t col = cols[i];
|
|
|
|
// skip unused attachments
|
|
if(col == VK_ATTACHMENT_UNUSED)
|
|
continue;
|
|
|
|
// the first valid attachment we find, use that as our reference point
|
|
if(col0 == VK_ATTACHMENT_UNUSED)
|
|
{
|
|
col0 = col;
|
|
continue;
|
|
}
|
|
|
|
// for any other attachments, compare them to the reference
|
|
if(store)
|
|
{
|
|
if(atts[col].storeOp != atts[col0].storeOp)
|
|
colsame = false;
|
|
}
|
|
else
|
|
{
|
|
if(atts[col].loadOp != atts[col0].loadOp)
|
|
colsame = false;
|
|
}
|
|
}
|
|
|
|
// handle depth only passes
|
|
if(depthonly)
|
|
{
|
|
opDesc = "";
|
|
}
|
|
else if(!colsame)
|
|
{
|
|
// if we have different storage for the colour, don't display
|
|
// the full details
|
|
|
|
opDesc = store ? "Different store ops" : "Different load ops";
|
|
}
|
|
else if(col0 == VK_ATTACHMENT_UNUSED)
|
|
{
|
|
// we're here if we didn't find any non-UNUSED color attachments at all
|
|
opDesc = "Unused";
|
|
}
|
|
else
|
|
{
|
|
// all colour ops are the same, print it
|
|
opDesc = store ? ToStr(atts[col0].storeOp) : ToStr(atts[col0].loadOp);
|
|
}
|
|
|
|
// do we have depth?
|
|
if(dsAttach != -1)
|
|
{
|
|
// could be empty if this is a depth-only pass
|
|
if(!opDesc.empty())
|
|
opDesc = "C=" + opDesc + ", ";
|
|
|
|
// if there's no stencil, just print depth op
|
|
if(!hasStencil)
|
|
{
|
|
opDesc += "D=" + (store ? ToStr(atts[dsAttach].storeOp) : ToStr(atts[dsAttach].loadOp));
|
|
}
|
|
else
|
|
{
|
|
if(store)
|
|
{
|
|
// if depth and stencil have same op, print together, otherwise separately
|
|
if(atts[dsAttach].storeOp == atts[dsAttach].stencilStoreOp)
|
|
opDesc += "DS=" + ToStr(atts[dsAttach].storeOp);
|
|
else
|
|
opDesc +=
|
|
"D=" + ToStr(atts[dsAttach].storeOp) + ", S=" + ToStr(atts[dsAttach].stencilStoreOp);
|
|
}
|
|
else
|
|
{
|
|
// if depth and stencil have same op, print together, otherwise separately
|
|
if(atts[dsAttach].loadOp == atts[dsAttach].stencilLoadOp)
|
|
opDesc += "DS=" + ToStr(atts[dsAttach].loadOp);
|
|
else
|
|
opDesc +=
|
|
"D=" + ToStr(atts[dsAttach].loadOp) + ", S=" + ToStr(atts[dsAttach].stencilLoadOp);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return opDesc;
|
|
}
|
|
|
|
// Command pool functions
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCreateCommandPool(SerialiserType &ser, VkDevice device,
|
|
const VkCommandPoolCreateInfo *pCreateInfo,
|
|
const VkAllocationCallbacks *pAllocator,
|
|
VkCommandPool *pCmdPool)
|
|
{
|
|
SERIALISE_ELEMENT(device);
|
|
SERIALISE_ELEMENT_LOCAL(CreateInfo, *pCreateInfo);
|
|
SERIALISE_ELEMENT_LOCAL(CmdPool, GetResID(*pCmdPool));
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
VkCommandPool pool = VK_NULL_HANDLE;
|
|
|
|
VkResult ret = ObjDisp(device)->CreateCommandPool(Unwrap(device), &CreateInfo, NULL, &pool);
|
|
|
|
if(ret != VK_SUCCESS)
|
|
{
|
|
RDCERR("Failed on resource serialise-creation, VkResult: 0x%08x", ret);
|
|
}
|
|
else
|
|
{
|
|
ResourceId live = GetResourceManager()->WrapResource(Unwrap(device), pool);
|
|
GetResourceManager()->AddLiveResource(CmdPool, pool);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
VkResult WrappedVulkan::vkCreateCommandPool(VkDevice device,
|
|
const VkCommandPoolCreateInfo *pCreateInfo,
|
|
const VkAllocationCallbacks *pAllocator,
|
|
VkCommandPool *pCmdPool)
|
|
{
|
|
VkResult ret =
|
|
ObjDisp(device)->CreateCommandPool(Unwrap(device), pCreateInfo, pAllocator, pCmdPool);
|
|
|
|
if(ret == VK_SUCCESS)
|
|
{
|
|
ResourceId id = GetResourceManager()->WrapResource(Unwrap(device), *pCmdPool);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
Chunk *chunk = NULL;
|
|
|
|
{
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCreateCommandPool);
|
|
Serialise_vkCreateCommandPool(ser, device, pCreateInfo, NULL, pCmdPool);
|
|
|
|
chunk = scope.Get();
|
|
}
|
|
|
|
VkResourceRecord *record = GetResourceManager()->AddResourceRecord(*pCmdPool);
|
|
record->AddChunk(chunk);
|
|
}
|
|
else
|
|
{
|
|
GetResourceManager()->AddLiveResource(id, *pCmdPool);
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
VkResult WrappedVulkan::vkResetCommandPool(VkDevice device, VkCommandPool cmdPool,
|
|
VkCommandPoolResetFlags flags)
|
|
{
|
|
return ObjDisp(device)->ResetCommandPool(Unwrap(device), Unwrap(cmdPool), flags);
|
|
}
|
|
|
|
void WrappedVulkan::vkTrimCommandPoolKHR(VkDevice device, VkCommandPool commandPool,
|
|
VkCommandPoolTrimFlagsKHR flags)
|
|
{
|
|
return ObjDisp(device)->TrimCommandPoolKHR(Unwrap(device), Unwrap(commandPool), flags);
|
|
}
|
|
|
|
// Command buffer functions
|
|
|
|
VkResult WrappedVulkan::vkAllocateCommandBuffers(VkDevice device,
|
|
const VkCommandBufferAllocateInfo *pAllocateInfo,
|
|
VkCommandBuffer *pCommandBuffers)
|
|
{
|
|
VkCommandBufferAllocateInfo unwrappedInfo = *pAllocateInfo;
|
|
unwrappedInfo.commandPool = Unwrap(unwrappedInfo.commandPool);
|
|
VkResult ret =
|
|
ObjDisp(device)->AllocateCommandBuffers(Unwrap(device), &unwrappedInfo, pCommandBuffers);
|
|
|
|
if(ret == VK_SUCCESS)
|
|
{
|
|
for(uint32_t i = 0; i < unwrappedInfo.commandBufferCount; i++)
|
|
{
|
|
VkCommandBuffer unwrappedReal = pCommandBuffers[i];
|
|
|
|
ResourceId id = GetResourceManager()->WrapResource(Unwrap(device), pCommandBuffers[i]);
|
|
|
|
// we set this *after* wrapping, so that the wrapped resource copies the 'uninitialised'
|
|
// loader table, since the loader expects to set the dispatch table onto an existing magic
|
|
// number in the trampoline function at the start of the chain.
|
|
if(m_SetDeviceLoaderData)
|
|
m_SetDeviceLoaderData(device, unwrappedReal);
|
|
else
|
|
SetDispatchTableOverMagicNumber(device, unwrappedReal);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetResourceManager()->AddResourceRecord(pCommandBuffers[i]);
|
|
|
|
record->bakedCommands = NULL;
|
|
|
|
record->pool = GetRecord(pAllocateInfo->commandPool);
|
|
record->AddParent(record->pool);
|
|
|
|
{
|
|
record->pool->LockChunks();
|
|
record->pool->pooledChildren.push_back(record);
|
|
record->pool->UnlockChunks();
|
|
}
|
|
|
|
// we don't serialise this as we never create this command buffer directly.
|
|
// Instead we create a command buffer for each baked list that we find.
|
|
|
|
// if pNext is non-NULL, need to do a deep copy
|
|
// we don't support any extensions on VkCommandBufferCreateInfo anyway
|
|
RDCASSERT(pAllocateInfo->pNext == NULL);
|
|
|
|
record->cmdInfo = new CmdBufferRecordingInfo();
|
|
|
|
record->cmdInfo->device = device;
|
|
record->cmdInfo->allocInfo = *pAllocateInfo;
|
|
record->cmdInfo->allocInfo.commandBufferCount = 1;
|
|
}
|
|
else
|
|
{
|
|
GetResourceManager()->AddLiveResource(id, pCommandBuffers[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkBeginCommandBuffer(SerialiserType &ser, VkCommandBuffer commandBuffer,
|
|
const VkCommandBufferBeginInfo *pBeginInfo)
|
|
{
|
|
ResourceId BakedCommandBuffer;
|
|
VkCommandBufferAllocateInfo AllocateInfo;
|
|
VkDevice device = VK_NULL_HANDLE;
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
RDCASSERT(record->bakedCommands);
|
|
if(record->bakedCommands)
|
|
BakedCommandBuffer = record->bakedCommands->GetResourceID();
|
|
|
|
RDCASSERT(record->cmdInfo);
|
|
device = record->cmdInfo->device;
|
|
AllocateInfo = record->cmdInfo->allocInfo;
|
|
}
|
|
|
|
SERIALISE_ELEMENT_LOCAL(CommandBuffer, GetResID(commandBuffer));
|
|
SERIALISE_ELEMENT_LOCAL(BeginInfo, *pBeginInfo);
|
|
SERIALISE_ELEMENT(BakedCommandBuffer);
|
|
SERIALISE_ELEMENT(device);
|
|
SERIALISE_ELEMENT(AllocateInfo);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = CommandBuffer;
|
|
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].level = m_BakedCmdBufferInfo[BakedCommandBuffer].level =
|
|
AllocateInfo.level;
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].beginFlags =
|
|
m_BakedCmdBufferInfo[BakedCommandBuffer].beginFlags = BeginInfo.flags;
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].markerCount = 0;
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
const uint32_t length = m_BakedCmdBufferInfo[BakedCommandBuffer].eventCount;
|
|
|
|
bool partial = false;
|
|
int partialType = ePartialNum;
|
|
|
|
// check for partial execution of this command buffer
|
|
for(int p = 0; p < ePartialNum; p++)
|
|
{
|
|
const std::vector<Submission> &submissions =
|
|
m_Partial[p].cmdBufferSubmits[BakedCommandBuffer];
|
|
|
|
for(auto it = submissions.begin(); it != submissions.end(); ++it)
|
|
{
|
|
if(it->baseEvent <= m_LastEventID && m_LastEventID < (it->baseEvent + length))
|
|
{
|
|
#if ENABLED(VERBOSE_PARTIAL_REPLAY)
|
|
RDCDEBUG("vkBegin - partial detected %u < %u < %u, %llu -> %llu", it->baseEvent,
|
|
m_LastEventID, it->baseEvent + length, m_LastCmdBufferID, BakedCommandBuffer);
|
|
#endif
|
|
|
|
m_Partial[p].partialParent = BakedCommandBuffer;
|
|
m_Partial[p].baseEvent = it->baseEvent;
|
|
m_Partial[p].renderPassActive = false;
|
|
m_Partial[p].partialDevice = device;
|
|
m_Partial[p].resultPartialCmdPool = AllocateInfo.commandPool;
|
|
|
|
partial = true;
|
|
partialType = p;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(partial || (m_DrawcallCallback && m_DrawcallCallback->RecordAllCmds()))
|
|
{
|
|
// pull all re-recorded commands from our own device and command pool for easier cleanup
|
|
if(!partial)
|
|
{
|
|
device = GetDev();
|
|
AllocateInfo.commandPool = Unwrap(m_InternalCmds.cmdpool);
|
|
}
|
|
|
|
VkCommandBuffer cmd = VK_NULL_HANDLE;
|
|
VkCommandBufferAllocateInfo unwrappedInfo = AllocateInfo;
|
|
unwrappedInfo.commandPool = Unwrap(unwrappedInfo.commandPool);
|
|
VkResult ret = ObjDisp(device)->AllocateCommandBuffers(Unwrap(device), &unwrappedInfo, &cmd);
|
|
|
|
if(ret != VK_SUCCESS)
|
|
{
|
|
RDCERR("Failed on resource serialise-creation, VkResult: 0x%08x", ret);
|
|
}
|
|
else
|
|
{
|
|
GetResourceManager()->WrapResource(Unwrap(device), cmd);
|
|
}
|
|
|
|
if(partial)
|
|
{
|
|
m_Partial[partialType].resultPartialCmdBuffer = cmd;
|
|
}
|
|
else
|
|
{
|
|
// we store under both baked and non baked ID.
|
|
// The baked ID is the 'real' entry, the non baked is simply so it
|
|
// can be found in the subsequent serialised commands that ref the
|
|
// non-baked ID. The baked ID is referenced by the submit itself.
|
|
//
|
|
// In vkEndCommandBuffer we erase the non-baked reference, and since
|
|
// we know you can only be recording a command buffer once at a time
|
|
// (even if it's baked to several command buffers in the frame)
|
|
// there's no issue with clashes here.
|
|
m_RerecordCmds[BakedCommandBuffer] = cmd;
|
|
m_RerecordCmds[m_LastCmdBufferID] = cmd;
|
|
}
|
|
|
|
m_BakedCmdBufferInfo[GetResID(cmd)].level = AllocateInfo.level;
|
|
m_BakedCmdBufferInfo[GetResID(cmd)].beginFlags = BeginInfo.flags;
|
|
|
|
// add one-time submit flag as this partial cmd buffer will only be submitted once
|
|
BeginInfo.flags |= VK_COMMAND_BUFFER_USAGE_ONE_TIME_SUBMIT_BIT;
|
|
if(AllocateInfo.level == VK_COMMAND_BUFFER_LEVEL_SECONDARY)
|
|
BeginInfo.flags |= VK_COMMAND_BUFFER_USAGE_SIMULTANEOUS_USE_BIT;
|
|
|
|
ObjDisp(cmd)->BeginCommandBuffer(Unwrap(cmd), &BeginInfo);
|
|
}
|
|
|
|
// whenever a vkCmd command-building chunk asks for the command buffer, it
|
|
// will get our baked version.
|
|
if(GetResourceManager()->HasReplacement(m_LastCmdBufferID))
|
|
GetResourceManager()->RemoveReplacement(m_LastCmdBufferID);
|
|
|
|
GetResourceManager()->ReplaceResource(m_LastCmdBufferID, BakedCommandBuffer);
|
|
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].curEventID = 0;
|
|
m_BakedCmdBufferInfo[BakedCommandBuffer].curEventID = 0;
|
|
}
|
|
else
|
|
{
|
|
// remove one-time submit flag as we will want to submit many
|
|
BeginInfo.flags &= ~VK_COMMAND_BUFFER_USAGE_ONE_TIME_SUBMIT_BIT;
|
|
if(AllocateInfo.level == VK_COMMAND_BUFFER_LEVEL_SECONDARY)
|
|
BeginInfo.flags |= VK_COMMAND_BUFFER_USAGE_SIMULTANEOUS_USE_BIT;
|
|
|
|
VkCommandBuffer cmd = VK_NULL_HANDLE;
|
|
|
|
if(!GetResourceManager()->HasLiveResource(BakedCommandBuffer))
|
|
{
|
|
VkCommandBufferAllocateInfo unwrappedInfo = AllocateInfo;
|
|
unwrappedInfo.commandPool = Unwrap(unwrappedInfo.commandPool);
|
|
VkResult ret = ObjDisp(device)->AllocateCommandBuffers(Unwrap(device), &unwrappedInfo, &cmd);
|
|
|
|
if(ret != VK_SUCCESS)
|
|
{
|
|
RDCERR("Failed on resource serialise-creation, VkResult: 0x%08x", ret);
|
|
}
|
|
else
|
|
{
|
|
ResourceId live = GetResourceManager()->WrapResource(Unwrap(device), cmd);
|
|
GetResourceManager()->AddLiveResource(BakedCommandBuffer, cmd);
|
|
}
|
|
|
|
// whenever a vkCmd command-building chunk asks for the command buffer, it
|
|
// will get our baked version.
|
|
if(GetResourceManager()->HasReplacement(m_LastCmdBufferID))
|
|
GetResourceManager()->RemoveReplacement(m_LastCmdBufferID);
|
|
|
|
GetResourceManager()->ReplaceResource(m_LastCmdBufferID, BakedCommandBuffer);
|
|
}
|
|
else
|
|
{
|
|
cmd = GetResourceManager()->GetLiveHandle<VkCommandBuffer>(BakedCommandBuffer);
|
|
}
|
|
|
|
// propagate any name there might be
|
|
if(m_CreationInfo.m_Names.find(m_LastCmdBufferID) != m_CreationInfo.m_Names.end())
|
|
m_CreationInfo.m_Names[GetResourceManager()->GetLiveID(BakedCommandBuffer)] =
|
|
m_CreationInfo.m_Names[m_LastCmdBufferID];
|
|
|
|
{
|
|
VulkanDrawcallTreeNode *draw = new VulkanDrawcallTreeNode;
|
|
m_BakedCmdBufferInfo[BakedCommandBuffer].draw = draw;
|
|
|
|
// On queue submit we increment all child events/drawcalls by
|
|
// m_RootEventID and insert them into the tree.
|
|
m_BakedCmdBufferInfo[BakedCommandBuffer].curEventID = 0;
|
|
m_BakedCmdBufferInfo[BakedCommandBuffer].eventCount = 0;
|
|
m_BakedCmdBufferInfo[BakedCommandBuffer].drawCount = 0;
|
|
|
|
m_BakedCmdBufferInfo[BakedCommandBuffer].drawStack.push_back(draw);
|
|
}
|
|
|
|
ObjDisp(device)->BeginCommandBuffer(Unwrap(cmd), &BeginInfo);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
VkResult WrappedVulkan::vkBeginCommandBuffer(VkCommandBuffer commandBuffer,
|
|
const VkCommandBufferBeginInfo *pBeginInfo)
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
RDCASSERT(record);
|
|
|
|
if(record)
|
|
{
|
|
// If a command bfufer was already recorded (ie we have some baked commands),
|
|
// then begin is spec'd to implicitly reset. That means we need to tidy up
|
|
// any existing baked commands before creating a new set.
|
|
if(record->bakedCommands)
|
|
record->bakedCommands->Delete(GetResourceManager());
|
|
|
|
record->bakedCommands = GetResourceManager()->AddResourceRecord(ResourceIDGen::GetNewUniqueID());
|
|
record->bakedCommands->SpecialResource = true;
|
|
record->bakedCommands->Resource = (WrappedVkRes *)commandBuffer;
|
|
record->bakedCommands->cmdInfo = new CmdBufferRecordingInfo();
|
|
|
|
record->bakedCommands->cmdInfo->device = record->cmdInfo->device;
|
|
record->bakedCommands->cmdInfo->allocInfo = record->cmdInfo->allocInfo;
|
|
|
|
{
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkBeginCommandBuffer);
|
|
Serialise_vkBeginCommandBuffer(ser, commandBuffer, pBeginInfo);
|
|
|
|
record->AddChunk(scope.Get());
|
|
}
|
|
|
|
if(pBeginInfo->pInheritanceInfo)
|
|
{
|
|
record->MarkResourceFrameReferenced(GetResID(pBeginInfo->pInheritanceInfo->renderPass),
|
|
eFrameRef_Read);
|
|
record->MarkResourceFrameReferenced(GetResID(pBeginInfo->pInheritanceInfo->framebuffer),
|
|
eFrameRef_Read);
|
|
}
|
|
}
|
|
|
|
VkCommandBufferInheritanceInfo unwrappedInfo;
|
|
if(pBeginInfo->pInheritanceInfo)
|
|
{
|
|
unwrappedInfo = *pBeginInfo->pInheritanceInfo;
|
|
unwrappedInfo.framebuffer = Unwrap(unwrappedInfo.framebuffer);
|
|
unwrappedInfo.renderPass = Unwrap(unwrappedInfo.renderPass);
|
|
|
|
VkCommandBufferBeginInfo beginInfo = *pBeginInfo;
|
|
beginInfo.pInheritanceInfo = &unwrappedInfo;
|
|
return ObjDisp(commandBuffer)->BeginCommandBuffer(Unwrap(commandBuffer), &beginInfo);
|
|
}
|
|
|
|
return ObjDisp(commandBuffer)->BeginCommandBuffer(Unwrap(commandBuffer), pBeginInfo);
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkEndCommandBuffer(SerialiserType &ser, VkCommandBuffer commandBuffer)
|
|
{
|
|
ResourceId BakedCommandBuffer;
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
RDCASSERT(record->bakedCommands);
|
|
if(record->bakedCommands)
|
|
BakedCommandBuffer = record->bakedCommands->GetResourceID();
|
|
}
|
|
|
|
SERIALISE_ELEMENT_LOCAL(CommandBuffer, GetResID(commandBuffer));
|
|
SERIALISE_ELEMENT(BakedCommandBuffer);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = CommandBuffer;
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(BakedCommandBuffer))
|
|
{
|
|
commandBuffer = RerecordCmdBuf(BakedCommandBuffer);
|
|
#if ENABLED(VERBOSE_PARTIAL_REPLAY)
|
|
RDCDEBUG("Ending partial command buffer for %llu baked to %llu", m_LastCmdBufferID,
|
|
BakedCommandBuffer);
|
|
#endif
|
|
|
|
bool recordAll = m_DrawcallCallback && m_DrawcallCallback->RecordAllCmds();
|
|
|
|
if(!recordAll && m_Partial[Primary].partialParent == BakedCommandBuffer &&
|
|
m_Partial[Primary].renderPassActive)
|
|
{
|
|
uint32_t numSubpasses =
|
|
(uint32_t)m_CreationInfo.m_RenderPass[m_RenderState.renderPass].subpasses.size();
|
|
|
|
for(uint32_t sub = m_RenderState.subpass; sub < numSubpasses - 1; sub++)
|
|
ObjDisp(commandBuffer)->CmdNextSubpass(Unwrap(commandBuffer), VK_SUBPASS_CONTENTS_INLINE);
|
|
|
|
ObjDisp(commandBuffer)->CmdEndRenderPass(Unwrap(commandBuffer));
|
|
}
|
|
|
|
if(ObjDisp(commandBuffer)->CmdDebugMarkerEndEXT)
|
|
for(int i = 0; i < m_BakedCmdBufferInfo[BakedCommandBuffer].markerCount; i++)
|
|
ObjDisp(commandBuffer)->CmdDebugMarkerEndEXT(Unwrap(commandBuffer));
|
|
|
|
ObjDisp(commandBuffer)->EndCommandBuffer(Unwrap(commandBuffer));
|
|
|
|
// erase the non-baked reference to this command buffer so that we don't have
|
|
// duplicates when it comes time to clean up. See above in vkBeginCommandBuffer
|
|
m_RerecordCmds.erase(m_LastCmdBufferID);
|
|
|
|
if(m_Partial[Primary].partialParent == BakedCommandBuffer)
|
|
m_Partial[Primary].partialParent = ResourceId();
|
|
}
|
|
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].curEventID = 0;
|
|
}
|
|
else
|
|
{
|
|
commandBuffer = GetResourceManager()->GetLiveHandle<VkCommandBuffer>(BakedCommandBuffer);
|
|
|
|
ObjDisp(commandBuffer)->EndCommandBuffer(Unwrap(commandBuffer));
|
|
|
|
if(!m_BakedCmdBufferInfo[m_LastCmdBufferID].curEvents.empty())
|
|
{
|
|
DrawcallDescription draw;
|
|
draw.name = "API Calls";
|
|
draw.flags |= DrawFlags::SetMarker | DrawFlags::APICalls;
|
|
|
|
AddDrawcall(draw, true);
|
|
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].curEventID++;
|
|
}
|
|
|
|
{
|
|
if(GetDrawcallStack().size() > 1)
|
|
GetDrawcallStack().pop_back();
|
|
}
|
|
|
|
{
|
|
m_BakedCmdBufferInfo[BakedCommandBuffer].eventCount =
|
|
m_BakedCmdBufferInfo[BakedCommandBuffer].curEventID;
|
|
m_BakedCmdBufferInfo[BakedCommandBuffer].curEventID = 0;
|
|
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].curEventID = 0;
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].eventCount = 0;
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].drawCount = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
VkResult WrappedVulkan::vkEndCommandBuffer(VkCommandBuffer commandBuffer)
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
RDCASSERT(record);
|
|
|
|
if(record)
|
|
{
|
|
// ensure that we have a matching begin
|
|
RDCASSERT(record->bakedCommands);
|
|
|
|
{
|
|
CACHE_THREAD_SERIALISER();
|
|
ser.SetDrawChunk();
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkEndCommandBuffer);
|
|
Serialise_vkEndCommandBuffer(ser, commandBuffer);
|
|
|
|
record->AddChunk(scope.Get());
|
|
}
|
|
|
|
record->Bake();
|
|
}
|
|
|
|
return ObjDisp(commandBuffer)->EndCommandBuffer(Unwrap(commandBuffer));
|
|
}
|
|
|
|
VkResult WrappedVulkan::vkResetCommandBuffer(VkCommandBuffer commandBuffer,
|
|
VkCommandBufferResetFlags flags)
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
RDCASSERT(record);
|
|
|
|
if(record)
|
|
{
|
|
// all we need to do is remove the existing baked commands.
|
|
// The application will still need to call begin command buffer itself.
|
|
// this function is essentially a driver hint as it cleans up implicitly
|
|
// on begin.
|
|
//
|
|
// Because it's totally legal for an application to record, submit, reset,
|
|
// record, submit again, and we need some way of referencing the two different
|
|
// sets of commands on replay, our command buffers are given new unique IDs
|
|
// each time they are begun, so on replay it looks like they were all unique
|
|
// (albeit with the same properties for those that share a 'parent'). Hence,
|
|
// we don't need to record or replay when a ResetCommandBuffer happens
|
|
if(record->bakedCommands)
|
|
record->bakedCommands->Delete(GetResourceManager());
|
|
|
|
record->bakedCommands = NULL;
|
|
}
|
|
|
|
return ObjDisp(commandBuffer)->ResetCommandBuffer(Unwrap(commandBuffer), flags);
|
|
}
|
|
|
|
// Command buffer building functions
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdBeginRenderPass(SerialiserType &ser, VkCommandBuffer commandBuffer,
|
|
const VkRenderPassBeginInfo *pRenderPassBegin,
|
|
VkSubpassContents contents)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT_LOCAL(RenderPassBegin, *pRenderPassBegin);
|
|
SERIALISE_ELEMENT(contents);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
VkRenderPassBeginInfo unwrappedInfo = RenderPassBegin;
|
|
unwrappedInfo.renderPass = Unwrap(unwrappedInfo.renderPass);
|
|
unwrappedInfo.framebuffer = Unwrap(unwrappedInfo.framebuffer);
|
|
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
{
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
|
|
m_Partial[Primary].renderPassActive = true;
|
|
ObjDisp(commandBuffer)->CmdBeginRenderPass(Unwrap(commandBuffer), &unwrappedInfo, contents);
|
|
|
|
m_RenderState.subpass = 0;
|
|
|
|
m_RenderState.renderPass = GetResID(RenderPassBegin.renderPass);
|
|
m_RenderState.framebuffer = GetResID(RenderPassBegin.framebuffer);
|
|
m_RenderState.renderArea = RenderPassBegin.renderArea;
|
|
|
|
std::vector<VkImageMemoryBarrier> imgBarriers = GetImplicitRenderPassBarriers();
|
|
|
|
ResourceId cmd = GetResID(commandBuffer);
|
|
GetResourceManager()->RecordBarriers(m_BakedCmdBufferInfo[cmd].imgbarriers, m_ImageLayouts,
|
|
(uint32_t)imgBarriers.size(), imgBarriers.data());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ObjDisp(commandBuffer)->CmdBeginRenderPass(Unwrap(commandBuffer), &unwrappedInfo, contents);
|
|
|
|
// track while reading, for fetching the right set of outputs in AddDrawcall
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].state.subpass = 0;
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].state.renderPass = GetResID(RenderPassBegin.renderPass);
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].state.framebuffer =
|
|
GetResID(RenderPassBegin.framebuffer);
|
|
|
|
std::vector<VkImageMemoryBarrier> imgBarriers = GetImplicitRenderPassBarriers();
|
|
|
|
ResourceId cmd = GetResID(commandBuffer);
|
|
GetResourceManager()->RecordBarriers(m_BakedCmdBufferInfo[cmd].imgbarriers, m_ImageLayouts,
|
|
(uint32_t)imgBarriers.size(), imgBarriers.data());
|
|
|
|
AddEvent();
|
|
DrawcallDescription draw;
|
|
draw.name =
|
|
StringFormat::Fmt("vkCmdBeginRenderPass(%s)", MakeRenderPassOpString(false).c_str());
|
|
draw.flags |= DrawFlags::PassBoundary | DrawFlags::BeginPass;
|
|
|
|
AddDrawcall(draw, true);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdBeginRenderPass(VkCommandBuffer commandBuffer,
|
|
const VkRenderPassBeginInfo *pRenderPassBegin,
|
|
VkSubpassContents contents)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
VkRenderPassBeginInfo unwrappedInfo = *pRenderPassBegin;
|
|
unwrappedInfo.renderPass = Unwrap(unwrappedInfo.renderPass);
|
|
unwrappedInfo.framebuffer = Unwrap(unwrappedInfo.framebuffer);
|
|
ObjDisp(commandBuffer)->CmdBeginRenderPass(Unwrap(commandBuffer), &unwrappedInfo, contents);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
ser.SetDrawChunk();
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdBeginRenderPass);
|
|
Serialise_vkCmdBeginRenderPass(ser, commandBuffer, pRenderPassBegin, contents);
|
|
|
|
record->AddChunk(scope.Get());
|
|
record->MarkResourceFrameReferenced(GetResID(pRenderPassBegin->renderPass), eFrameRef_Read);
|
|
|
|
VkResourceRecord *fb = GetRecord(pRenderPassBegin->framebuffer);
|
|
|
|
record->MarkResourceFrameReferenced(fb->GetResourceID(), eFrameRef_Read);
|
|
for(size_t i = 0; i < VkResourceRecord::MaxImageAttachments; i++)
|
|
{
|
|
VkResourceRecord *att = fb->imageAttachments[i].record;
|
|
if(att == NULL)
|
|
break;
|
|
|
|
record->MarkResourceFrameReferenced(att->baseResource, eFrameRef_Write);
|
|
if(att->baseResourceMem != ResourceId())
|
|
record->MarkResourceFrameReferenced(att->baseResourceMem, eFrameRef_Read);
|
|
if(att->sparseInfo)
|
|
record->cmdInfo->sparse.insert(att->sparseInfo);
|
|
record->cmdInfo->dirtied.insert(att->baseResource);
|
|
}
|
|
|
|
record->cmdInfo->framebuffer = fb;
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdNextSubpass(SerialiserType &ser, VkCommandBuffer commandBuffer,
|
|
VkSubpassContents contents)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT(contents);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
// don't do anything if we're executing a single draw, NextSubpass is meaningless (and invalid
|
|
// on a partial render pass)
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID) &&
|
|
m_FirstEventID != m_LastEventID)
|
|
{
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
|
|
m_RenderState.subpass++;
|
|
|
|
ObjDisp(commandBuffer)->CmdNextSubpass(Unwrap(commandBuffer), contents);
|
|
|
|
std::vector<VkImageMemoryBarrier> imgBarriers = GetImplicitRenderPassBarriers();
|
|
|
|
ResourceId cmd = GetResID(commandBuffer);
|
|
GetResourceManager()->RecordBarriers(m_BakedCmdBufferInfo[cmd].imgbarriers, m_ImageLayouts,
|
|
(uint32_t)imgBarriers.size(), imgBarriers.data());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ObjDisp(commandBuffer)->CmdNextSubpass(Unwrap(commandBuffer), contents);
|
|
|
|
// track while reading, for fetching the right set of outputs in AddDrawcall
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].state.subpass++;
|
|
|
|
std::vector<VkImageMemoryBarrier> imgBarriers = GetImplicitRenderPassBarriers();
|
|
|
|
ResourceId cmd = GetResID(commandBuffer);
|
|
GetResourceManager()->RecordBarriers(m_BakedCmdBufferInfo[cmd].imgbarriers, m_ImageLayouts,
|
|
(uint32_t)imgBarriers.size(), &imgBarriers[0]);
|
|
|
|
AddEvent();
|
|
DrawcallDescription draw;
|
|
draw.name = StringFormat::Fmt("vkCmdNextSubpass() => %u",
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].state.subpass);
|
|
draw.flags |= DrawFlags::PassBoundary | DrawFlags::BeginPass | DrawFlags::EndPass;
|
|
|
|
AddDrawcall(draw, true);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdNextSubpass(VkCommandBuffer commandBuffer, VkSubpassContents contents)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)->CmdNextSubpass(Unwrap(commandBuffer), contents);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
ser.SetDrawChunk();
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdNextSubpass);
|
|
Serialise_vkCmdNextSubpass(ser, commandBuffer, contents);
|
|
|
|
record->AddChunk(scope.Get());
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdEndRenderPass(SerialiserType &ser, VkCommandBuffer commandBuffer)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
{
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
|
|
m_Partial[Primary].renderPassActive = false;
|
|
ObjDisp(commandBuffer)->CmdEndRenderPass(Unwrap(commandBuffer));
|
|
|
|
std::vector<VkImageMemoryBarrier> imgBarriers = GetImplicitRenderPassBarriers(~0U);
|
|
|
|
ResourceId cmd = GetResID(commandBuffer);
|
|
GetResourceManager()->RecordBarriers(m_BakedCmdBufferInfo[cmd].imgbarriers, m_ImageLayouts,
|
|
(uint32_t)imgBarriers.size(), imgBarriers.data());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ObjDisp(commandBuffer)->CmdEndRenderPass(Unwrap(commandBuffer));
|
|
|
|
std::vector<VkImageMemoryBarrier> imgBarriers = GetImplicitRenderPassBarriers(~0U);
|
|
|
|
ResourceId cmd = GetResID(commandBuffer);
|
|
GetResourceManager()->RecordBarriers(m_BakedCmdBufferInfo[cmd].imgbarriers, m_ImageLayouts,
|
|
(uint32_t)imgBarriers.size(), &imgBarriers[0]);
|
|
|
|
AddEvent();
|
|
DrawcallDescription draw;
|
|
draw.name = StringFormat::Fmt("vkCmdEndRenderPass(%s)", MakeRenderPassOpString(true).c_str());
|
|
draw.flags |= DrawFlags::PassBoundary | DrawFlags::EndPass;
|
|
|
|
AddDrawcall(draw, true);
|
|
|
|
// track while reading, reset this to empty so AddDrawcall sets no outputs,
|
|
// but only AFTER the above AddDrawcall (we want it grouped together)
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].state.renderPass = ResourceId();
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].state.framebuffer = ResourceId();
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdEndRenderPass(VkCommandBuffer commandBuffer)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)->CmdEndRenderPass(Unwrap(commandBuffer));
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
ser.SetDrawChunk();
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdEndRenderPass);
|
|
Serialise_vkCmdEndRenderPass(ser, commandBuffer);
|
|
|
|
record->AddChunk(scope.Get());
|
|
|
|
VkResourceRecord *fb = record->cmdInfo->framebuffer;
|
|
|
|
std::vector<VkImageMemoryBarrier> barriers;
|
|
|
|
for(size_t i = 0; i < VkResourceRecord::MaxImageAttachments; i++)
|
|
{
|
|
if(fb->imageAttachments[i].barrier.oldLayout == fb->imageAttachments[i].barrier.newLayout)
|
|
continue;
|
|
|
|
barriers.push_back(fb->imageAttachments[i].barrier);
|
|
}
|
|
|
|
// apply the implicit layout transitions here
|
|
{
|
|
SCOPED_LOCK(m_ImageLayoutsLock);
|
|
GetResourceManager()->RecordBarriers(GetRecord(commandBuffer)->cmdInfo->imgbarriers,
|
|
m_ImageLayouts, (uint32_t)barriers.size(), &barriers[0]);
|
|
}
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdBindPipeline(SerialiserType &ser, VkCommandBuffer commandBuffer,
|
|
VkPipelineBindPoint pipelineBindPoint,
|
|
VkPipeline pipeline)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT(pipelineBindPoint);
|
|
SERIALISE_ELEMENT(pipeline);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
{
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
|
|
ResourceId liveid = GetResID(pipeline);
|
|
|
|
if(pipelineBindPoint == VK_PIPELINE_BIND_POINT_COMPUTE)
|
|
{
|
|
m_RenderState.compute.pipeline = liveid;
|
|
}
|
|
else
|
|
{
|
|
m_RenderState.graphics.pipeline = liveid;
|
|
|
|
if(!m_CreationInfo.m_Pipeline[liveid].dynamicStates[VK_DYNAMIC_STATE_VIEWPORT])
|
|
{
|
|
m_RenderState.views = m_CreationInfo.m_Pipeline[liveid].viewports;
|
|
}
|
|
if(!m_CreationInfo.m_Pipeline[liveid].dynamicStates[VK_DYNAMIC_STATE_SCISSOR])
|
|
{
|
|
m_RenderState.scissors = m_CreationInfo.m_Pipeline[liveid].scissors;
|
|
}
|
|
if(!m_CreationInfo.m_Pipeline[liveid].dynamicStates[VK_DYNAMIC_STATE_LINE_WIDTH])
|
|
{
|
|
m_RenderState.lineWidth = m_CreationInfo.m_Pipeline[liveid].lineWidth;
|
|
}
|
|
if(!m_CreationInfo.m_Pipeline[liveid].dynamicStates[VK_DYNAMIC_STATE_DEPTH_BIAS])
|
|
{
|
|
m_RenderState.bias.depth = m_CreationInfo.m_Pipeline[liveid].depthBiasConstantFactor;
|
|
m_RenderState.bias.biasclamp = m_CreationInfo.m_Pipeline[liveid].depthBiasClamp;
|
|
m_RenderState.bias.slope = m_CreationInfo.m_Pipeline[liveid].depthBiasSlopeFactor;
|
|
}
|
|
if(!m_CreationInfo.m_Pipeline[liveid].dynamicStates[VK_DYNAMIC_STATE_BLEND_CONSTANTS])
|
|
{
|
|
memcpy(m_RenderState.blendConst, m_CreationInfo.m_Pipeline[liveid].blendConst,
|
|
sizeof(float) * 4);
|
|
}
|
|
if(!m_CreationInfo.m_Pipeline[liveid].dynamicStates[VK_DYNAMIC_STATE_DEPTH_BOUNDS])
|
|
{
|
|
m_RenderState.mindepth = m_CreationInfo.m_Pipeline[liveid].minDepthBounds;
|
|
m_RenderState.maxdepth = m_CreationInfo.m_Pipeline[liveid].maxDepthBounds;
|
|
}
|
|
if(!m_CreationInfo.m_Pipeline[liveid].dynamicStates[VK_DYNAMIC_STATE_STENCIL_COMPARE_MASK])
|
|
{
|
|
m_RenderState.front.compare = m_CreationInfo.m_Pipeline[liveid].front.compareMask;
|
|
m_RenderState.back.compare = m_CreationInfo.m_Pipeline[liveid].back.compareMask;
|
|
}
|
|
if(!m_CreationInfo.m_Pipeline[liveid].dynamicStates[VK_DYNAMIC_STATE_STENCIL_WRITE_MASK])
|
|
{
|
|
m_RenderState.front.write = m_CreationInfo.m_Pipeline[liveid].front.writeMask;
|
|
m_RenderState.back.write = m_CreationInfo.m_Pipeline[liveid].back.writeMask;
|
|
}
|
|
if(!m_CreationInfo.m_Pipeline[liveid].dynamicStates[VK_DYNAMIC_STATE_STENCIL_REFERENCE])
|
|
{
|
|
m_RenderState.front.ref = m_CreationInfo.m_Pipeline[liveid].front.reference;
|
|
m_RenderState.back.ref = m_CreationInfo.m_Pipeline[liveid].back.reference;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
commandBuffer = VK_NULL_HANDLE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// track while reading, as we need to bind current topology & index byte width in AddDrawcall
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].state.pipeline = GetResID(pipeline);
|
|
}
|
|
|
|
if(commandBuffer != VK_NULL_HANDLE)
|
|
ObjDisp(commandBuffer)->CmdBindPipeline(Unwrap(commandBuffer), pipelineBindPoint, Unwrap(pipeline));
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdBindPipeline(VkCommandBuffer commandBuffer,
|
|
VkPipelineBindPoint pipelineBindPoint, VkPipeline pipeline)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)->CmdBindPipeline(Unwrap(commandBuffer), pipelineBindPoint, Unwrap(pipeline));
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdBindPipeline);
|
|
Serialise_vkCmdBindPipeline(ser, commandBuffer, pipelineBindPoint, pipeline);
|
|
|
|
record->AddChunk(scope.Get());
|
|
record->MarkResourceFrameReferenced(GetResID(pipeline), eFrameRef_Read);
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdBindDescriptorSets(
|
|
SerialiserType &ser, VkCommandBuffer commandBuffer, VkPipelineBindPoint pipelineBindPoint,
|
|
VkPipelineLayout layout, uint32_t firstSet, uint32_t setCount,
|
|
const VkDescriptorSet *pDescriptorSets, uint32_t dynamicOffsetCount,
|
|
const uint32_t *pDynamicOffsets)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT(pipelineBindPoint);
|
|
SERIALISE_ELEMENT(layout);
|
|
SERIALISE_ELEMENT(firstSet);
|
|
SERIALISE_ELEMENT_ARRAY(pDescriptorSets, setCount);
|
|
SERIALISE_ELEMENT_ARRAY(pDynamicOffsets, dynamicOffsetCount);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
{
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
|
|
ObjDisp(commandBuffer)
|
|
->CmdBindDescriptorSets(Unwrap(commandBuffer), pipelineBindPoint, Unwrap(layout),
|
|
firstSet, setCount, UnwrapArray(pDescriptorSets, setCount),
|
|
dynamicOffsetCount, pDynamicOffsets);
|
|
|
|
std::vector<VulkanRenderState::Pipeline::DescriptorAndOffsets> &descsets =
|
|
(pipelineBindPoint == VK_PIPELINE_BIND_POINT_GRAPHICS) ? m_RenderState.graphics.descSets
|
|
: m_RenderState.compute.descSets;
|
|
|
|
// expand as necessary
|
|
if(descsets.size() < firstSet + setCount)
|
|
descsets.resize(firstSet + setCount);
|
|
|
|
const std::vector<ResourceId> &descSetLayouts =
|
|
m_CreationInfo.m_PipelineLayout[GetResID(layout)].descSetLayouts;
|
|
|
|
const uint32_t *offsIter = pDynamicOffsets;
|
|
uint32_t dynConsumed = 0;
|
|
|
|
// consume the offsets linearly along the descriptor set layouts
|
|
for(uint32_t i = 0; i < setCount; i++)
|
|
{
|
|
descsets[firstSet + i].descSet = GetResID(pDescriptorSets[i]);
|
|
uint32_t dynCount =
|
|
m_CreationInfo.m_DescSetLayout[descSetLayouts[firstSet + i]].dynamicCount;
|
|
descsets[firstSet + i].offsets.assign(offsIter, offsIter + dynCount);
|
|
offsIter += dynCount;
|
|
dynConsumed += dynCount;
|
|
RDCASSERT(dynConsumed <= dynamicOffsetCount);
|
|
}
|
|
|
|
// if there are dynamic offsets, bake them into the current bindings by alias'ing
|
|
// the image layout member (which is never used for buffer views).
|
|
// This lets us look it up easily when we want to show the current pipeline state
|
|
RDCCOMPILE_ASSERT(sizeof(VkImageLayout) >= sizeof(uint32_t),
|
|
"Can't alias image layout for dynamic offset!");
|
|
if(dynamicOffsetCount > 0)
|
|
{
|
|
uint32_t o = 0;
|
|
|
|
// spec states that dynamic offsets precisely match all the offsets needed for these
|
|
// sets, in order of set N before set N+1, binding X before binding X+1 within a set,
|
|
// and in array element order within a binding
|
|
for(uint32_t i = 0; i < setCount; i++)
|
|
{
|
|
ResourceId descId = GetResID(pDescriptorSets[i]);
|
|
const DescSetLayout &layoutinfo =
|
|
m_CreationInfo.m_DescSetLayout[descSetLayouts[firstSet + i]];
|
|
|
|
for(size_t b = 0; b < layoutinfo.bindings.size(); b++)
|
|
{
|
|
// not dynamic, doesn't need an offset
|
|
if(layoutinfo.bindings[b].descriptorType != VK_DESCRIPTOR_TYPE_STORAGE_BUFFER_DYNAMIC &&
|
|
layoutinfo.bindings[b].descriptorType != VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER_DYNAMIC)
|
|
continue;
|
|
|
|
// assign every array element an offset according to array size
|
|
for(uint32_t a = 0; a < layoutinfo.bindings[b].descriptorCount; a++)
|
|
{
|
|
RDCASSERT(o < dynamicOffsetCount);
|
|
uint32_t *alias =
|
|
(uint32_t *)&m_DescriptorSetState[descId].currentBindings[b][a].imageInfo.imageLayout;
|
|
*alias = pDynamicOffsets[o++];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// track while reading, as we need to track resource usage
|
|
std::vector<BakedCmdBufferInfo::CmdBufferState::DescriptorAndOffsets> &descsets =
|
|
(pipelineBindPoint == VK_PIPELINE_BIND_POINT_GRAPHICS)
|
|
? m_BakedCmdBufferInfo[m_LastCmdBufferID].state.graphicsDescSets
|
|
: m_BakedCmdBufferInfo[m_LastCmdBufferID].state.computeDescSets;
|
|
|
|
// expand as necessary
|
|
if(descsets.size() < firstSet + setCount)
|
|
descsets.resize(firstSet + setCount);
|
|
|
|
for(uint32_t i = 0; i < setCount; i++)
|
|
descsets[firstSet + i].descSet = GetResID(pDescriptorSets[i]);
|
|
|
|
ObjDisp(commandBuffer)
|
|
->CmdBindDescriptorSets(Unwrap(commandBuffer), pipelineBindPoint, Unwrap(layout),
|
|
firstSet, setCount, UnwrapArray(pDescriptorSets, setCount),
|
|
dynamicOffsetCount, pDynamicOffsets);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdBindDescriptorSets(VkCommandBuffer commandBuffer,
|
|
VkPipelineBindPoint pipelineBindPoint,
|
|
VkPipelineLayout layout, uint32_t firstSet,
|
|
uint32_t setCount, const VkDescriptorSet *pDescriptorSets,
|
|
uint32_t dynamicOffsetCount,
|
|
const uint32_t *pDynamicOffsets)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)
|
|
->CmdBindDescriptorSets(Unwrap(commandBuffer), pipelineBindPoint, Unwrap(layout), firstSet,
|
|
setCount, UnwrapArray(pDescriptorSets, setCount), dynamicOffsetCount,
|
|
pDynamicOffsets);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdBindDescriptorSets);
|
|
Serialise_vkCmdBindDescriptorSets(ser, commandBuffer, pipelineBindPoint, layout, firstSet,
|
|
setCount, pDescriptorSets, dynamicOffsetCount, pDynamicOffsets);
|
|
|
|
record->AddChunk(scope.Get());
|
|
record->MarkResourceFrameReferenced(GetResID(layout), eFrameRef_Read);
|
|
record->cmdInfo->boundDescSets.insert(pDescriptorSets, pDescriptorSets + setCount);
|
|
|
|
// conservatively mark all writeable objects in the descriptor set as dirty here.
|
|
// Technically not all might be written although that required verifying what the
|
|
// shader does and is a large problem space. The binding could be overridden though
|
|
// but per Vulkan ethos we consider that the application's problem to solve. Plus,
|
|
// it would mean we'd need to dirty every drawcall instead of just every bind at
|
|
// lower frequency.
|
|
for(uint32_t i = 0; i < setCount; i++)
|
|
{
|
|
VkResourceRecord *descSet = GetRecord(pDescriptorSets[i]);
|
|
|
|
map<ResourceId, pair<uint32_t, FrameRefType> > &frameRefs = descSet->descInfo->bindFrameRefs;
|
|
|
|
for(auto it = frameRefs.begin(); it != frameRefs.end(); ++it)
|
|
{
|
|
if(it->second.second == eFrameRef_Write || it->second.second == eFrameRef_ReadBeforeWrite)
|
|
record->cmdInfo->dirtied.insert(it->first);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdBindVertexBuffers(SerialiserType &ser,
|
|
VkCommandBuffer commandBuffer,
|
|
uint32_t firstBinding, uint32_t bindingCount,
|
|
const VkBuffer *pBuffers,
|
|
const VkDeviceSize *pOffsets)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT(firstBinding);
|
|
SERIALISE_ELEMENT_ARRAY(pBuffers, bindingCount);
|
|
SERIALISE_ELEMENT_ARRAY(pOffsets, bindingCount);
|
|
SERIALISE_ELEMENT(bindingCount);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
{
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
ObjDisp(commandBuffer)
|
|
->CmdBindVertexBuffers(Unwrap(commandBuffer), firstBinding, bindingCount,
|
|
UnwrapArray(pBuffers, bindingCount), pOffsets);
|
|
|
|
if(m_RenderState.vbuffers.size() < firstBinding + bindingCount)
|
|
m_RenderState.vbuffers.resize(firstBinding + bindingCount);
|
|
|
|
for(uint32_t i = 0; i < bindingCount; i++)
|
|
{
|
|
m_RenderState.vbuffers[firstBinding + i].buf = GetResID(pBuffers[i]);
|
|
m_RenderState.vbuffers[firstBinding + i].offs = pOffsets[i];
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// track while reading, as we need to track resource usage
|
|
if(m_BakedCmdBufferInfo[m_LastCmdBufferID].state.vbuffers.size() < firstBinding + bindingCount)
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].state.vbuffers.resize(firstBinding + bindingCount);
|
|
|
|
for(uint32_t i = 0; i < bindingCount; i++)
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].state.vbuffers[firstBinding + i] =
|
|
GetResID(pBuffers[i]);
|
|
|
|
ObjDisp(commandBuffer)
|
|
->CmdBindVertexBuffers(Unwrap(commandBuffer), firstBinding, bindingCount,
|
|
UnwrapArray(pBuffers, bindingCount), pOffsets);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdBindVertexBuffers(VkCommandBuffer commandBuffer, uint32_t firstBinding,
|
|
uint32_t bindingCount, const VkBuffer *pBuffers,
|
|
const VkDeviceSize *pOffsets)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)
|
|
->CmdBindVertexBuffers(Unwrap(commandBuffer), firstBinding, bindingCount,
|
|
UnwrapArray(pBuffers, bindingCount), pOffsets);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdBindVertexBuffers);
|
|
Serialise_vkCmdBindVertexBuffers(ser, commandBuffer, firstBinding, bindingCount, pBuffers,
|
|
pOffsets);
|
|
|
|
record->AddChunk(scope.Get());
|
|
for(uint32_t i = 0; i < bindingCount; i++)
|
|
{
|
|
record->MarkResourceFrameReferenced(GetResID(pBuffers[i]), eFrameRef_Read);
|
|
record->MarkResourceFrameReferenced(GetRecord(pBuffers[i])->baseResource, eFrameRef_Read);
|
|
if(GetRecord(pBuffers[i])->sparseInfo)
|
|
record->cmdInfo->sparse.insert(GetRecord(pBuffers[i])->sparseInfo);
|
|
}
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdBindIndexBuffer(SerialiserType &ser,
|
|
VkCommandBuffer commandBuffer, VkBuffer buffer,
|
|
VkDeviceSize offset, VkIndexType indexType)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT(buffer);
|
|
SERIALISE_ELEMENT(offset);
|
|
SERIALISE_ELEMENT(indexType);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
{
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
ObjDisp(commandBuffer)
|
|
->CmdBindIndexBuffer(Unwrap(commandBuffer), Unwrap(buffer), offset, indexType);
|
|
|
|
m_RenderState.ibuffer.buf = GetResID(buffer);
|
|
m_RenderState.ibuffer.offs = offset;
|
|
m_RenderState.ibuffer.bytewidth = indexType == VK_INDEX_TYPE_UINT32 ? 4 : 2;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// track while reading, as we need to bind current topology & index byte width in AddDrawcall
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].state.idxWidth =
|
|
(indexType == VK_INDEX_TYPE_UINT32 ? 4 : 2);
|
|
|
|
// track while reading, as we need to track resource usage
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].state.ibuffer = GetResID(buffer);
|
|
|
|
ObjDisp(commandBuffer)->CmdBindIndexBuffer(Unwrap(commandBuffer), Unwrap(buffer), offset, indexType);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdBindIndexBuffer(VkCommandBuffer commandBuffer, VkBuffer buffer,
|
|
VkDeviceSize offset, VkIndexType indexType)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)->CmdBindIndexBuffer(Unwrap(commandBuffer), Unwrap(buffer), offset, indexType);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdBindIndexBuffer);
|
|
Serialise_vkCmdBindIndexBuffer(ser, commandBuffer, buffer, offset, indexType);
|
|
|
|
record->AddChunk(scope.Get());
|
|
record->MarkResourceFrameReferenced(GetResID(buffer), eFrameRef_Read);
|
|
record->MarkResourceFrameReferenced(GetRecord(buffer)->baseResource, eFrameRef_Read);
|
|
if(GetRecord(buffer)->sparseInfo)
|
|
record->cmdInfo->sparse.insert(GetRecord(buffer)->sparseInfo);
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdUpdateBuffer(SerialiserType &ser, VkCommandBuffer commandBuffer,
|
|
VkBuffer destBuffer, VkDeviceSize destOffset,
|
|
VkDeviceSize dataSize, const uint32_t *pData)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT(destBuffer);
|
|
SERIALISE_ELEMENT(destOffset);
|
|
|
|
// serialise as void* so it goes through as a buffer, not an actual array of integers.
|
|
const void *Data = (const void *)pData;
|
|
SERIALISE_ELEMENT_ARRAY(Data, dataSize);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
else
|
|
commandBuffer = VK_NULL_HANDLE;
|
|
}
|
|
|
|
if(commandBuffer != VK_NULL_HANDLE)
|
|
{
|
|
ObjDisp(commandBuffer)
|
|
->CmdUpdateBuffer(Unwrap(commandBuffer), Unwrap(destBuffer), destOffset, dataSize, Data);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdUpdateBuffer(VkCommandBuffer commandBuffer, VkBuffer destBuffer,
|
|
VkDeviceSize destOffset, VkDeviceSize dataSize,
|
|
const uint32_t *pData)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)
|
|
->CmdUpdateBuffer(Unwrap(commandBuffer), Unwrap(destBuffer), destOffset, dataSize, pData);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdUpdateBuffer);
|
|
Serialise_vkCmdUpdateBuffer(ser, commandBuffer, destBuffer, destOffset, dataSize, pData);
|
|
|
|
record->AddChunk(scope.Get());
|
|
|
|
VkResourceRecord *buf = GetRecord(destBuffer);
|
|
|
|
// mark buffer just as read, and memory behind as write & dirtied
|
|
record->MarkResourceFrameReferenced(buf->GetResourceID(), eFrameRef_Read);
|
|
record->MarkResourceFrameReferenced(buf->baseResource, eFrameRef_Write);
|
|
if(buf->baseResource != ResourceId())
|
|
record->cmdInfo->dirtied.insert(buf->baseResource);
|
|
if(buf->sparseInfo)
|
|
record->cmdInfo->sparse.insert(buf->sparseInfo);
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdFillBuffer(SerialiserType &ser, VkCommandBuffer commandBuffer,
|
|
VkBuffer destBuffer, VkDeviceSize destOffset,
|
|
VkDeviceSize fillSize, uint32_t data)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT(destBuffer);
|
|
SERIALISE_ELEMENT(destOffset);
|
|
SERIALISE_ELEMENT(fillSize);
|
|
SERIALISE_ELEMENT(data);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
else
|
|
commandBuffer = VK_NULL_HANDLE;
|
|
}
|
|
|
|
if(commandBuffer != VK_NULL_HANDLE)
|
|
{
|
|
ObjDisp(commandBuffer)
|
|
->CmdFillBuffer(Unwrap(commandBuffer), Unwrap(destBuffer), destOffset, fillSize, data);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdFillBuffer(VkCommandBuffer commandBuffer, VkBuffer destBuffer,
|
|
VkDeviceSize destOffset, VkDeviceSize fillSize, uint32_t data)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)
|
|
->CmdFillBuffer(Unwrap(commandBuffer), Unwrap(destBuffer), destOffset, fillSize, data);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdFillBuffer);
|
|
Serialise_vkCmdFillBuffer(ser, commandBuffer, destBuffer, destOffset, fillSize, data);
|
|
|
|
record->AddChunk(scope.Get());
|
|
|
|
VkResourceRecord *buf = GetRecord(destBuffer);
|
|
|
|
// mark buffer just as read, and memory behind as write & dirtied
|
|
record->MarkResourceFrameReferenced(buf->GetResourceID(), eFrameRef_Read);
|
|
record->MarkResourceFrameReferenced(buf->baseResource, eFrameRef_Write);
|
|
if(buf->baseResource != ResourceId())
|
|
record->cmdInfo->dirtied.insert(buf->baseResource);
|
|
if(buf->sparseInfo)
|
|
record->cmdInfo->sparse.insert(buf->sparseInfo);
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdPushConstants(SerialiserType &ser, VkCommandBuffer commandBuffer,
|
|
VkPipelineLayout layout,
|
|
VkShaderStageFlags stageFlags, uint32_t start,
|
|
uint32_t length, const void *values)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT(layout);
|
|
SERIALISE_ELEMENT_TYPED(VkShaderStageFlagBits, stageFlags);
|
|
SERIALISE_ELEMENT(start);
|
|
SERIALISE_ELEMENT_ARRAY(values, length);
|
|
SERIALISE_ELEMENT(length);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
{
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
ObjDisp(commandBuffer)
|
|
->CmdPushConstants(Unwrap(commandBuffer), Unwrap(layout), stageFlags, start, length,
|
|
values);
|
|
|
|
RDCASSERT(start + length < (uint32_t)ARRAY_COUNT(m_RenderState.pushconsts));
|
|
|
|
memcpy(m_RenderState.pushconsts + start, values, length);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ObjDisp(commandBuffer)
|
|
->CmdPushConstants(Unwrap(commandBuffer), Unwrap(layout), stageFlags, start, length,
|
|
values);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdPushConstants(VkCommandBuffer commandBuffer, VkPipelineLayout layout,
|
|
VkShaderStageFlags stageFlags, uint32_t start,
|
|
uint32_t length, const void *values)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)
|
|
->CmdPushConstants(Unwrap(commandBuffer), Unwrap(layout), stageFlags, start, length, values);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdPushConstants);
|
|
Serialise_vkCmdPushConstants(ser, commandBuffer, layout, stageFlags, start, length, values);
|
|
|
|
record->AddChunk(scope.Get());
|
|
record->MarkResourceFrameReferenced(GetResID(layout), eFrameRef_Read);
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdPipelineBarrier(
|
|
SerialiserType &ser, VkCommandBuffer commandBuffer, VkPipelineStageFlags srcStageMask,
|
|
VkPipelineStageFlags destStageMask, VkDependencyFlags dependencyFlags,
|
|
uint32_t memoryBarrierCount, const VkMemoryBarrier *pMemoryBarriers,
|
|
uint32_t bufferMemoryBarrierCount, const VkBufferMemoryBarrier *pBufferMemoryBarriers,
|
|
uint32_t imageMemoryBarrierCount, const VkImageMemoryBarrier *pImageMemoryBarriers)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT_TYPED(VkPipelineStageFlagBits, srcStageMask);
|
|
SERIALISE_ELEMENT_TYPED(VkPipelineStageFlagBits, destStageMask);
|
|
SERIALISE_ELEMENT_TYPED(VkDependencyFlagBits, dependencyFlags);
|
|
SERIALISE_ELEMENT_ARRAY(pMemoryBarriers, memoryBarrierCount);
|
|
SERIALISE_ELEMENT_ARRAY(pBufferMemoryBarriers, bufferMemoryBarrierCount);
|
|
SERIALISE_ELEMENT_ARRAY(pImageMemoryBarriers, imageMemoryBarrierCount);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
std::vector<VkImageMemoryBarrier> imgBarriers;
|
|
std::vector<VkBufferMemoryBarrier> bufBarriers;
|
|
|
|
// it's possible for buffer or image to be NULL if it refers to a resource that is otherwise
|
|
// not in the log (barriers do not mark resources referenced). If the resource in question does
|
|
// not exist, then it's safe to skip this barrier.
|
|
//
|
|
// Since it's a convenient place, we unwrap at the same time.
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
for(uint32_t i = 0; i < bufferMemoryBarrierCount; i++)
|
|
{
|
|
if(pBufferMemoryBarriers[i].buffer != VK_NULL_HANDLE)
|
|
{
|
|
bufBarriers.push_back(pBufferMemoryBarriers[i]);
|
|
bufBarriers.back().buffer = Unwrap(bufBarriers.back().buffer);
|
|
}
|
|
}
|
|
|
|
for(uint32_t i = 0; i < imageMemoryBarrierCount; i++)
|
|
{
|
|
if(pImageMemoryBarriers[i].image != VK_NULL_HANDLE)
|
|
{
|
|
imgBarriers.push_back(pImageMemoryBarriers[i]);
|
|
imgBarriers.back().image = Unwrap(imgBarriers.back().image);
|
|
ReplacePresentableImageLayout(imgBarriers.back().oldLayout);
|
|
ReplacePresentableImageLayout(imgBarriers.back().newLayout);
|
|
|
|
ReplaceExternalQueueFamily(imgBarriers.back().srcQueueFamilyIndex,
|
|
imgBarriers.back().dstQueueFamilyIndex);
|
|
}
|
|
}
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
else
|
|
commandBuffer = VK_NULL_HANDLE;
|
|
}
|
|
else
|
|
{
|
|
ResourceId cmd = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
for(size_t i = 0; i < imgBarriers.size(); i++)
|
|
{
|
|
m_BakedCmdBufferInfo[cmd].resourceUsage.push_back(std::make_pair(
|
|
GetResID(imgBarriers[i].image),
|
|
EventUsage(m_BakedCmdBufferInfo[cmd].curEventID, ResourceUsage::Barrier)));
|
|
}
|
|
}
|
|
|
|
if(commandBuffer != VK_NULL_HANDLE)
|
|
{
|
|
ObjDisp(commandBuffer)
|
|
->CmdPipelineBarrier(Unwrap(commandBuffer), srcStageMask, destStageMask, dependencyFlags,
|
|
memoryBarrierCount, pMemoryBarriers, (uint32_t)bufBarriers.size(),
|
|
bufBarriers.data(), (uint32_t)imgBarriers.size(), imgBarriers.data());
|
|
|
|
ResourceId cmd = GetResID(commandBuffer);
|
|
GetResourceManager()->RecordBarriers(m_BakedCmdBufferInfo[cmd].imgbarriers, m_ImageLayouts,
|
|
(uint32_t)imgBarriers.size(), imgBarriers.data());
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdPipelineBarrier(
|
|
VkCommandBuffer commandBuffer, VkPipelineStageFlags srcStageMask,
|
|
VkPipelineStageFlags destStageMask, VkDependencyFlags dependencyFlags,
|
|
uint32_t memoryBarrierCount, const VkMemoryBarrier *pMemoryBarriers,
|
|
uint32_t bufferMemoryBarrierCount, const VkBufferMemoryBarrier *pBufferMemoryBarriers,
|
|
uint32_t imageMemoryBarrierCount, const VkImageMemoryBarrier *pImageMemoryBarriers)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
{
|
|
byte *memory = GetTempMemory(sizeof(VkBufferMemoryBarrier) * bufferMemoryBarrierCount +
|
|
sizeof(VkImageMemoryBarrier) * imageMemoryBarrierCount);
|
|
|
|
VkImageMemoryBarrier *im = (VkImageMemoryBarrier *)memory;
|
|
VkBufferMemoryBarrier *buf = (VkBufferMemoryBarrier *)(im + imageMemoryBarrierCount);
|
|
|
|
for(uint32_t i = 0; i < bufferMemoryBarrierCount; i++)
|
|
{
|
|
buf[i] = pBufferMemoryBarriers[i];
|
|
buf[i].buffer = Unwrap(buf[i].buffer);
|
|
}
|
|
|
|
for(uint32_t i = 0; i < imageMemoryBarrierCount; i++)
|
|
{
|
|
im[i] = pImageMemoryBarriers[i];
|
|
im[i].image = Unwrap(im[i].image);
|
|
}
|
|
|
|
ObjDisp(commandBuffer)
|
|
->CmdPipelineBarrier(Unwrap(commandBuffer), srcStageMask, destStageMask, dependencyFlags,
|
|
memoryBarrierCount, pMemoryBarriers, bufferMemoryBarrierCount, buf,
|
|
imageMemoryBarrierCount, im);
|
|
}
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdPipelineBarrier);
|
|
Serialise_vkCmdPipelineBarrier(ser, commandBuffer, srcStageMask, destStageMask, dependencyFlags,
|
|
memoryBarrierCount, pMemoryBarriers, bufferMemoryBarrierCount,
|
|
pBufferMemoryBarriers, imageMemoryBarrierCount,
|
|
pImageMemoryBarriers);
|
|
|
|
record->AddChunk(scope.Get());
|
|
|
|
if(imageMemoryBarrierCount > 0)
|
|
{
|
|
SCOPED_LOCK(m_ImageLayoutsLock);
|
|
GetResourceManager()->RecordBarriers(GetRecord(commandBuffer)->cmdInfo->imgbarriers,
|
|
m_ImageLayouts, imageMemoryBarrierCount,
|
|
pImageMemoryBarriers);
|
|
}
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdWriteTimestamp(SerialiserType &ser, VkCommandBuffer commandBuffer,
|
|
VkPipelineStageFlagBits pipelineStage,
|
|
VkQueryPool queryPool, uint32_t query)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT(pipelineStage);
|
|
SERIALISE_ELEMENT(queryPool);
|
|
SERIALISE_ELEMENT(query);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
else
|
|
commandBuffer = VK_NULL_HANDLE;
|
|
}
|
|
|
|
if(commandBuffer != VK_NULL_HANDLE)
|
|
{
|
|
ObjDisp(commandBuffer)
|
|
->CmdWriteTimestamp(Unwrap(commandBuffer), pipelineStage, Unwrap(queryPool), query);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdWriteTimestamp(VkCommandBuffer commandBuffer,
|
|
VkPipelineStageFlagBits pipelineStage,
|
|
VkQueryPool queryPool, uint32_t query)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)
|
|
->CmdWriteTimestamp(Unwrap(commandBuffer), pipelineStage, Unwrap(queryPool), query);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdWriteTimestamp);
|
|
Serialise_vkCmdWriteTimestamp(ser, commandBuffer, pipelineStage, queryPool, query);
|
|
|
|
record->AddChunk(scope.Get());
|
|
|
|
record->MarkResourceFrameReferenced(GetResID(queryPool), eFrameRef_Read);
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdCopyQueryPoolResults(
|
|
SerialiserType &ser, VkCommandBuffer commandBuffer, VkQueryPool queryPool, uint32_t firstQuery,
|
|
uint32_t queryCount, VkBuffer destBuffer, VkDeviceSize destOffset, VkDeviceSize destStride,
|
|
VkQueryResultFlags flags)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT(queryPool);
|
|
SERIALISE_ELEMENT(firstQuery);
|
|
SERIALISE_ELEMENT(queryCount);
|
|
SERIALISE_ELEMENT(destBuffer);
|
|
SERIALISE_ELEMENT(destOffset);
|
|
SERIALISE_ELEMENT(destStride);
|
|
SERIALISE_ELEMENT_TYPED(VkQueryResultFlagBits, flags);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
else
|
|
commandBuffer = VK_NULL_HANDLE;
|
|
}
|
|
|
|
if(commandBuffer != VK_NULL_HANDLE)
|
|
{
|
|
ObjDisp(commandBuffer)
|
|
->CmdCopyQueryPoolResults(Unwrap(commandBuffer), Unwrap(queryPool), firstQuery,
|
|
queryCount, Unwrap(destBuffer), destOffset, destStride, flags);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdCopyQueryPoolResults(VkCommandBuffer commandBuffer, VkQueryPool queryPool,
|
|
uint32_t firstQuery, uint32_t queryCount,
|
|
VkBuffer destBuffer, VkDeviceSize destOffset,
|
|
VkDeviceSize destStride, VkQueryResultFlags flags)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)
|
|
->CmdCopyQueryPoolResults(Unwrap(commandBuffer), Unwrap(queryPool), firstQuery, queryCount,
|
|
Unwrap(destBuffer), destOffset, destStride, flags);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdCopyQueryPoolResults);
|
|
Serialise_vkCmdCopyQueryPoolResults(ser, commandBuffer, queryPool, firstQuery, queryCount,
|
|
destBuffer, destOffset, destStride, flags);
|
|
|
|
record->AddChunk(scope.Get());
|
|
record->MarkResourceFrameReferenced(GetResID(queryPool), eFrameRef_Read);
|
|
|
|
VkResourceRecord *buf = GetRecord(destBuffer);
|
|
|
|
// mark buffer just as read, and memory behind as write & dirtied
|
|
record->MarkResourceFrameReferenced(buf->GetResourceID(), eFrameRef_Read);
|
|
record->MarkResourceFrameReferenced(buf->baseResource, eFrameRef_Write);
|
|
if(buf->baseResource != ResourceId())
|
|
record->cmdInfo->dirtied.insert(buf->baseResource);
|
|
if(buf->sparseInfo)
|
|
record->cmdInfo->sparse.insert(buf->sparseInfo);
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdBeginQuery(SerialiserType &ser, VkCommandBuffer commandBuffer,
|
|
VkQueryPool queryPool, uint32_t query,
|
|
VkQueryControlFlags flags)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT(queryPool);
|
|
SERIALISE_ELEMENT(query);
|
|
SERIALISE_ELEMENT_TYPED(VkQueryControlFlagBits, flags);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
else
|
|
commandBuffer = VK_NULL_HANDLE;
|
|
}
|
|
|
|
if(commandBuffer != VK_NULL_HANDLE)
|
|
ObjDisp(commandBuffer)->CmdBeginQuery(Unwrap(commandBuffer), Unwrap(queryPool), query, flags);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdBeginQuery(VkCommandBuffer commandBuffer, VkQueryPool queryPool,
|
|
uint32_t query, VkQueryControlFlags flags)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)->CmdBeginQuery(Unwrap(commandBuffer), Unwrap(queryPool), query, flags);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdBeginQuery);
|
|
Serialise_vkCmdBeginQuery(ser, commandBuffer, queryPool, query, flags);
|
|
|
|
record->AddChunk(scope.Get());
|
|
record->MarkResourceFrameReferenced(GetResID(queryPool), eFrameRef_Read);
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdEndQuery(SerialiserType &ser, VkCommandBuffer commandBuffer,
|
|
VkQueryPool queryPool, uint32_t query)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT(queryPool);
|
|
SERIALISE_ELEMENT(query);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
else
|
|
commandBuffer = VK_NULL_HANDLE;
|
|
}
|
|
|
|
if(commandBuffer != VK_NULL_HANDLE)
|
|
ObjDisp(commandBuffer)->CmdEndQuery(Unwrap(commandBuffer), Unwrap(queryPool), query);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdEndQuery(VkCommandBuffer commandBuffer, VkQueryPool queryPool, uint32_t query)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)->CmdEndQuery(Unwrap(commandBuffer), Unwrap(queryPool), query);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdEndQuery);
|
|
Serialise_vkCmdEndQuery(ser, commandBuffer, queryPool, query);
|
|
|
|
record->AddChunk(scope.Get());
|
|
record->MarkResourceFrameReferenced(GetResID(queryPool), eFrameRef_Read);
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdResetQueryPool(SerialiserType &ser, VkCommandBuffer commandBuffer,
|
|
VkQueryPool queryPool, uint32_t firstQuery,
|
|
uint32_t queryCount)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT(queryPool);
|
|
SERIALISE_ELEMENT(firstQuery);
|
|
SERIALISE_ELEMENT(queryCount);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
else
|
|
commandBuffer = VK_NULL_HANDLE;
|
|
}
|
|
|
|
if(commandBuffer != VK_NULL_HANDLE)
|
|
{
|
|
ObjDisp(commandBuffer)
|
|
->CmdResetQueryPool(Unwrap(commandBuffer), Unwrap(queryPool), firstQuery, queryCount);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdResetQueryPool(VkCommandBuffer commandBuffer, VkQueryPool queryPool,
|
|
uint32_t firstQuery, uint32_t queryCount)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)
|
|
->CmdResetQueryPool(Unwrap(commandBuffer), Unwrap(queryPool), firstQuery, queryCount);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdResetQueryPool);
|
|
Serialise_vkCmdResetQueryPool(ser, commandBuffer, queryPool, firstQuery, queryCount);
|
|
|
|
record->AddChunk(scope.Get());
|
|
record->MarkResourceFrameReferenced(GetResID(queryPool), eFrameRef_Read);
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdExecuteCommands(SerialiserType &ser, VkCommandBuffer commandBuffer,
|
|
uint32_t commandBufferCount,
|
|
const VkCommandBuffer *pCommandBuffers)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT_ARRAY(pCommandBuffers, commandBufferCount);
|
|
|
|
Serialise_DebugMessages(ser);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsLoading(m_State))
|
|
{
|
|
// execute the commands
|
|
ObjDisp(commandBuffer)
|
|
->CmdExecuteCommands(Unwrap(commandBuffer), commandBufferCount,
|
|
UnwrapArray(pCommandBuffers, commandBufferCount));
|
|
|
|
// apply barriers
|
|
for(uint32_t i = 0; i < commandBufferCount; i++)
|
|
GetResourceManager()->ApplyBarriers(
|
|
m_BakedCmdBufferInfo[GetResID(pCommandBuffers[i])].imgbarriers, m_ImageLayouts);
|
|
|
|
AddEvent();
|
|
|
|
DrawcallDescription draw;
|
|
draw.name = StringFormat::Fmt("vkCmdExecuteCommands(%u)", commandBufferCount);
|
|
draw.flags = DrawFlags::CmdList | DrawFlags::PushMarker;
|
|
|
|
AddDrawcall(draw, true);
|
|
|
|
BakedCmdBufferInfo &parentCmdBufInfo = m_BakedCmdBufferInfo[m_LastCmdBufferID];
|
|
|
|
parentCmdBufInfo.curEventID++;
|
|
|
|
for(uint32_t c = 0; c < commandBufferCount; c++)
|
|
{
|
|
ResourceId cmd = GetResourceManager()->GetOriginalID(GetResID(pCommandBuffers[c]));
|
|
|
|
// add a fake marker
|
|
DrawcallDescription marker;
|
|
marker.name = StringFormat::Fmt("=> vkCmdExecuteCommands()[%u]: vkBeginCommandBuffer(%s)",
|
|
c, ToStr(cmd).c_str());
|
|
marker.flags = DrawFlags::PassBoundary | DrawFlags::BeginPass;
|
|
AddEvent();
|
|
AddDrawcall(marker, true);
|
|
parentCmdBufInfo.curEventID++;
|
|
|
|
BakedCmdBufferInfo &cmdBufInfo = m_BakedCmdBufferInfo[cmd];
|
|
|
|
if(m_BakedCmdBufferInfo[m_LastCmdBufferID].state.renderPass == ResourceId() &&
|
|
(cmdBufInfo.beginFlags & VK_COMMAND_BUFFER_USAGE_RENDER_PASS_CONTINUE_BIT))
|
|
{
|
|
AddDebugMessage(
|
|
MessageCategory::Execution, MessageSeverity::High, MessageSource::IncorrectAPIUse,
|
|
"Executing a command buffer with RENDER_PASS_CONTINUE_BIT outside of render pass");
|
|
}
|
|
|
|
// insert the baked command buffer in-line into this list of notes, assigning new event and
|
|
// drawIDs
|
|
parentCmdBufInfo.draw->InsertAndUpdateIDs(*cmdBufInfo.draw, parentCmdBufInfo.curEventID,
|
|
parentCmdBufInfo.drawCount);
|
|
|
|
for(size_t i = 0; i < cmdBufInfo.debugMessages.size(); i++)
|
|
{
|
|
parentCmdBufInfo.debugMessages.push_back(cmdBufInfo.debugMessages[i]);
|
|
parentCmdBufInfo.debugMessages.back().eventID += parentCmdBufInfo.curEventID;
|
|
}
|
|
|
|
// only primary command buffers can be submitted
|
|
m_Partial[Secondary].cmdBufferSubmits[cmd].push_back(parentCmdBufInfo.curEventID);
|
|
|
|
parentCmdBufInfo.draw->executedCmds.push_back(cmd);
|
|
|
|
parentCmdBufInfo.curEventID += cmdBufInfo.eventCount;
|
|
parentCmdBufInfo.drawCount += cmdBufInfo.drawCount;
|
|
|
|
marker.name = StringFormat::Fmt("=> vkCmdExecuteCommands()[%u]: vkEndCommandBuffer(%s)", c,
|
|
ToStr(cmd).c_str());
|
|
marker.flags = DrawFlags::PassBoundary | DrawFlags::EndPass;
|
|
AddEvent();
|
|
AddDrawcall(marker, true);
|
|
parentCmdBufInfo.curEventID++;
|
|
}
|
|
|
|
// add an extra pop marker
|
|
draw = DrawcallDescription();
|
|
draw.flags = DrawFlags::PopMarker;
|
|
|
|
AddDrawcall(draw, true);
|
|
|
|
// don't change curEventID here, as it will be incremented outside in the outer
|
|
// loop for the EXEC_CMDS event. in vkQueueSubmit we need to decrement curEventID
|
|
// because we don't have the extra popmarker event to 'absorb' the outer loop's
|
|
// increment, and it incremented once too many for the last vkEndCommandBuffer
|
|
// setmarker event in the loop over all commands
|
|
}
|
|
else
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
{
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
|
|
BakedCmdBufferInfo &parentCmdBufInfo = m_BakedCmdBufferInfo[m_LastCmdBufferID];
|
|
|
|
// if we're replaying a range but not from the start, we are guaranteed to only be replaying
|
|
// one of our executed command buffers and doing it to an outside command buffer. The outer
|
|
// loop will be doing SetOffset() to jump to each event, and any time we land here is just
|
|
// for
|
|
// the markers we've added, which have this file offset, so just skip all of our work.
|
|
if(m_FirstEventID > 1 && m_FirstEventID + 1 < m_LastEventID)
|
|
return true;
|
|
|
|
// account for the execute commands event
|
|
parentCmdBufInfo.curEventID++;
|
|
|
|
uint32_t startEID = parentCmdBufInfo.curEventID + m_Partial[Primary].baseEvent;
|
|
|
|
// advance m_CurEventID to match the events added when reading
|
|
for(uint32_t c = 0; c < commandBufferCount; c++)
|
|
{
|
|
ResourceId cmd = GetResourceManager()->GetOriginalID(GetResID(pCommandBuffers[c]));
|
|
|
|
// 2 extra for the virtual labels around the command buffer
|
|
parentCmdBufInfo.curEventID += 2 + m_BakedCmdBufferInfo[cmd].eventCount;
|
|
}
|
|
|
|
// same accounting for the outer loop as above means no need to change anything here
|
|
|
|
if(commandBufferCount == 0)
|
|
{
|
|
// do nothing, don't bother with the logic below
|
|
}
|
|
else if(m_FirstEventID == m_LastEventID)
|
|
{
|
|
#if ENABLED(VERBOSE_PARTIAL_REPLAY)
|
|
RDCDEBUG("ExecuteCommands no OnlyDraw %u", m_FirstEventID);
|
|
#endif
|
|
}
|
|
else if(m_LastEventID <= startEID)
|
|
{
|
|
#if ENABLED(VERBOSE_PARTIAL_REPLAY)
|
|
RDCDEBUG("ExecuteCommands no replay %u == %u", m_LastEventID, startEID);
|
|
#endif
|
|
}
|
|
else if(m_DrawcallCallback && m_DrawcallCallback->RecordAllCmds())
|
|
{
|
|
#if ENABLED(VERBOSE_PARTIAL_REPLAY)
|
|
RDCDEBUG("ExecuteCommands re-recording from %u", startEID);
|
|
#endif
|
|
|
|
std::vector<VkCommandBuffer> rerecordedCmds;
|
|
|
|
for(uint32_t c = 0; c < commandBufferCount; c++)
|
|
{
|
|
ResourceId cmdid = GetResourceManager()->GetOriginalID(GetResID(pCommandBuffers[c]));
|
|
VkCommandBuffer cmd = RerecordCmdBuf(cmdid);
|
|
ResourceId rerecord = GetResID(cmd);
|
|
#if ENABLED(VERBOSE_PARTIAL_REPLAY)
|
|
RDCDEBUG("ExecuteCommands fully re-recorded replay of %llu, using %llu", cmdid, rerecord);
|
|
#endif
|
|
rerecordedCmds.push_back(Unwrap(cmd));
|
|
|
|
GetResourceManager()->ApplyBarriers(m_BakedCmdBufferInfo[rerecord].imgbarriers,
|
|
m_ImageLayouts);
|
|
}
|
|
|
|
ObjDisp(commandBuffer)
|
|
->CmdExecuteCommands(Unwrap(commandBuffer), commandBufferCount, &rerecordedCmds[0]);
|
|
}
|
|
else if(m_LastEventID > startEID &&
|
|
m_LastEventID < parentCmdBufInfo.curEventID + m_Partial[Primary].baseEvent)
|
|
{
|
|
#if ENABLED(VERBOSE_PARTIAL_REPLAY)
|
|
RDCDEBUG("ExecuteCommands partial replay %u < %u", m_LastEventID,
|
|
parentCmdBufInfo.curEventID + m_Partial[Primary].baseEvent);
|
|
#endif
|
|
|
|
uint32_t eid = startEID;
|
|
|
|
std::vector<ResourceId> trimmedCmdIds;
|
|
std::vector<VkCommandBuffer> trimmedCmds;
|
|
|
|
for(uint32_t c = 0; c < commandBufferCount; c++)
|
|
{
|
|
ResourceId cmdid = GetResourceManager()->GetOriginalID(GetResID(pCommandBuffers[c]));
|
|
|
|
// account for the virtual vkBeginCommandBuffer label at the start of the events here
|
|
// so it matches up to baseEvent
|
|
eid++;
|
|
|
|
uint32_t end = eid + m_BakedCmdBufferInfo[cmdid].eventCount;
|
|
|
|
if(eid == m_Partial[Secondary].baseEvent)
|
|
{
|
|
ResourceId partial = GetResID(RerecordCmdBuf(cmdid, Secondary));
|
|
#if ENABLED(VERBOSE_PARTIAL_REPLAY)
|
|
RDCDEBUG("ExecuteCommands partial replay of %llu at %u, using %llu", cmdid, eid,
|
|
partial);
|
|
#endif
|
|
trimmedCmdIds.push_back(partial);
|
|
trimmedCmds.push_back(Unwrap(RerecordCmdBuf(cmdid, Secondary)));
|
|
}
|
|
else if(m_LastEventID >= end)
|
|
{
|
|
#if ENABLED(VERBOSE_PARTIAL_REPLAY)
|
|
RDCDEBUG("ExecuteCommands full replay %llu", cmdid);
|
|
#endif
|
|
trimmedCmdIds.push_back(cmdid);
|
|
trimmedCmds.push_back(
|
|
Unwrap(GetResourceManager()->GetLiveHandle<VkCommandBuffer>(cmdid)));
|
|
}
|
|
else
|
|
{
|
|
#if ENABLED(VERBOSE_PARTIAL_REPLAY)
|
|
RDCDEBUG("not executing %llu", cmdid);
|
|
#endif
|
|
}
|
|
|
|
// 1 extra to account for the virtual end command buffer label (begin is accounted for
|
|
// above)
|
|
eid += 1 + m_BakedCmdBufferInfo[cmdid].eventCount;
|
|
}
|
|
|
|
if(trimmedCmds.size() > 0)
|
|
ObjDisp(commandBuffer)
|
|
->CmdExecuteCommands(Unwrap(commandBuffer), (uint32_t)trimmedCmds.size(),
|
|
&trimmedCmds[0]);
|
|
|
|
for(uint32_t i = 0; i < trimmedCmdIds.size(); i++)
|
|
{
|
|
ResourceId cmd = trimmedCmdIds[i];
|
|
GetResourceManager()->ApplyBarriers(m_BakedCmdBufferInfo[cmd].imgbarriers,
|
|
m_ImageLayouts);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
#if ENABLED(VERBOSE_PARTIAL_REPLAY)
|
|
RDCDEBUG("ExecuteCommands full replay %u >= %u", m_LastEventID,
|
|
parentCmdBufInfo.curEventID + m_Partial[Primary].baseEvent);
|
|
#endif
|
|
|
|
// execute the commands
|
|
ObjDisp(commandBuffer)
|
|
->CmdExecuteCommands(Unwrap(commandBuffer), commandBufferCount,
|
|
UnwrapArray(pCommandBuffers, commandBufferCount));
|
|
|
|
for(uint32_t i = 0; i < commandBufferCount; i++)
|
|
{
|
|
GetResourceManager()->ApplyBarriers(
|
|
m_BakedCmdBufferInfo[GetResID(pCommandBuffers[i])].imgbarriers, m_ImageLayouts);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdExecuteCommands(VkCommandBuffer commandBuffer, uint32_t commandBufferCount,
|
|
const VkCommandBuffer *pCommandBuffers)
|
|
{
|
|
SCOPED_DBG_SINK();
|
|
|
|
ObjDisp(commandBuffer)
|
|
->CmdExecuteCommands(Unwrap(commandBuffer), commandBufferCount,
|
|
UnwrapArray(pCommandBuffers, commandBufferCount));
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
ser.SetDrawChunk();
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdExecuteCommands);
|
|
Serialise_vkCmdExecuteCommands(ser, commandBuffer, commandBufferCount, pCommandBuffers);
|
|
|
|
record->AddChunk(scope.Get());
|
|
|
|
for(uint32_t i = 0; i < commandBufferCount; i++)
|
|
{
|
|
VkResourceRecord *execRecord = GetRecord(pCommandBuffers[i]);
|
|
if(execRecord->bakedCommands)
|
|
{
|
|
record->cmdInfo->dirtied.insert(execRecord->bakedCommands->cmdInfo->dirtied.begin(),
|
|
execRecord->bakedCommands->cmdInfo->dirtied.end());
|
|
record->cmdInfo->boundDescSets.insert(
|
|
execRecord->bakedCommands->cmdInfo->boundDescSets.begin(),
|
|
execRecord->bakedCommands->cmdInfo->boundDescSets.end());
|
|
record->cmdInfo->subcmds.push_back(execRecord);
|
|
|
|
GetResourceManager()->MergeBarriers(record->cmdInfo->imgbarriers,
|
|
execRecord->bakedCommands->cmdInfo->imgbarriers);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdDebugMarkerBeginEXT(SerialiserType &ser,
|
|
VkCommandBuffer commandBuffer,
|
|
const VkDebugMarkerMarkerInfoEXT *pMarker)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT_LOCAL(Marker, *pMarker);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
{
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
|
|
m_BakedCmdBufferInfo[m_LastCmdBufferID].markerCount++;
|
|
|
|
if(ObjDisp(commandBuffer)->CmdDebugMarkerBeginEXT)
|
|
ObjDisp(commandBuffer)->CmdDebugMarkerBeginEXT(Unwrap(commandBuffer), &Marker);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if(ObjDisp(commandBuffer)->CmdDebugMarkerBeginEXT)
|
|
ObjDisp(commandBuffer)->CmdDebugMarkerBeginEXT(Unwrap(commandBuffer), &Marker);
|
|
|
|
DrawcallDescription draw;
|
|
draw.name = Marker.pMarkerName;
|
|
draw.flags |= DrawFlags::PushMarker;
|
|
|
|
draw.markerColor[0] = RDCCLAMP(Marker.color[0], 0.0f, 1.0f);
|
|
draw.markerColor[1] = RDCCLAMP(Marker.color[1], 0.0f, 1.0f);
|
|
draw.markerColor[2] = RDCCLAMP(Marker.color[2], 0.0f, 1.0f);
|
|
draw.markerColor[3] = RDCCLAMP(Marker.color[3], 0.0f, 1.0f);
|
|
|
|
AddDrawcall(draw, false);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdDebugMarkerBeginEXT(VkCommandBuffer commandBuffer,
|
|
const VkDebugMarkerMarkerInfoEXT *pMarker)
|
|
{
|
|
if(ObjDisp(commandBuffer)->CmdDebugMarkerBeginEXT)
|
|
ObjDisp(commandBuffer)->CmdDebugMarkerBeginEXT(Unwrap(commandBuffer), pMarker);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
ser.SetDrawChunk();
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdDebugMarkerBeginEXT);
|
|
Serialise_vkCmdDebugMarkerBeginEXT(ser, commandBuffer, pMarker);
|
|
|
|
record->AddChunk(scope.Get());
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdDebugMarkerEndEXT(SerialiserType &ser,
|
|
VkCommandBuffer commandBuffer)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
{
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
|
|
int &markerCount = m_BakedCmdBufferInfo[m_LastCmdBufferID].markerCount;
|
|
markerCount = RDCMAX(0, markerCount - 1);
|
|
|
|
if(ObjDisp(commandBuffer)->CmdDebugMarkerEndEXT)
|
|
ObjDisp(commandBuffer)->CmdDebugMarkerEndEXT(Unwrap(commandBuffer));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if(ObjDisp(commandBuffer)->CmdDebugMarkerEndEXT)
|
|
ObjDisp(commandBuffer)->CmdDebugMarkerEndEXT(Unwrap(commandBuffer));
|
|
|
|
if(!m_BakedCmdBufferInfo[m_LastCmdBufferID].curEvents.empty())
|
|
{
|
|
DrawcallDescription draw;
|
|
draw.name = "API Calls";
|
|
draw.flags = DrawFlags::SetMarker | DrawFlags::APICalls;
|
|
|
|
AddDrawcall(draw, true);
|
|
}
|
|
|
|
// dummy draw that is consumed when this command buffer
|
|
// is being in-lined into the call stream
|
|
DrawcallDescription draw;
|
|
draw.name = "Pop()";
|
|
draw.flags = DrawFlags::PopMarker;
|
|
|
|
AddDrawcall(draw, false);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdDebugMarkerEndEXT(VkCommandBuffer commandBuffer)
|
|
{
|
|
if(ObjDisp(commandBuffer)->CmdDebugMarkerEndEXT)
|
|
ObjDisp(commandBuffer)->CmdDebugMarkerEndEXT(Unwrap(commandBuffer));
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
ser.SetDrawChunk();
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdDebugMarkerEndEXT);
|
|
Serialise_vkCmdDebugMarkerEndEXT(ser, commandBuffer);
|
|
|
|
record->AddChunk(scope.Get());
|
|
}
|
|
}
|
|
|
|
template <typename SerialiserType>
|
|
bool WrappedVulkan::Serialise_vkCmdDebugMarkerInsertEXT(SerialiserType &ser,
|
|
VkCommandBuffer commandBuffer,
|
|
const VkDebugMarkerMarkerInfoEXT *pMarker)
|
|
{
|
|
SERIALISE_ELEMENT(commandBuffer);
|
|
SERIALISE_ELEMENT_LOCAL(Marker, *pMarker);
|
|
|
|
if(IsReplayingAndReading())
|
|
{
|
|
m_LastCmdBufferID = GetResourceManager()->GetOriginalID(GetResID(commandBuffer));
|
|
|
|
if(IsActiveReplaying(m_State))
|
|
{
|
|
if(ShouldRerecordCmd(m_LastCmdBufferID) && InRerecordRange(m_LastCmdBufferID))
|
|
{
|
|
commandBuffer = RerecordCmdBuf(m_LastCmdBufferID);
|
|
|
|
if(ObjDisp(commandBuffer)->CmdDebugMarkerInsertEXT)
|
|
ObjDisp(commandBuffer)->CmdDebugMarkerInsertEXT(Unwrap(commandBuffer), &Marker);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if(ObjDisp(commandBuffer)->CmdDebugMarkerInsertEXT)
|
|
ObjDisp(commandBuffer)->CmdDebugMarkerInsertEXT(Unwrap(commandBuffer), &Marker);
|
|
|
|
DrawcallDescription draw;
|
|
draw.name = Marker.pMarkerName;
|
|
draw.flags |= DrawFlags::SetMarker;
|
|
|
|
draw.markerColor[0] = RDCCLAMP(Marker.color[0], 0.0f, 1.0f);
|
|
draw.markerColor[1] = RDCCLAMP(Marker.color[1], 0.0f, 1.0f);
|
|
draw.markerColor[2] = RDCCLAMP(Marker.color[2], 0.0f, 1.0f);
|
|
draw.markerColor[3] = RDCCLAMP(Marker.color[3], 0.0f, 1.0f);
|
|
|
|
AddDrawcall(draw, false);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void WrappedVulkan::vkCmdDebugMarkerInsertEXT(VkCommandBuffer commandBuffer,
|
|
const VkDebugMarkerMarkerInfoEXT *pMarker)
|
|
{
|
|
if(ObjDisp(commandBuffer)->CmdDebugMarkerInsertEXT)
|
|
ObjDisp(commandBuffer)->CmdDebugMarkerInsertEXT(Unwrap(commandBuffer), pMarker);
|
|
|
|
if(IsCaptureMode(m_State))
|
|
{
|
|
VkResourceRecord *record = GetRecord(commandBuffer);
|
|
|
|
CACHE_THREAD_SERIALISER();
|
|
ser.SetDrawChunk();
|
|
SCOPED_SERIALISE_CHUNK(VulkanChunk::vkCmdDebugMarkerInsertEXT);
|
|
Serialise_vkCmdDebugMarkerInsertEXT(ser, commandBuffer, pMarker);
|
|
|
|
record->AddChunk(scope.Get());
|
|
}
|
|
}
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(VkResult, vkCreateCommandPool, VkDevice device,
|
|
const VkCommandPoolCreateInfo *pCreateInfo,
|
|
const VkAllocationCallbacks *pAllocator, VkCommandPool *pCommandPool);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(VkResult, vkBeginCommandBuffer, VkCommandBuffer commandBuffer,
|
|
const VkCommandBufferBeginInfo *pBeginInfo);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(VkResult, vkEndCommandBuffer, VkCommandBuffer commandBuffer);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdBeginRenderPass, VkCommandBuffer commandBuffer,
|
|
const VkRenderPassBeginInfo *pRenderPassBegin,
|
|
VkSubpassContents contents);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdNextSubpass, VkCommandBuffer commandBuffer,
|
|
VkSubpassContents contents);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdEndRenderPass, VkCommandBuffer commandBuffer);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdBindPipeline, VkCommandBuffer commandBuffer,
|
|
VkPipelineBindPoint pipelineBindPoint, VkPipeline pipeline);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdBindDescriptorSets, VkCommandBuffer commandBuffer,
|
|
VkPipelineBindPoint pipelineBindPoint, VkPipelineLayout layout,
|
|
uint32_t firstSet, uint32_t setCount,
|
|
const VkDescriptorSet *pDescriptorSets, uint32_t dynamicOffsetCount,
|
|
const uint32_t *pDynamicOffsets);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdBindIndexBuffer, VkCommandBuffer commandBuffer,
|
|
VkBuffer buffer, VkDeviceSize offset, VkIndexType indexType);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdBindVertexBuffers, VkCommandBuffer commandBuffer,
|
|
uint32_t firstBinding, uint32_t bindingCount,
|
|
const VkBuffer *pBuffers, const VkDeviceSize *pOffsets);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdUpdateBuffer, VkCommandBuffer commandBuffer,
|
|
VkBuffer dstBuffer, VkDeviceSize dstOffset, VkDeviceSize dataSize,
|
|
const uint32_t *pData);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdFillBuffer, VkCommandBuffer commandBuffer,
|
|
VkBuffer dstBuffer, VkDeviceSize dstOffset, VkDeviceSize fillSize,
|
|
uint32_t data);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdPushConstants, VkCommandBuffer commandBuffer,
|
|
VkPipelineLayout layout, VkShaderStageFlags stageFlags,
|
|
uint32_t offset, uint32_t size, const void *pValues);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdPipelineBarrier, VkCommandBuffer commandBuffer,
|
|
VkPipelineStageFlags srcStageMask, VkPipelineStageFlags dstStageMask,
|
|
VkDependencyFlags dependencyFlags, uint32_t memoryBarrierCount,
|
|
const VkMemoryBarrier *pMemoryBarriers,
|
|
uint32_t bufferMemoryBarrierCount,
|
|
const VkBufferMemoryBarrier *pBufferMemoryBarriers,
|
|
uint32_t imageMemoryBarrierCount,
|
|
const VkImageMemoryBarrier *pImageMemoryBarriers);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdWriteTimestamp, VkCommandBuffer commandBuffer,
|
|
VkPipelineStageFlagBits pipelineStage, VkQueryPool queryPool,
|
|
uint32_t query);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdCopyQueryPoolResults, VkCommandBuffer commandBuffer,
|
|
VkQueryPool queryPool, uint32_t firstQuery, uint32_t queryCount,
|
|
VkBuffer dstBuffer, VkDeviceSize dstOffset, VkDeviceSize stride,
|
|
VkQueryResultFlags flags);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdBeginQuery, VkCommandBuffer commandBuffer,
|
|
VkQueryPool queryPool, uint32_t query, VkQueryControlFlags flags);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdEndQuery, VkCommandBuffer commandBuffer,
|
|
VkQueryPool queryPool, uint32_t query);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdResetQueryPool, VkCommandBuffer commandBuffer,
|
|
VkQueryPool queryPool, uint32_t firstQuery, uint32_t queryCount);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdExecuteCommands, VkCommandBuffer commandBuffer,
|
|
uint32_t commandBufferCount, const VkCommandBuffer *pCommandBuffers);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdDebugMarkerBeginEXT, VkCommandBuffer commandBuffer,
|
|
const VkDebugMarkerMarkerInfoEXT *pMarker);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdDebugMarkerEndEXT, VkCommandBuffer commandBuffer);
|
|
|
|
INSTANTIATE_FUNCTION_SERIALISED(void, vkCmdDebugMarkerInsertEXT, VkCommandBuffer commandBuffer,
|
|
const VkDebugMarkerMarkerInfoEXT *pMarker);
|