/****************************************************************************** * The MIT License (MIT) * * Copyright (c) 2016-2026 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 "VulkanPipelineStateViewer.h" #include #include #include #include #include #include #include "Code/Resources.h" #include "Widgets/ComputeDebugSelector.h" #include "Widgets/Extended/RDHeaderView.h" #include "flowlayout/FlowLayout.h" #include "toolwindowmanager/ToolWindowManager.h" #include "PipelineStateViewer.h" #include "ui_VulkanPipelineStateViewer.h" Q_DECLARE_METATYPE(CombinedSamplerData); namespace { QString getTextureRenderSamples(const TextureDescription *tex, const VKPipe::RenderPass &renderpass) { const uint32_t texSamples = tex ? tex->msSamp : 1; const uint32_t renderSamples = renderpass.tileOnlyMSAASampleCount; QString result = lit("%1x").arg(std::max(texSamples, renderSamples)); // With VK_EXT_multisampled_render_to_single_sampled, attachments can either have N or 1 samples, // where N is the same number of samples specified for MSRTSS. Attachments with Nx samples are // rendered to normally, while 1x ones are implicitly rendered with N samples. The latter are // specifically tagged as such. if(renderSamples > 1 && texSamples == 1) { result += lit(" (tile-only)"); } return result; } } // namespace struct VulkanVBIBTag { VulkanVBIBTag() { offset = 0; } VulkanVBIBTag(ResourceId i, uint64_t offs, QString f = QString()) { id = i; offset = offs; format = f; } ResourceId id; uint64_t offset; QString format; }; Q_DECLARE_METATYPE(VulkanVBIBTag); struct VulkanCBufferTag { VulkanCBufferTag() { index = DescriptorAccess::NoShaderBinding; } VulkanCBufferTag(uint32_t index, uint32_t arrayElement, uint32_t dynOffset) : index(index), arrayElement(arrayElement), dynamicOffset(dynOffset) { } VulkanCBufferTag(Descriptor descriptor, uint32_t dynOffset) : index(DescriptorAccess::NoShaderBinding), arrayElement(0), descriptor(descriptor), dynamicOffset(dynOffset) { } Descriptor descriptor; uint32_t dynamicOffset; uint32_t index, arrayElement; }; Q_DECLARE_METATYPE(VulkanCBufferTag); struct VulkanBufferTag { VulkanBufferTag() {} VulkanBufferTag(const DescriptorAccess &access, const Descriptor &desc, uint32_t dynOffset) : access(access), descriptor(desc), dynamicOffset(dynOffset) { } VulkanBufferTag(ResourceId id, uint64_t offset, uint64_t length) { access.index = DescriptorAccess::NoShaderBinding; descriptor.resource = id; descriptor.byteOffset = offset; descriptor.byteSize = length; dynamicOffset = 0; } DescriptorAccess access; Descriptor descriptor; uint32_t dynamicOffset = 0; }; Q_DECLARE_METATYPE(VulkanBufferTag); struct VulkanTextureTag { VulkanTextureTag() { compType = CompType::Typeless; } VulkanTextureTag(ResourceId id, CompType ty) { ID = id; compType = ty; } ResourceId ID; CompType compType; }; Q_DECLARE_METATYPE(VulkanTextureTag); VulkanPipelineStateViewer::VulkanPipelineStateViewer(ICaptureContext &ctx, PipelineStateViewer &common, QWidget *parent) : QFrame(parent), ui(new Ui::VulkanPipelineStateViewer), m_Ctx(ctx), m_Common(common) { ui->setupUi(this); m_ComputeDebugSelector = new ComputeDebugSelector(this); const QIcon &action = Icons::action(); const QIcon &action_hover = Icons::action_hover(); RDLabel *shaderLabels[] = { ui->tsPipeline, ui->msPipeline, ui->vsPipeline, ui->tcsPipeline, ui->tesPipeline, ui->gsPipeline, ui->fsPipeline, ui->csPipeline, ui->tsShader, ui->msShader, ui->vsShader, ui->tcsShader, ui->tesShader, ui->gsShader, ui->fsShader, ui->csShader, ui->tsShaderDebug, ui->msShaderDebug, ui->vsShaderDebug, ui->tcsShaderDebug, ui->tesShaderDebug, ui->gsShaderDebug, ui->fsShaderDebug, ui->csShaderDebug, }; RDLabel *pipeLayoutLabels[] = { ui->tsPipeLayout, ui->msPipeLayout, ui->vsPipeLayout, ui->tcsPipeLayout, ui->tesPipeLayout, ui->gsPipeLayout, ui->fsPipeLayout, ui->csPipeLayout, }; QToolButton *viewButtons[] = { ui->tsShaderViewButton, ui->msShaderViewButton, ui->vsShaderViewButton, ui->tcsShaderViewButton, ui->tesShaderViewButton, ui->gsShaderViewButton, ui->fsShaderViewButton, ui->csShaderViewButton, }; QToolButton *editButtons[] = { ui->tsShaderEditButton, ui->msShaderEditButton, ui->vsShaderEditButton, ui->tcsShaderEditButton, ui->tesShaderEditButton, ui->gsShaderEditButton, ui->fsShaderEditButton, ui->csShaderEditButton, }; QToolButton *saveButtons[] = { ui->tsShaderSaveButton, ui->msShaderSaveButton, ui->vsShaderSaveButton, ui->tcsShaderSaveButton, ui->tesShaderSaveButton, ui->gsShaderSaveButton, ui->fsShaderSaveButton, ui->csShaderSaveButton, }; QToolButton *messageButtons[] = { ui->tsShaderMessagesButton, ui->msShaderMessagesButton, ui->vsShaderMessagesButton, ui->tcsShaderMessagesButton, ui->tesShaderMessagesButton, ui->gsShaderMessagesButton, ui->fsShaderMessagesButton, ui->csShaderMessagesButton, }; QToolButton *viewPredicateBufferButtons[] = { ui->predicateBufferViewButton, ui->csPredicateBufferViewButton, }; RDTreeWidget *resources[] = { ui->tsResources, ui->msResources, ui->vsResources, ui->tcsResources, ui->tesResources, ui->gsResources, ui->fsResources, ui->csResources, }; RDTreeWidget *ubos[] = { ui->tsUBOs, ui->msUBOs, ui->vsUBOs, ui->tcsUBOs, ui->tesUBOs, ui->gsUBOs, ui->fsUBOs, ui->csUBOs, }; RDTreeWidget *descSets[] = { ui->tsDescSets, ui->msDescSets, ui->vsDescSets, ui->tcsDescSets, ui->tesDescSets, ui->gsDescSets, ui->fsDescSets, ui->csDescSets, }; // setup FlowLayout for shader groups QWidget *shaderGroups[] = { ui->tsShaderGroup, ui->msShaderGroup, ui->vsShaderGroup, ui->tcsShaderGroup, ui->tesShaderGroup, ui->gsShaderGroup, ui->fsShaderGroup, ui->csShaderGroup, }; // setup FlowLayout for shader groups for(QWidget *shaderGroup : shaderGroups) { QLayout *oldLayout = shaderGroup->layout(); QObjectList childs = shaderGroup->children(); childs.removeOne((QObject *)oldLayout); delete oldLayout; FlowLayout *shaderFlow = new FlowLayout(shaderGroup, -1, 3, 3); for(QObject *o : childs) shaderFlow->addWidget(qobject_cast(o)); shaderGroup->setLayout(shaderFlow); } for(QToolButton *b : viewButtons) QObject::connect(b, &QToolButton::clicked, this, &VulkanPipelineStateViewer::shaderView_clicked); for(RDLabel *b : shaderLabels) { b->setAutoFillBackground(true); b->setBackgroundRole(QPalette::ToolTipBase); b->setForegroundRole(QPalette::ToolTipText); b->setMinimumSizeHint(QSize(250, 0)); b->setFont(Formatter::PreferredFont()); } for(RDLabel *b : pipeLayoutLabels) { b->setAutoFillBackground(true); b->setBackgroundRole(QPalette::ToolTipBase); b->setForegroundRole(QPalette::ToolTipText); b->setMinimumSizeHint(QSize(250, ui->vsShaderViewButton->minimumSizeHint().height())); b->setFont(Formatter::PreferredFont()); } // collapse the descriptor groups by default ui->vsDescGroup->setCollapsed(true); ui->tcsDescGroup->setCollapsed(true); ui->tesDescGroup->setCollapsed(true); ui->gsDescGroup->setCollapsed(true); ui->fsDescGroup->setCollapsed(true); ui->csDescGroup->setCollapsed(true); ui->tsDescGroup->setCollapsed(true); ui->msDescGroup->setCollapsed(true); QObject::connect(m_ComputeDebugSelector, &ComputeDebugSelector::beginDebug, this, &VulkanPipelineStateViewer::computeDebugSelector_beginDebug); for(QToolButton *b : editButtons) QObject::connect(b, &QToolButton::clicked, &m_Common, &PipelineStateViewer::shaderEdit_clicked); for(QToolButton *b : saveButtons) QObject::connect(b, &QToolButton::clicked, this, &VulkanPipelineStateViewer::shaderSave_clicked); for(QToolButton *b : messageButtons) QObject::connect(b, &QToolButton::clicked, this, &VulkanPipelineStateViewer::shaderMessages_clicked); for(QToolButton *b : viewPredicateBufferButtons) QObject::connect(b, &QToolButton::clicked, this, &VulkanPipelineStateViewer::predicateBufferView_clicked); QObject::connect(ui->viAttrs, &RDTreeWidget::leave, this, &VulkanPipelineStateViewer::vertex_leave); QObject::connect(ui->viBuffers, &RDTreeWidget::leave, this, &VulkanPipelineStateViewer::vertex_leave); QObject::connect(ui->xfbBuffers, &RDTreeWidget::itemActivated, this, &VulkanPipelineStateViewer::resource_itemActivated); QObject::connect(ui->fbAttach, &RDTreeWidget::itemActivated, this, &VulkanPipelineStateViewer::resource_itemActivated); for(RDTreeWidget *res : resources) { QObject::connect(res, &RDTreeWidget::itemActivated, this, &VulkanPipelineStateViewer::resource_itemActivated); QObject::connect(res, &RDTreeWidget::hoverItemChanged, this, &VulkanPipelineStateViewer::resource_hoverItemChanged); } for(RDTreeWidget *ubo : ubos) QObject::connect(ubo, &RDTreeWidget::itemActivated, this, &VulkanPipelineStateViewer::ubo_itemActivated); for(RDTreeWidget *desc : descSets) QObject::connect(desc, &RDTreeWidget::itemActivated, this, &VulkanPipelineStateViewer::descSet_itemActivated); { QMenu *extensionsMenu = new QMenu(this); ui->extensions->setMenu(extensionsMenu); ui->extensions->setPopupMode(QToolButton::InstantPopup); QObject::connect(extensionsMenu, &QMenu::aboutToShow, [this, extensionsMenu]() { extensionsMenu->clear(); m_Ctx.Extensions().MenuDisplaying(PanelMenu::PipelineStateViewer, extensionsMenu, ui->extensions, {}); }); } addGridLines(ui->rasterizerGridLayout, palette().color(QPalette::WindowText)); addGridLines(ui->MSAAGridLayout, palette().color(QPalette::WindowText)); addGridLines(ui->blendStateGridLayout, palette().color(QPalette::WindowText)); addGridLines(ui->depthStateGridLayout, palette().color(QPalette::WindowText)); { RDHeaderView *header = new RDHeaderView(Qt::Horizontal, this); ui->viAttrs->setHeader(header); ui->viAttrs->setColumns({tr("Index"), tr("Name"), tr("Location"), tr("Binding"), tr("Format"), tr("Offset"), tr("Go")}); header->setColumnStretchHints({1, 4, 1, 2, 3, 2, -1}); ui->viAttrs->setHoverIconColumn(6, action, action_hover); ui->viAttrs->setClearSelectionOnFocusLoss(true); ui->viAttrs->setInstantTooltips(true); } { RDHeaderView *header = new RDHeaderView(Qt::Horizontal, this); ui->viBuffers->setHeader(header); ui->viBuffers->setColumns({tr("Slot"), tr("Buffer"), tr("Rate"), tr("Divisor"), tr("Offset"), tr("Stride"), tr("Byte Length"), tr("Go")}); header->setColumnStretchHints({1, 4, 2, 2, 2, 2, 3, -1}); ui->viBuffers->setHoverIconColumn(7, action, action_hover); ui->viBuffers->setClearSelectionOnFocusLoss(true); ui->viBuffers->setInstantTooltips(true); m_Common.SetupResourceView(ui->viBuffers); } for(RDTreeWidget *res : resources) { RDHeaderView *header = new RDHeaderView(Qt::Horizontal, this); res->setHeader(header); res->setColumns( {tr("Binding"), tr("Type"), tr("Resource"), tr("Contents"), tr("Additional"), tr("Go")}); header->setColumnStretchHints({2, 2, 2, 4, 4, -1}); res->setHoverIconColumn(5, action, action_hover); res->setClearSelectionOnFocusLoss(true); res->setInstantTooltips(true); m_Common.SetupResourceView(res); } for(RDTreeWidget *ubo : ubos) { RDHeaderView *header = new RDHeaderView(Qt::Horizontal, this); ubo->setHeader(header); ubo->setColumns({tr("Binding"), tr("Buffer"), tr("Byte Range"), tr("Size"), tr("Go")}); header->setColumnStretchHints({2, 4, 3, 3, -1}); ubo->setHoverIconColumn(4, action, action_hover); ubo->setClearSelectionOnFocusLoss(true); ubo->setInstantTooltips(true); m_Common.SetupResourceView(ubo); } for(RDTreeWidget *desc : descSets) { RDHeaderView *header = new RDHeaderView(Qt::Horizontal, this); desc->setHeader(header); // Buffer / Set desc->setColumns({tr("Index"), tr("Layout"), tr("Buffer"), tr("Buffer Offset"), tr("Set Offset"), tr("Type"), tr("Go")}); header->setColumnStretchHints({-1, 4, 4, 2, 2, 2, -1}); desc->setHoverIconColumn(6, action, action_hover); desc->setClearSelectionOnFocusLoss(true); desc->setInstantTooltips(true); m_Common.SetupResourceView(desc); } ui->vsDescGroupVLayout->activate(); { RDHeaderView *header = new RDHeaderView(Qt::Horizontal, this); ui->xfbBuffers->setHeader(header); ui->xfbBuffers->setColumns({tr("Slot"), tr("Active"), tr("Data Buffer"), tr("Byte Offset"), tr("Byte Length"), tr("Written Count Buffer"), tr("Written Count Offset"), tr("Go")}); header->setColumnStretchHints({1, 1, 4, 2, 3, 4, 2, -1}); header->setMinimumSectionSize(40); ui->xfbBuffers->setHoverIconColumn(7, action, action_hover); ui->xfbBuffers->setClearSelectionOnFocusLoss(true); ui->xfbBuffers->setInstantTooltips(true); m_Common.SetupResourceView(ui->xfbBuffers); } { RDHeaderView *header = new RDHeaderView(Qt::Horizontal, this); ui->viewports->setHeader(header); ui->viewports->setColumns({tr("Slot"), tr("X"), tr("Y"), tr("Width"), tr("Height"), tr("MinDepth"), tr("MaxDepth"), tr("NDCDepthRange")}); header->setColumnStretchHints({-1, -1, -1, -1, -1, -1, -1, 1}); header->setMinimumSectionSize(40); ui->viewports->setClearSelectionOnFocusLoss(true); ui->viewports->setInstantTooltips(true); } { RDHeaderView *header = new RDHeaderView(Qt::Horizontal, this); ui->scissors->setHeader(header); ui->scissors->setColumns({tr("Slot"), tr("X"), tr("Y"), tr("Width"), tr("Height")}); header->setColumnStretchHints({-1, -1, -1, -1, 1}); header->setMinimumSectionSize(40); ui->scissors->setClearSelectionOnFocusLoss(true); ui->scissors->setInstantTooltips(true); } { RDHeaderView *header = new RDHeaderView(Qt::Horizontal, this); ui->discards->setHeader(header); ui->discards->setColumns({tr("Slot"), tr("X"), tr("Y"), tr("Width"), tr("Height")}); header->setColumnStretchHints({-1, -1, -1, -1, 1}); header->setMinimumSectionSize(40); ui->discards->setClearSelectionOnFocusLoss(true); ui->discards->setInstantTooltips(true); } for(RDLabel *rp : {ui->renderpass, ui->framebuffer, ui->predicateBuffer, ui->csPredicateBuffer}) { rp->setAutoFillBackground(true); rp->setBackgroundRole(QPalette::ToolTipBase); rp->setForegroundRole(QPalette::ToolTipText); rp->setMinimumSizeHint(QSize(250, 0)); } { RDHeaderView *header = new RDHeaderView(Qt::Horizontal, this); ui->fbAttach->setHeader(header); ui->fbAttach->setColumns({tr("Slot"), tr("Resource"), tr("Type"), tr("Dimensions"), tr("Format"), tr("Samples"), tr("Go")}); header->setColumnStretchHints({2, 4, 2, 2, 3, 1, -1}); ui->fbAttach->setHoverIconColumn(6, action, action_hover); ui->fbAttach->setClearSelectionOnFocusLoss(true); ui->fbAttach->setInstantTooltips(true); m_Common.SetupResourceView(ui->fbAttach); } { RDHeaderView *header = new RDHeaderView(Qt::Horizontal, this); ui->blends->setHeader(header); ui->blends->setColumns({tr("Slot"), tr("Enabled"), tr("Col Src"), tr("Col Dst"), tr("Col Op"), tr("Alpha Src"), tr("Alpha Dst"), tr("Alpha Op"), tr("Write Mask")}); header->setColumnStretchHints({-1, 1, 2, 2, 2, 2, 2, 2, 1}); ui->blends->setClearSelectionOnFocusLoss(true); ui->blends->setInstantTooltips(true); } { RDHeaderView *header = new RDHeaderView(Qt::Horizontal, this); ui->stencils->setHeader(header); ui->stencils->setColumns({tr("Face"), tr("Func"), tr("Fail Op"), tr("Depth Fail Op"), tr("Pass Op"), tr("Write Mask"), tr("Comp Mask"), tr("Ref")}); header->setColumnStretchHints({1, 2, 2, 2, 2, 1, 1, 1}); ui->stencils->setClearSelectionOnFocusLoss(true); ui->stencils->setInstantTooltips(true); } // this is often changed just because we're changing some tab in the designer. ui->stagesTabs->setCurrentIndex(0); ui->stagesTabs->tabBar()->setVisible(false); setOldMeshPipeFlow(); ui->pipeFlow->setStagesEnabled({true, true, true, true, true, true, true, true, true}); m_Common.setMeshViewPixmap(ui->meshView); ui->viAttrs->setFont(Formatter::PreferredFont()); ui->viBuffers->setFont(Formatter::PreferredFont()); ui->tsResources->setFont(Formatter::PreferredFont()); ui->tsUBOs->setFont(Formatter::PreferredFont()); ui->msResources->setFont(Formatter::PreferredFont()); ui->msUBOs->setFont(Formatter::PreferredFont()); ui->vsResources->setFont(Formatter::PreferredFont()); ui->vsUBOs->setFont(Formatter::PreferredFont()); ui->gsResources->setFont(Formatter::PreferredFont()); ui->gsUBOs->setFont(Formatter::PreferredFont()); ui->tcsResources->setFont(Formatter::PreferredFont()); ui->tcsUBOs->setFont(Formatter::PreferredFont()); ui->tesResources->setFont(Formatter::PreferredFont()); ui->tesUBOs->setFont(Formatter::PreferredFont()); ui->fsResources->setFont(Formatter::PreferredFont()); ui->fsUBOs->setFont(Formatter::PreferredFont()); ui->csResources->setFont(Formatter::PreferredFont()); ui->csUBOs->setFont(Formatter::PreferredFont()); ui->xfbBuffers->setFont(Formatter::PreferredFont()); ui->viewports->setFont(Formatter::PreferredFont()); ui->scissors->setFont(Formatter::PreferredFont()); ui->renderpass->setFont(Formatter::PreferredFont()); ui->framebuffer->setFont(Formatter::PreferredFont()); ui->fbAttach->setFont(Formatter::PreferredFont()); ui->blends->setFont(Formatter::PreferredFont()); m_ExportMenu = new QMenu(this); m_ExportHTML = new QAction(tr("Export current state to &HTML"), this); m_ExportHTML->setIcon(Icons::save()); m_ExportFOZ = new QAction(tr("Export to &Fossilize database"), this); m_ExportFOZ->setIcon(Icons::save()); m_ExportMenu->addAction(m_ExportHTML); m_ExportMenu->addAction(m_ExportFOZ); ui->exportDrop->setMenu(m_ExportMenu); QObject::connect(m_ExportHTML, &QAction::triggered, this, &VulkanPipelineStateViewer::exportHTML_clicked); QObject::connect(m_ExportFOZ, &QAction::triggered, this, &VulkanPipelineStateViewer::exportFOZ_clicked); QObject::connect(ui->exportDrop, &QToolButton::clicked, this, &VulkanPipelineStateViewer::exportHTML_clicked); // reset everything back to defaults clearState(); } VulkanPipelineStateViewer::~VulkanPipelineStateViewer() { m_CombinedImageSamplers.clear(); delete ui; delete m_ComputeDebugSelector; } void VulkanPipelineStateViewer::OnCaptureLoaded() { OnEventChanged(m_Ctx.CurEvent()); } void VulkanPipelineStateViewer::OnCaptureClosed() { setOldMeshPipeFlow(); ui->pipeFlow->setStagesEnabled({true, true, true, true, true, true, true, true, true}); clearState(); } void VulkanPipelineStateViewer::OnEventChanged(uint32_t eventId) { setState(); } void VulkanPipelineStateViewer::SelectPipelineStage(PipelineStage stage) { if(stage == PipelineStage::SampleMask) ui->pipeFlow->setSelectedStage((int)PipelineStage::Rasterizer); else ui->pipeFlow->setSelectedStage((int)stage); } ResourceId VulkanPipelineStateViewer::GetResource(RDTreeWidgetItem *item) { QVariant tag = item->tag(); if(tag.canConvert()) { return tag.value(); } else if(tag.canConvert()) { VulkanTextureTag texTag = tag.value(); return texTag.ID; } else if(tag.canConvert()) { VulkanBufferTag buf = tag.value(); return buf.descriptor.resource; } else if(tag.canConvert()) { VulkanVBIBTag buf = tag.value(); return buf.id; } else if(tag.canConvert()) { const VKPipe::Shader *stage = stageForSender(item->treeWidget()); if(stage == NULL) return ResourceId(); VulkanCBufferTag cb = tag.value(); if(cb.index == DescriptorAccess::NoShaderBinding) return cb.descriptor.resource; return m_Ctx.CurPipelineState() .GetConstantBlock(stage->stage, cb.index, cb.arrayElement) .descriptor.resource; } return ResourceId(); } void VulkanPipelineStateViewer::on_showUnused_toggled(bool checked) { setState(); } void VulkanPipelineStateViewer::on_showEmpty_toggled(bool checked) { setState(); } void VulkanPipelineStateViewer::setInactiveRow(RDTreeWidgetItem *node) { node->setItalic(true); } void VulkanPipelineStateViewer::setEmptyRow(RDTreeWidgetItem *node) { node->setBackgroundColor(QColor(255, 70, 70)); node->setForegroundColor(QColor(0, 0, 0)); } bool VulkanPipelineStateViewer::setViewDetails(RDTreeWidgetItem *node, const Descriptor &descriptor, TextureDescription *tex, const QString &hiddenCombinedSampler, bool includeSampleLocations, bool includeOffsets) { if(tex == NULL) return false; QString text; bool viewdetails = false; const VKPipe::State &state = *m_Ctx.CurVulkanPipelineState(); { for(const VKPipe::ImageData &im : state.images) { if(im.resourceId == tex->resourceId) { text += tr("Texture is in the '%1' layout\n").arg(im.layouts[0].name); break; } } text += hiddenCombinedSampler; text += lit("\n"); if(descriptor.format != tex->format) { text += tr("The texture is format %1, the view treats it as %2.\n") .arg(tex->format.Name()) .arg(descriptor.format.Name()); viewdetails = true; } if(tex->mips > 1 && (tex->mips != descriptor.numMips || descriptor.firstMip > 0)) { if(descriptor.numMips == 1) text += tr("The texture has %1 mips, the view covers mip %2.\n") .arg(tex->mips) .arg(descriptor.firstMip); else text += tr("The texture has %1 mips, the view covers mips %2-%3.\n") .arg(tex->mips) .arg(descriptor.firstMip) .arg(descriptor.firstMip + descriptor.numMips - 1); viewdetails = true; } if(tex->arraysize > 1 && (tex->arraysize != descriptor.numSlices || descriptor.firstSlice > 0)) { if(descriptor.numSlices == 1) text += tr("The texture has %1 array slices, the view covers slice %2.\n") .arg(tex->arraysize) .arg(descriptor.firstSlice); else text += tr("The texture has %1 array slices, the view covers slices %2-%3.\n") .arg(tex->arraysize) .arg(descriptor.firstSlice) .arg(descriptor.firstSlice + descriptor.numSlices - 1); viewdetails = true; } if(tex->depth > 1 && ((tex->depth != descriptor.numSlices && descriptor.numSlices > 0) || descriptor.firstSlice > 0)) { if(descriptor.numSlices == 1) text += tr("The texture has %1 3D slices, the view covers slice %2.\n") .arg(tex->depth) .arg(descriptor.firstSlice); else text += tr("The texture has %1 3D slices, the view covers slices %2-%3.\n") .arg(tex->depth) .arg(descriptor.firstSlice) .arg(descriptor.firstSlice + descriptor.numSlices - 1); viewdetails = true; } } if(descriptor.minLODClamp != 0.0f) { text += tr("Clamped to a minimum LOD of %1\n").arg(descriptor.minLODClamp); } if(includeSampleLocations && state.multisample.rasterSamples > 1 && !state.multisample.sampleLocations.customLocations.isEmpty()) { text += tr("Rendering with custom sample locations over %1x%2 grid:\n") .arg(state.multisample.sampleLocations.gridWidth) .arg(state.multisample.sampleLocations.gridHeight); const rdcarray &locations = state.multisample.sampleLocations.customLocations; for(int i = 0; i < locations.count(); i++) { text += QFormatStr(" [%1]: %2, %3\n") .arg(i) .arg(Formatter::Format(locations[i].x)) .arg(Formatter::Format(locations[i].y)); } viewdetails = true; } if(includeOffsets) { text += tr("Rendering with %1 offsets:\n") .arg(state.currentPass.renderpass.fragmentDensityOffsets.size()); for(uint32_t j = 0; j < state.currentPass.renderpass.fragmentDensityOffsets.size(); j++) { const Offset &o = state.currentPass.renderpass.fragmentDensityOffsets[j]; if(j > 0) text += tr(", "); text += tr(" %1x%2").arg(o.x).arg(o.y); } text += lit("\n"); } text = text.trimmed(); node->setToolTip(text); if(viewdetails) { node->setBackgroundColor(m_Common.GetViewDetailsColor()); } return viewdetails; } bool VulkanPipelineStateViewer::setViewDetails(RDTreeWidgetItem *node, const Descriptor &descriptor, BufferDescription *buf, uint32_t dynamicOffset) { if(buf == NULL) return false; QString text; if(descriptor.byteOffset + dynamicOffset > 0 || descriptor.byteSize < buf->length) { uint64_t effectiveSize = descriptor.byteSize; if(descriptor.byteSize == UINT64_MAX) effectiveSize = buf->length - (descriptor.byteOffset + dynamicOffset); text += tr("The view covers bytes %1-%2.\nThe buffer is %3 bytes in length.\n") .arg(Formatter::HumanFormat(descriptor.byteOffset + dynamicOffset, Formatter::OffsetSize)) .arg(Formatter::HumanFormat(descriptor.byteOffset + dynamicOffset + effectiveSize, Formatter::OffsetSize)) .arg(Formatter::HumanFormat(buf->length, Formatter::OffsetSize)); } else { return false; } node->setToolTip(text); node->setBackgroundColor(m_Common.GetViewDetailsColor()); return true; } bool VulkanPipelineStateViewer::showNode(bool usedSlot, bool filledSlot) { // show if it's referenced by the shader - regardless of empty or not if(usedSlot) return true; // it's not referenced, but if it's bound and we have "show unused" then show it if(m_ShowUnused && filledSlot) return true; // it's empty, and we have "show empty" if(m_ShowEmpty && !filledSlot) return true; return false; } QString VulkanPipelineStateViewer::formatByteRange(const BufferDescription *buf, const Descriptor &descriptor, uint32_t dynamicOffset) { if(buf == NULL) return lit("-"); uint64_t byteOffset = descriptor.byteOffset + dynamicOffset; if(descriptor.byteSize == 0) { return tr("%1 - %2 (empty view)").arg(byteOffset).arg(byteOffset); } else if(descriptor.byteSize == UINT64_MAX) { return QFormatStr("%1 - %2 (VK_WHOLE_SIZE)") .arg(Formatter::HumanFormat(byteOffset, Formatter::OffsetSize)) .arg(Formatter::HumanFormat(byteOffset + (buf->length - byteOffset), Formatter::OffsetSize)); } else { return QFormatStr("%1 - %2") .arg(Formatter::HumanFormat(byteOffset, Formatter::OffsetSize)) .arg(Formatter::HumanFormat(byteOffset + descriptor.byteSize, Formatter::OffsetSize)); } } const VKPipe::Shader *VulkanPipelineStateViewer::stageForSender(QWidget *widget) { if(!m_Ctx.IsCaptureLoaded()) return NULL; while(widget) { if(widget == ui->stagesTabs->widget(0)) return &m_Ctx.CurVulkanPipelineState()->vertexShader; if(widget == ui->stagesTabs->widget(1)) return &m_Ctx.CurVulkanPipelineState()->vertexShader; if(widget == ui->stagesTabs->widget(2)) return &m_Ctx.CurVulkanPipelineState()->tessControlShader; if(widget == ui->stagesTabs->widget(3)) return &m_Ctx.CurVulkanPipelineState()->tessEvalShader; if(widget == ui->stagesTabs->widget(4)) return &m_Ctx.CurVulkanPipelineState()->geometryShader; if(widget == ui->stagesTabs->widget(5)) return &m_Ctx.CurVulkanPipelineState()->fragmentShader; if(widget == ui->stagesTabs->widget(6)) return &m_Ctx.CurVulkanPipelineState()->fragmentShader; if(widget == ui->stagesTabs->widget(7)) return &m_Ctx.CurVulkanPipelineState()->fragmentShader; if(widget == ui->stagesTabs->widget(8)) return &m_Ctx.CurVulkanPipelineState()->computeShader; if(widget == ui->stagesTabs->widget(9)) return &m_Ctx.CurVulkanPipelineState()->taskShader; if(widget == ui->stagesTabs->widget(10)) return &m_Ctx.CurVulkanPipelineState()->meshShader; widget = widget->parentWidget(); } qCritical() << "Unrecognised control calling event handler"; return NULL; } void VulkanPipelineStateViewer::setOldMeshPipeFlow() { m_MeshPipe = false; ui->pipeFlow->setStages( { lit("VTX"), lit("VS"), lit("TCS"), lit("TES"), lit("GS"), lit("RS"), lit("FS"), lit("FB"), lit("CS"), }, { tr("Vertex Input"), tr("Vertex Shader"), tr("Tess. Control Shader"), tr("Tess. Eval. Shader"), tr("Geometry Shader"), tr("Rasterizer"), tr("Fragment Shader"), tr("Framebuffer Output"), tr("Compute Shader"), }); ui->pipeFlow->setIsolatedStage(8); // compute shader isolated } void VulkanPipelineStateViewer::setNewMeshPipeFlow() { m_MeshPipe = true; ui->pipeFlow->setStages( { lit("TS"), lit("MS"), lit("RS"), lit("FS"), lit("FB"), lit("CS"), }, { tr("Task Shader"), tr("Mesh Shader"), tr("Rasterizer"), tr("Fragment Shader"), tr("Framebuffer Output"), tr("Compute Shader"), }); ui->pipeFlow->setIsolatedStage(5); // compute shader isolated } void VulkanPipelineStateViewer::clearShaderState(RDLabel *pipeline, RDLabel *shader, RDLabel *shaderDebug, RDLabel *pipeLayout, RDTreeWidget *resources, RDTreeWidget *cbuffers, RDTreeWidget *descSets) { pipeLayout->setText(tr("Pipeline Layout")); pipeline->show(); pipeline->setText(ToQStr(ResourceId())); shader->setText(ToQStr(ResourceId())); shaderDebug->hide(); resources->clear(); cbuffers->clear(); descSets->clear(); } void VulkanPipelineStateViewer::clearState() { m_CombinedImageSamplers.clear(); m_VBNodes.clear(); m_BindNodes.clear(); m_EmptyNodes.clear(); ui->viAttrs->clear(); ui->viBuffers->clear(); ui->topology->setText(QString()); ui->primRestart->setVisible(false); ui->topologyDiagram->setPixmap(QPixmap()); clearShaderState(ui->tsPipeline, ui->tsShader, ui->tsShaderDebug, ui->tsPipeLayout, ui->tsResources, ui->tsUBOs, ui->tsDescSets); clearShaderState(ui->msPipeline, ui->msShader, ui->msShaderDebug, ui->msPipeLayout, ui->msResources, ui->msUBOs, ui->msDescSets); clearShaderState(ui->vsPipeline, ui->vsShader, ui->vsShaderDebug, ui->vsPipeLayout, ui->vsResources, ui->vsUBOs, ui->vsDescSets); clearShaderState(ui->tcsPipeline, ui->tcsShader, ui->tcsShaderDebug, ui->tcsPipeLayout, ui->tcsResources, ui->tcsUBOs, ui->tcsDescSets); clearShaderState(ui->tesPipeline, ui->tesShader, ui->tesShaderDebug, ui->tesPipeLayout, ui->tesResources, ui->tesUBOs, ui->tesDescSets); clearShaderState(ui->gsPipeline, ui->gsShader, ui->gsShaderDebug, ui->gsPipeLayout, ui->gsResources, ui->gsUBOs, ui->gsDescSets); clearShaderState(ui->fsPipeline, ui->fsShader, ui->fsShaderDebug, ui->fsPipeLayout, ui->fsResources, ui->fsUBOs, ui->fsDescSets); clearShaderState(ui->csPipeline, ui->csShader, ui->csShaderDebug, ui->csPipeLayout, ui->csResources, ui->csUBOs, ui->csDescSets); ui->xfbBuffers->clear(); QToolButton *shaderButtons[] = { // view buttons ui->tsShaderViewButton, ui->msShaderViewButton, ui->vsShaderViewButton, ui->tcsShaderViewButton, ui->tesShaderViewButton, ui->gsShaderViewButton, ui->fsShaderViewButton, ui->csShaderViewButton, // edit buttons ui->tsShaderEditButton, ui->msShaderEditButton, ui->vsShaderEditButton, ui->tcsShaderEditButton, ui->tesShaderEditButton, ui->gsShaderEditButton, ui->fsShaderEditButton, ui->csShaderEditButton, // save buttons ui->tsShaderSaveButton, ui->msShaderSaveButton, ui->vsShaderSaveButton, ui->tcsShaderSaveButton, ui->tesShaderSaveButton, ui->gsShaderSaveButton, ui->fsShaderSaveButton, ui->csShaderSaveButton, }; for(QToolButton *b : shaderButtons) b->setEnabled(false); QToolButton *messageButtons[] = { ui->tsShaderMessagesButton, ui->msShaderMessagesButton, ui->vsShaderMessagesButton, ui->tcsShaderMessagesButton, ui->tesShaderMessagesButton, ui->gsShaderMessagesButton, ui->fsShaderMessagesButton, ui->csShaderMessagesButton, }; for(QToolButton *b : messageButtons) b->setVisible(false); const QPixmap &tick = Pixmaps::tick(this); const QPixmap &cross = Pixmaps::cross(this); ui->fillMode->setText(tr("Solid", "Fill Mode")); ui->cullMode->setText(tr("Front", "Cull Mode")); ui->frontCCW->setPixmap(tick); ui->depthBias->setText(lit("0.0")); ui->depthBiasClamp->setText(lit("0.0")); ui->slopeScaledBias->setText(lit("0.0")); ui->depthClamp->setPixmap(tick); ui->depthClip->setPixmap(cross); ui->rasterizerDiscard->setPixmap(tick); ui->lineWidth->setText(lit("1.0")); ui->conservativeRaster->setText(tr("Disabled")); ui->multiview->setText(tr("Disabled")); ui->stippleFactor->setText(QString()); ui->stippleFactor->setPixmap(cross); ui->stipplePattern->setText(QString()); ui->stipplePattern->setPixmap(cross); ui->pipelineShadingRate->setText(tr("1x1")); ui->shadingRateCombiners->setText(tr("Keep, Keep")); ui->provokingVertex->setText(tr("First")); ui->sampleCount->setText(lit("1")); ui->sampleShading->setPixmap(tick); ui->minSampleShading->setText(lit("0.0")); ui->sampleMask->setText(lit("FFFFFFFF")); ui->viewports->clear(); ui->scissors->clear(); ui->discards->clear(); ui->discardMode->setText(tr("Inclusive")); ui->discardGroup->setVisible(false); ui->renderpass->setText(QFormatStr("Render Pass: %1").arg(ToQStr(ResourceId()))); ui->framebuffer->setText(QFormatStr("Framebuffer: %1").arg(ToQStr(ResourceId()))); ui->fbAttach->clear(); ui->blends->clear(); ui->blendFactor->setText(lit("0.00, 0.00, 0.00, 0.00")); ui->logicOp->setText(lit("-")); ui->alphaToOne->setPixmap(tick); ui->depthEnabled->setPixmap(tick); ui->depthFunc->setText(lit("GREATER_EQUAL")); ui->depthWrite->setPixmap(tick); ui->depthBounds->setPixmap(QPixmap()); ui->depthBounds->setText(lit("0.0-1.0")); ui->stencils->clear(); ui->computeDebugSelector->setEnabled(false); ui->conditionalRenderingGroup->setVisible(false); ui->csConditionalRenderingGroup->setVisible(false); } QVariantList VulkanPipelineStateViewer::makeSampler(const QString &slotname, const SamplerDescriptor &descriptor) { QString addressing; QString addPrefix; QString addVal; QString filter; QString addr[] = {ToQStr(descriptor.addressU, GraphicsAPI::Vulkan), ToQStr(descriptor.addressV, GraphicsAPI::Vulkan), ToQStr(descriptor.addressW, GraphicsAPI::Vulkan)}; // arrange like either UVW: WRAP or UV: WRAP, W: CLAMP for(int a = 0; a < 3; a++) { const char *uvw = "UVW"; QString prefix = QString(QLatin1Char(uvw[a])); if(a == 0 || addr[a] == addr[a - 1]) { addPrefix += prefix; } else { addressing += addPrefix + lit(": ") + addVal + lit(", "); addPrefix = prefix; } addVal = addr[a]; } addressing += addPrefix + lit(": ") + addVal; if(descriptor.UseBorder()) { if(descriptor.borderColorType == CompType::Float) addressing += QFormatStr(" <%1, %2, %3, %4>") .arg(descriptor.borderColorValue.floatValue[0]) .arg(descriptor.borderColorValue.floatValue[1]) .arg(descriptor.borderColorValue.floatValue[2]) .arg(descriptor.borderColorValue.floatValue[3]); else addressing += QFormatStr(" <%1, %2, %3, %4>") .arg(descriptor.borderColorValue.uintValue[0]) .arg(descriptor.borderColorValue.uintValue[1]) .arg(descriptor.borderColorValue.uintValue[2]) .arg(descriptor.borderColorValue.uintValue[3]); } if(descriptor.unnormalized) addressing += lit(" (Un-norm)"); filter = ToQStr(descriptor.filter); if(descriptor.maxAnisotropy >= 1.0f) filter += lit(" Aniso %1x").arg(descriptor.maxAnisotropy); if(descriptor.filter.filter == FilterFunction::Comparison) filter += QFormatStr(" (%1)").arg(ToQStr(descriptor.compareFunction)); else if(descriptor.filter.filter != FilterFunction::Normal) filter += QFormatStr(" (%1)").arg(ToQStr(descriptor.filter.filter)); QString minLOD = QString::number(descriptor.minLOD); QString maxLOD = QString::number(descriptor.maxLOD); if(descriptor.minLOD == -FLT_MAX) minLOD = lit("0"); if(descriptor.minLOD == -1000.0) minLOD = lit("VK_LOD_CLAMP_NONE"); if(descriptor.maxLOD == FLT_MAX) maxLOD = lit("FLT_MAX"); if(descriptor.maxLOD == 1000.0) maxLOD = lit("VK_LOD_CLAMP_NONE"); QString lod = lit("LODs: %1 - %2").arg(minLOD).arg(maxLOD); if(descriptor.mipBias != 0.0f) lod += lit(" Bias %1").arg(descriptor.mipBias); if(!lod.isEmpty()) lod = lit(", ") + lod; QString obj = ToQStr(descriptor.object); if(descriptor.swizzle.red != TextureSwizzle::Red || descriptor.swizzle.green != TextureSwizzle::Green || descriptor.swizzle.blue != TextureSwizzle::Blue || descriptor.swizzle.alpha != TextureSwizzle::Alpha) { obj += tr(" swizzle[%1%2%3%4]") .arg(ToQStr(descriptor.swizzle.red)) .arg(ToQStr(descriptor.swizzle.green)) .arg(ToQStr(descriptor.swizzle.blue)) .arg(ToQStr(descriptor.swizzle.alpha)); } if(!descriptor.seamlessCubemaps) addressing += tr(" Non-Seamless"); if(descriptor.ycbcrSampler != ResourceId()) { obj += lit(" ") + ToQStr(descriptor.ycbcrSampler); filter += QFormatStr(", %1 %2").arg(ToQStr(descriptor.ycbcrModel)).arg(ToQStr(descriptor.ycbcrRange)); addressing += tr(", Chroma %1 [%2,%3]") .arg(ToQStr(descriptor.chromaFilter)) .arg(ToQStr(descriptor.xChromaOffset)) .arg(ToQStr(descriptor.yChromaOffset)); if(descriptor.forceExplicitReconstruction) addressing += tr(" Explicit"); } return {slotname, descriptor.creationTimeConstant ? tr("Immutable Sampler") : tr("Sampler"), obj, addressing, filter + lod, QString()}; } void VulkanPipelineStateViewer::addResourceRow(const ShaderResource *shaderRes, const ShaderSampler *shaderSamp, const UsedDescriptor &used, uint32_t dynamicOffset, RDTreeWidget *resources, QMap &samplers) { const Descriptor &descriptor = used.descriptor; const SamplerDescriptor &samplerDescriptor = used.sampler; bool filledSlot = (descriptor.resource != ResourceId() || samplerDescriptor.object != ResourceId() || samplerDescriptor.creationTimeConstant); // Vulkan does not report unused elements at all because we enumerate exclusively from the // perspective of which descriptors are used bool usedSlot = true; if(showNode(usedSlot, filledSlot)) { QString slotname; if(used.access.index == DescriptorAccess::NoShaderBinding) { slotname = m_Locations[{used.access.descriptorStore, used.access.byteOffset}].logicalBindName; slotname += QFormatStr("[%1]").arg(used.access.arrayElement); } else if(shaderRes) { if(IsPushSet(used.access.stage, used.access.descriptorStore)) slotname = tr("Push "); slotname += QFormatStr("Set %1, %2").arg(shaderRes->fixedBindSetOrSpace).arg(shaderRes->fixedBindNumber); if(!shaderRes->name.empty()) slotname += lit(": ") + shaderRes->name; if(shaderRes->bindArraySize > 1) slotname += QFormatStr("[%1]").arg(used.access.arrayElement); } else if(shaderSamp) { if(IsPushSet(used.access.stage, used.access.descriptorStore)) slotname = tr("Push "); slotname += QFormatStr("Set %1, %2").arg(shaderSamp->fixedBindSetOrSpace).arg(shaderSamp->fixedBindNumber); if(!shaderSamp->name.empty()) slotname += lit(": ") + shaderSamp->name; if(shaderSamp->bindArraySize > 1) slotname += QFormatStr("[%1]").arg(used.access.arrayElement); } bool isbuf = false; uint32_t w = 1, h = 1, d = 1; uint32_t a = 1; uint32_t samples = 1; uint64_t resourceByteSize = 0; QString format = descriptor.format.Name(); TextureType restype = TextureType::Unknown; QVariant tag; TextureDescription *tex = NULL; BufferDescription *buf = NULL; if(descriptor.resource != ResourceId()) { // check to see if it's a texture tex = m_Ctx.GetTexture(descriptor.resource); if(tex) { w = tex->width; h = tex->height; d = tex->depth; a = tex->arraysize; restype = tex->type; samples = tex->msSamp; tag = QVariant::fromValue(VulkanTextureTag(descriptor.resource, descriptor.format.compType)); } // if not a texture, it must be a buffer buf = m_Ctx.GetBuffer(descriptor.resource); if(buf) { resourceByteSize = buf->length; w = 0; h = 0; d = 0; a = 0; restype = TextureType::Buffer; tag = QVariant::fromValue(VulkanBufferTag(used.access, used.descriptor, dynamicOffset)); isbuf = true; } } else { format = lit("-"); w = h = d = a = 0; } RDTreeWidgetItem *node = NULL; RDTreeWidgetItem *samplerNode = NULL; QString bindType = ToQStr(used.access.type); if(shaderRes && shaderRes->isInputAttachment) bindType = tr("Input Attachment"); if(used.access.type == DescriptorType::ReadWriteBuffer) { if(!isbuf) { node = new RDTreeWidgetItem({ slotname, bindType, ResourceId(), lit("-"), QString(), QString(), }); setEmptyRow(node); } else { node = new RDTreeWidgetItem({ slotname, bindType, descriptor.resource, tr("%1 bytes").arg(Formatter::HumanFormat(resourceByteSize, Formatter::OffsetSize)), QFormatStr("Viewing bytes %1").arg(formatByteRange(buf, descriptor, dynamicOffset)), QString(), }); node->setTag(tag); if(!filledSlot) setEmptyRow(node); } } else if(used.access.type == DescriptorType::TypedBuffer || used.access.type == DescriptorType::ReadWriteTypedBuffer) { node = new RDTreeWidgetItem({ slotname, bindType, descriptor.resource, format, QFormatStr("bytes %1").arg(formatByteRange(buf, descriptor, dynamicOffset)), QString(), }); node->setTag(tag); if(!filledSlot) setEmptyRow(node); } else if(used.access.type == DescriptorType::AccelerationStructure) { node = new RDTreeWidgetItem({ slotname, bindType, descriptor.resource, QString(), QFormatStr("%1 bytes").arg(Formatter::HumanFormat(descriptor.byteSize, Formatter::OffsetSize)), QString(), }); node->setTag(tag); if(!filledSlot) setEmptyRow(node); } else if(used.access.type == DescriptorType::Sampler) { if(samplerDescriptor.object == ResourceId()) { node = new RDTreeWidgetItem({ slotname, bindType, ResourceId(), lit("-"), QString(), QString(), }); setEmptyRow(node); } else { node = new RDTreeWidgetItem(makeSampler(slotname, samplerDescriptor)); if(!filledSlot) setEmptyRow(node); } } else { if(descriptor.resource == ResourceId()) { node = new RDTreeWidgetItem({ slotname, bindType, ResourceId(), lit("-"), QString(), QString(), }); setEmptyRow(node); } else { QString typeName = ToQStr(restype) + lit(" ") + bindType; QString dim; if(restype == TextureType::Texture3D) dim = QFormatStr("%1x%2x%3").arg(w).arg(h).arg(d); else if(restype == TextureType::Texture1D || restype == TextureType::Texture1DArray) dim = QString::number(w); else dim = QFormatStr("%1x%2").arg(w).arg(h); if(descriptor.swizzle.red != TextureSwizzle::Red || descriptor.swizzle.green != TextureSwizzle::Green || descriptor.swizzle.blue != TextureSwizzle::Blue || descriptor.swizzle.alpha != TextureSwizzle::Alpha) { format += tr(" swizzle[%1%2%3%4]") .arg(ToQStr(descriptor.swizzle.red)) .arg(ToQStr(descriptor.swizzle.green)) .arg(ToQStr(descriptor.swizzle.blue)) .arg(ToQStr(descriptor.swizzle.alpha)); } if(restype == TextureType::Texture1DArray || restype == TextureType::Texture2DArray || restype == TextureType::Texture2DMSArray || restype == TextureType::TextureCubeArray) { dim += QFormatStr(" %1[%2]").arg(ToQStr(restype)).arg(a); } if(restype == TextureType::Texture2DMS || restype == TextureType::Texture2DMSArray) dim += QFormatStr(", %1x MSAA").arg(samples); node = new RDTreeWidgetItem({ slotname, typeName, descriptor.resource, dim, format, QString(), }); node->setTag(tag); if(!filledSlot) setEmptyRow(node); if(used.access.type == DescriptorType::ImageSampler) { if(samplerDescriptor.object == ResourceId()) { samplerNode = new RDTreeWidgetItem({ slotname, bindType, ResourceId(), lit("-"), QString(), QString(), }); setEmptyRow(samplerNode); } else { if(!samplers.contains(samplerDescriptor.object)) { samplerNode = new RDTreeWidgetItem(makeSampler(QString(), samplerDescriptor)); if(!filledSlot) setEmptyRow(samplerNode); CombinedSamplerData sampData; sampData.node = samplerNode; samplerNode->setTag(QVariant::fromValue(sampData)); samplers.insert(samplerDescriptor.object, samplerNode); } { RDTreeWidgetItem *combinedSamp = m_CombinedImageSamplers[node] = samplers[samplerDescriptor.object]; CombinedSamplerData sampData = combinedSamp->tag().value(); sampData.images.push_back(node); combinedSamp->setTag(QVariant::fromValue(sampData)); } } } } } if(tex) { // for rows with view details we can't highlight used combined samplers, so instead we put // it in the tooltip for that row and remove it from the m_CombinedImageSamplers list. QString samplerString = used.access.type == DescriptorType::ImageSampler ? tr("Image combined with sampler %1\n").arg(m_Ctx.GetResourceName(descriptor.secondary)) : QString(); bool hasViewDetails = setViewDetails(node, descriptor, tex, samplerString); if(hasViewDetails) { node->setText( 4, tr("%1 viewed by %2").arg(ToQStr(descriptor.resource)).arg(ToQStr(descriptor.view))); if(used.access.type == DescriptorType::ImageSampler) { RDTreeWidgetItem *combinedSamp = m_CombinedImageSamplers[node]; if(combinedSamp) { CombinedSamplerData sampData = combinedSamp->tag().value(); sampData.images.removeOne(node); combinedSamp->setTag(QVariant::fromValue(sampData)); m_CombinedImageSamplers.remove(node); } } } } else if(buf) { setViewDetails(node, descriptor, buf, dynamicOffset); } resources->addTopLevelItem(node); if(samplerNode) resources->addTopLevelItem(samplerNode); } } void VulkanPipelineStateViewer::addConstantBlockRow(const ConstantBlock *cblock, const UsedDescriptor &used, uint32_t dynamicOffset, RDTreeWidget *ubos) { const Descriptor &descriptor = used.descriptor; VulkanCBufferTag tag(used.access.index, used.access.arrayElement, dynamicOffset); bool filledSlot = (descriptor.resource != ResourceId()); // Vulkan does not report unused elements at all because we enumerate exclusively from the // perspective of which descriptors are used bool usedSlot = true; if(showNode(usedSlot, filledSlot)) { QString slotname; if(used.access.index == DescriptorAccess::NoShaderBinding) { slotname = m_Locations[{used.access.descriptorStore, used.access.byteOffset}].logicalBindName; slotname += QFormatStr("[%1]").arg(used.access.arrayElement); } else if(cblock) { if(IsPushSet(used.access.stage, used.access.descriptorStore)) slotname = tr("Push "); slotname += QFormatStr("Set %1, %2").arg(cblock->fixedBindSetOrSpace).arg(cblock->fixedBindNumber); if(!cblock->name.empty()) slotname += lit(": ") + cblock->name; if(cblock->bindArraySize > 1) slotname += QFormatStr("[%1]").arg(used.access.arrayElement); } uint64_t bufferByteSize = descriptor.byteSize; int numvars = cblock != NULL ? cblock->variables.count() : 0; uint64_t declaredByteSize = cblock != NULL ? cblock->byteSize : 0; QString byteRange = lit("-"); uint64_t byteOffset = descriptor.byteOffset + dynamicOffset; { BufferDescription *buf = m_Ctx.GetBuffer(descriptor.resource); if(buf && bufferByteSize == UINT64_MAX) bufferByteSize = buf->length - byteOffset; byteRange = formatByteRange(buf, descriptor, dynamicOffset); } QString sizestr; QVariant name = descriptor.resource; // push constants or specialization constants if(cblock && !cblock->bufferBacked) { slotname = cblock->name; if(cblock->compileConstants) { name = tr("Specialization constants"); byteRange = QString(); } else { name = tr("Push constants"); uint32_t minOffset = getMinOffset(cblock->variables); if(minOffset == ~0U) minOffset = 0; byteRange = QFormatStr("%1 - %2 bytes") .arg(Formatter::HumanFormat(byteOffset + minOffset, Formatter::OffsetSize)) .arg(Formatter::HumanFormat(byteOffset + cblock->byteSize, Formatter::OffsetSize)); if(byteOffset + descriptor.byteSize > m_Ctx.CurVulkanPipelineState()->pushconsts.size()) { filledSlot = false; byteRange += tr(", only %1 bytes pushed") .arg(Formatter::HumanFormat(m_Ctx.CurVulkanPipelineState()->pushconsts.size(), Formatter::OffsetSize)); } } sizestr = tr("%1 Variables").arg(numvars); } else { if(descriptor.flags & DescriptorFlags::InlineData) { name = tr("Inline block"); byteRange = tr("%1 bytes").arg(Formatter::HumanFormat(bufferByteSize, Formatter::OffsetSize)); } if(bufferByteSize == declaredByteSize) sizestr = tr("%1 Variables, %2 bytes") .arg(numvars) .arg(Formatter::HumanFormat(bufferByteSize, Formatter::OffsetSize)); else sizestr = tr("%1 Variables, %2 bytes needed, %3 provided") .arg(numvars) .arg(Formatter::HumanFormat(declaredByteSize, Formatter::OffsetSize)) .arg(Formatter::HumanFormat(bufferByteSize, Formatter::OffsetSize)); if(bufferByteSize < declaredByteSize) filledSlot = false; } RDTreeWidgetItem *node = new RDTreeWidgetItem({slotname, name, byteRange, sizestr, QString()}); node->setTag(QVariant::fromValue(tag)); if(!filledSlot) setEmptyRow(node); if(!usedSlot) setInactiveRow(node); ubos->addTopLevelItem(node); } } void VulkanPipelineStateViewer::setShaderState(const VKPipe::Pipeline &pipe, const VKPipe::Shader &stage, RDLabel *pipeline, RDLabel *shader, RDLabel *shaderDebug, RDLabel *pipeLayout, RDTreeWidget *descSets) { const ShaderReflection *shaderDetails = stage.reflection; if(stage.shaderObject) { pipeline->hide(); shader->setText(ToQStr(stage.resourceId)); } else { pipeline->show(); pipeline->setText(ToQStr(pipe.pipelineResourceId)); shader->setText(ToQStr(stage.resourceId)); } if(shaderDetails != NULL) { QString entryFunc = shaderDetails->entryPoint; QString shText = entryFunc + lit("()"); const ShaderDebugInfo &dbg = shaderDetails->debugInfo; int entryFile = qMax(0, dbg.entryLocation.fileIndex); if(!dbg.files.isEmpty()) { QString filename = QFileInfo(dbg.files[entryFile].filename).fileName(); TruncateStringFromEnd(filename); shText += lit(" - ") + filename; } if(stage.requiredSubgroupSize != 0) shText += tr(" (Subgroup size %1)").arg(stage.requiredSubgroupSize); shaderDebug->show(); shaderDebug->setText(shText); } else { shaderDebug->hide(); } if(pipe.pipelineComputeLayoutResourceId != ResourceId()) { pipeLayout->setText(tr("Pipeline Layout: %1").arg(ToQStr(pipe.pipelineComputeLayoutResourceId))); } else if(pipe.pipelinePreRastLayoutResourceId == pipe.pipelineFragmentLayoutResourceId) { pipeLayout->setText(tr("Pipeline Layout: %1").arg(ToQStr(pipe.pipelineFragmentLayoutResourceId))); } else { pipeLayout->setText(tr("Pipeline Layouts: %1 and %2") .arg(ToQStr(pipe.pipelinePreRastLayoutResourceId)) .arg(ToQStr(pipe.pipelineFragmentLayoutResourceId))); } bool descBuf = false; for(uint32_t i = 0; i < pipe.descriptorSets.size(); i++) { if(pipe.descriptorSets[i].descriptorBufferIndex >= 0) { descBuf = true; break; } } if(descBuf) { descSets->setHeaderText(2, tr(" Buffer")); descSets->header()->showSection(3); // buffer offset descSets->header()->showSection(4); // set offset descSets->header()->showSection(5); // type } else { descSets->setHeaderText(2, tr("Set")); descSets->header()->hideSection(3); // buffer offset descSets->header()->hideSection(4); // set offset descSets->header()->hideSection(5); // type } descSets->clear(); if(descBuf) { for(uint32_t i = 0; i < pipe.descriptorSets.size(); i++) { RDTreeWidgetItem *item = new RDTreeWidgetItem( {i, pipe.descriptorSets[i].layoutResourceId, QString(), QString(), qulonglong(pipe.descriptorSets[i].descriptorBufferByteOffset), QString(), QString()}); item->setTag(i); if(pipe.descriptorSets[i].descriptorBufferEmbeddedSamplers) { item->setText(2, tr("Embedded Samplers")); item->setText(4, QString()); item->setText(5, tr("Immutable Samplers")); } else if(pipe.descriptorSets[i].pushDescriptor) { item->setText(2, tr("Bufferless")); for(size_t b = 0; b < pipe.descriptorBuffers.size(); b++) if(pipe.descriptorBuffers[b].pushBuffer != ResourceId()) item->setText(2, ToQStr(pipe.descriptorBuffers[b].pushBuffer)); item->setText(4, QString()); item->setText(5, tr("Push Descriptors")); } else if(pipe.descriptorSets[i].descriptorBufferIndex < 0 || pipe.descriptorSets[i].descriptorBufferIndex >= pipe.descriptorBuffers.count()) { setEmptyRow(item); } else { const VKPipe::DescriptorBuffer &buf = pipe.descriptorBuffers[pipe.descriptorSets[i].descriptorBufferIndex]; item->setText(2, buf.buffer); item->setText(3, qulonglong(buf.offset)); QString type = tr("None"); if(buf.resourceBuffer && buf.samplerBuffer) type = tr("Resources+Samplers"); else if(buf.resourceBuffer) type = tr("Resources"); else if(buf.samplerBuffer) type = tr("Samplers"); item->setText(5, type); } descSets->addTopLevelItem(item); } } else { for(uint32_t i = 0; i < pipe.descriptorSets.size(); i++) { RDTreeWidgetItem *item = new RDTreeWidgetItem({i, pipe.descriptorSets[i].layoutResourceId, pipe.descriptorSets[i].descriptorSetResourceId, QString(), QString(), QString(), QString()}); item->setTag(i); descSets->addTopLevelItem(item); } } } rdcpair GetSetAndBind(const UsedDescriptor &a, const ShaderReflection *refl) { rdcpair ret = {~0U, ~0U}; if(!refl) return ret; DescriptorCategory category = CategoryForDescriptorType(a.access.type); if(category == DescriptorCategory::ConstantBlock) { ret = { refl->constantBlocks[a.access.index].fixedBindSetOrSpace, refl->constantBlocks[a.access.index].fixedBindNumber, }; } else if(category == DescriptorCategory::ReadOnlyResource) { ret = { refl->readOnlyResources[a.access.index].fixedBindSetOrSpace, refl->readOnlyResources[a.access.index].fixedBindNumber, }; } else if(category == DescriptorCategory::ReadWriteResource) { ret = { refl->readWriteResources[a.access.index].fixedBindSetOrSpace, refl->readWriteResources[a.access.index].fixedBindNumber, }; } else if(category == DescriptorCategory::Sampler) { ret = { refl->samplers[a.access.index].fixedBindSetOrSpace, refl->samplers[a.access.index].fixedBindNumber, }; } return ret; } void VulkanPipelineStateViewer::setState() { if(!m_Ctx.IsCaptureLoaded()) { clearState(); return; } // cache latest state of these checkboxes m_ShowUnused = ui->showUnused->isChecked(); m_ShowEmpty = ui->showEmpty->isChecked(); m_CombinedImageSamplers.clear(); const VKPipe::State &state = *m_Ctx.CurVulkanPipelineState(); const ActionDescription *action = m_Ctx.CurAction(); bool showUnused = ui->showUnused->isChecked(); bool showEmpty = ui->showEmpty->isChecked(); const QPixmap &tick = Pixmaps::tick(this); const QPixmap &cross = Pixmaps::cross(this); bool usedBindings[128] = {}; // highlight the appropriate stages in the flowchart if(action == NULL) { QList allOn; for(int i = 0; i < ui->pipeFlow->stageNames().count(); i++) allOn.append(true); ui->pipeFlow->setStagesEnabled(allOn); } else if(action->flags & ActionFlags::Dispatch) { QList computeOnly; for(int i = 0; i < ui->pipeFlow->stageNames().count(); i++) computeOnly.append(false); computeOnly.back() = true; ui->pipeFlow->setStagesEnabled(computeOnly); } else if(action->flags & ActionFlags::MeshDispatch) { setNewMeshPipeFlow(); ui->pipeFlow->setStagesEnabled( {state.taskShader.resourceId != ResourceId(), true, true, true, true, false}); } else { bool xfbActive = !state.transformFeedback.buffers.isEmpty(); bool raster = true; if(state.rasterizer.rasterizerDiscardEnable) { raster = false; } setOldMeshPipeFlow(); if(state.geometryShader.resourceId == ResourceId() && xfbActive) { ui->pipeFlow->setStageName(4, lit("XFB"), tr("Transform Feedback")); } else { ui->pipeFlow->setStageName(4, lit("GS"), tr("Geometry Shader")); } ui->pipeFlow->setStagesEnabled( {true, true, state.tessControlShader.resourceId != ResourceId(), state.tessEvalShader.resourceId != ResourceId(), state.geometryShader.resourceId != ResourceId() || xfbActive, raster, raster && state.fragmentShader.resourceId != ResourceId(), raster, false}); } //////////////////////////////////////////////// // Vertex Input int vs = 0; if(m_MeshPipe) { setShaderState(state.graphics, state.taskShader, ui->tsPipeline, ui->tsShader, ui->tsShaderDebug, ui->tsPipeLayout, ui->tsDescSets); setShaderState(state.graphics, state.meshShader, ui->msPipeline, ui->msShader, ui->msShaderDebug, ui->msPipeLayout, ui->msDescSets); if(state.meshShader.reflection) ui->msTopology->setText(ToQStr(state.meshShader.reflection->outputTopology)); else ui->msTopology->setText(QString()); } else { vs = ui->viAttrs->verticalScrollBar()->value(); ui->viAttrs->beginUpdate(); ui->viAttrs->clear(); { int i = 0; for(const VKPipe::VertexAttribute &a : state.vertexInput.attributes) { bool usedSlot = false; QString name = tr("Attribute %1").arg(i); if(state.vertexShader.resourceId != ResourceId()) { for(const SigParameter &sigParam : state.vertexShader.reflection->inputSignature) { if(sigParam.regIndex == a.location && sigParam.systemValue == ShaderBuiltin::Undefined) { name = sigParam.varName; usedSlot = true; break; } } } if(showNode(usedSlot, /*filledSlot*/ true)) { RDTreeWidgetItem *node = new RDTreeWidgetItem( {i, name, a.location, a.binding, a.format.Name(), a.byteOffset, QString()}); node->setTag(i); usedBindings[a.binding] = true; if(!usedSlot) setInactiveRow(node); ui->viAttrs->addTopLevelItem(node); } i++; } } ui->viAttrs->clearSelection(); ui->viAttrs->endUpdate(); ui->viAttrs->verticalScrollBar()->setValue(vs); m_BindNodes.clear(); m_VBNodes.clear(); m_EmptyNodes.clear(); int numCPs = PatchList_Count(state.inputAssembly.topology); if(numCPs > 0) { ui->topology->setText(tr("PatchList (%1 Control Points)").arg(numCPs)); } else { ui->topology->setText(ToQStr(state.inputAssembly.topology)); } m_Common.setTopologyDiagram(ui->topologyDiagram, state.inputAssembly.topology); ui->primRestart->setVisible(state.inputAssembly.primitiveRestartEnable); vs = ui->viBuffers->verticalScrollBar()->value(); ui->viBuffers->beginUpdate(); ui->viBuffers->clear(); bool ibufferUsed = action != NULL && (action->flags & ActionFlags::Indexed); if(state.inputAssembly.indexBuffer.resourceId != ResourceId()) { if(ibufferUsed || showUnused) { uint64_t length = 1; if(!ibufferUsed) length = 0; BufferDescription *buf = m_Ctx.GetBuffer(state.inputAssembly.indexBuffer.resourceId); if(buf) length = qMin(state.inputAssembly.indexBuffer.byteSize, buf->length); RDTreeWidgetItem *node = new RDTreeWidgetItem( {tr("Index"), state.inputAssembly.indexBuffer.resourceId, tr("Index"), lit("-"), (qulonglong)state.inputAssembly.indexBuffer.byteOffset, (qulonglong)state.inputAssembly.indexBuffer.byteStride, (qulonglong)length, QString()}); QString iformat; if(state.inputAssembly.indexBuffer.byteStride == 1) iformat = lit("ubyte"); else if(state.inputAssembly.indexBuffer.byteStride == 2) iformat = lit("ushort"); else if(state.inputAssembly.indexBuffer.byteStride == 4) iformat = lit("uint"); iformat += lit(" indices[%1]").arg(RENDERDOC_NumVerticesPerPrimitive(state.inputAssembly.topology)); node->setTag(QVariant::fromValue(VulkanVBIBTag( state.inputAssembly.indexBuffer.resourceId, state.inputAssembly.indexBuffer.byteOffset + (action ? action->indexOffset * state.inputAssembly.indexBuffer.byteStride : 0), iformat))); if(!ibufferUsed) setInactiveRow(node); if(state.inputAssembly.indexBuffer.resourceId == ResourceId()) { setEmptyRow(node); m_EmptyNodes.push_back(node); } ui->viBuffers->addTopLevelItem(node); } } else { if(ibufferUsed || showEmpty) { RDTreeWidgetItem *node = new RDTreeWidgetItem({tr("Index"), ResourceId(), tr("Index"), lit("-"), lit("-"), lit("-"), lit("-"), QString()}); QString iformat; if(state.inputAssembly.indexBuffer.byteStride == 1) iformat = lit("ubyte"); else if(state.inputAssembly.indexBuffer.byteStride == 2) iformat = lit("ushort"); else if(state.inputAssembly.indexBuffer.byteStride == 4) iformat = lit("uint"); iformat += lit(" indices[%1]").arg(RENDERDOC_NumVerticesPerPrimitive(state.inputAssembly.topology)); node->setTag(QVariant::fromValue(VulkanVBIBTag( state.inputAssembly.indexBuffer.resourceId, state.inputAssembly.indexBuffer.byteOffset + (action ? action->indexOffset * state.inputAssembly.indexBuffer.byteStride : 0), iformat))); setEmptyRow(node); m_EmptyNodes.push_back(node); if(!ibufferUsed) setInactiveRow(node); ui->viBuffers->addTopLevelItem(node); } } { int i = 0; for(; i < qMax(state.vertexInput.vertexBuffers.count(), state.vertexInput.bindings.count()); i++) { const VKPipe::VertexBuffer *vbuff = (i < state.vertexInput.vertexBuffers.count() ? &state.vertexInput.vertexBuffers[i] : NULL); const VKPipe::VertexBinding *bind = NULL; for(int b = 0; b < state.vertexInput.bindings.count(); b++) { if(state.vertexInput.bindings[b].vertexBufferBinding == (uint32_t)i) bind = &state.vertexInput.bindings[b]; } bool filledSlot = ((vbuff != NULL && vbuff->resourceId != ResourceId()) || bind != NULL); bool usedSlot = (usedBindings[i]); if(showNode(usedSlot, filledSlot)) { QString rate = lit("-"); uint64_t length = 1; uint64_t offset = 0; uint32_t stride = 0; uint32_t divisor = 1; if(vbuff != NULL) { offset = vbuff->byteOffset; stride = vbuff->byteStride; length = vbuff->byteSize; BufferDescription *buf = m_Ctx.GetBuffer(vbuff->resourceId); if(buf && length >= ULONG_MAX) length = buf->length; } if(bind != NULL) { rate = bind->perInstance ? tr("Instance") : tr("Vertex"); if(bind->perInstance) divisor = bind->instanceDivisor; } else { rate += tr("No Binding"); } RDTreeWidgetItem *node = NULL; if(filledSlot) node = new RDTreeWidgetItem({i, vbuff->resourceId, rate, divisor, (qulonglong)offset, stride, (qulonglong)length, QString()}); else node = new RDTreeWidgetItem( {i, tr("No Binding"), lit("-"), lit("-"), lit("-"), lit("-"), lit("-"), QString()}); node->setTag(QVariant::fromValue(VulkanVBIBTag( vbuff != NULL ? vbuff->resourceId : ResourceId(), vbuff != NULL ? vbuff->byteOffset : 0, m_Common.GetVBufferFormatString(i)))); if(!filledSlot || bind == NULL || vbuff == NULL || vbuff->resourceId == ResourceId()) { setEmptyRow(node); m_EmptyNodes.push_back(node); } if(!usedSlot) setInactiveRow(node); m_VBNodes.push_back(node); ui->viBuffers->addTopLevelItem(node); } else { m_VBNodes.push_back(NULL); } } for(; i < (int)ARRAY_COUNT(usedBindings); i++) { if(usedBindings[i]) { RDTreeWidgetItem *node = new RDTreeWidgetItem( {i, tr("No Binding"), lit("-"), lit("-"), lit("-"), lit("-"), lit("-"), QString()}); node->setTag(QVariant::fromValue(VulkanVBIBTag(ResourceId(), 0))); setEmptyRow(node); m_EmptyNodes.push_back(node); setInactiveRow(node); ui->viBuffers->addTopLevelItem(node); m_VBNodes.push_back(node); } else { m_VBNodes.push_back(NULL); } } } ui->viBuffers->clearSelection(); ui->viBuffers->endUpdate(); ui->viBuffers->verticalScrollBar()->setValue(vs); setShaderState(state.graphics, state.vertexShader, ui->vsPipeline, ui->vsShader, ui->vsShaderDebug, ui->vsPipeLayout, ui->vsDescSets); setShaderState(state.graphics, state.geometryShader, ui->gsPipeline, ui->gsShader, ui->gsShaderDebug, ui->gsPipeLayout, ui->gsDescSets); setShaderState(state.graphics, state.tessControlShader, ui->tcsPipeline, ui->tcsShader, ui->tcsShaderDebug, ui->tcsPipeLayout, ui->tcsDescSets); setShaderState(state.graphics, state.tessEvalShader, ui->tesPipeline, ui->tesShader, ui->tesShaderDebug, ui->tesPipeLayout, ui->tesDescSets); } setShaderState(state.graphics, state.fragmentShader, ui->fsPipeline, ui->fsShader, ui->fsShaderDebug, ui->fsPipeLayout, ui->fsDescSets); setShaderState(state.compute, state.computeShader, ui->csPipeline, ui->csShader, ui->csShaderDebug, ui->csPipeLayout, ui->csDescSets); // fill in descriptor access { RDTreeWidget *resources[] = { ui->vsResources, ui->tcsResources, ui->tesResources, ui->gsResources, ui->fsResources, ui->csResources, ui->tsResources, ui->msResources, }; RDTreeWidget *ubos[] = { ui->vsUBOs, ui->tcsUBOs, ui->tesUBOs, ui->gsUBOs, ui->fsUBOs, ui->csUBOs, ui->tsUBOs, ui->msUBOs, }; ScopedTreeUpdater restorers[] = { ui->vsResources, ui->tcsResources, ui->tesResources, ui->gsResources, ui->fsResources, ui->csResources, ui->tsResources, ui->msResources, ui->vsUBOs, ui->tcsUBOs, ui->tesUBOs, ui->gsUBOs, ui->fsUBOs, ui->csUBOs, ui->tsUBOs, ui->msUBOs, }; // samplers we only deduplicate within a stage QMap samplers[NumShaderStages]; const ShaderReflection *shaderRefls[NumShaderStages]; for(ShaderStage stage : values()) shaderRefls[(uint32_t)stage] = m_Ctx.CurPipelineState().GetShaderReflection(stage); rdcarray descriptors = m_Ctx.CurPipelineState().GetAllUsedDescriptors(); const VKPipe::Pipeline &pipeline = (action && (action->flags & ActionFlags::Dispatch)) ? state.compute : state.graphics; QMap, uint32_t> dynamicOffsets; for(const VKPipe::DescriptorSet &set : pipeline.descriptorSets) { for(const VKPipe::DynamicOffset &offs : set.dynamicOffsets) { dynamicOffsets[{set.descriptorSetResourceId, offs.descriptorByteOffset}] = offs.dynamicBufferByteOffset; } } std::sort(descriptors.begin(), descriptors.end(), [&shaderRefls](const UsedDescriptor &a, const UsedDescriptor &b) { // sort stages together, not really needed but keeps the code below simple if(a.access.stage != b.access.stage) return a.access.stage < b.access.stage; const ShaderReflection *refl = shaderRefls[(uint32_t)a.access.stage]; rdcpair aBind = GetSetAndBind(a, refl); rdcpair bBind = GetSetAndBind(b, refl); // non-set associated things (specialisation constants, push constants, etc) to the end if(a.access.type == DescriptorType::ConstantBuffer && !refl->constantBlocks[a.access.index].bufferBacked) aBind.first = ~0U; if(b.access.type == DescriptorType::ConstantBuffer && !refl->constantBlocks[b.access.index].bufferBacked) bBind.first = ~0U; // most things will have a set and binding, that sorting is enough if(aBind.first != bBind.first) return aBind.first < bBind.first; if(aBind.second != bBind.second) return aBind.second < bBind.second; // for non-sets, sort by interface index return a.access.index < b.access.index; }); for(const UsedDescriptor &used : descriptors) { if(used.access.type == DescriptorType::Unknown || used.access.stage == ShaderStage::Count) continue; const ShaderReflection *refl = shaderRefls[(uint32_t)used.access.stage]; uint32_t dynamicOffset = 0; auto dynIt = dynamicOffsets.find({used.access.descriptorStore, (uint64_t)used.access.byteOffset}); if(dynIt != dynamicOffsets.end()) dynamicOffset = *dynIt; if(IsConstantBlockDescriptor(used.access.type)) { const ConstantBlock *shaderBind = NULL; if(refl && used.access.index < refl->constantBlocks.size()) shaderBind = &refl->constantBlocks[used.access.index]; addConstantBlockRow(shaderBind, used, dynamicOffset, ubos[(uint32_t)used.access.stage]); } else { const bool ro = IsReadOnlyDescriptor(used.access.type); const ShaderResource *shaderRes = NULL; const ShaderSampler *shaderSamp = NULL; if(IsSamplerDescriptor(used.access.type)) { if(refl && used.access.index < refl->samplers.size()) shaderSamp = &refl->samplers[used.access.index]; } else if(IsReadOnlyDescriptor(used.access.type)) { if(refl && used.access.index < refl->readOnlyResources.size()) shaderRes = &refl->readOnlyResources[used.access.index]; } else { if(refl && used.access.index < refl->readWriteResources.size()) shaderRes = &refl->readWriteResources[used.access.index]; } addResourceRow(shaderRes, shaderSamp, used, dynamicOffset, resources[(uint32_t)used.access.stage], samplers[(uint32_t)used.access.stage]); } } } QToolButton *shaderButtons[] = { // view buttons ui->tsShaderViewButton, ui->msShaderViewButton, ui->vsShaderViewButton, ui->tcsShaderViewButton, ui->tesShaderViewButton, ui->gsShaderViewButton, ui->fsShaderViewButton, ui->csShaderViewButton, // edit buttons ui->tsShaderEditButton, ui->msShaderEditButton, ui->vsShaderEditButton, ui->tcsShaderEditButton, ui->tesShaderEditButton, ui->gsShaderEditButton, ui->fsShaderEditButton, ui->csShaderEditButton, // save buttons ui->tsShaderSaveButton, ui->msShaderSaveButton, ui->vsShaderSaveButton, ui->tcsShaderSaveButton, ui->tesShaderSaveButton, ui->gsShaderSaveButton, ui->fsShaderSaveButton, ui->csShaderSaveButton, }; for(QToolButton *b : shaderButtons) { const VKPipe::Shader *stage = stageForSender(b); if(stage == NULL || stage->resourceId == ResourceId()) continue; ResourceId pipe = stage->stage == ShaderStage::Compute ? state.compute.pipelineResourceId : state.graphics.pipelineResourceId; b->setEnabled(stage->reflection && (pipe != ResourceId() || stage->shaderObject)); m_Common.SetupShaderEditButton(b, pipe, stage->resourceId, stage->reflection); } QToolButton *messageButtons[] = { ui->vsShaderMessagesButton, ui->tcsShaderMessagesButton, ui->tesShaderMessagesButton, ui->gsShaderMessagesButton, ui->fsShaderMessagesButton, ui->csShaderMessagesButton, ui->tsShaderMessagesButton, ui->msShaderMessagesButton, }; int numMessages[NumShaderStages] = {}; for(const ShaderMessage &msg : state.shaderMessages) numMessages[(uint32_t)msg.stage]++; static_assert(ARRAY_COUNT(messageButtons) <= ARRAY_COUNT(numMessages), "More buttons than shader stages"); for(uint32_t i = 0; i < ARRAY_COUNT(messageButtons); i++) { messageButtons[i]->setVisible(numMessages[i] > 0); messageButtons[i]->setText(tr("%n Message(s)", "", numMessages[i])); } bool xfbSet = false; vs = ui->xfbBuffers->verticalScrollBar()->value(); ui->xfbBuffers->beginUpdate(); ui->xfbBuffers->clear(); for(int i = 0; i < state.transformFeedback.buffers.count(); i++) { const VKPipe::XFBBuffer &s = state.transformFeedback.buffers[i]; bool filledSlot = (s.bufferResourceId != ResourceId()); bool usedSlot = (s.active); if(showNode(usedSlot, filledSlot)) { qulonglong length = s.byteSize; BufferDescription *buf = m_Ctx.GetBuffer(s.bufferResourceId); if(buf && length == UINT64_MAX) length = buf->length - s.byteOffset; RDTreeWidgetItem *node = new RDTreeWidgetItem({ i, s.active ? tr("Active") : tr("Inactive"), s.bufferResourceId, Formatter::HumanFormat(s.byteOffset, Formatter::OffsetSize), Formatter::HumanFormat(length, Formatter::OffsetSize), s.counterBufferResourceId, Formatter::HumanFormat(s.counterBufferOffset, Formatter::OffsetSize), QString(), }); node->setTag(QVariant::fromValue(VulkanBufferTag(s.bufferResourceId, s.byteOffset, length))); if(!filledSlot) setEmptyRow(node); if(!usedSlot) setInactiveRow(node); xfbSet = true; ui->xfbBuffers->addTopLevelItem(node); } } ui->xfbBuffers->verticalScrollBar()->setValue(vs); ui->xfbBuffers->clearSelection(); ui->xfbBuffers->endUpdate(); ui->xfbBuffers->setVisible(xfbSet); ui->xfbGroup->setVisible(xfbSet); //////////////////////////////////////////////// // Rasterizer vs = ui->discards->verticalScrollBar()->value(); ui->discards->beginUpdate(); ui->discards->clear(); { int i = 0; for(const VKPipe::RenderArea &v : state.viewportScissor.discardRectangles) { RDTreeWidgetItem *node = new RDTreeWidgetItem({i, v.x, v.y, v.width, v.height}); ui->discards->addTopLevelItem(node); if(v.width == 0 || v.height == 0) setEmptyRow(node); i++; } } ui->discards->verticalScrollBar()->setValue(vs); ui->discards->clearSelection(); ui->discards->endUpdate(); ui->discardMode->setText(state.viewportScissor.discardRectanglesExclusive ? tr("Exclusive") : tr("Inclusive")); ui->discardGroup->setVisible(!state.viewportScissor.discardRectanglesExclusive || !state.viewportScissor.discardRectangles.isEmpty()); vs = ui->viewports->verticalScrollBar()->value(); ui->viewports->beginUpdate(); ui->viewports->clear(); int vs2 = ui->scissors->verticalScrollBar()->value(); ui->scissors->beginUpdate(); ui->scissors->clear(); if(state.currentPass.renderpass.resourceId != ResourceId() || state.currentPass.renderpass.dynamic) { ui->scissors->addTopLevelItem(new RDTreeWidgetItem( {tr("Render Area"), state.currentPass.renderArea.x, state.currentPass.renderArea.y, state.currentPass.renderArea.width, state.currentPass.renderArea.height})); } { const QString ndcDepthRange = state.viewportScissor.depthNegativeOneToOne ? lit("[-1, 1]") : lit("[0, 1]"); int i = 0; for(const VKPipe::ViewportScissor &v : state.viewportScissor.viewportScissors) { RDTreeWidgetItem *node = new RDTreeWidgetItem({i, v.vp.x, v.vp.y, v.vp.width, v.vp.height, v.vp.minDepth, v.vp.maxDepth, ndcDepthRange}); ui->viewports->addTopLevelItem(node); if(v.vp.width == 0 || v.vp.height == 0) setEmptyRow(node); node = new RDTreeWidgetItem({i, v.scissor.x, v.scissor.y, v.scissor.width, v.scissor.height}); ui->scissors->addTopLevelItem(node); if(v.scissor.width == 0 || v.scissor.height == 0) setEmptyRow(node); i++; } } ui->viewports->verticalScrollBar()->setValue(vs); ui->viewports->clearSelection(); ui->scissors->clearSelection(); ui->scissors->verticalScrollBar()->setValue(vs2); ui->viewports->endUpdate(); ui->scissors->endUpdate(); ui->fillMode->setText(ToQStr(state.rasterizer.fillMode)); ui->cullMode->setText(ToQStr(state.rasterizer.cullMode)); ui->frontCCW->setPixmap(state.rasterizer.frontCCW ? tick : cross); if(state.rasterizer.depthBiasEnable) { ui->depthBias->setPixmap(QPixmap()); ui->depthBiasClamp->setPixmap(QPixmap()); ui->slopeScaledBias->setPixmap(QPixmap()); ui->depthBias->setText(Formatter::Format(state.rasterizer.depthBias)); ui->depthBiasClamp->setText(Formatter::Format(state.rasterizer.depthBiasClamp)); ui->slopeScaledBias->setText(Formatter::Format(state.rasterizer.slopeScaledDepthBias)); } else { ui->depthBias->setText(QString()); ui->depthBiasClamp->setText(QString()); ui->slopeScaledBias->setText(QString()); ui->depthBias->setPixmap(cross); ui->depthBiasClamp->setPixmap(cross); ui->slopeScaledBias->setPixmap(cross); } ui->depthClamp->setPixmap(state.rasterizer.depthClampEnable ? tick : cross); ui->depthClip->setPixmap(state.rasterizer.depthClipEnable ? tick : cross); ui->rasterizerDiscard->setPixmap(state.rasterizer.rasterizerDiscardEnable ? tick : cross); ui->lineWidth->setText(Formatter::Format(state.rasterizer.lineWidth)); QString conservRaster = ToQStr(state.rasterizer.conservativeRasterization); if(state.rasterizer.conservativeRasterization == ConservativeRaster::Overestimate && state.rasterizer.extraPrimitiveOverestimationSize > 0.0f) conservRaster += QFormatStr(" (+%1)").arg(state.rasterizer.extraPrimitiveOverestimationSize); ui->conservativeRaster->setText(conservRaster); if(state.rasterizer.lineStippleFactor == 0) { ui->stippleFactor->setText(QString()); ui->stippleFactor->setPixmap(cross); ui->stipplePattern->setText(QString()); ui->stipplePattern->setPixmap(cross); } else { ui->stippleFactor->setPixmap(QPixmap()); ui->stippleFactor->setText(ToQStr(state.rasterizer.lineStippleFactor)); ui->stipplePattern->setPixmap(QPixmap()); ui->stipplePattern->setText(QString::number(state.rasterizer.lineStipplePattern, 2)); } ui->pipelineShadingRate->setText(QFormatStr("%1x%2") .arg(state.rasterizer.pipelineShadingRate.first) .arg(state.rasterizer.pipelineShadingRate.second)); ui->shadingRateCombiners->setText( QFormatStr("%1, %2") .arg(ToQStr(state.rasterizer.shadingRateCombiners.first, GraphicsAPI::Vulkan)) .arg(ToQStr(state.rasterizer.shadingRateCombiners.second, GraphicsAPI::Vulkan))); ui->provokingVertex->setText(state.rasterizer.provokingVertexFirst ? tr("First") : tr("Last")); if(state.currentPass.renderpass.multiviews.isEmpty()) { ui->multiview->setText(tr("Disabled")); } else { QString views = tr("Views: "); for(int i = 0; i < state.currentPass.renderpass.multiviews.count(); i++) { if(i > 0) views += lit(", "); views += QString::number(state.currentPass.renderpass.multiviews[i]); } ui->multiview->setText(views); } ui->sampleCount->setText(QString::number(state.multisample.rasterSamples)); ui->sampleShading->setPixmap(state.multisample.sampleShadingEnable ? tick : cross); ui->minSampleShading->setText(Formatter::Format(state.multisample.minSampleShading)); ui->sampleMask->setText(Formatter::Format(state.multisample.sampleMask, true)); ui->alphaToOne->setPixmap(state.colorBlend.alphaToOneEnable ? tick : cross); ui->alphaToCoverage->setPixmap(state.colorBlend.alphaToCoverageEnable ? tick : cross); //////////////////////////////////////////////// // Conditional Rendering if(state.conditionalRendering.bufferId == ResourceId()) { ui->conditionalRenderingGroup->setVisible(false); ui->csConditionalRenderingGroup->setVisible(false); } else { ui->conditionalRenderingGroup->setVisible(true); ui->predicateBuffer->setText(QFormatStr("%1 (Byte Offset %2)") .arg(ToQStr(state.conditionalRendering.bufferId)) .arg(state.conditionalRendering.byteOffset)); ui->predicatePassing->setPixmap(state.conditionalRendering.isPassing ? tick : cross); ui->predicateInverted->setPixmap(state.conditionalRendering.isInverted ? tick : cross); ui->csConditionalRenderingGroup->setVisible(true); ui->csPredicateBuffer->setText(QFormatStr("%1 (Byte Offset %2)") .arg(ToQStr(state.conditionalRendering.bufferId)) .arg(state.conditionalRendering.byteOffset)); ui->csPredicatePassing->setPixmap(state.conditionalRendering.isPassing ? tick : cross); ui->csPredicateInverted->setPixmap(state.conditionalRendering.isInverted ? tick : cross); } //////////////////////////////////////////////// // Output Merger if(state.currentPass.renderpass.dynamic) { QString dynamic = tr("Dynamic", "Dynamic rendering renderpass name"); QString text = QFormatStr("Render Pass: %1").arg(dynamic); if(state.currentPass.renderpass.suspended) text += tr(" (Suspended)", "Dynamic rendering renderpass name"); ui->renderpass->setText(text); ui->framebuffer->setText(tr("Framebuffer: %1").arg(dynamic)); } else { QString text = QFormatStr("Render Pass: %1 (Subpass %2)") .arg(ToQStr(state.currentPass.renderpass.resourceId)) .arg(state.currentPass.renderpass.subpass); if(state.currentPass.renderpass.feedbackLoop) text += tr(" (Feedback Loop)"); ui->renderpass->setText(text); ui->framebuffer->setText( QFormatStr("Framebuffer: %1").arg(ToQStr(state.currentPass.framebuffer.resourceId))); } vs = ui->fbAttach->verticalScrollBar()->value(); ui->fbAttach->beginUpdate(); ui->fbAttach->clear(); vs2 = ui->blends->verticalScrollBar()->value(); ui->blends->beginUpdate(); ui->blends->clear(); { const VKPipe::Framebuffer &fb = state.currentPass.framebuffer; const VKPipe::RenderPass &rp = state.currentPass.renderpass; enum class AttType { Color, Resolve, Depth, DepthResolve, Density, ShadingRate }; struct AttachRef { int32_t fbIdx; int32_t localIdx; AttType type; }; rdcarray attachs; // iterate the attachments in logical order, checking each index into the framebuffer for(int c = 0; c < rp.colorAttachments.count(); c++) attachs.push_back({int32_t(rp.colorAttachments[c]), c, AttType::Color}); for(int c = 0; c < rp.resolveAttachments.count(); c++) attachs.push_back({int32_t(rp.resolveAttachments[c]), c, AttType::Resolve}); attachs.push_back({rp.depthstencilAttachment, 0, AttType::Depth}); attachs.push_back({rp.depthstencilResolveAttachment, 0, AttType::DepthResolve}); attachs.push_back({rp.fragmentDensityAttachment, 0, AttType::Density}); attachs.push_back({rp.shadingRateAttachment, 0, AttType::ShadingRate}); for(const AttachRef &a : attachs) { int32_t attIdx = a.fbIdx; // negative index means unused bool usedSlot = (attIdx >= 0); bool filledSlot = false; if(usedSlot && attIdx < fb.attachments.count()) filledSlot = fb.attachments[attIdx].resource != ResourceId(); if(showNode(usedSlot, filledSlot)) { QString slotname; if(a.type == AttType::Color) { slotname = QFormatStr("Color %1").arg(a.localIdx); // With dynamic rendering, the API references the framebuffer index everywhere, for // example when specifying blend state for attachments or with vkCmdClearAttachments. As // such, RenderDoc shows the same index in Color attachments (i.e. fbIdx == localIdx) to // avoid confusion, even when VK_KHR_dynamic_rendering_local_read maps these attachments // to different "locations" used by the shader. In that case, the mapped location is // shown besides the attachment index. uint32_t location = a.localIdx; if(a.fbIdx < rp.colorAttachmentLocations.count()) { location = rp.colorAttachmentLocations[a.fbIdx]; if(location == VKPipe::RenderPass::AttachmentUnused) { slotname += QFormatStr(" [disabled]"); } else { slotname += QFormatStr(" [location %1]").arg(location); } } if(state.fragmentShader.reflection != NULL) { const rdcarray &outSig = state.fragmentShader.reflection->outputSignature; for(int s = 0; s < outSig.count(); s++) { if(outSig[s].regIndex == location && (outSig[s].systemValue == ShaderBuiltin::Undefined || outSig[s].systemValue == ShaderBuiltin::ColorOutput)) { slotname += QFormatStr(": %1").arg(outSig[s].varName); } } } } else if(a.type == AttType::Resolve) { slotname = QFormatStr("Resolve %1").arg(a.localIdx); } else if(a.type == AttType::Depth) { slotname = lit("Depth/Stencil"); if(filledSlot) { const Descriptor &p = fb.attachments[attIdx]; slotname = lit("Depth"); if(p.format.type == ResourceFormatType::D16S8 || p.format.type == ResourceFormatType::D24S8 || p.format.type == ResourceFormatType::D32S8) slotname = lit("Depth/Stencil"); else if(p.format.type == ResourceFormatType::S8) slotname = lit("Stencil"); } } else if(a.type == AttType::DepthResolve) { slotname = lit("Depth/Stencil Resolve"); } else if(a.type == AttType::Density) { slotname = lit("Fragment Density Map"); } else if(a.type == AttType::ShadingRate) { slotname = lit("Fragment Shading Rate Map"); } RDTreeWidgetItem *node; if(filledSlot) { const Descriptor &p = fb.attachments[attIdx]; QString format; QString typeName; QString dimensions; QString samples; bool tooltipOffsets = false; if(p.resource != ResourceId()) { format = p.format.Name(); typeName = tr("Unknown"); } else { format = lit("-"); typeName = lit("-"); dimensions = lit("-"); samples = lit("-"); } TextureDescription *tex = m_Ctx.GetTexture(p.resource); if(tex) { dimensions += tr("%1x%2").arg(tex->width).arg(tex->height); if(tex->depth > 1) dimensions += tr("x%1").arg(tex->depth); if(tex->arraysize > 1) dimensions += tr("[%1]").arg(tex->arraysize); typeName = ToQStr(tex->type); } samples = getTextureRenderSamples(tex, state.currentPass.renderpass); if(p.swizzle.red != TextureSwizzle::Red || p.swizzle.green != TextureSwizzle::Green || p.swizzle.blue != TextureSwizzle::Blue || p.swizzle.alpha != TextureSwizzle::Alpha) { format += tr(" swizzle[%1%2%3%4]") .arg(ToQStr(p.swizzle.red)) .arg(ToQStr(p.swizzle.green)) .arg(ToQStr(p.swizzle.blue)) .arg(ToQStr(p.swizzle.alpha)); } rdcpair shadingRateTexelSize = {0, 0}; if(a.type == AttType::Density) { if(state.currentPass.renderpass.fragmentDensityOffsets.size() > 2) { tooltipOffsets = true; } else if(state.currentPass.renderpass.fragmentDensityOffsets.size() > 0) { dimensions += tr(" : offsets"); for(uint32_t j = 0; j < state.currentPass.renderpass.fragmentDensityOffsets.size(); j++) { const Offset &o = state.currentPass.renderpass.fragmentDensityOffsets[j]; if(j > 0) dimensions += tr(", "); dimensions += tr(" %1x%2").arg(o.x).arg(o.y); } } } else if(a.type == AttType::ShadingRate) { shadingRateTexelSize = state.currentPass.renderpass.shadingRateTexelSize; } QString resName = ToQStr(p.resource); if(shadingRateTexelSize.first > 0) resName += tr(" (%1x%2 texels)").arg(shadingRateTexelSize.first).arg(shadingRateTexelSize.second); // append if colour or depth/stencil feedback is allowed if(a.type == AttType::Color && state.currentPass.colorFeedbackAllowed) { resName += tr(" (Feedback)"); } else if(a.type == AttType::Depth && state.currentPass.depthFeedbackAllowed && state.currentPass.stencilFeedbackAllowed) { resName += tr(" (Feedback)"); } else if(a.type == AttType::Depth && (state.currentPass.depthFeedbackAllowed || state.currentPass.stencilFeedbackAllowed)) { // if only one of depth or stencil is allowed, display that specifically if(tex->format.type == ResourceFormatType::D16S8 || tex->format.type == ResourceFormatType::D24S8 || tex->format.type == ResourceFormatType::D32S8) { if(state.currentPass.depthFeedbackAllowed) resName += tr(" (Depth Feedback)"); else if(state.currentPass.stencilFeedbackAllowed) resName += tr(" (Depth Feedback)"); } else if(tex->format.type == ResourceFormatType::S8 && state.currentPass.stencilFeedbackAllowed) { resName += tr(" (Feedback)"); } // this case must be depth-only, since depth/stencil and stencil-only are covered above. else if(state.currentPass.depthFeedbackAllowed) { resName += tr(" (Feedback)"); } } node = new RDTreeWidgetItem( {slotname, resName, typeName, dimensions, format, samples, QString()}); if(tex) node->setTag(QVariant::fromValue(VulkanTextureTag(p.resource, p.format.compType))); if(p.resource == ResourceId()) setEmptyRow(node); else if(!usedSlot) setInactiveRow(node); bool hasViewDetails = setViewDetails( node, p, tex, QString(), a.type == AttType::Resolve || a.type == AttType::DepthResolve, tooltipOffsets); if(hasViewDetails) node->setText(1, tr("%1 viewed by %2").arg(ToQStr(p.resource)).arg(ToQStr(p.view))); } else { // special simple case for an attachment that's not used. No framebuffer to look up so // just display the name and empty contents. node = new RDTreeWidgetItem({slotname, usedSlot ? ToQStr(ResourceId()) : tr("Unused"), QString(), QString(), QString(), QString(), QString()}); setEmptyRow(node); } ui->fbAttach->addTopLevelItem(node); } } int i = 0; for(const ColorBlend &blend : state.colorBlend.blends) { bool usedSlot = (i < rp.colorAttachments.count() && rp.colorAttachments[i] < fb.attachments.size()); if(showNode(usedSlot, /*filledSlot*/ true)) { QString writemask = QFormatStr("%1%2%3%4") .arg((blend.writeMask & 0x1) == 0 ? lit("_") : lit("R")) .arg((blend.writeMask & 0x2) == 0 ? lit("_") : lit("G")) .arg((blend.writeMask & 0x4) == 0 ? lit("_") : lit("B")) .arg((blend.writeMask & 0x8) == 0 ? lit("_") : lit("A")); // With VK_KHR_dynamic_rendering_local_read, if a color attachment is mapped to // VK_ATTACHMENT_UNUSED, it is implicitly disabled. The Slot name in the "Render Pass" // pane already tags the attachment with [disabled], but for clarity the write mask is also // set to DISABLED here. if(i < rp.colorAttachmentLocations.count() && rp.colorAttachmentLocations[i] == VKPipe::RenderPass::AttachmentUnused) { writemask = lit("DISABLED"); } RDTreeWidgetItem *node = new RDTreeWidgetItem( {i, blend.enabled ? tr("True") : tr("False"), ToQStr(blend.colorBlend.source), ToQStr(blend.colorBlend.destination), ToQStr(blend.colorBlend.operation), ToQStr(blend.alphaBlend.source), ToQStr(blend.alphaBlend.destination), ToQStr(blend.alphaBlend.operation), writemask}); if(!usedSlot) setInactiveRow(node); ui->blends->addTopLevelItem(node); } i++; } } ui->fbAttach->clearSelection(); ui->fbAttach->endUpdate(); ui->fbAttach->verticalScrollBar()->setValue(vs); ui->blends->clearSelection(); ui->blends->endUpdate(); ui->blends->verticalScrollBar()->setValue(vs2); ui->blendFactor->setText(QFormatStr("%1, %2, %3, %4") .arg(state.colorBlend.blendFactor[0], 0, 'f', 2) .arg(state.colorBlend.blendFactor[1], 0, 'f', 2) .arg(state.colorBlend.blendFactor[2], 0, 'f', 2) .arg(state.colorBlend.blendFactor[3], 0, 'f', 2)); if(state.colorBlend.blends.count() > 0) ui->logicOp->setText(state.colorBlend.blends[0].logicOperationEnabled ? ToQStr(state.colorBlend.blends[0].logicOperation) : lit("-")); else ui->logicOp->setText(lit("-")); if(state.depthStencil.depthTestEnable) { ui->depthEnabled->setPixmap(tick); ui->depthFunc->setText(ToQStr(state.depthStencil.depthFunction)); ui->depthWrite->setPixmap(state.depthStencil.depthWriteEnable ? tick : cross); ui->depthWrite->setText(QString()); } else { ui->depthEnabled->setPixmap(cross); ui->depthFunc->setText(tr("Disabled")); ui->depthWrite->setPixmap(QPixmap()); ui->depthWrite->setText(tr("Disabled")); } if(state.depthStencil.depthBoundsEnable) { ui->depthBounds->setPixmap(QPixmap()); ui->depthBounds->setText(Formatter::Format(state.depthStencil.minDepthBounds) + lit("-") + Formatter::Format(state.depthStencil.maxDepthBounds)); } else { ui->depthBounds->setText(QString()); ui->depthBounds->setPixmap(cross); } ui->stencils->beginUpdate(); ui->stencils->clear(); if(state.depthStencil.stencilTestEnable) { ui->stencils->addTopLevelItem(new RDTreeWidgetItem({ tr("Front"), ToQStr(state.depthStencil.frontFace.function), ToQStr(state.depthStencil.frontFace.failOperation), ToQStr(state.depthStencil.frontFace.depthFailOperation), ToQStr(state.depthStencil.frontFace.passOperation), QVariant(), QVariant(), QVariant(), })); m_Common.SetStencilTreeItemValue(ui->stencils->topLevelItem(0), 5, state.depthStencil.frontFace.writeMask); m_Common.SetStencilTreeItemValue(ui->stencils->topLevelItem(0), 6, state.depthStencil.frontFace.compareMask); m_Common.SetStencilTreeItemValue(ui->stencils->topLevelItem(0), 7, state.depthStencil.frontFace.reference); ui->stencils->addTopLevelItem(new RDTreeWidgetItem({ tr("Back"), ToQStr(state.depthStencil.backFace.function), ToQStr(state.depthStencil.backFace.failOperation), ToQStr(state.depthStencil.backFace.depthFailOperation), ToQStr(state.depthStencil.backFace.passOperation), QVariant(), QVariant(), QVariant(), })); m_Common.SetStencilTreeItemValue(ui->stencils->topLevelItem(1), 5, state.depthStencil.backFace.writeMask); m_Common.SetStencilTreeItemValue(ui->stencils->topLevelItem(1), 6, state.depthStencil.backFace.compareMask); m_Common.SetStencilTreeItemValue(ui->stencils->topLevelItem(1), 7, state.depthStencil.backFace.reference); } else { ui->stencils->addTopLevelItem(new RDTreeWidgetItem( {tr("Front"), lit("-"), lit("-"), lit("-"), lit("-"), lit("-"), lit("-"), lit("-")})); ui->stencils->addTopLevelItem(new RDTreeWidgetItem( {tr("Back"), lit("-"), lit("-"), lit("-"), lit("-"), lit("-"), lit("-"), lit("-")})); } ui->stencils->clearSelection(); ui->stencils->endUpdate(); // set up thread debugging inputs bool enableDebug = m_Ctx.APIProps().shaderDebugging && state.computeShader.reflection && state.computeShader.reflection->debugInfo.debuggable && action && (action->flags & ActionFlags::Dispatch); if(enableDebug) { // Validate dispatch/threadgroup dimensions enableDebug &= action->dispatchDimension[0] > 0; enableDebug &= action->dispatchDimension[1] > 0; enableDebug &= action->dispatchDimension[2] > 0; const rdcfixedarray &threadDims = (action->dispatchThreadsDimension[0] == 0) ? state.computeShader.reflection->dispatchThreadsDimension : action->dispatchThreadsDimension; enableDebug &= threadDims[0] > 0; enableDebug &= threadDims[1] > 0; enableDebug &= threadDims[2] > 0; } if(enableDebug) { ui->computeDebugSelector->setEnabled(true); // set maximums for CS debugging m_ComputeDebugSelector->SetThreadBounds( action->dispatchDimension, (action->dispatchThreadsDimension[0] == 0) ? state.computeShader.reflection->dispatchThreadsDimension : action->dispatchThreadsDimension); ui->computeDebugSelector->setToolTip( tr("Debug this compute shader by specifying group/thread ID or dispatch ID")); } else { ui->computeDebugSelector->setEnabled(false); if(!m_Ctx.APIProps().shaderDebugging) ui->computeDebugSelector->setToolTip(tr("This API does not support shader debugging")); else if(!action || !(action->flags & ActionFlags::Dispatch)) ui->computeDebugSelector->setToolTip(tr("No dispatch selected")); else if(!state.computeShader.reflection) ui->computeDebugSelector->setToolTip(tr("No compute shader bound")); else if(!state.computeShader.reflection->debugInfo.debuggable) ui->computeDebugSelector->setToolTip( tr("This shader doesn't support debugging: %1") .arg(state.computeShader.reflection->debugInfo.debugStatus)); else ui->computeDebugSelector->setToolTip(tr("Invalid dispatch/threadgroup dimensions.")); } } void VulkanPipelineStateViewer::resource_itemActivated(RDTreeWidgetItem *item, int column) { const VKPipe::Shader *stage = stageForSender(item->treeWidget()); if(stage == NULL) return; QVariant tag = item->tag(); if(tag.canConvert()) { VulkanTextureTag vtex = tag.value(); TextureDescription *tex = m_Ctx.GetTexture(vtex.ID); if(tex) { if(tex->type == TextureType::Buffer) { IBufferViewer *viewer = m_Ctx.ViewTextureAsBuffer( tex->resourceId, Subresource(), BufferFormatter::GetTextureFormatString(*tex)); m_Ctx.AddDockWindow(viewer->Widget(), DockReference::AddTo, this); } else { if(!m_Ctx.HasTextureViewer()) m_Ctx.ShowTextureViewer(); ITextureViewer *viewer = m_Ctx.GetTextureViewer(); viewer->ViewTexture(tex->resourceId, vtex.compType, true); } return; } } else if(tag.canConvert()) { VulkanBufferTag buf = tag.value(); QString format; if(stage->reflection) { const rdcarray &resArray = (IsReadWriteDescriptor(buf.access.type) ? stage->reflection->readWriteResources : stage->reflection->readOnlyResources); if(buf.access.index < resArray.size()) { const ShaderResource &shaderRes = resArray[buf.access.index]; format = BufferFormatter::GetBufferFormatString( BufferFormatter::EstimatePackingRules(stage->resourceId, shaderRes.variableType.members), stage->resourceId, shaderRes, buf.descriptor.format); } } if(buf.descriptor.resource != ResourceId()) { IBufferViewer *viewer = m_Ctx.ViewBuffer(buf.descriptor.byteOffset + buf.dynamicOffset, buf.descriptor.byteSize, buf.descriptor.resource, format); m_Ctx.AddDockWindow(viewer->Widget(), DockReference::AddTo, this); } } } void VulkanPipelineStateViewer::resource_hoverItemChanged(RDTreeWidgetItem *hover) { // first make all rows transparent. for(RDTreeWidgetItem *item : m_CombinedImageSamplers.keys()) { item->setBackground(QBrush()); m_CombinedImageSamplers[item]->setBackground(QBrush()); } if(hover) { // try to get combined sampler data from the current row CombinedSamplerData sampData = hover->tag().value(); // or try to see if it's a combined image if(m_CombinedImageSamplers.contains(hover)) sampData = m_CombinedImageSamplers[hover]->tag().value(); // if we got a sampler, highlight it and all images using it if(sampData.node) { sampData.node->setBackgroundColor(QColor(127, 212, 255, 100)); for(RDTreeWidgetItem *item : sampData.images) item->setBackgroundColor(QColor(127, 212, 255, 100)); } } } void VulkanPipelineStateViewer::ubo_itemActivated(RDTreeWidgetItem *item, int column) { const VKPipe::Shader *stage = stageForSender(item->treeWidget()); if(stage == NULL) return; QVariant tag = item->tag(); if(!tag.canConvert()) return; VulkanCBufferTag cb = tag.value(); if(cb.index == DescriptorAccess::NoShaderBinding) { if(cb.descriptor.resource != ResourceId()) { IBufferViewer *viewer = m_Ctx.ViewBuffer(cb.descriptor.byteOffset + cb.dynamicOffset, cb.descriptor.byteSize, cb.descriptor.resource); m_Ctx.AddDockWindow(viewer->Widget(), DockReference::AddTo, this); } return; } IBufferViewer *prev = m_Ctx.ViewConstantBuffer(stage->stage, cb.index, cb.arrayElement); m_Ctx.AddDockWindow(prev->Widget(), DockReference::TransientPopupArea, this, 0.3f); } void VulkanPipelineStateViewer::descSet_itemActivated(RDTreeWidgetItem *item, int column) { const VKPipe::Shader *stage = stageForSender(item->treeWidget()); if(stage == NULL) return; int index = item->tag().toInt(); const rdcarray &descSets = stage->stage == ShaderStage::Compute ? m_Ctx.CurVulkanPipelineState()->compute.descriptorSets : m_Ctx.CurVulkanPipelineState()->graphics.descriptorSets; const rdcarray &descBufs = stage->stage == ShaderStage::Compute ? m_Ctx.CurVulkanPipelineState()->compute.descriptorBuffers : m_Ctx.CurVulkanPipelineState()->graphics.descriptorBuffers; if(index < descSets.count()) { if(descSets[index].descriptorBufferIndex >= 0) { if(descSets[index].descriptorBufferIndex < descBufs.count()) { IBufferViewer *viewer = m_Ctx.ViewBuffer(descBufs[descSets[index].descriptorBufferIndex].offset + descSets[index].descriptorBufferByteOffset, 0, descBufs[descSets[index].descriptorBufferIndex].buffer); m_Ctx.AddDockWindow(viewer->Widget(), DockReference::AddTo, this); } } else { IDescriptorViewer *viewer = m_Ctx.ViewDescriptorStore(descSets[index].descriptorSetResourceId); m_Ctx.AddDockWindow(viewer->Widget(), DockReference::AddTo, this); } } } void VulkanPipelineStateViewer::on_viAttrs_itemActivated(RDTreeWidgetItem *item, int column) { on_meshView_clicked(); } void VulkanPipelineStateViewer::on_viBuffers_itemActivated(RDTreeWidgetItem *item, int column) { QVariant tag = item->tag(); if(tag.canConvert()) { VulkanVBIBTag buf = tag.value(); if(buf.id != ResourceId()) { IBufferViewer *viewer = m_Ctx.ViewBuffer(buf.offset, UINT64_MAX, buf.id, buf.format); m_Ctx.AddDockWindow(viewer->Widget(), DockReference::AddTo, this); } } } void VulkanPipelineStateViewer::highlightIABind(int slot) { int idx = ((slot + 1) * 21) % 32; // space neighbouring colours reasonably distinctly const VKPipe::VertexInput &VI = m_Ctx.CurVulkanPipelineState()->vertexInput; QColor col = QColor::fromHslF(float(idx) / 32.0f, 1.0f, qBound(0.05, palette().color(QPalette::Base).lightnessF(), 0.95)); ui->viAttrs->beginUpdate(); ui->viBuffers->beginUpdate(); if(slot < m_VBNodes.count()) { if(m_VBNodes[slot] && !m_EmptyNodes.contains(m_VBNodes[slot])) { m_VBNodes[slot]->setBackgroundColor(col); m_VBNodes[slot]->setForegroundColor(contrastingColor(col, QColor(0, 0, 0))); } } if(slot < m_BindNodes.count()) { m_BindNodes[slot]->setBackgroundColor(col); m_BindNodes[slot]->setForegroundColor(contrastingColor(col, QColor(0, 0, 0))); } for(int i = 0; i < ui->viAttrs->topLevelItemCount(); i++) { RDTreeWidgetItem *item = ui->viAttrs->topLevelItem(i); if((int)VI.attributes[item->tag().toUInt()].binding != slot) { item->setBackground(QBrush()); item->setForeground(QBrush()); } else { item->setBackgroundColor(col); item->setForegroundColor(contrastingColor(col, QColor(0, 0, 0))); } } ui->viAttrs->endUpdate(); ui->viBuffers->endUpdate(); } bool VulkanPipelineStateViewer::IsPushSet(ShaderStage stage, ResourceId id) { if(stage == ShaderStage::Compute) { for(const VKPipe::DescriptorSet &set : m_Ctx.CurVulkanPipelineState()->compute.descriptorSets) if(set.descriptorSetResourceId == id) return set.pushDescriptor; } else { for(const VKPipe::DescriptorSet &set : m_Ctx.CurVulkanPipelineState()->graphics.descriptorSets) if(set.descriptorSetResourceId == id) return set.pushDescriptor; } return false; } void VulkanPipelineStateViewer::on_viAttrs_mouseMove(QMouseEvent *e) { if(!m_Ctx.IsCaptureLoaded()) return; RDTreeWidgetItem *item = ui->viAttrs->itemAt(e->pos()); vertex_leave(NULL); const VKPipe::VertexInput &VI = m_Ctx.CurVulkanPipelineState()->vertexInput; if(item) { uint32_t binding = VI.attributes[item->tag().toUInt()].binding; highlightIABind((int)binding); } } void VulkanPipelineStateViewer::on_viBuffers_mouseMove(QMouseEvent *e) { if(!m_Ctx.IsCaptureLoaded()) return; RDTreeWidgetItem *item = ui->viBuffers->itemAt(e->pos()); vertex_leave(NULL); if(item) { int idx = m_VBNodes.indexOf(item); if(idx >= 0) { highlightIABind(idx); } else { if(!m_EmptyNodes.contains(item)) { item->setBackground(ui->viBuffers->palette().brush(QPalette::Window)); item->setForeground(ui->viBuffers->palette().brush(QPalette::WindowText)); } } } } void VulkanPipelineStateViewer::vertex_leave(QEvent *e) { ui->viAttrs->beginUpdate(); ui->viBuffers->beginUpdate(); for(int i = 0; i < ui->viAttrs->topLevelItemCount(); i++) { ui->viAttrs->topLevelItem(i)->setBackground(QBrush()); ui->viAttrs->topLevelItem(i)->setForeground(QBrush()); } for(int i = 0; i < ui->viBuffers->topLevelItemCount(); i++) { RDTreeWidgetItem *item = ui->viBuffers->topLevelItem(i); if(m_EmptyNodes.contains(item)) continue; item->setBackground(QBrush()); item->setForeground(QBrush()); } ui->viAttrs->endUpdate(); ui->viBuffers->endUpdate(); } void VulkanPipelineStateViewer::on_pipeFlow_stageSelected(int index) { if(m_MeshPipe) { // remap since TS/MS are the last tabs but appear first in the flow switch(index) { // TS case 0: ui->stagesTabs->setCurrentIndex(9); break; // MS case 1: ui->stagesTabs->setCurrentIndex(10); break; // raster onwards are the same, just skipping VTX,VS,TCS,TES,GS case 2: ui->stagesTabs->setCurrentIndex(5); break; case 3: ui->stagesTabs->setCurrentIndex(6); break; case 4: ui->stagesTabs->setCurrentIndex(7); break; case 5: ui->stagesTabs->setCurrentIndex(8); break; } } else { ui->stagesTabs->setCurrentIndex(index); } } void VulkanPipelineStateViewer::shaderView_clicked() { const VKPipe::Shader *stage = stageForSender(qobject_cast(QObject::sender())); if(stage == NULL || stage->resourceId == ResourceId()) return; const ShaderReflection *shaderDetails = stage->reflection; ResourceId pipe = stage->stage == ShaderStage::Compute ? m_Ctx.CurVulkanPipelineState()->compute.pipelineResourceId : m_Ctx.CurVulkanPipelineState()->graphics.pipelineResourceId; if(!shaderDetails) return; IShaderViewer *shad = m_Ctx.ViewShader(shaderDetails, pipe); m_Ctx.AddDockWindow(shad->Widget(), DockReference::AddTo, this); } void VulkanPipelineStateViewer::shaderSave_clicked() { const VKPipe::Shader *stage = stageForSender(qobject_cast(QObject::sender())); if(stage == NULL) return; const ShaderReflection *shaderDetails = stage->reflection; if(stage->resourceId == ResourceId()) return; m_Common.SaveShaderFile(shaderDetails); } void VulkanPipelineStateViewer::shaderMessages_clicked() { const VKPipe::Shader *stage = stageForSender(qobject_cast(QObject::sender())); if(stage == NULL) return; IShaderMessageViewer *shad = m_Ctx.ViewShaderMessages(MaskForStage(stage->stage)); m_Ctx.AddDockWindow(shad->Widget(), DockReference::AddTo, this); } void VulkanPipelineStateViewer::predicateBufferView_clicked() { const VKPipe::ConditionalRendering &cr = m_Ctx.CurVulkanPipelineState()->conditionalRendering; IBufferViewer *viewer = m_Ctx.ViewBuffer(cr.byteOffset, sizeof(uint32_t), cr.bufferId, "uint"); m_Ctx.AddDockWindow(viewer->Widget(), DockReference::AddTo, this); } void VulkanPipelineStateViewer::exportHTML(QXmlStreamWriter &xml, const VKPipe::VertexInput &vi) { { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Attributes")); xml.writeEndElement(); QList rows; for(const VKPipe::VertexAttribute &attr : vi.attributes) rows.push_back({attr.location, attr.binding, attr.format.Name(), attr.byteOffset}); m_Common.exportHTMLTable(xml, {tr("Location"), tr("Binding"), tr("Format"), tr("Offset")}, rows); } { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Bindings")); xml.writeEndElement(); QList rows; for(const VKPipe::VertexBinding &attr : vi.bindings) rows.push_back( {attr.vertexBufferBinding, attr.perInstance ? tr("PER_INSTANCE") : tr("PER_VERTEX")}); m_Common.exportHTMLTable(xml, {tr("Binding"), tr("Step Rate")}, rows); } { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Vertex Buffers")); xml.writeEndElement(); QList rows; int i = 0; for(const VKPipe::VertexBuffer &vb : vi.vertexBuffers) { uint64_t length = vb.byteSize; if(vb.resourceId == ResourceId()) { continue; } else { BufferDescription *buf = m_Ctx.GetBuffer(vb.resourceId); if(buf && length >= ULONG_MAX) length = buf->length; } rows.push_back({i, m_Ctx.GetResourceName(vb.resourceId), (qulonglong)vb.byteOffset, (qulonglong)vb.byteStride, (qulonglong)length}); i++; } m_Common.exportHTMLTable( xml, {tr("Binding"), tr("Buffer"), tr("Offset"), tr("Byte Stride"), tr("Byte Length")}, rows); } } void VulkanPipelineStateViewer::exportHTML(QXmlStreamWriter &xml, const VKPipe::InputAssembly &ia) { const ActionDescription *action = m_Ctx.CurAction(); { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Index Buffer")); xml.writeEndElement(); BufferDescription *ib = m_Ctx.GetBuffer(ia.indexBuffer.resourceId); QString name = tr("Empty"); uint64_t length = 0; if(ib) { name = m_Ctx.GetResourceName(ia.indexBuffer.resourceId); length = qMin(ib->length, ia.indexBuffer.byteSize); } QString ifmt = lit("UNKNOWN"); if(ia.indexBuffer.byteStride == 1) ifmt = lit("UINT8"); else if(ia.indexBuffer.byteStride == 2) ifmt = lit("UINT16"); else if(ia.indexBuffer.byteStride == 4) ifmt = lit("UINT32"); m_Common.exportHTMLTable( xml, {tr("Buffer"), tr("Format"), tr("Offset"), tr("Byte Length"), tr("Primitive Restart")}, {name, ifmt, (qulonglong)ia.indexBuffer.byteOffset, (qulonglong)length, ia.primitiveRestartEnable ? tr("Yes") : tr("No")}); } xml.writeStartElement(lit("p")); xml.writeEndElement(); m_Common.exportHTMLTable( xml, {tr("Primitive Topology"), tr("Tessellation Control Points")}, {ToQStr(ia.topology), m_Ctx.CurVulkanPipelineState()->tessellation.numControlPoints}); } void VulkanPipelineStateViewer::exportHTML(QXmlStreamWriter &xml, const VKPipe::Shader &sh) { const ShaderReflection *shaderDetails = sh.reflection; const VKPipe::Pipeline &pipeline = (sh.stage == ShaderStage::Compute ? m_Ctx.CurVulkanPipelineState()->compute : m_Ctx.CurVulkanPipelineState()->graphics); { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Shader")); xml.writeEndElement(); QString shadername = tr("Unknown"); if(sh.resourceId == ResourceId()) shadername = tr("Unbound"); else shadername = m_Ctx.GetResourceName(sh.resourceId); QString entryname = tr("Unknown"); if(shaderDetails) { QString entryFunc = shaderDetails->entryPoint; const ShaderDebugInfo &dbg = shaderDetails->debugInfo; int entryFile = qMax(0, dbg.entryLocation.fileIndex); if(!dbg.files.isEmpty()) entryname = QFormatStr("%1() - %2") .arg(entryFunc) .arg(QFileInfo(dbg.files[entryFile].filename).fileName()); else entryname = QFormatStr("%1()").arg(entryFunc); } m_Common.exportHTMLTable( xml, {tr("Pipeline"), tr("Shader"), tr("Entry")}, {m_Ctx.GetResourceName(pipeline.pipelineResourceId), shadername, entryname}); if(sh.resourceId == ResourceId()) return; } if(!shaderDetails) return; QList uboRows; QList roRows; QList rwRows; QList sampRows; for(const UsedDescriptor &used : m_Ctx.CurPipelineState().GetConstantBlocks(sh.stage)) { if(used.access.stage != sh.stage) continue; const Descriptor &descriptor = used.descriptor; uint32_t dynamicOffset = 0; for(const VKPipe::DescriptorSet &set : pipeline.descriptorSets) { for(const VKPipe::DynamicOffset &offs : set.dynamicOffsets) { if(set.descriptorSetResourceId == used.access.descriptorStore && offs.descriptorByteOffset == used.access.byteOffset) { dynamicOffset += offs.dynamicBufferByteOffset; } } } QString name = m_Ctx.GetResourceName(descriptor.resource); uint64_t byteOffset = descriptor.byteOffset + dynamicOffset; uint64_t length = descriptor.byteSize; int numvars = 0; uint32_t bindByteSize = 0; QString slotname; if(used.access.index == DescriptorAccess::NoShaderBinding) { slotname = m_Locations[{used.access.descriptorStore, used.access.byteOffset}].logicalBindName; slotname += QFormatStr("[%1]").arg(used.access.arrayElement); } else { const ConstantBlock &b = shaderDetails->constantBlocks[used.access.index]; // push constants if(!b.bufferBacked) { if(b.compileConstants) name = tr("Specialization constants"); else name = tr("Push constants"); qulonglong offset = 0, size = 0; offset = byteOffset; size = descriptor.byteSize; // could maybe get range/size from ShaderVariable.reg if it's filled out // from SPIR-V side. uboRows.push_back({b.name, name, offset, size, b.variables.count(), b.byteSize}); continue; } if(IsPushSet(used.access.stage, used.access.descriptorStore)) slotname = tr("Push "); slotname += QFormatStr("Set %1, %2").arg(b.fixedBindSetOrSpace).arg(b.fixedBindNumber); if(!b.name.empty()) slotname += lit(": ") + b.name; if(b.bindArraySize > 1) slotname += QFormatStr("[%1]").arg(used.access.arrayElement); numvars = b.variables.count(); bindByteSize = b.byteSize; } if(descriptor.flags & DescriptorFlags::InlineData) name = tr("Inline block"); if(descriptor.resource == ResourceId()) { name = tr("Empty"); length = 0; } BufferDescription *buf = m_Ctx.GetBuffer(descriptor.resource); if(buf) { if(length == UINT64_MAX) length = buf->length - byteOffset; } uboRows.push_back( {slotname, name, (qulonglong)byteOffset, (qulonglong)length, numvars, bindByteSize}); } for(const UsedDescriptor &used : m_Ctx.CurPipelineState().GetReadOnlyResources(sh.stage)) { if(used.access.stage != sh.stage) continue; const Descriptor &descriptor = used.descriptor; uint32_t dynamicOffset = 0; for(const VKPipe::DescriptorSet &set : pipeline.descriptorSets) { for(const VKPipe::DynamicOffset &offs : set.dynamicOffsets) { if(set.descriptorSetResourceId == used.access.descriptorStore && offs.descriptorByteOffset == used.access.byteOffset) { dynamicOffset += offs.dynamicBufferByteOffset; } } } exportDescriptorHTML(used, sh.reflection, descriptor, dynamicOffset, roRows); } for(const UsedDescriptor &used : m_Ctx.CurPipelineState().GetReadWriteResources(sh.stage)) { if(used.access.stage != sh.stage) continue; const Descriptor &descriptor = used.descriptor; uint32_t dynamicOffset = 0; for(const VKPipe::DescriptorSet &set : pipeline.descriptorSets) { for(const VKPipe::DynamicOffset &offs : set.dynamicOffsets) { if(set.descriptorSetResourceId == used.access.descriptorStore && offs.descriptorByteOffset == used.access.byteOffset) { dynamicOffset += offs.dynamicBufferByteOffset; } } } exportDescriptorHTML(used, sh.reflection, descriptor, dynamicOffset, rwRows); } for(const UsedDescriptor &used : m_Ctx.CurPipelineState().GetSamplers(sh.stage)) { if(used.access.stage != sh.stage) continue; const SamplerDescriptor &descriptor = used.sampler; const ShaderSampler *shaderSamp = NULL; if(used.access.index < sh.reflection->samplers.size()) shaderSamp = &sh.reflection->samplers[used.access.index]; { QString borderColor; if(descriptor.borderColorType == CompType::Float) borderColor = QFormatStr("%1, %2, %3, %4") .arg(descriptor.borderColorValue.floatValue[0]) .arg(descriptor.borderColorValue.floatValue[1]) .arg(descriptor.borderColorValue.floatValue[2]) .arg(descriptor.borderColorValue.floatValue[3]); else borderColor = QFormatStr("%1, %2, %3, %4") .arg(descriptor.borderColorValue.uintValue[0]) .arg(descriptor.borderColorValue.uintValue[1]) .arg(descriptor.borderColorValue.uintValue[2]) .arg(descriptor.borderColorValue.uintValue[3]); QString addressing; QString addPrefix; QString addVal; QString addr[] = {ToQStr(descriptor.addressU, GraphicsAPI::D3D12), ToQStr(descriptor.addressV, GraphicsAPI::D3D12), ToQStr(descriptor.addressW, GraphicsAPI::D3D12)}; // arrange like either UVW: WRAP or UV: WRAP, W: CLAMP for(int a = 0; a < 3; a++) { const QString str[] = {lit("U"), lit("V"), lit("W")}; QString prefix = str[a]; if(a == 0 || addr[a] == addr[a - 1]) { addPrefix += prefix; } else { addressing += QFormatStr("%1: %2, ").arg(addPrefix).arg(addVal); addPrefix = prefix; } addVal = addr[a]; } addressing += addPrefix + lit(": ") + addVal; if(descriptor.UseBorder()) addressing += QFormatStr("<%1>").arg(borderColor); if(descriptor.unnormalized) addressing += lit(" (Un-norm)"); QString filter = ToQStr(descriptor.filter); if(descriptor.maxAnisotropy > 1) filter += QFormatStr(" %1x").arg(descriptor.maxAnisotropy); if(descriptor.filter.filter == FilterFunction::Comparison) filter += QFormatStr(" (%1)").arg(ToQStr(descriptor.compareFunction)); else if(descriptor.filter.filter != FilterFunction::Normal) filter += QFormatStr(" (%1)").arg(ToQStr(descriptor.filter.filter)); QString slotname; if(used.access.index == DescriptorAccess::NoShaderBinding) { slotname = m_Locations[{used.access.descriptorStore, used.access.byteOffset}].logicalBindName; slotname += QFormatStr("[%1]").arg(used.access.arrayElement); } else if(shaderSamp) { if(IsPushSet(used.access.stage, used.access.descriptorStore)) slotname = tr("Push "); slotname += QFormatStr("Set %1, %2").arg(shaderSamp->fixedBindSetOrSpace).arg(shaderSamp->fixedBindNumber); if(!shaderSamp->name.empty()) slotname += lit(": ") + shaderSamp->name; if(shaderSamp->bindArraySize > 1) slotname += QFormatStr("[%1]").arg(used.access.arrayElement); } sampRows.push_back( {slotname, m_Ctx.GetResourceName(descriptor.object), addressing, filter, QFormatStr("%1 - %2") .arg(descriptor.minLOD == -FLT_MAX ? lit("0") : QString::number(descriptor.minLOD)) .arg(descriptor.maxLOD == FLT_MAX ? lit("FLT_MAX") : QString::number(descriptor.maxLOD)), descriptor.mipBias}); } } if(!roRows.empty()) { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Read-only Resources")); xml.writeEndElement(); m_Common.exportHTMLTable( xml, {tr("Binding"), tr("Resource"), tr("Type"), tr("Width"), tr("Height"), tr("Depth"), tr("Array Size"), tr("Resource Format"), tr("View Parameters")}, roRows); } if(!rwRows.empty()) { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Read-write Resources")); xml.writeEndElement(); m_Common.exportHTMLTable( xml, {tr("Binding"), tr("Resource"), tr("Type"), tr("Width"), tr("Height"), tr("Depth"), tr("Array Size"), tr("Resource Format"), tr("View Parameters")}, rwRows); } if(!sampRows.empty()) { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Samplers")); xml.writeEndElement(); m_Common.exportHTMLTable(xml, {tr("Binding"), tr("Sampler"), tr("Addressing"), tr("Filter"), tr("LOD Clamp"), tr("LOD Bias")}, sampRows); } if(!uboRows.empty()) { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("UBOs")); xml.writeEndElement(); m_Common.exportHTMLTable(xml, {tr("Binding"), tr("Buffer"), tr("Byte Offset"), tr("Byte Size"), tr("Number of Variables"), tr("Bytes Needed")}, uboRows); } } void VulkanPipelineStateViewer::exportHTML(QXmlStreamWriter &xml, const VKPipe::TransformFeedback &xfb) { { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Transform Feedback Bindings")); xml.writeEndElement(); QList rows; int i = 0; for(const VKPipe::XFBBuffer &b : xfb.buffers) { QString name = m_Ctx.GetResourceName(b.bufferResourceId); uint64_t length = b.byteSize; QString counterName = m_Ctx.GetResourceName(b.counterBufferResourceId); if(b.bufferResourceId == ResourceId()) { name = tr("Empty"); } else { BufferDescription *buf = m_Ctx.GetBuffer(b.bufferResourceId); if(buf && length == UINT64_MAX) length = buf->length - b.byteOffset; } if(b.counterBufferResourceId == ResourceId()) { counterName = tr("Empty"); } rows.push_back({i, name, (qulonglong)b.byteOffset, (qulonglong)length, counterName, (qulonglong)b.counterBufferOffset}); i++; } m_Common.exportHTMLTable(xml, {tr("Slot"), tr("Buffer"), tr("Byte Offset"), tr("Byte Length"), tr("Counter Buffer"), tr("Counter Offset")}, rows); } } void VulkanPipelineStateViewer::exportHTML(QXmlStreamWriter &xml, const VKPipe::Rasterizer &rs) { { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Raster State")); xml.writeEndElement(); m_Common.exportHTMLTable( xml, {tr("Fill Mode"), tr("Cull Mode"), tr("Front CCW"), tr("Provoking Vertex")}, {ToQStr(rs.fillMode), ToQStr(rs.cullMode), rs.frontCCW ? tr("Yes") : tr("No"), rs.provokingVertexFirst ? tr("First") : tr("Last")}); xml.writeStartElement(lit("p")); xml.writeEndElement(); m_Common.exportHTMLTable(xml, { tr("Depth Clamp Enable"), tr("Depth Clip Enable"), tr("Rasterizer Discard Enable"), }, { rs.depthClampEnable ? tr("Yes") : tr("No"), rs.depthClipEnable ? tr("Yes") : tr("No"), rs.rasterizerDiscardEnable ? tr("Yes") : tr("No"), }); xml.writeStartElement(lit("p")); xml.writeEndElement(); m_Common.exportHTMLTable(xml, {tr("Depth Bias Enable"), tr("Depth Bias"), tr("Depth Bias Clamp"), tr("Slope Scaled Bias"), tr("Line Width")}, { rs.depthBiasEnable ? tr("Yes") : tr("No"), Formatter::Format(rs.depthBias), Formatter::Format(rs.depthBiasClamp), Formatter::Format(rs.slopeScaledDepthBias), Formatter::Format(rs.lineWidth), }); } { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Line Stipple")); xml.writeEndElement(); m_Common.exportHTMLTable( xml, {tr("Factor"), tr("Pattern")}, {ToQStr(rs.lineStippleFactor), QString::number(rs.lineStipplePattern, 2)}); } { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Conservative Raster")); xml.writeEndElement(); m_Common.exportHTMLTable(xml, {tr("Mode"), tr("Over-estimation size")}, {ToQStr(rs.conservativeRasterization), Formatter::Format(rs.extraPrimitiveOverestimationSize)}); } { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Variable Shading Rate")); xml.writeEndElement(); m_Common.exportHTMLTable( xml, {tr("Pipeline Shading Rate"), tr("Combiners")}, {QFormatStr("%1x%2").arg(rs.pipelineShadingRate.first).arg(rs.pipelineShadingRate.second), QFormatStr("%1, %2") .arg(ToQStr(rs.shadingRateCombiners.first, GraphicsAPI::Vulkan)) .arg(ToQStr(rs.shadingRateCombiners.second, GraphicsAPI::Vulkan))}); } const VKPipe::MultiSample &msaa = m_Ctx.CurVulkanPipelineState()->multisample; { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Multisampling State")); xml.writeEndElement(); m_Common.exportHTMLTable( xml, {tr("Raster Samples"), tr("Sample-rate shading"), tr("Min Sample Shading Rate"), tr("Sample Mask")}, {msaa.rasterSamples, msaa.sampleShadingEnable ? tr("Yes") : tr("No"), Formatter::Format(msaa.minSampleShading), Formatter::Format(msaa.sampleMask, true)}); } const VKPipe::ViewState &vp = m_Ctx.CurVulkanPipelineState()->viewportScissor; { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Viewports")); xml.writeEndElement(); const QString ndcDepthRange = vp.depthNegativeOneToOne ? lit("[-1, 1]") : lit("[0, 1]"); QList rows; int i = 0; for(const VKPipe::ViewportScissor &vs : vp.viewportScissors) { const Viewport &v = vs.vp; rows.push_back({i, v.x, v.y, v.width, v.height, v.minDepth, v.maxDepth, ndcDepthRange}); i++; } QStringList header = {tr("Slot"), tr("X"), tr("Y"), tr("Width"), tr("Height"), tr("Min Depth"), tr("Max Depth"), tr("NDC Depth Range")}; m_Common.exportHTMLTable(xml, header, rows); } { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Scissors")); xml.writeEndElement(); QList rows; int i = 0; for(const VKPipe::ViewportScissor &vs : vp.viewportScissors) { const Scissor &s = vs.scissor; rows.push_back({i, s.x, s.y, s.width, s.height}); i++; } m_Common.exportHTMLTable(xml, {tr("Slot"), tr("X"), tr("Y"), tr("Width"), tr("Height")}, rows); } } void VulkanPipelineStateViewer::exportHTML(QXmlStreamWriter &xml, const VKPipe::ColorBlendState &cb) { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Color Blend State")); xml.writeEndElement(); QString blendConst = QFormatStr("%1, %2, %3, %4") .arg(cb.blendFactor[0], 0, 'f', 2) .arg(cb.blendFactor[1], 0, 'f', 2) .arg(cb.blendFactor[2], 0, 'f', 2) .arg(cb.blendFactor[3], 0, 'f', 2); bool logic = !cb.blends.isEmpty() && cb.blends[0].logicOperationEnabled; m_Common.exportHTMLTable( xml, {tr("Alpha to Coverage"), tr("Alpha to One"), tr("Logic Op"), tr("Blend Constant")}, { cb.alphaToCoverageEnable ? tr("Yes") : tr("No"), cb.alphaToOneEnable ? tr("Yes") : tr("No"), logic ? ToQStr(cb.blends[0].logicOperation) : tr("Disabled"), blendConst, }); xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Attachment Blends")); xml.writeEndElement(); QList rows; int i = 0; for(const ColorBlend &b : cb.blends) { rows.push_back({i, b.enabled ? tr("Yes") : tr("No"), ToQStr(b.colorBlend.source), ToQStr(b.colorBlend.destination), ToQStr(b.colorBlend.operation), ToQStr(b.alphaBlend.source), ToQStr(b.alphaBlend.destination), ToQStr(b.alphaBlend.operation), ((b.writeMask & 0x1) == 0 ? lit("_") : lit("R")) + ((b.writeMask & 0x2) == 0 ? lit("_") : lit("G")) + ((b.writeMask & 0x4) == 0 ? lit("_") : lit("B")) + ((b.writeMask & 0x8) == 0 ? lit("_") : lit("A"))}); i++; } m_Common.exportHTMLTable(xml, { tr("Slot"), tr("Blend Enable"), tr("Blend Source"), tr("Blend Destination"), tr("Blend Operation"), tr("Alpha Blend Source"), tr("Alpha Blend Destination"), tr("Alpha Blend Operation"), tr("Write Mask"), }, rows); } void VulkanPipelineStateViewer::exportHTML(QXmlStreamWriter &xml, const VKPipe::DepthStencil &ds) { { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Depth State")); xml.writeEndElement(); m_Common.exportHTMLTable( xml, {tr("Depth Test Enable"), tr("Depth Writes Enable"), tr("Depth Function"), tr("Depth Bounds")}, { ds.depthTestEnable ? tr("Yes") : tr("No"), ds.depthWriteEnable ? tr("Yes") : tr("No"), ToQStr(ds.depthFunction), ds.depthBoundsEnable ? QFormatStr("%1 - %2") .arg(Formatter::Format(ds.minDepthBounds)) .arg(Formatter::Format(ds.maxDepthBounds)) : tr("Disabled"), }); } { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Stencil State")); xml.writeEndElement(); if(ds.stencilTestEnable) { QList rows; rows.push_back({ tr("Front"), Formatter::Format(ds.frontFace.reference, true), Formatter::Format(ds.frontFace.compareMask, true), Formatter::Format(ds.frontFace.writeMask, true), ToQStr(ds.frontFace.function), ToQStr(ds.frontFace.passOperation), ToQStr(ds.frontFace.failOperation), ToQStr(ds.frontFace.depthFailOperation), }); rows.push_back({ tr("back"), Formatter::Format(ds.backFace.reference, true), Formatter::Format(ds.backFace.compareMask, true), Formatter::Format(ds.backFace.writeMask, true), ToQStr(ds.backFace.function), ToQStr(ds.backFace.passOperation), ToQStr(ds.backFace.failOperation), ToQStr(ds.backFace.depthFailOperation), }); m_Common.exportHTMLTable(xml, {tr("Face"), tr("Ref"), tr("Compare Mask"), tr("Write Mask"), tr("Function"), tr("Pass Op"), tr("Fail Op"), tr("Depth Fail Op")}, rows); } else { xml.writeStartElement(lit("p")); xml.writeCharacters(tr("Disabled")); xml.writeEndElement(); } } } void VulkanPipelineStateViewer::exportHTML(QXmlStreamWriter &xml, const VKPipe::CurrentPass &pass) { { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Framebuffer")); xml.writeEndElement(); m_Common.exportHTMLTable( xml, {tr("Width"), tr("Height"), tr("Layers")}, {pass.framebuffer.width, pass.framebuffer.height, pass.framebuffer.layers}); xml.writeStartElement(lit("p")); xml.writeEndElement(); QList rows; int i = 0; for(const Descriptor &a : pass.framebuffer.attachments) { TextureDescription *tex = m_Ctx.GetTexture(a.resource); QString name = m_Ctx.GetResourceName(a.resource); rows.push_back({i, name, tex->width, tex->height, tex->depth, tex->arraysize, a.firstMip, a.numMips, a.firstSlice, a.numSlices, getTextureRenderSamples(tex, pass.renderpass)}); i++; } m_Common.exportHTMLTable(xml, { tr("Slot"), tr("Image"), tr("Width"), tr("Height"), tr("Depth"), tr("Array Size"), tr("First mip"), tr("Number of mips"), tr("First array layer"), tr("Number of layers"), tr("Sample Count"), }, rows); } { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Render Pass")); xml.writeEndElement(); if(!pass.renderpass.inputAttachments.isEmpty()) { QList inputs; for(int i = 0; i < pass.renderpass.inputAttachments.count(); i++) inputs.push_back({pass.renderpass.inputAttachments[i]}); m_Common.exportHTMLTable(xml, { tr("Input Attachment"), }, inputs); xml.writeStartElement(lit("p")); xml.writeEndElement(); } if(!pass.renderpass.colorAttachments.isEmpty()) { QList colors; for(int i = 0; i < pass.renderpass.colorAttachments.count(); i++) colors.push_back({pass.renderpass.colorAttachments[i]}); m_Common.exportHTMLTable(xml, { tr("Color Attachment"), }, colors); xml.writeStartElement(lit("p")); xml.writeEndElement(); } if(!pass.renderpass.colorAttachmentLocations.isEmpty()) { QList locations; for(int i = 0; i < pass.renderpass.colorAttachmentLocations.count(); i++) locations.push_back({pass.renderpass.colorAttachmentLocations[i]}); m_Common.exportHTMLTable(xml, { tr("Color Attachment Location"), }, locations); xml.writeStartElement(lit("p")); xml.writeEndElement(); } if(!pass.renderpass.colorAttachmentInputIndices.isEmpty()) { QList inputIndices; for(int i = 0; i < pass.renderpass.colorAttachmentInputIndices.count(); i++) inputIndices.push_back({pass.renderpass.colorAttachmentInputIndices[i]}); m_Common.exportHTMLTable(xml, { tr("Color Attachment Input Index"), }, inputIndices); xml.writeStartElement(lit("p")); xml.writeEndElement(); } if(!pass.renderpass.resolveAttachments.isEmpty()) { QList resolves; for(int i = 0; i < pass.renderpass.resolveAttachments.count(); i++) resolves.push_back({pass.renderpass.resolveAttachments[i]}); m_Common.exportHTMLTable(xml, { tr("Resolve Attachment"), }, resolves); xml.writeStartElement(lit("p")); xml.writeEndElement(); } if(!pass.renderpass.multiviews.isEmpty()) { QList colors; for(int i = 0; i < pass.renderpass.multiviews.count(); i++) colors.push_back({pass.renderpass.multiviews[i]}); m_Common.exportHTMLTable(xml, { tr("Multiview Mask"), }, colors); xml.writeStartElement(lit("p")); xml.writeEndElement(); } if(pass.renderpass.depthstencilAttachment >= 0) { xml.writeStartElement(lit("p")); xml.writeCharacters( tr("Depth-stencil Attachment: %1").arg(pass.renderpass.depthstencilAttachment)); xml.writeEndElement(); } if(!pass.renderpass.isDepthInputAttachmentIndexImplicit) { xml.writeStartElement(lit("p")); xml.writeCharacters( tr("Depth Input Attachment Index: %1").arg(pass.renderpass.depthInputAttachmentIndex)); xml.writeEndElement(); } if(!pass.renderpass.isStencilInputAttachmentIndexImplicit) { xml.writeStartElement(lit("p")); xml.writeCharacters( tr("Stencil Input Attachment Index: %1").arg(pass.renderpass.stencilInputAttachmentIndex)); xml.writeEndElement(); } if(pass.renderpass.depthstencilResolveAttachment >= 0) { xml.writeStartElement(lit("p")); xml.writeCharacters(tr("Depth-stencil Resolve Attachment: %1") .arg(pass.renderpass.depthstencilResolveAttachment)); xml.writeEndElement(); } if(pass.renderpass.fragmentDensityAttachment >= 0) { xml.writeStartElement(lit("p")); xml.writeCharacters( tr("Fragment Density Attachment: %1").arg(pass.renderpass.fragmentDensityAttachment)); if(pass.renderpass.fragmentDensityOffsets.size() > 0) { xml.writeCharacters( tr(". Rendering with %1 offsets : ").arg(pass.renderpass.fragmentDensityOffsets.size())); for(uint32_t j = 0; j < pass.renderpass.fragmentDensityOffsets.size(); j++) { const Offset &o = pass.renderpass.fragmentDensityOffsets[j]; if(j > 0) xml.writeCharacters(tr(", ")); xml.writeCharacters(tr(" %1x%2").arg(o.x).arg(o.y)); } } xml.writeEndElement(); } if(pass.renderpass.shadingRateAttachment >= 0) { xml.writeStartElement(lit("p")); xml.writeCharacters(tr("Fragment Shading Rate Attachment: %1 (texel size %2x%3)") .arg(pass.renderpass.shadingRateAttachment) .arg(pass.renderpass.shadingRateTexelSize.first) .arg(pass.renderpass.shadingRateTexelSize.second)); xml.writeEndElement(); } } { xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Render Area")); xml.writeEndElement(); m_Common.exportHTMLTable( xml, {tr("X"), tr("Y"), tr("Width"), tr("Height")}, {pass.renderArea.x, pass.renderArea.y, pass.renderArea.width, pass.renderArea.height}); } } void VulkanPipelineStateViewer::exportHTML(QXmlStreamWriter &xml, const VKPipe::ConditionalRendering &cr) { if(cr.bufferId == ResourceId()) return; xml.writeStartElement(lit("h3")); xml.writeCharacters(tr("Conditional Rendering")); xml.writeEndElement(); QString bufferName = m_Ctx.GetResourceName(cr.bufferId); m_Common.exportHTMLTable( xml, {tr("Predicate Passing"), tr("Is Inverted"), tr("Buffer"), tr("Byte Offset")}, { cr.isPassing ? tr("Yes") : tr("No"), cr.isInverted ? tr("Yes") : tr("No"), bufferName, (qulonglong)cr.byteOffset, }); } const ShaderResource *VulkanPipelineStateViewer::exportDescriptorHTML(const UsedDescriptor &used, const ShaderReflection *refl, const Descriptor &descriptor, uint32_t dynamicOffset, QList &rows) { const ShaderResource *shaderRes = NULL; if(IsReadOnlyDescriptor(used.access.type)) { if(used.access.index < refl->readOnlyResources.size()) shaderRes = &refl->readOnlyResources[used.access.index]; } else { if(used.access.index < refl->readWriteResources.size()) shaderRes = &refl->readWriteResources[used.access.index]; } QString slotname; if(used.access.index == DescriptorAccess::NoShaderBinding) { slotname = m_Locations[{used.access.descriptorStore, used.access.byteOffset}].logicalBindName; slotname += QFormatStr("[%1]").arg(used.access.arrayElement); } else if(shaderRes) { if(IsPushSet(used.access.stage, used.access.descriptorStore)) slotname = tr("Push "); slotname += QFormatStr("Set %1, %2").arg(shaderRes->fixedBindSetOrSpace).arg(shaderRes->fixedBindNumber); if(!shaderRes->name.empty()) slotname += lit(": ") + shaderRes->name; if(shaderRes->bindArraySize > 1) slotname += QFormatStr("[%1]").arg(used.access.arrayElement); } ResourceId id = descriptor.resource; QString name = m_Ctx.GetResourceName(id); if(id == ResourceId()) name = tr("Empty"); BufferDescription *buf = m_Ctx.GetBuffer(id); TextureDescription *tex = m_Ctx.GetTexture(id); uint64_t w = 1; uint32_t h = 1, d = 1; uint32_t arr = 0; QString format = tr("Unknown"); QString viewParams; if(tex) { w = tex->width; h = tex->height; d = tex->depth; arr = tex->arraysize; format = tex->format.Name(); if(tex->mips > 1) { viewParams = tr("Mips: %1-%2").arg(descriptor.firstMip).arg(descriptor.firstMip + descriptor.numMips - 1); } if(tex->arraysize > 1) { if(!viewParams.isEmpty()) viewParams += lit(", "); viewParams += tr("Layers: %1-%2") .arg(descriptor.firstSlice) .arg(descriptor.firstSlice + descriptor.numSlices - 1); } if(descriptor.swizzle.red != TextureSwizzle::Red || descriptor.swizzle.green != TextureSwizzle::Green || descriptor.swizzle.blue != TextureSwizzle::Blue || descriptor.swizzle.alpha != TextureSwizzle::Alpha) { if(!viewParams.isEmpty()) viewParams += lit(", "); viewParams += tr("swizzle[%1%2%3%4]") .arg(ToQStr(descriptor.swizzle.red)) .arg(ToQStr(descriptor.swizzle.green)) .arg(ToQStr(descriptor.swizzle.blue)) .arg(ToQStr(descriptor.swizzle.alpha)); } } if(buf) { w = buf->length; h = 0; d = 0; arr = 0; format = lit("-"); viewParams = tr("Byte Range: %1").arg(formatByteRange(buf, descriptor, dynamicOffset)); } if(descriptor.type != DescriptorType::Sampler) rows.push_back( {slotname, name, ToQStr(descriptor.type), (qulonglong)w, h, d, arr, format, viewParams}); if(descriptor.type == DescriptorType::ImageSampler) { QString samplerName = m_Ctx.GetResourceName(used.sampler.object); if(used.sampler.object == ResourceId()) samplerName = tr("Empty"); QVariantList sampDetails = makeSampler(QString(), used.sampler); rows.push_back({slotname, samplerName, ToQStr(descriptor.type), QString(), QString(), QString(), QString(), sampDetails[3], sampDetails[4]}); } return shaderRes; } QString VulkanPipelineStateViewer::GetFossilizeHash(ResourceId id) { uint h = qHash(ToQStr(id)); if(id == ResourceId()) h = 0; return QFormatStr("%1").arg(h, 16, 16, QLatin1Char('0')); } QString VulkanPipelineStateViewer::GetFossilizeFilename(QDir d, uint32_t tag, ResourceId id) { return d.absoluteFilePath( lit("%1.%2.json").arg(tag, 2, 16, QLatin1Char('0')).arg(GetFossilizeHash(id))); } QByteArray VulkanPipelineStateViewer::ReconstructSpecializationData(const VKPipe::Shader &sh, const SDObject *mapEntries) { bytebuf specData; if(mapEntries->NumChildren() == 0) return specData; if(sh.reflection == NULL) { qCritical("Tried to reconstruct specialization constants but reflection data is missing"); return specData; } auto specBlockIt = std::find_if(sh.reflection->constantBlocks.begin(), sh.reflection->constantBlocks.end(), [](const ConstantBlock &block) { return block.compileConstants; }); if(specBlockIt == sh.reflection->constantBlocks.end()) { qCritical("Cannot find the constant block for specialization constants"); return specData; } const rdcarray &specVars = specBlockIt->variables; // We don't have access to the buffers in the original creation info, so we try to reconstruct // from our preprocessed pipeline state instead. Note that this data might have a different order // from the original call or have unused entries eliminated based on shader reflection. const bytebuf &src = sh.specializationData; for(size_t i = 0; i < mapEntries->NumChildren(); i++) { const SDObject *map = mapEntries->GetChild(i); size_t dstByteOffset = map->FindChild("offset")->AsUInt32(); size_t size = map->FindChild("size")->AsUInt32(); specData.resize_for_index(dstByteOffset + size - 1); uint32_t constantId = map->FindChild("constantID")->AsUInt32(); int32_t idx = sh.specializationIds.indexOf(constantId); if(idx == -1) continue; // Entry was eliminated as it was probably unused --- skip it size_t srcByteOffset = specVars[idx].byteOffset; Q_ASSERT(srcByteOffset + size <= src.size()); memcpy(specData.data() + dstByteOffset, src.data() + srcByteOffset, size); } return specData; } QString VulkanPipelineStateViewer::GetBufferForFossilize(const SDObject *obj) { const VKPipe::State *pipe = m_Ctx.CurVulkanPipelineState(); QByteArray ret; if(obj->name == "pData" && obj->GetParent() && obj->GetParent()->name == "pSpecializationInfo") { const SDObject *shad = obj->GetParent()->GetParent(); const SDObject *stage = NULL; if(shad) stage = shad->FindChild("stage"); const SDObject *mapEntries = obj->GetParent()->FindChild("pMapEntries"); if(stage) { switch(ShaderStageMask(stage->AsUInt32())) { case ShaderStageMask::Vertex: ret = ReconstructSpecializationData(pipe->vertexShader, mapEntries); break; case ShaderStageMask::Tess_Control: ret = ReconstructSpecializationData(pipe->tessControlShader, mapEntries); break; case ShaderStageMask::Tess_Eval: ret = ReconstructSpecializationData(pipe->tessEvalShader, mapEntries); break; case ShaderStageMask::Geometry: ret = ReconstructSpecializationData(pipe->geometryShader, mapEntries); break; case ShaderStageMask::Pixel: ret = ReconstructSpecializationData(pipe->fragmentShader, mapEntries); break; case ShaderStageMask::Compute: ret = ReconstructSpecializationData(pipe->computeShader, mapEntries); break; case ShaderStageMask::Task: ret = ReconstructSpecializationData(pipe->taskShader, mapEntries); break; case ShaderStageMask::Mesh: ret = ReconstructSpecializationData(pipe->meshShader, mapEntries); break; default: break; } } const SDObject *size = obj->GetParent()->FindChild("dataSize"); if(size) { Q_ASSERT((uint32_t)ret.size() <= size->AsUInt32()); ret.resize(size->AsUInt32()); } } return QString::fromLatin1(ret.toBase64()); } void VulkanPipelineStateViewer::AddFossilizeNexts(QVariantMap &info, const SDObject *baseStruct) { QVariantList nexts; while(baseStruct) { const SDObject *next = baseStruct->FindChild("pNext"); if(next && next->type.basetype != SDBasic::Null) { QVariant v = ConvertSDObjectToFossilizeJSON( next, { // VkPipelineVertexInputDivisorStateCreateInfo {"pVertexBindingDivisors", "vertexBindingDivisors"}, // VkRenderPassMultiviewCreateInfo {"subpassCount", ""}, {"pViewMasks", "viewMasks"}, {"dependencyCount", ""}, {"pViewOffsets", "viewOffsets"}, {"correlationMaskCount", ""}, {"pCorrelationMasks", "correlationMasks"}, // VkDescriptorSetLayoutBindingFlagsCreateInfoEXT {"bindingCount", ""}, {"pBindingFlags", "bindingFlags"}, // VkMutableDescriptorTypeCreateInfoEXT {"mutableDescriptorTypeListCount", ""}, {"pMutableDescriptorTypeLists", "mutableDescriptorTypeLists"}, // VkSubpassDescriptionDepthStencilResolve {"pDepthStencilResolveAttachment", "depthStencilResolveAttachment"}, // VkFragmentShadingRateAttachmentInfoKHR {"pFragmentShadingRateAttachment", "fragmentShadingRateAttachment"}, // VkPipelineRenderingCreateInfo {"colorAttachmentCount", ""}, {"pColorAttachmentFormats", "colorAttachmentFormats"}, }); QVariantMap &vm = (QVariantMap &)v.data_ptr(); vm[lit("sType")] = next->FindChild("sType")->AsUInt32(); nexts.push_back(v); baseStruct = next; } else { break; } } if(!nexts.empty()) { info[lit("pNext")] = nexts; } } QVariant VulkanPipelineStateViewer::ConvertSDObjectToFossilizeJSON(const SDObject *obj, QMap renames) { switch(obj->type.basetype) { case SDBasic::Chunk: case SDBasic::Struct: { QVariantMap map; for(size_t i = 0; i < obj->NumChildren(); i++) { const SDObject *ch = obj->GetChild(i); if(ch->name == "sType" || ch->name == "pNext" || ch->name == "pNextType") continue; QByteArray name(ch->name.c_str(), (int)ch->name.size()); auto it = renames.find(name); if(it != renames.end()) name = it.value(); if(name.isEmpty()) continue; QString key = QString::fromLatin1(name); QVariant v = ConvertSDObjectToFossilizeJSON(ch, renames); if(v.isValid()) map[key] = v; } // VkMutableDescriptorTypeListEXT if(map.contains(lit("pDescriptorTypes"))) return map[lit("pDescriptorTypes")]; else if(map.contains(lit("descriptorTypeCount"))) return QVariantList(); AddFossilizeNexts(map, obj); if(map.contains(lit("sampleMask"))) { QVariantList sampleMaskArray = QVariantList({map[lit("sampleMask")]}); map[lit("sampleMask")] = sampleMaskArray; } return map; } case SDBasic::Null: break; case SDBasic::Buffer: return GetBufferForFossilize(obj); case SDBasic::Array: { if(obj->NumChildren() == 0) return QVariant(); QVariantList list; for(size_t j = 0; j < obj->NumChildren(); j++) list.push_back(ConvertSDObjectToFossilizeJSON(obj->GetChild(j), renames)); return list; } case SDBasic::String: return QString(obj->AsString()); case SDBasic::GPUAddress: case SDBasic::Enum: case SDBasic::UnsignedInteger: return (qulonglong)obj->AsUInt64(); case SDBasic::SignedInteger: return (qlonglong)obj->AsInt64(); case SDBasic::Float: return obj->AsDouble(); case SDBasic::Boolean: return obj->AsBool() ? 1U : 0U; case SDBasic::Character: return QString(QLatin1Char(obj->AsChar())); case SDBasic::Resource: return GetFossilizeHash(obj->AsResourceId()); } return QVariant(); } void VulkanPipelineStateViewer::EncodeFossilizeVarint(const bytebuf &spirv, bytebuf &varint) { if((spirv.size() % 4) != 0) return; const uint32_t *curWord = (const uint32_t *)spirv.data(); varint.reserve(spirv.size() / 2); for(size_t i = 0; i < spirv.size(); i += 4) { uint32_t w = *curWord; do { if(w <= 0x7f) varint.push_back(uint8_t(w)); else varint.push_back(uint8_t(w & 0x7fU) | 0x80U); w >>= 7; } while(w); curWord++; } } void VulkanPipelineStateViewer::WriteFossilizeJSON(QIODevice &f, QVariantMap &contents) { contents[lit("version")] = 6; QJsonDocument doc = QJsonDocument::fromVariant(contents); QByteArray jsontext = doc.toJson(QJsonDocument::Compact); f.write(jsontext); } void VulkanPipelineStateViewer::exportFOZ(QString dir, ResourceId pso) { enum { TagAppInfo = 0, TagSampler = 1, TagDescriptorSetLayout = 2, TagPipelineLayout = 3, TagShaderModule = 4, TagRenderPass = 5, TagGraphicsPipe = 6, TagComputePipe = 7, }; QDir d(dir); const SDFile &sdfile = m_Ctx.GetStructuredFile(); const VKPipe::State *pipe = m_Ctx.CurVulkanPipelineState(); // enumerate all the parents of the pipeline, and cache the name of the first initialisation // chunk (easy way to find things by type) rdcarray> resources; { rdcarray todo; rdcarray done; todo.push_back(pso); while(!todo.empty()) { ResourceId cur = todo.back(); todo.pop_back(); const ResourceDescription *desc = m_Ctx.GetResource(cur); resources.push_back({sdfile.chunks[desc->initialisationChunks[0]]->name, desc}); done.push_back(cur); for(ResourceId parent : desc->parentResources) { if(!done.contains(parent)) todo.push_back(parent); } } } { const ResourceDescription *instance = NULL; const ResourceDescription *device = NULL; for(size_t i = 0; i < resources.size(); i++) { if(resources[i].first == "vkCreateInstance") instance = resources[i].second; else if(resources[i].first == "vkCreateDevice") device = resources[i].second; } if(!instance || instance->type != ResourceType::Device) { RDDialog::critical(this, tr("Couldn't locate instance"), tr("Couldn't locate VkInstance from current PSO!")); return; } if(!device || device->type != ResourceType::Device) { RDDialog::critical(this, tr("Couldn't locate device"), tr("Couldn't locate VkDevice from current PSO!")); return; } QFile f(GetFossilizeFilename(d, TagAppInfo, instance->resourceId)); if(f.open(QIODevice::WriteOnly | QIODevice::Truncate | QIODevice::Text)) { QVariantMap instanceData; const SDChunk *instCreate = sdfile.chunks[instance->initialisationChunks[0]]; QVariantMap appInfo; QVariantMap physicalDeviceFeatures; const SDObject *apiVersion = instCreate->FindChildRecursively("APIVersion"); if(apiVersion && apiVersion->AsUInt32() > 0) { appInfo[lit("applicationName")] = instCreate->FindChildRecursively("AppName")->AsString(); appInfo[lit("engineName")] = instCreate->FindChildRecursively("EngineName")->AsString(); appInfo[lit("applicationVersion")] = instCreate->FindChildRecursively("AppVersion")->AsUInt32(); appInfo[lit("engineVersion")] = instCreate->FindChildRecursively("EngineVersion")->AsUInt32(); appInfo[lit("apiVersion")] = apiVersion->AsUInt32(); } const SDChunk *devCreate = sdfile.chunks[device->initialisationChunks[0]]; // this is a recursive search so we don't need to care if it's in PDF or PDF2 const SDObject *robustBufferAccess = devCreate->FindChildRecursively("robustBufferAccess"); if(robustBufferAccess) { physicalDeviceFeatures[lit("robustBufferAccess")] = robustBufferAccess->AsUInt32(); } instanceData[lit("applicationInfo")] = appInfo; instanceData[lit("physicalDeviceFeatures")] = physicalDeviceFeatures; WriteFossilizeJSON(f, instanceData); } } for(size_t i = 0; i < resources.size(); i++) { const SDChunk *create = sdfile.chunks[resources[i].second->initialisationChunks[0]]; ResourceId id = resources[i].second->resourceId; if(resources[i].first == "vkCreateSampler") { QFile f(GetFossilizeFilename(d, TagSampler, id)); if(f.open(QIODevice::WriteOnly | QIODevice::Truncate | QIODevice::Text)) { const SDObject *createInfo = create->FindChildRecursively("CreateInfo"); QVariant samplerData = ConvertSDObjectToFossilizeJSON(createInfo, {}); QVariantMap root({{lit("samplers"), QVariantMap({{GetFossilizeHash(id), samplerData}})}}); WriteFossilizeJSON(f, root); } } else if(resources[i].first == "vkCreateDescriptorSetLayout") { QFile f(GetFossilizeFilename(d, TagDescriptorSetLayout, id)); if(f.open(QIODevice::WriteOnly | QIODevice::Truncate | QIODevice::Text)) { const SDObject *createInfo = create->FindChildRecursively("CreateInfo"); QVariant layoutData = ConvertSDObjectToFossilizeJSON(createInfo, { {"bindingCount", ""}, {"pBindings", "bindings"}, }); QVariantMap root({{lit("setLayouts"), QVariantMap({{GetFossilizeHash(id), layoutData}})}}); WriteFossilizeJSON(f, root); } } else if(resources[i].first == "vkCreatePipelineLayout") { QFile f(GetFossilizeFilename(d, TagPipelineLayout, id)); if(f.open(QIODevice::WriteOnly | QIODevice::Truncate | QIODevice::Text)) { const SDObject *createInfo = create->FindChildRecursively("CreateInfo"); QVariant layoutData = ConvertSDObjectToFossilizeJSON( createInfo, { {"setLayoutCount", ""}, {"pSetLayouts", "setLayouts"}, {"pushConstantRangeCount", ""}, {"pPushConstantRanges", "pushConstantRanges"}, }); QVariantMap root( {{lit("pipelineLayouts"), QVariantMap({{GetFossilizeHash(id), layoutData}})}}); WriteFossilizeJSON(f, root); } } else if(resources[i].first == "vkCreateRenderPass" || resources[i].first == "vkCreateRenderPass2") { QFile f(GetFossilizeFilename(d, TagRenderPass, id)); if(f.open(QIODevice::WriteOnly | QIODevice::Truncate | QIODevice::Text)) { const SDObject *createInfo = create->FindChildRecursively("CreateInfo"); QVariant layoutData = ConvertSDObjectToFossilizeJSON( createInfo, { {"attachmentCount", ""}, {"pAttachments", "attachments"}, {"dependencyCount", ""}, {"pDependencies", "dependencies"}, {"subpassCount", ""}, {"pSubpasses", "subpasses"}, {"pDepthStencilAttachment", "depthStencilAttachment"}, {"colorAttachmentCount", ""}, {"pColorAttachments", "colorAttachments"}, {"inputAttachmentCount", ""}, {"pInputAttachments", "inputAttachments"}, {"preserveAttachmentCount", ""}, {"pPreserveAttachments", "preserveAttachments"}, {"resolveAttachmentCount", ""}, {"pResolveAttachments", "resolveAttachments"}, }); QVariantMap root({{lit("renderPasses"), QVariantMap({{GetFossilizeHash(id), layoutData}})}}); WriteFossilizeJSON(f, root); } } else if(resources[i].first == "vkCreateGraphicsPipelines") { QFile f(GetFossilizeFilename(d, TagGraphicsPipe, id)); if(f.open(QIODevice::WriteOnly | QIODevice::Truncate | QIODevice::Text)) { const SDObject *createInfo = create->FindChildRecursively("CreateInfo"); QVariant layoutData = ConvertSDObjectToFossilizeJSON( createInfo, { {"pName", "name"}, {"mapEntryCount", ""}, {"pMapEntries", "mapEntries"}, {"pSpecializationInfo", "specializationInfo"}, {"pData", "data"}, {"pTessellationState", "tessellationState"}, {"pDynamicState", "dynamicState"}, {"pMultisampleState", "multisampleState"}, {"pSampleMask", "sampleMask"}, {"pVertexInputState", "vertexInputState"}, {"vertexAttributeDescriptionCount", ""}, {"vertexBindingDescriptionCount", ""}, {"pVertexAttributeDescriptions", "attributes"}, {"pVertexBindingDescriptions", "bindings"}, {"pRasterizationState", "rasterizationState"}, {"pInputAssemblyState", "inputAssemblyState"}, {"pColorBlendState", "colorBlendState"}, {"attachmentCount", ""}, {"pAttachments", "attachments"}, {"pViewportState", "viewportState"}, {"dynamicStateCount", ""}, {"pDynamicStates", "dynamicState"}, {"pViewports", "viewports"}, {"pScissors", "scissors"}, {"pDepthStencilState", "depthStencilState"}, {"stageCount", ""}, {"pStages", "stages"}, }); QVariantMap root( {{lit("graphicsPipelines"), QVariantMap({{GetFossilizeHash(id), layoutData}})}}); WriteFossilizeJSON(f, root); } } else if(resources[i].first == "vkCreateComputePipelines") { QFile f(GetFossilizeFilename(d, TagComputePipe, id)); if(f.open(QIODevice::WriteOnly | QIODevice::Truncate | QIODevice::Text)) { const SDObject *createInfo = create->FindChildRecursively("CreateInfo"); QVariant layoutData = ConvertSDObjectToFossilizeJSON( createInfo, { {"pName", "name"}, {"mapEntryCount", ""}, {"pMapEntries", "mapEntries"}, {"pSpecializationInfo", "specializationInfo"}, {"pData", "data"}, }); QVariantMap root( {{lit("computePipelines"), QVariantMap({{GetFossilizeHash(id), layoutData}})}}); WriteFossilizeJSON(f, root); } } else if(resources[i].first == "vkCreateShaderModule") { QFile f(GetFossilizeFilename(d, TagShaderModule, id)); if(f.open(QIODevice::WriteOnly | QIODevice::Truncate)) { // shaders we handle specially QVariantMap shaderData; const bytebuf *spirv = NULL; // we don't care which reflection we get, as long as the ID matches for(const VKPipe::Shader *sh : {&pipe->taskShader, &pipe->meshShader, &pipe->vertexShader, &pipe->tessControlShader, &pipe->tessEvalShader, &pipe->geometryShader, &pipe->fragmentShader, &pipe->computeShader}) { if(sh->resourceId == id) spirv = &sh->reflection->rawBytes; } if(!spirv) { RDDialog::critical( this, tr("Shader not found"), tr("Couldn't get SPIR-V bytes for bound shader %1").arg(m_Ctx.GetResourceName(id))); return; } bytebuf varint; EncodeFossilizeVarint(*spirv, varint); shaderData[lit("varintOffset")] = 0; shaderData[lit("varintSize")] = qulonglong(varint.size()); shaderData[lit("codeSize")] = qulonglong(spirv->size()); shaderData[lit("flags")] = create->FindChildRecursively("CreateInfo")->FindChild("flags")->AsUInt32(); QVariantMap root({{lit("shaderModules"), QVariantMap({{GetFossilizeHash(id), shaderData}})}}); WriteFossilizeJSON(f, root); f.write(QByteArray(1, '\0')); f.write((const char *)varint.data(), (qint64)varint.size()); } } } } uint32_t VulkanPipelineStateViewer::getMinOffset(const rdcarray &variables) { uint32_t minOffset = ~0U; for(const ShaderConstant &v : variables) minOffset = qMin(v.byteOffset, minOffset); return minOffset; } void VulkanPipelineStateViewer::exportFOZ_clicked() { if(!m_Ctx.IsCaptureLoaded()) return; if(!m_Ctx.CurAction()) { RDDialog::critical(this, tr("No action selected"), tr("To export the pipeline as FOZ an action must be selected.")); return; } ResourceId pso; if(m_Ctx.CurAction()->flags & ActionFlags::Dispatch) pso = m_Ctx.CurVulkanPipelineState()->compute.pipelineResourceId; else if(m_Ctx.CurAction()->flags & (ActionFlags::MeshDispatch | ActionFlags::Drawcall)) pso = m_Ctx.CurVulkanPipelineState()->graphics.pipelineResourceId; if(pso == ResourceId()) { RDDialog::critical( this, tr("No pipeline bound"), tr("To export the pipeline as FOZ an action must be selected which has a pipeline bound.")); return; } QString dir = RDDialog::getExistingDirectory(this, tr("Export pipeline state as fossilize DB")); if(!dir.isEmpty()) exportFOZ(dir, pso); } void VulkanPipelineStateViewer::exportHTML_clicked() { if(!m_Ctx.IsCaptureLoaded()) return; QXmlStreamWriter *xmlptr = m_Common.beginHTMLExport(); if(xmlptr) { QXmlStreamWriter &xml = *xmlptr; const QStringList &stageNames = ui->pipeFlow->stageNames(); const QStringList &stageAbbrevs = ui->pipeFlow->stageAbbreviations(); int stage = 0; for(const QString &sn : stageNames) { xml.writeStartElement(lit("div")); xml.writeStartElement(lit("a")); xml.writeAttribute(lit("name"), stageAbbrevs[stage]); xml.writeEndElement(); xml.writeEndElement(); xml.writeStartElement(lit("div")); xml.writeAttribute(lit("class"), lit("stage")); xml.writeStartElement(lit("h1")); xml.writeCharacters(sn); xml.writeEndElement(); if(m_MeshPipe) { switch(stage) { case 0: exportHTML(xml, m_Ctx.CurVulkanPipelineState()->taskShader); break; case 1: exportHTML(xml, m_Ctx.CurVulkanPipelineState()->meshShader); break; case 2: exportHTML(xml, m_Ctx.CurVulkanPipelineState()->rasterizer); exportHTML(xml, m_Ctx.CurVulkanPipelineState()->conditionalRendering); break; case 3: exportHTML(xml, m_Ctx.CurVulkanPipelineState()->fragmentShader); break; case 4: // FB xml.writeStartElement(lit("h2")); xml.writeCharacters(tr("Color Blend")); xml.writeEndElement(); exportHTML(xml, m_Ctx.CurVulkanPipelineState()->colorBlend); xml.writeStartElement(lit("h2")); xml.writeCharacters(tr("Depth Stencil")); xml.writeEndElement(); exportHTML(xml, m_Ctx.CurVulkanPipelineState()->depthStencil); xml.writeStartElement(lit("h2")); xml.writeCharacters(tr("Current Pass")); xml.writeEndElement(); exportHTML(xml, m_Ctx.CurVulkanPipelineState()->currentPass); break; case 5: exportHTML(xml, m_Ctx.CurVulkanPipelineState()->computeShader); exportHTML(xml, m_Ctx.CurVulkanPipelineState()->conditionalRendering); break; } } else { switch(stage) { case 0: // VTX xml.writeStartElement(lit("h2")); xml.writeCharacters(tr("Input Assembly")); xml.writeEndElement(); exportHTML(xml, m_Ctx.CurVulkanPipelineState()->inputAssembly); xml.writeStartElement(lit("h2")); xml.writeCharacters(tr("Vertex Input")); xml.writeEndElement(); exportHTML(xml, m_Ctx.CurVulkanPipelineState()->vertexInput); break; case 1: exportHTML(xml, m_Ctx.CurVulkanPipelineState()->vertexShader); break; case 2: exportHTML(xml, m_Ctx.CurVulkanPipelineState()->tessControlShader); break; case 3: exportHTML(xml, m_Ctx.CurVulkanPipelineState()->tessEvalShader); break; case 4: exportHTML(xml, m_Ctx.CurVulkanPipelineState()->geometryShader); exportHTML(xml, m_Ctx.CurVulkanPipelineState()->transformFeedback); break; case 5: exportHTML(xml, m_Ctx.CurVulkanPipelineState()->rasterizer); exportHTML(xml, m_Ctx.CurVulkanPipelineState()->conditionalRendering); break; case 6: exportHTML(xml, m_Ctx.CurVulkanPipelineState()->fragmentShader); break; case 7: // FB xml.writeStartElement(lit("h2")); xml.writeCharacters(tr("Color Blend")); xml.writeEndElement(); exportHTML(xml, m_Ctx.CurVulkanPipelineState()->colorBlend); xml.writeStartElement(lit("h2")); xml.writeCharacters(tr("Depth Stencil")); xml.writeEndElement(); exportHTML(xml, m_Ctx.CurVulkanPipelineState()->depthStencil); xml.writeStartElement(lit("h2")); xml.writeCharacters(tr("Current Pass")); xml.writeEndElement(); exportHTML(xml, m_Ctx.CurVulkanPipelineState()->currentPass); break; case 8: exportHTML(xml, m_Ctx.CurVulkanPipelineState()->computeShader); exportHTML(xml, m_Ctx.CurVulkanPipelineState()->conditionalRendering); break; } } xml.writeEndElement(); stage++; } m_Common.endHTMLExport(xmlptr); } } void VulkanPipelineStateViewer::on_msMeshButton_clicked() { if(!m_Ctx.HasMeshPreview()) m_Ctx.ShowMeshPreview(); ToolWindowManager::raiseToolWindow(m_Ctx.GetMeshPreview()->Widget()); } void VulkanPipelineStateViewer::on_meshView_clicked() { if(!m_Ctx.HasMeshPreview()) m_Ctx.ShowMeshPreview(); ToolWindowManager::raiseToolWindow(m_Ctx.GetMeshPreview()->Widget()); } void VulkanPipelineStateViewer::on_computeDebugSelector_clicked() { // Check whether debugging is valid for this event before showing the dialog if(!m_Ctx.APIProps().shaderDebugging) return; if(!m_Ctx.IsCaptureLoaded()) return; const ActionDescription *action = m_Ctx.CurAction(); if(!action) return; const ShaderReflection *shaderDetails = m_Ctx.CurPipelineState().GetShaderReflection(ShaderStage::Compute); if(!shaderDetails) return; RDDialog::show(m_ComputeDebugSelector); } void VulkanPipelineStateViewer::computeDebugSelector_beginDebug( const rdcfixedarray &group, const rdcfixedarray &thread) { const ActionDescription *action = m_Ctx.CurAction(); if(!action) return; const ShaderReflection *shaderDetails = m_Ctx.CurPipelineState().GetShaderReflection(ShaderStage::Compute); if(!shaderDetails) return; struct threadSelect { rdcfixedarray g; rdcfixedarray t; } debugThread = { // g[] {group[0], group[1], group[2]}, // t[] {thread[0], thread[1], thread[2]}, }; bool done = false; ShaderDebugTrace *trace = NULL; m_Ctx.Replay().AsyncInvoke([&trace, &done, debugThread](IReplayController *r) { trace = r->DebugThread(debugThread.g, debugThread.t); if(trace->debugger == NULL) { r->FreeTrace(trace); trace = NULL; } done = true; }); QString debugContext = lit("Group [%1,%2,%3] Thread [%4,%5,%6]") .arg(group[0]) .arg(group[1]) .arg(group[2]) .arg(thread[0]) .arg(thread[1]) .arg(thread[2]); // wait a short while before displaying the progress dialog (which won't show if we're already // done by the time we reach it) for(int i = 0; !done && i < 100; i++) QThread::msleep(5); ShowProgressDialog(this, tr("Debugging %1").arg(debugContext), [&done]() { return done; }); if(!trace) { RDDialog::critical( this, tr("Error debugging"), tr("Error debugging thread - make sure a valid group and thread is selected")); return; } // viewer takes ownership of the trace IShaderViewer *s = m_Ctx.DebugShader( shaderDetails, m_Ctx.CurPipelineState().GetComputePipelineObject(), trace, debugContext); m_Ctx.AddDockWindow(s->Widget(), DockReference::AddTo, this); }