blob: ebc8305e29ec65b5075794915e1de588c238efc5 [file] [log] [blame]
// Copyright (C) 2024 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#include <string>
#include "GfxstreamEnd2EndTestUtils.h"
#include "GfxstreamEnd2EndTests.h"
#include "shaders/simple_shader_frag.h"
#include "shaders/simple_shader_vert.h"
namespace gfxstream {
namespace tests {
namespace {
using testing::Eq;
using testing::Ge;
using testing::IsEmpty;
using testing::IsNull;
using testing::Not;
using testing::NotNull;
struct PipelineInfo {
vkhpp::UniqueRenderPass renderPass;
vkhpp::UniqueDescriptorSetLayout descriptorSetLayout;
vkhpp::UniquePipelineLayout pipelineLayout;
vkhpp::UniqueShaderModule vertexShaderModule;
vkhpp::UniqueShaderModule fragmentShaderModule;
vkhpp::UniquePipeline pipeline;
};
struct ImageInfo {
vkhpp::UniqueImage image;
vkhpp::UniqueDeviceMemory memory;
vkhpp::UniqueImageView imageView;
};
struct BufferInfo {
vkhpp::UniqueBuffer buffer;
vkhpp::UniqueDeviceMemory memory;
};
class GfxstreamEnd2EndVkSnapshotPipelineTest : public GfxstreamEnd2EndTest {
protected:
vkhpp::UniqueRenderPass createRenderPass(vkhpp::Device device);
std::unique_ptr<ImageInfo> createColorAttachment(vkhpp::PhysicalDevice physicalDevice,
vkhpp::Device device);
std::unique_ptr<PipelineInfo> createPipeline(vkhpp::Device device);
Result<BufferInfo> createAndPopulateBuffer(vkhpp::PhysicalDevice physicalDevice,
vkhpp::Device device, vkhpp::BufferUsageFlags usage,
const void* data, uint64_t dataSize);
static const uint32_t kFbWidth = 32;
static const uint32_t kFbHeight = 32;
};
class GfxstreamEnd2EndVkSnapshotPipelineWithMultiSamplingTest
: public GfxstreamEnd2EndVkSnapshotPipelineTest {};
template <typename DurationType>
constexpr uint64_t AsVkTimeout(DurationType duration) {
return static_cast<uint64_t>(
std::chrono::duration_cast<std::chrono::nanoseconds>(duration).count());
}
// Full screen blue rectangle
const float kFullscreenBlueRectangleVertexData[] = {
// clang-format off
/*pos=*/ -1.0f, -1.0f, 0.0f, 1.0f, /*color=*/ 0.0f, 0.0f, 1.0f, 1.0f,
/*pos=*/ 1.0f, -1.0f, 0.0f, 1.0f, /*color=*/ 0.0f, 0.0f, 1.0f, 1.0f,
/*pos=*/ 1.0f, 1.0f, 0.0f, 1.0f, /*color=*/ 0.0f, 0.0f, 1.0f, 1.0f,
/*pos=*/ 1.0f, 1.0f, 0.0f, 1.0f, /*color=*/ 0.0f, 0.0f, 1.0f, 1.0f,
/*pos=*/ -1.0f, 1.0f, 0.0f, 1.0f, /*color=*/ 0.0f, 0.0f, 1.0f, 1.0f,
/*pos=*/ -1.0f, -1.0f, 0.0f, 1.0f, /*color=*/ 0.0f, 0.0f, 1.0f, 1.0f,
// clang-format on
};
Result<BufferInfo> GfxstreamEnd2EndVkSnapshotPipelineTest::createAndPopulateBuffer(
vkhpp::PhysicalDevice physicalDevice, vkhpp::Device device, vkhpp::BufferUsageFlags usage,
const void* data, uint64_t dataSize) {
const vkhpp::BufferCreateInfo vertexBufferCreateInfo = {
.size = dataSize,
.usage = usage,
.sharingMode = vkhpp::SharingMode::eExclusive,
};
auto vertexBuffer =
GFXSTREAM_EXPECT_VKHPP_RV(device.createBufferUnique(vertexBufferCreateInfo));
vkhpp::MemoryRequirements vertexBufferMemoryRequirements{};
device.getBufferMemoryRequirements(*vertexBuffer, &vertexBufferMemoryRequirements);
const auto vertexBufferMemoryType = utils::getMemoryType(
physicalDevice, vertexBufferMemoryRequirements,
vkhpp::MemoryPropertyFlagBits::eHostVisible | vkhpp::MemoryPropertyFlagBits::eHostCoherent);
if (vertexBufferMemoryType == -1) {
return gfxstream::unexpected("Failed to allocate buffer memory.");
}
// Vertex memory
const vkhpp::MemoryAllocateInfo vertexBufferMemoryAllocateInfo = {
.allocationSize = vertexBufferMemoryRequirements.size,
.memoryTypeIndex = vertexBufferMemoryType,
};
auto vertexBufferMemory =
GFXSTREAM_EXPECT_VKHPP_RV(device.allocateMemoryUnique(vertexBufferMemoryAllocateInfo));
device.bindBufferMemory(*vertexBuffer, *vertexBufferMemory, 0);
void* mapped;
device.mapMemory(*vertexBufferMemory, 0, VK_WHOLE_SIZE, vkhpp::MemoryMapFlags{}, &mapped);
memcpy(mapped, data, dataSize);
device.unmapMemory(*vertexBufferMemory);
BufferInfo res;
res.buffer = std::move(vertexBuffer);
res.memory = std::move(vertexBufferMemory);
return res;
}
vkhpp::UniqueRenderPass GfxstreamEnd2EndVkSnapshotPipelineTest::createRenderPass(
vkhpp::Device device) {
vkhpp::AttachmentDescription colorAttachmentDescription = {
.format = vkhpp::Format::eR8G8B8A8Unorm,
.samples = static_cast<vkhpp::SampleCountFlagBits>(GetParam().samples),
.loadOp = vkhpp::AttachmentLoadOp::eLoad,
.storeOp = vkhpp::AttachmentStoreOp::eStore,
.initialLayout = vkhpp::ImageLayout::eColorAttachmentOptimal,
.finalLayout = vkhpp::ImageLayout::eColorAttachmentOptimal,
};
vkhpp::AttachmentReference attachmentReference = {
.attachment = 0,
.layout = vkhpp::ImageLayout::eColorAttachmentOptimal,
};
vkhpp::SubpassDescription subpassDescription = {
.colorAttachmentCount = 1,
.pColorAttachments = &attachmentReference,
};
vkhpp::RenderPassCreateInfo renderPassCreateInfo = {
.attachmentCount = 1,
.pAttachments = &colorAttachmentDescription,
.subpassCount = 1,
.pSubpasses = &subpassDescription,
};
return device.createRenderPassUnique(renderPassCreateInfo).value;
}
std::unique_ptr<PipelineInfo> GfxstreamEnd2EndVkSnapshotPipelineTest::createPipeline(
vkhpp::Device device) {
std::unique_ptr<PipelineInfo> res(new PipelineInfo);
res->renderPass = createRenderPass(device);
vkhpp::DescriptorSetLayoutBinding bindings[1] = {
{
.binding = 0,
.descriptorType = vkhpp::DescriptorType::eUniformBuffer,
.descriptorCount = 1,
.stageFlags = vkhpp::ShaderStageFlagBits::eFragment,
},
};
vkhpp::DescriptorSetLayoutCreateInfo descriptorSetLayoutInfo = {
.bindingCount = 1,
.pBindings = bindings,
};
res->descriptorSetLayout =
device.createDescriptorSetLayoutUnique(descriptorSetLayoutInfo).value;
res->pipelineLayout = device
.createPipelineLayoutUnique(vkhpp::PipelineLayoutCreateInfo{
.setLayoutCount = 1,
.pSetLayouts = &res->descriptorSetLayout.get(),
})
.value;
vkhpp::ShaderModuleCreateInfo vertexShaderModuleCreateInfo = {
.codeSize = kSimpleShaderVert.size() * sizeof(uint32_t),
.pCode = kSimpleShaderVert.data(),
};
vkhpp::ShaderModuleCreateInfo fragmentShaderModuleCreateInfo = {
.codeSize = kSimpleShaderFrag.size() * sizeof(uint32_t),
.pCode = kSimpleShaderFrag.data(),
};
res->vertexShaderModule = device.createShaderModuleUnique(vertexShaderModuleCreateInfo).value;
res->fragmentShaderModule =
device.createShaderModuleUnique(fragmentShaderModuleCreateInfo).value;
vkhpp::PipelineShaderStageCreateInfo pipelineShaderStageCreateInfos[2] = {
{
.stage = vkhpp::ShaderStageFlagBits::eVertex,
.module = *(res->vertexShaderModule),
.pName = "main",
},
{
.stage = vkhpp::ShaderStageFlagBits::eFragment,
.module = *(res->fragmentShaderModule),
.pName = "main",
},
};
const vkhpp::VertexInputBindingDescription vertexInputBindingDescription = {
.stride = 32,
};
vkhpp::VertexInputAttributeDescription vertexInputAttributeDescriptions[2] = {
{
.location = 0,
.format = vkhpp::Format::eR32G32B32A32Sfloat,
.offset = 0,
},
{
.location = 1,
.format = vkhpp::Format::eR32G32B32A32Sfloat,
.offset = 16,
},
};
const vkhpp::PipelineVertexInputStateCreateInfo pipelineVertexInputStateCreateInfo = {
.vertexBindingDescriptionCount = 1,
.pVertexBindingDescriptions = &vertexInputBindingDescription,
.vertexAttributeDescriptionCount = 2,
.pVertexAttributeDescriptions = vertexInputAttributeDescriptions,
};
const vkhpp::PipelineInputAssemblyStateCreateInfo pipelineInputAssemblyStateCreateInfo = {
.topology = vkhpp::PrimitiveTopology::eTriangleList,
};
const vkhpp::PipelineViewportStateCreateInfo pipelineViewportStateCreateInfo = {
.viewportCount = 1,
.scissorCount = 1,
};
const vkhpp::PipelineRasterizationStateCreateInfo pipelineRasterizationStateCreateInfo = {
.cullMode = vkhpp::CullModeFlagBits::eNone,
.lineWidth = 1.0f,
};
const vkhpp::PipelineMultisampleStateCreateInfo pipelineMultisampleStateCreateInfo = {
.rasterizationSamples = static_cast<vkhpp::SampleCountFlagBits>(GetParam().samples),
};
const vkhpp::PipelineDepthStencilStateCreateInfo pipelineDepthStencilStateCreateInfo = {};
const vkhpp::PipelineColorBlendAttachmentState pipelineColorBlendAttachmentState = {
.colorBlendOp = vkhpp::BlendOp::eAdd,
.srcAlphaBlendFactor = vkhpp::BlendFactor::eZero,
.dstAlphaBlendFactor = vkhpp::BlendFactor::eZero,
.alphaBlendOp = vkhpp::BlendOp::eAdd,
.colorWriteMask = vkhpp::ColorComponentFlagBits::eR | vkhpp::ColorComponentFlagBits::eG |
vkhpp::ColorComponentFlagBits::eB | vkhpp::ColorComponentFlagBits::eA};
const vkhpp::PipelineColorBlendStateCreateInfo pipelineColorBlendStateCreateInfo = {
.attachmentCount = 1,
.pAttachments = &pipelineColorBlendAttachmentState,
};
const vkhpp::DynamicState dynamicStates[2] = {vkhpp::DynamicState::eViewport,
vkhpp::DynamicState::eScissor};
const vkhpp::PipelineDynamicStateCreateInfo pipelineDynamicStateCreateInfo = {
.dynamicStateCount = 2,
.pDynamicStates = dynamicStates,
};
vkhpp::GraphicsPipelineCreateInfo graphicsPipelineCreateInfo = {
.stageCount = 2,
.pStages = pipelineShaderStageCreateInfos,
.pVertexInputState = &pipelineVertexInputStateCreateInfo,
.pInputAssemblyState = &pipelineInputAssemblyStateCreateInfo,
.pViewportState = &pipelineViewportStateCreateInfo,
.pRasterizationState = &pipelineRasterizationStateCreateInfo,
.pMultisampleState = &pipelineMultisampleStateCreateInfo,
.pDepthStencilState = &pipelineDepthStencilStateCreateInfo,
.pColorBlendState = &pipelineColorBlendStateCreateInfo,
.pDynamicState = &pipelineDynamicStateCreateInfo,
.layout = *(res->pipelineLayout),
.renderPass = *(res->renderPass),
};
res->pipeline = device.createGraphicsPipelineUnique(nullptr, graphicsPipelineCreateInfo).value;
return res;
}
std::unique_ptr<ImageInfo> GfxstreamEnd2EndVkSnapshotPipelineTest::createColorAttachment(
vkhpp::PhysicalDevice physicalDevice, vkhpp::Device device) {
std::unique_ptr<ImageInfo> res(new ImageInfo);
const vkhpp::ImageCreateInfo imageCreateInfo = {
.pNext = nullptr,
.imageType = vkhpp::ImageType::e2D,
.extent.width = kFbWidth,
.extent.height = kFbHeight,
.extent.depth = 1,
.mipLevels = 1,
.arrayLayers = 1,
.format = vkhpp::Format::eR8G8B8A8Unorm,
.tiling = vkhpp::ImageTiling::eOptimal,
.initialLayout = vkhpp::ImageLayout::eUndefined,
.usage = vkhpp::ImageUsageFlagBits::eColorAttachment | vkhpp::ImageUsageFlagBits::eSampled |
vkhpp::ImageUsageFlagBits::eTransferDst | vkhpp::ImageUsageFlagBits::eTransferSrc,
.sharingMode = vkhpp::SharingMode::eExclusive,
.samples = static_cast<vkhpp::SampleCountFlagBits>(GetParam().samples),
};
res->image = device.createImageUnique(imageCreateInfo).value;
vkhpp::MemoryRequirements imageMemoryRequirements{};
device.getImageMemoryRequirements(*(res->image), &imageMemoryRequirements);
const uint32_t imageMemoryIndex = utils::getMemoryType(
physicalDevice, imageMemoryRequirements, vkhpp::MemoryPropertyFlagBits::eDeviceLocal);
const vkhpp::MemoryAllocateInfo imageMemoryAllocateInfo = {
.allocationSize = imageMemoryRequirements.size,
.memoryTypeIndex = imageMemoryIndex,
};
res->memory = device.allocateMemoryUnique(imageMemoryAllocateInfo).value;
device.bindImageMemory(*(res->image), *(res->memory), 0);
const vkhpp::ImageViewCreateInfo imageViewCreateInfo = {
.image = *(res->image),
.viewType = vkhpp::ImageViewType::e2D,
.format = vkhpp::Format::eR8G8B8A8Unorm,
.subresourceRange =
{
.aspectMask = vkhpp::ImageAspectFlagBits::eColor,
.baseMipLevel = 0,
.levelCount = 1,
.baseArrayLayer = 0,
.layerCount = 1,
},
};
res->imageView = device.createImageViewUnique(imageViewCreateInfo).value;
return res;
}
TEST_P(GfxstreamEnd2EndVkSnapshotPipelineTest, CanRecreateShaderModule) {
auto [instance, physicalDevice, device, queue, queueFamilyIndex] =
GFXSTREAM_ASSERT(SetUpTypicalVkTestEnvironment());
auto pipelineInfo = createPipeline(device.get());
ASSERT_THAT(pipelineInfo->renderPass, IsValidHandle());
ASSERT_THAT(pipelineInfo->descriptorSetLayout, IsValidHandle());
ASSERT_THAT(pipelineInfo->pipelineLayout, IsValidHandle());
ASSERT_THAT(pipelineInfo->vertexShaderModule, IsValidHandle());
ASSERT_THAT(pipelineInfo->fragmentShaderModule, IsValidHandle());
ASSERT_THAT(pipelineInfo->pipeline, IsValidHandle());
// Check if snapshot can restore the pipeline even after shaders are destroyed.
pipelineInfo->vertexShaderModule.reset();
pipelineInfo->fragmentShaderModule.reset();
SnapshotSaveAndLoad();
// Don't crash
// TODO(b/330763497): try to render something
// TODO(b/330766521): fix dangling shader modules after snapshot load
}
// vkCreateDescriptorPool injects extra handles into the internal handle map, thus add
// a test for it.
TEST_P(GfxstreamEnd2EndVkSnapshotPipelineTest, CanSnapshotDescriptorPool) {
auto [instance, physicalDevice, device, queue, queueFamilyIndex] =
GFXSTREAM_ASSERT(SetUpTypicalVkTestEnvironment());
std::vector<vkhpp::DescriptorPoolSize> sizes = {
{
.descriptorCount = 10,
},
};
vkhpp::DescriptorPoolCreateInfo descriptorPoolCreateInfo = {
.maxSets = 10,
.poolSizeCount = static_cast<uint32_t>(sizes.size()),
.pPoolSizes = sizes.data(),
};
auto descriptorPool0 = device->createDescriptorPoolUnique(descriptorPoolCreateInfo).value;
ASSERT_THAT(descriptorPool0, IsValidHandle());
auto descriptorPool1 = device->createDescriptorPoolUnique(descriptorPoolCreateInfo).value;
ASSERT_THAT(descriptorPool1, IsValidHandle());
vkhpp::DescriptorSetLayoutCreateInfo descriptorSetLayoutInfo = {};
auto descriptorSetLayout =
device->createDescriptorSetLayoutUnique(descriptorSetLayoutInfo).value;
ASSERT_THAT(descriptorSetLayout, IsValidHandle());
SnapshotSaveAndLoad();
const std::vector<vkhpp::DescriptorSetLayout> descriptorSetLayouts(1, *descriptorSetLayout);
vkhpp::DescriptorSetAllocateInfo descriptorSetAllocateInfo0 = {
.descriptorPool = *descriptorPool0,
.descriptorSetCount = 1,
.pSetLayouts = descriptorSetLayouts.data(),
};
auto descriptorSets0 = device->allocateDescriptorSetsUnique(descriptorSetAllocateInfo0);
EXPECT_THAT(descriptorSets0.result, Eq(vkhpp::Result::eSuccess));
vkhpp::DescriptorSetAllocateInfo descriptorSetAllocateInfo1 = {
.descriptorPool = *descriptorPool1,
.descriptorSetCount = 1,
.pSetLayouts = descriptorSetLayouts.data(),
};
auto descriptorSets1 = device->allocateDescriptorSetsUnique(descriptorSetAllocateInfo1);
EXPECT_THAT(descriptorSets1.result, Eq(vkhpp::Result::eSuccess));
}
TEST_P(GfxstreamEnd2EndVkSnapshotPipelineTest, CanSnapshotFramebuffer) {
auto [instance, physicalDevice, device, queue, queueFamilyIndex] =
GFXSTREAM_ASSERT(SetUpTypicalVkTestEnvironment());
auto renderPass = createRenderPass(device.get());
ASSERT_THAT(renderPass, IsValidHandle());
auto colorAttachmentInfo = createColorAttachment(physicalDevice, device.get());
ASSERT_THAT(colorAttachmentInfo->image, IsValidHandle());
ASSERT_THAT(colorAttachmentInfo->memory, IsValidHandle());
ASSERT_THAT(colorAttachmentInfo->imageView, IsValidHandle());
const std::vector<vkhpp::ImageView> attachments(1, *colorAttachmentInfo->imageView);
vkhpp::FramebufferCreateInfo framebufferCreateInfo = {
.renderPass = *renderPass,
.attachmentCount = 1,
.pAttachments = attachments.data(),
.width = kFbWidth,
.height = kFbHeight,
.layers = 1,
};
auto framebuffer = device->createFramebufferUnique(framebufferCreateInfo).value;
ASSERT_THAT(framebuffer, IsValidHandle());
SnapshotSaveAndLoad();
}
TEST_P(GfxstreamEnd2EndVkSnapshotPipelineWithMultiSamplingTest, CanSubmitQueue) {
TypicalVkTestEnvironment testEnvironment = GFXSTREAM_ASSERT(SetUpTypicalVkTestEnvironment());
auto& [instance, physicalDevice, device, queue, queueFamilyIndex] = testEnvironment;
auto pipelineInfo = createPipeline(device.get());
auto colorAttachmentInfo = createColorAttachment(physicalDevice, device.get());
ASSERT_THAT(colorAttachmentInfo->image, IsValidHandle());
ASSERT_THAT(colorAttachmentInfo->memory, IsValidHandle());
ASSERT_THAT(colorAttachmentInfo->imageView, IsValidHandle());
const std::vector<vkhpp::ImageView> attachments(1, *colorAttachmentInfo->imageView);
vkhpp::FramebufferCreateInfo framebufferCreateInfo = {
.renderPass = *pipelineInfo->renderPass,
.attachmentCount = 1,
.pAttachments = attachments.data(),
.width = kFbWidth,
.height = kFbHeight,
.layers = 1,
};
auto framebuffer = device->createFramebufferUnique(framebufferCreateInfo).value;
ASSERT_THAT(framebuffer, IsValidHandle());
auto fence = device->createFenceUnique(vkhpp::FenceCreateInfo()).value;
ASSERT_THAT(fence, IsValidHandle());
const vkhpp::CommandPoolCreateInfo commandPoolCreateInfo = {
.flags = vkhpp::CommandPoolCreateFlagBits::eResetCommandBuffer,
.queueFamilyIndex = queueFamilyIndex,
};
auto commandPool = device->createCommandPoolUnique(commandPoolCreateInfo).value;
ASSERT_THAT(commandPool, IsValidHandle());
const vkhpp::CommandBufferAllocateInfo commandBufferAllocateInfo = {
.level = vkhpp::CommandBufferLevel::ePrimary,
.commandPool = *commandPool,
.commandBufferCount = 1,
};
auto commandBuffers = device->allocateCommandBuffersUnique(commandBufferAllocateInfo).value;
ASSERT_THAT(commandBuffers, Not(IsEmpty()));
auto commandBuffer = std::move(commandBuffers[0]);
ASSERT_THAT(commandBuffer, IsValidHandle());
vkhpp::ClearColorValue clearColor(std::array<float, 4>{1.0f, 0.0f, 1.0f, 1.0f});
vkhpp::ClearValue clearValue{
.color = clearColor,
};
vkhpp::RenderPassBeginInfo renderPassBeginInfo{
.renderPass = *pipelineInfo->renderPass,
.framebuffer = *framebuffer,
.renderArea = vkhpp::Rect2D(vkhpp::Offset2D(0, 0), vkhpp::Extent2D(kFbWidth, kFbHeight)),
.clearValueCount = 1,
.pClearValues = &clearValue,
};
const vkhpp::CommandBufferBeginInfo commandBufferBeginInfo = {
.flags = vkhpp::CommandBufferUsageFlagBits::eOneTimeSubmit,
};
commandBuffer->begin(commandBufferBeginInfo);
const vkhpp::ImageMemoryBarrier colorAttachmentBarrier{
.oldLayout = vkhpp::ImageLayout::eUndefined,
.newLayout = vkhpp::ImageLayout::eColorAttachmentOptimal,
.dstAccessMask = vkhpp::AccessFlagBits::eColorAttachmentRead |
vkhpp::AccessFlagBits::eColorAttachmentWrite,
.srcQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED,
.dstQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED,
.image = *colorAttachmentInfo->image,
.subresourceRange =
{
.aspectMask = vkhpp::ImageAspectFlagBits::eColor,
.levelCount = 1,
.layerCount = 1,
},
};
commandBuffer->pipelineBarrier(
vkhpp::PipelineStageFlagBits::eTopOfPipe | vkhpp::PipelineStageFlagBits::eTransfer,
vkhpp::PipelineStageFlagBits::eColorAttachmentOutput, vkhpp::DependencyFlags(), nullptr,
nullptr, colorAttachmentBarrier);
commandBuffer->end();
std::vector<vkhpp::CommandBuffer> commandBufferHandles;
commandBufferHandles.push_back(*commandBuffer);
const vkhpp::SubmitInfo submitInfo = {
.commandBufferCount = static_cast<uint32_t>(commandBufferHandles.size()),
.pCommandBuffers = commandBufferHandles.data(),
};
queue.submit(submitInfo, *fence);
auto waitResult = device->waitForFences(*fence, VK_TRUE, 3000000000L);
ASSERT_THAT(waitResult, IsVkSuccess());
commandBuffer->reset();
SnapshotSaveAndLoad();
ASSERT_THAT(device->getFenceStatus(*fence), IsVkSuccess());
// TODO(b/332763326): fix validation layer complain about unreleased pipeline layout
// Try to draw something.
// Color attachment layout must be snapshotted, otherwise validation layer will complain.
commandBuffer->begin(commandBufferBeginInfo);
commandBuffer->beginRenderPass(renderPassBeginInfo, vkhpp::SubpassContents::eInline);
commandBuffer->bindPipeline(vkhpp::PipelineBindPoint::eGraphics, *pipelineInfo->pipeline);
vkhpp::ClearAttachment clearAttachment{
.aspectMask = vkhpp::ImageAspectFlagBits::eColor,
.colorAttachment = 0,
.clearValue = clearValue,
};
vkhpp::ClearRect clearRect{
.rect = vkhpp::Rect2D(vkhpp::Offset2D(0, 0), vkhpp::Extent2D(kFbWidth, kFbHeight)),
.baseArrayLayer = 0,
.layerCount = 1,
};
commandBuffer->clearAttachments(1, &clearAttachment, 1, &clearRect);
commandBuffer->endRenderPass();
commandBuffer->end();
queue.submit(submitInfo, *fence);
waitResult = device->waitForFences(*fence, VK_TRUE, 3000000000L);
ASSERT_THAT(waitResult, IsVkSuccess());
if (GetParam().samples != 1) {
return;
}
std::vector<uint32_t> dst(kFbWidth * kFbHeight);
utils::readImageData(*colorAttachmentInfo->image, kFbWidth, kFbHeight,
vkhpp::ImageLayout::eColorAttachmentOptimal, dst.data(),
dst.size() * sizeof(uint32_t), testEnvironment);
for (int i = 0; i < dst.size(); i++) {
ASSERT_THAT(dst[i], Eq(0xffff00ff));
}
}
TEST_P(GfxstreamEnd2EndVkSnapshotPipelineTest, CanSnapshotCommandBuffer) {
TypicalVkTestEnvironment testEnvironment = GFXSTREAM_ASSERT(SetUpTypicalVkTestEnvironment());
auto& [instance, physicalDevice, device, queue, queueFamilyIndex] = testEnvironment;
auto pipelineInfo = createPipeline(device.get());
auto colorAttachmentInfo = createColorAttachment(physicalDevice, device.get());
ASSERT_THAT(colorAttachmentInfo->image, IsValidHandle());
ASSERT_THAT(colorAttachmentInfo->memory, IsValidHandle());
ASSERT_THAT(colorAttachmentInfo->imageView, IsValidHandle());
const std::vector<vkhpp::ImageView> attachments(1, *colorAttachmentInfo->imageView);
vkhpp::FramebufferCreateInfo framebufferCreateInfo = {
.renderPass = *pipelineInfo->renderPass,
.attachmentCount = 1,
.pAttachments = attachments.data(),
.width = kFbWidth,
.height = kFbHeight,
.layers = 1,
};
auto framebuffer = device->createFramebufferUnique(framebufferCreateInfo).value;
ASSERT_THAT(framebuffer, IsValidHandle());
auto fence = device->createFenceUnique(vkhpp::FenceCreateInfo()).value;
ASSERT_THAT(fence, IsValidHandle());
const vkhpp::CommandPoolCreateInfo commandPoolCreateInfo = {
.flags = vkhpp::CommandPoolCreateFlagBits::eResetCommandBuffer,
.queueFamilyIndex = queueFamilyIndex,
};
auto commandPool = device->createCommandPoolUnique(commandPoolCreateInfo).value;
ASSERT_THAT(commandPool, IsValidHandle());
const vkhpp::CommandBufferAllocateInfo commandBufferAllocateInfo = {
.level = vkhpp::CommandBufferLevel::ePrimary,
.commandPool = *commandPool,
.commandBufferCount = 1,
};
auto commandBuffers = device->allocateCommandBuffersUnique(commandBufferAllocateInfo).value;
ASSERT_THAT(commandBuffers, Not(IsEmpty()));
auto commandBuffer = std::move(commandBuffers[0]);
ASSERT_THAT(commandBuffer, IsValidHandle());
vkhpp::ClearColorValue clearColor(std::array<float, 4>{1.0f, 0.0f, 1.0f, 1.0f});
vkhpp::ClearValue clearValue{
.color = clearColor,
};
vkhpp::RenderPassBeginInfo renderPassBeginInfo{
.renderPass = *pipelineInfo->renderPass,
.framebuffer = *framebuffer,
.renderArea = vkhpp::Rect2D(vkhpp::Offset2D(0, 0), vkhpp::Extent2D(kFbWidth, kFbHeight)),
.clearValueCount = 1,
.pClearValues = &clearValue,
};
const vkhpp::CommandBufferBeginInfo commandBufferBeginInfo = {
.flags = vkhpp::CommandBufferUsageFlagBits::eOneTimeSubmit,
};
commandBuffer->begin(commandBufferBeginInfo);
const vkhpp::ImageMemoryBarrier colorAttachmentBarrier{
.oldLayout = vkhpp::ImageLayout::eUndefined,
.newLayout = vkhpp::ImageLayout::eColorAttachmentOptimal,
.dstAccessMask = vkhpp::AccessFlagBits::eColorAttachmentRead |
vkhpp::AccessFlagBits::eColorAttachmentWrite,
.srcQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED,
.dstQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED,
.image = *colorAttachmentInfo->image,
.subresourceRange =
{
.aspectMask = vkhpp::ImageAspectFlagBits::eColor,
.levelCount = 1,
.layerCount = 1,
},
};
commandBuffer->pipelineBarrier(
vkhpp::PipelineStageFlagBits::eTopOfPipe | vkhpp::PipelineStageFlagBits::eTransfer,
vkhpp::PipelineStageFlagBits::eColorAttachmentOutput, vkhpp::DependencyFlags(), nullptr,
nullptr, colorAttachmentBarrier);
commandBuffer->beginRenderPass(renderPassBeginInfo, vkhpp::SubpassContents::eInline);
commandBuffer->bindPipeline(vkhpp::PipelineBindPoint::eGraphics, *pipelineInfo->pipeline);
vkhpp::ClearAttachment clearAttachment{
.aspectMask = vkhpp::ImageAspectFlagBits::eColor,
.colorAttachment = 0,
.clearValue = clearValue,
};
vkhpp::ClearRect clearRect{
.rect = vkhpp::Rect2D(vkhpp::Offset2D(0, 0), vkhpp::Extent2D(kFbWidth, kFbHeight)),
.baseArrayLayer = 0,
.layerCount = 1,
};
commandBuffer->clearAttachments(1, &clearAttachment, 1, &clearRect);
commandBuffer->endRenderPass();
commandBuffer->end();
std::vector<vkhpp::CommandBuffer> commandBufferHandles;
commandBufferHandles.push_back(*commandBuffer);
const vkhpp::SubmitInfo submitInfo = {
.commandBufferCount = static_cast<uint32_t>(commandBufferHandles.size()),
.pCommandBuffers = commandBufferHandles.data(),
};
SnapshotSaveAndLoad();
// Command buffer should still work after snapshot.
queue.submit(submitInfo, *fence);
auto waitResult = device->waitForFences(*fence, VK_TRUE, 3000000000L);
ASSERT_THAT(waitResult, IsVkSuccess());
std::vector<uint32_t> dst(kFbWidth * kFbHeight);
utils::readImageData(*colorAttachmentInfo->image, kFbWidth, kFbHeight,
vkhpp::ImageLayout::eColorAttachmentOptimal, dst.data(),
dst.size() * sizeof(uint32_t), testEnvironment);
for (int i = 0; i < dst.size(); i++) {
ASSERT_THAT(dst[i], Eq(0xffff00ff));
}
}
TEST_P(GfxstreamEnd2EndVkSnapshotPipelineTest, CanSnapshotDescriptors) {
TypicalVkTestEnvironment testEnvironment = GFXSTREAM_ASSERT(SetUpTypicalVkTestEnvironment());
auto& [instance, physicalDevice, device, queue, queueFamilyIndex] = testEnvironment;
auto pipelineInfo = createPipeline(device.get());
auto vertexBufferInfo = createAndPopulateBuffer(
physicalDevice, device.get(), vkhpp::BufferUsageFlagBits::eVertexBuffer,
kFullscreenBlueRectangleVertexData, sizeof(kFullscreenBlueRectangleVertexData));
auto colorAttachmentInfo = createColorAttachment(physicalDevice, device.get());
ASSERT_THAT(colorAttachmentInfo->image, IsValidHandle());
ASSERT_THAT(colorAttachmentInfo->memory, IsValidHandle());
ASSERT_THAT(colorAttachmentInfo->imageView, IsValidHandle());
// Descriptor
std::vector<vkhpp::DescriptorPoolSize> sizes = {
{
.type = vkhpp::DescriptorType::eUniformBuffer,
.descriptorCount = 10,
},
};
vkhpp::DescriptorPoolCreateInfo descriptorPoolCreateInfo = {
.maxSets = 10,
.poolSizeCount = static_cast<uint32_t>(sizes.size()),
.pPoolSizes = sizes.data(),
};
auto descriptorPool = device->createDescriptorPoolUnique(descriptorPoolCreateInfo).value;
ASSERT_THAT(descriptorPool, IsValidHandle());
const std::vector<vkhpp::DescriptorSetLayout> descriptorSetLayouts(
1, *pipelineInfo->descriptorSetLayout);
vkhpp::DescriptorSetAllocateInfo descriptorSetAllocateInfo = {
.descriptorPool = *descriptorPool,
.descriptorSetCount = 1,
.pSetLayouts = descriptorSetLayouts.data(),
};
auto descriptorSets = device->allocateDescriptorSetsUnique(descriptorSetAllocateInfo);
EXPECT_THAT(descriptorSets.result, Eq(vkhpp::Result::eSuccess));
auto descriptorSet = *descriptorSets.value[0];
// A uniform for red color
float kColor1[] = {1.0f, 0.0f, 0.0f, 0.0f};
auto uniformBufferInfo = createAndPopulateBuffer(physicalDevice, device.get(),
vkhpp::BufferUsageFlagBits::eUniformBuffer,
kColor1, sizeof(kColor1));
std::vector<vkhpp::WriteDescriptorSet> writeDescriptorSets;
std::vector<vkhpp::DescriptorBufferInfo> bufferInfos;
bufferInfos.emplace_back(vkhpp::DescriptorBufferInfo{
.buffer = *uniformBufferInfo->buffer,
.offset = 0,
.range = VK_WHOLE_SIZE,
});
writeDescriptorSets.emplace_back(vkhpp::WriteDescriptorSet{
.dstSet = descriptorSet,
.dstBinding = 0,
.descriptorCount = 1,
.descriptorType = vkhpp::DescriptorType::eUniformBuffer,
.pBufferInfo = bufferInfos.data(),
});
device->updateDescriptorSets(writeDescriptorSets, nullptr);
const std::vector<vkhpp::ImageView> attachments(1, *colorAttachmentInfo->imageView);
vkhpp::FramebufferCreateInfo framebufferCreateInfo = {
.renderPass = *pipelineInfo->renderPass,
.attachmentCount = 1,
.pAttachments = attachments.data(),
.width = kFbWidth,
.height = kFbHeight,
.layers = 1,
};
auto framebuffer = device->createFramebufferUnique(framebufferCreateInfo).value;
ASSERT_THAT(framebuffer, IsValidHandle());
auto fence = device->createFenceUnique(vkhpp::FenceCreateInfo()).value;
ASSERT_THAT(fence, IsValidHandle());
const vkhpp::CommandPoolCreateInfo commandPoolCreateInfo = {
.flags = vkhpp::CommandPoolCreateFlagBits::eResetCommandBuffer,
.queueFamilyIndex = queueFamilyIndex,
};
auto commandPool = device->createCommandPoolUnique(commandPoolCreateInfo).value;
ASSERT_THAT(commandPool, IsValidHandle());
const vkhpp::CommandBufferAllocateInfo commandBufferAllocateInfo = {
.level = vkhpp::CommandBufferLevel::ePrimary,
.commandPool = *commandPool,
.commandBufferCount = 1,
};
auto commandBuffers = device->allocateCommandBuffersUnique(commandBufferAllocateInfo).value;
ASSERT_THAT(commandBuffers, Not(IsEmpty()));
auto commandBuffer = std::move(commandBuffers[0]);
ASSERT_THAT(commandBuffer, IsValidHandle());
vkhpp::ClearColorValue clearColor(std::array<float, 4>{0.0f, 0.0f, 0.0f, 1.0f});
vkhpp::ClearValue clearValue{
.color = clearColor,
};
vkhpp::RenderPassBeginInfo renderPassBeginInfo{
.renderPass = *pipelineInfo->renderPass,
.framebuffer = *framebuffer,
.renderArea = vkhpp::Rect2D(vkhpp::Offset2D(0, 0), vkhpp::Extent2D(kFbWidth, kFbHeight)),
.clearValueCount = 1,
.pClearValues = &clearValue,
};
// Descriptor updates are cached on the guest, for testing purpose we need to submit a queue to
// commit descriptor updates.
const vkhpp::CommandBufferBeginInfo commandBufferBeginInfo = {
.flags = vkhpp::CommandBufferUsageFlagBits::eOneTimeSubmit,
};
commandBuffer->begin(commandBufferBeginInfo);
const vkhpp::ImageMemoryBarrier colorAttachmentBarrier{
.oldLayout = vkhpp::ImageLayout::eUndefined,
.newLayout = vkhpp::ImageLayout::eColorAttachmentOptimal,
.dstAccessMask = vkhpp::AccessFlagBits::eColorAttachmentRead |
vkhpp::AccessFlagBits::eColorAttachmentWrite,
.srcQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED,
.dstQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED,
.image = *colorAttachmentInfo->image,
.subresourceRange =
{
.aspectMask = vkhpp::ImageAspectFlagBits::eColor,
.levelCount = 1,
.layerCount = 1,
},
};
commandBuffer->pipelineBarrier(
vkhpp::PipelineStageFlagBits::eTopOfPipe | vkhpp::PipelineStageFlagBits::eTransfer,
vkhpp::PipelineStageFlagBits::eColorAttachmentOutput, vkhpp::DependencyFlags(), nullptr,
nullptr, colorAttachmentBarrier);
commandBuffer->beginRenderPass(renderPassBeginInfo, vkhpp::SubpassContents::eInline);
commandBuffer->bindPipeline(vkhpp::PipelineBindPoint::eGraphics, *pipelineInfo->pipeline);
commandBuffer->bindDescriptorSets(vkhpp::PipelineBindPoint::eGraphics,
*pipelineInfo->pipelineLayout, 0, descriptorSet, nullptr);
commandBuffer->bindVertexBuffers(0, {*vertexBufferInfo->buffer}, {0});
commandBuffer->setViewport(0, vkhpp::Viewport(0.0f, 0.0f, static_cast<float>(kFbWidth),
static_cast<float>(kFbHeight), 0.0f, 1.0f));
commandBuffer->setScissor(
0, vkhpp::Rect2D(vkhpp::Offset2D(0, 0), vkhpp::Extent2D(kFbWidth, kFbHeight)));
commandBuffer->draw(6, 1, 0, 0);
commandBuffer->endRenderPass();
commandBuffer->end();
std::vector<vkhpp::CommandBuffer> commandBufferHandles;
commandBufferHandles.push_back(*commandBuffer);
const vkhpp::SubmitInfo submitInfo = {
.commandBufferCount = static_cast<uint32_t>(commandBufferHandles.size()),
.pCommandBuffers = commandBufferHandles.data(),
};
queue.submit(submitInfo, *fence);
auto waitResult = device->waitForFences(*fence, VK_TRUE, 3000000000L);
ASSERT_THAT(waitResult, IsVkSuccess());
commandBuffer->reset();
// Clear the rendering
commandBuffer->begin(commandBufferBeginInfo);
commandBuffer->beginRenderPass(renderPassBeginInfo, vkhpp::SubpassContents::eInline);
commandBuffer->bindPipeline(vkhpp::PipelineBindPoint::eGraphics, *pipelineInfo->pipeline);
vkhpp::ClearAttachment clearAttachment{
.aspectMask = vkhpp::ImageAspectFlagBits::eColor,
.colorAttachment = 0,
.clearValue = clearValue,
};
vkhpp::ClearRect clearRect{
.rect = vkhpp::Rect2D(vkhpp::Offset2D(0, 0), vkhpp::Extent2D(kFbWidth, kFbHeight)),
.baseArrayLayer = 0,
.layerCount = 1,
};
commandBuffer->clearAttachments(1, &clearAttachment, 1, &clearRect);
commandBuffer->endRenderPass();
commandBuffer->end();
device->resetFences(1, &fence.get());
queue.submit(submitInfo, *fence);
waitResult = device->waitForFences(*fence, VK_TRUE, 3000000000L);
ASSERT_THAT(waitResult, IsVkSuccess());
commandBuffer->reset();
SnapshotSaveAndLoad();
// Redraw after snapshot, verify descriptors keep their value
// Command buffer snapshot is not implemented yet, so we need to re-make the command buffer.
commandBuffer->begin(commandBufferBeginInfo);
commandBuffer->beginRenderPass(renderPassBeginInfo, vkhpp::SubpassContents::eInline);
commandBuffer->bindPipeline(vkhpp::PipelineBindPoint::eGraphics, *pipelineInfo->pipeline);
commandBuffer->bindDescriptorSets(vkhpp::PipelineBindPoint::eGraphics,
*pipelineInfo->pipelineLayout, 0, descriptorSet, nullptr);
commandBuffer->bindVertexBuffers(0, {*vertexBufferInfo->buffer}, {0});
commandBuffer->setViewport(0, vkhpp::Viewport(0.0f, 0.0f, static_cast<float>(kFbWidth),
static_cast<float>(kFbHeight), 0.0f, 1.0f));
commandBuffer->setScissor(
0, vkhpp::Rect2D(vkhpp::Offset2D(0, 0), vkhpp::Extent2D(kFbWidth, kFbHeight)));
commandBuffer->draw(6, 1, 0, 0);
commandBuffer->endRenderPass();
commandBuffer->end();
device->resetFences(1, &fence.get());
queue.submit(submitInfo, *fence);
waitResult = device->waitForFences(*fence, VK_TRUE, 3000000000L);
ASSERT_THAT(waitResult, IsVkSuccess());
std::vector<uint32_t> dst(kFbWidth * kFbHeight);
utils::readImageData(*colorAttachmentInfo->image, kFbWidth, kFbHeight,
vkhpp::ImageLayout::eColorAttachmentOptimal, dst.data(),
dst.size() * sizeof(uint32_t), testEnvironment);
for (int i = 0; i < dst.size(); i++) {
// The shader adds a blue color (from vertex buffer) with a red color (from uniform) and get
// purple.
ASSERT_THAT(dst[i], Eq(0xffff00ff));
}
}
TEST_P(GfxstreamEnd2EndVkSnapshotPipelineTest, DeleteBufferBeforeCommit) {
TypicalVkTestEnvironment testEnvironment = GFXSTREAM_ASSERT(SetUpTypicalVkTestEnvironment());
auto& [instance, physicalDevice, device, queue, queueFamilyIndex] = testEnvironment;
auto pipelineInfo = createPipeline(device.get());
auto vertexBufferInfo = createAndPopulateBuffer(
physicalDevice, device.get(), vkhpp::BufferUsageFlagBits::eVertexBuffer,
kFullscreenBlueRectangleVertexData, sizeof(kFullscreenBlueRectangleVertexData));
auto colorAttachmentInfo = createColorAttachment(physicalDevice, device.get());
ASSERT_THAT(colorAttachmentInfo->image, IsValidHandle());
ASSERT_THAT(colorAttachmentInfo->memory, IsValidHandle());
ASSERT_THAT(colorAttachmentInfo->imageView, IsValidHandle());
// Descriptor
std::vector<vkhpp::DescriptorPoolSize> sizes = {
{
.type = vkhpp::DescriptorType::eUniformBuffer,
.descriptorCount = 10,
},
};
vkhpp::DescriptorPoolCreateInfo descriptorPoolCreateInfo = {
.maxSets = 10,
.poolSizeCount = static_cast<uint32_t>(sizes.size()),
.pPoolSizes = sizes.data(),
};
auto descriptorPool = device->createDescriptorPoolUnique(descriptorPoolCreateInfo).value;
ASSERT_THAT(descriptorPool, IsValidHandle());
const std::vector<vkhpp::DescriptorSetLayout> descriptorSetLayouts(
1, *pipelineInfo->descriptorSetLayout);
vkhpp::DescriptorSetAllocateInfo descriptorSetAllocateInfo = {
.descriptorPool = *descriptorPool,
.descriptorSetCount = 1,
.pSetLayouts = descriptorSetLayouts.data(),
};
auto descriptorSets = device->allocateDescriptorSetsUnique(descriptorSetAllocateInfo);
EXPECT_THAT(descriptorSets.result, Eq(vkhpp::Result::eSuccess));
auto descriptorSet = *descriptorSets.value[0];
// A uniform for red color
float kColor1[] = {1.0f, 0.0f, 0.0f, 0.0f};
auto uniformBufferInfo = createAndPopulateBuffer(physicalDevice, device.get(),
vkhpp::BufferUsageFlagBits::eUniformBuffer,
kColor1, sizeof(kColor1));
std::vector<vkhpp::WriteDescriptorSet> writeDescriptorSets;
std::vector<vkhpp::DescriptorBufferInfo> bufferInfos;
bufferInfos.emplace_back(vkhpp::DescriptorBufferInfo{
.buffer = *uniformBufferInfo->buffer,
.offset = 0,
.range = VK_WHOLE_SIZE,
});
writeDescriptorSets.emplace_back(vkhpp::WriteDescriptorSet{
.dstSet = descriptorSet,
.dstBinding = 0,
.descriptorCount = 1,
.descriptorType = vkhpp::DescriptorType::eUniformBuffer,
.pBufferInfo = bufferInfos.data(),
});
device->updateDescriptorSets(writeDescriptorSets, nullptr);
// Delete the underlying buffer, should not crash.
uniformBufferInfo->buffer.reset();
const std::vector<vkhpp::ImageView> attachments(1, *colorAttachmentInfo->imageView);
vkhpp::FramebufferCreateInfo framebufferCreateInfo = {
.renderPass = *pipelineInfo->renderPass,
.attachmentCount = 1,
.pAttachments = attachments.data(),
.width = kFbWidth,
.height = kFbHeight,
.layers = 1,
};
auto framebuffer = device->createFramebufferUnique(framebufferCreateInfo).value;
ASSERT_THAT(framebuffer, IsValidHandle());
auto fence = device->createFenceUnique(vkhpp::FenceCreateInfo()).value;
ASSERT_THAT(fence, IsValidHandle());
const vkhpp::CommandPoolCreateInfo commandPoolCreateInfo = {
.flags = vkhpp::CommandPoolCreateFlagBits::eResetCommandBuffer,
.queueFamilyIndex = queueFamilyIndex,
};
auto commandPool = device->createCommandPoolUnique(commandPoolCreateInfo).value;
ASSERT_THAT(commandPool, IsValidHandle());
const vkhpp::CommandBufferAllocateInfo commandBufferAllocateInfo = {
.level = vkhpp::CommandBufferLevel::ePrimary,
.commandPool = *commandPool,
.commandBufferCount = 1,
};
auto commandBuffers = device->allocateCommandBuffersUnique(commandBufferAllocateInfo).value;
ASSERT_THAT(commandBuffers, Not(IsEmpty()));
auto commandBuffer = std::move(commandBuffers[0]);
ASSERT_THAT(commandBuffer, IsValidHandle());
vkhpp::ClearColorValue clearColor(std::array<float, 4>{0.0f, 0.0f, 0.0f, 1.0f});
vkhpp::ClearValue clearValue{
.color = clearColor,
};
// Descriptor updates are cached on the guest, for testing purpose we need to submit a queue to
// commit descriptor updates.
const vkhpp::CommandBufferBeginInfo commandBufferBeginInfo = {
.flags = vkhpp::CommandBufferUsageFlagBits::eOneTimeSubmit,
};
commandBuffer->begin(commandBufferBeginInfo);
const vkhpp::ImageMemoryBarrier colorAttachmentBarrier{
.oldLayout = vkhpp::ImageLayout::eUndefined,
.newLayout = vkhpp::ImageLayout::eColorAttachmentOptimal,
.dstAccessMask = vkhpp::AccessFlagBits::eColorAttachmentRead |
vkhpp::AccessFlagBits::eColorAttachmentWrite,
.srcQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED,
.dstQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED,
.image = *colorAttachmentInfo->image,
.subresourceRange =
{
.aspectMask = vkhpp::ImageAspectFlagBits::eColor,
.levelCount = 1,
.layerCount = 1,
},
};
commandBuffer->pipelineBarrier(
vkhpp::PipelineStageFlagBits::eTopOfPipe | vkhpp::PipelineStageFlagBits::eTransfer,
vkhpp::PipelineStageFlagBits::eColorAttachmentOutput, vkhpp::DependencyFlags(), nullptr,
nullptr, colorAttachmentBarrier);
vkhpp::RenderPassBeginInfo renderPassBeginInfo{
.renderPass = *pipelineInfo->renderPass,
.framebuffer = *framebuffer,
.renderArea = vkhpp::Rect2D(vkhpp::Offset2D(0, 0), vkhpp::Extent2D(kFbWidth, kFbHeight)),
.clearValueCount = 1,
.pClearValues = &clearValue,
};
commandBuffer->beginRenderPass(renderPassBeginInfo, vkhpp::SubpassContents::eInline);
vkhpp::ClearAttachment clearAttachment{
.aspectMask = vkhpp::ImageAspectFlagBits::eColor,
.colorAttachment = 0,
.clearValue = clearValue,
};
vkhpp::ClearRect clearRect{
.rect = vkhpp::Rect2D(vkhpp::Offset2D(0, 0), vkhpp::Extent2D(kFbWidth, kFbHeight)),
.baseArrayLayer = 0,
.layerCount = 1,
};
commandBuffer->clearAttachments(1, &clearAttachment, 1, &clearRect);
commandBuffer->endRenderPass();
commandBuffer->end();
std::vector<vkhpp::CommandBuffer> commandBufferHandles;
commandBufferHandles.push_back(*commandBuffer);
const vkhpp::SubmitInfo submitInfo = {
.commandBufferCount = static_cast<uint32_t>(commandBufferHandles.size()),
.pCommandBuffers = commandBufferHandles.data(),
};
// Submit will update the stale descriptor. Should not crash.
queue.submit(submitInfo, *fence);
auto waitResult = device->waitForFences(*fence, VK_TRUE, 3000000000L);
ASSERT_THAT(waitResult, IsVkSuccess());
commandBuffer->reset();
// Snapshot should not crash.
SnapshotSaveAndLoad();
}
TEST_P(GfxstreamEnd2EndVkSnapshotPipelineTest, DeleteBufferAfterWriteDescriptor) {
TypicalVkTestEnvironment testEnvironment = GFXSTREAM_ASSERT(SetUpTypicalVkTestEnvironment());
auto& [instance, physicalDevice, device, queue, queueFamilyIndex] = testEnvironment;
auto pipelineInfo = createPipeline(device.get());
auto vertexBufferInfo = createAndPopulateBuffer(
physicalDevice, device.get(), vkhpp::BufferUsageFlagBits::eVertexBuffer,
kFullscreenBlueRectangleVertexData, sizeof(kFullscreenBlueRectangleVertexData));
auto colorAttachmentInfo = createColorAttachment(physicalDevice, device.get());
ASSERT_THAT(colorAttachmentInfo->image, IsValidHandle());
ASSERT_THAT(colorAttachmentInfo->memory, IsValidHandle());
ASSERT_THAT(colorAttachmentInfo->imageView, IsValidHandle());
// Descriptor
std::vector<vkhpp::DescriptorPoolSize> sizes = {
{
.type = vkhpp::DescriptorType::eUniformBuffer,
.descriptorCount = 10,
},
};
vkhpp::DescriptorPoolCreateInfo descriptorPoolCreateInfo = {
.maxSets = 10,
.poolSizeCount = static_cast<uint32_t>(sizes.size()),
.pPoolSizes = sizes.data(),
};
auto descriptorPool = device->createDescriptorPoolUnique(descriptorPoolCreateInfo).value;
ASSERT_THAT(descriptorPool, IsValidHandle());
const std::vector<vkhpp::DescriptorSetLayout> descriptorSetLayouts(
1, *pipelineInfo->descriptorSetLayout);
vkhpp::DescriptorSetAllocateInfo descriptorSetAllocateInfo = {
.descriptorPool = *descriptorPool,
.descriptorSetCount = 1,
.pSetLayouts = descriptorSetLayouts.data(),
};
auto descriptorSets = device->allocateDescriptorSetsUnique(descriptorSetAllocateInfo);
EXPECT_THAT(descriptorSets.result, Eq(vkhpp::Result::eSuccess));
auto descriptorSet = *descriptorSets.value[0];
// A uniform for red color
float kColor1[] = {1.0f, 0.0f, 0.0f, 0.0f};
auto uniformBufferInfo = createAndPopulateBuffer(physicalDevice, device.get(),
vkhpp::BufferUsageFlagBits::eUniformBuffer,
kColor1, sizeof(kColor1));
std::vector<vkhpp::WriteDescriptorSet> writeDescriptorSets;
std::vector<vkhpp::DescriptorBufferInfo> bufferInfos;
bufferInfos.emplace_back(vkhpp::DescriptorBufferInfo{
.buffer = *uniformBufferInfo->buffer,
.offset = 0,
.range = VK_WHOLE_SIZE,
});
writeDescriptorSets.emplace_back(vkhpp::WriteDescriptorSet{
.dstSet = descriptorSet,
.dstBinding = 0,
.descriptorCount = 1,
.descriptorType = vkhpp::DescriptorType::eUniformBuffer,
.pBufferInfo = bufferInfos.data(),
});
device->updateDescriptorSets(writeDescriptorSets, nullptr);
const std::vector<vkhpp::ImageView> attachments(1, *colorAttachmentInfo->imageView);
vkhpp::FramebufferCreateInfo framebufferCreateInfo = {
.renderPass = *pipelineInfo->renderPass,
.attachmentCount = 1,
.pAttachments = attachments.data(),
.width = kFbWidth,
.height = kFbHeight,
.layers = 1,
};
auto framebuffer = device->createFramebufferUnique(framebufferCreateInfo).value;
ASSERT_THAT(framebuffer, IsValidHandle());
auto fence = device->createFenceUnique(vkhpp::FenceCreateInfo()).value;
ASSERT_THAT(fence, IsValidHandle());
const vkhpp::CommandPoolCreateInfo commandPoolCreateInfo = {
.flags = vkhpp::CommandPoolCreateFlagBits::eResetCommandBuffer,
.queueFamilyIndex = queueFamilyIndex,
};
auto commandPool = device->createCommandPoolUnique(commandPoolCreateInfo).value;
ASSERT_THAT(commandPool, IsValidHandle());
const vkhpp::CommandBufferAllocateInfo commandBufferAllocateInfo = {
.level = vkhpp::CommandBufferLevel::ePrimary,
.commandPool = *commandPool,
.commandBufferCount = 1,
};
auto commandBuffers = device->allocateCommandBuffersUnique(commandBufferAllocateInfo).value;
ASSERT_THAT(commandBuffers, Not(IsEmpty()));
auto commandBuffer = std::move(commandBuffers[0]);
ASSERT_THAT(commandBuffer, IsValidHandle());
vkhpp::ClearColorValue clearColor(std::array<float, 4>{0.0f, 0.0f, 0.0f, 1.0f});
vkhpp::ClearValue clearValue{
.color = clearColor,
};
vkhpp::RenderPassBeginInfo renderPassBeginInfo{
.renderPass = *pipelineInfo->renderPass,
.framebuffer = *framebuffer,
.renderArea = vkhpp::Rect2D(vkhpp::Offset2D(0, 0), vkhpp::Extent2D(kFbWidth, kFbHeight)),
.clearValueCount = 1,
.pClearValues = &clearValue,
};
// Descriptor updates are cached on the guest, for testing purpose we need to submit a queue to
// commit descriptor updates.
const vkhpp::CommandBufferBeginInfo commandBufferBeginInfo = {
.flags = vkhpp::CommandBufferUsageFlagBits::eOneTimeSubmit,
};
commandBuffer->begin(commandBufferBeginInfo);
const vkhpp::ImageMemoryBarrier colorAttachmentBarrier{
.oldLayout = vkhpp::ImageLayout::eUndefined,
.newLayout = vkhpp::ImageLayout::eColorAttachmentOptimal,
.dstAccessMask = vkhpp::AccessFlagBits::eColorAttachmentRead |
vkhpp::AccessFlagBits::eColorAttachmentWrite,
.srcQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED,
.dstQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED,
.image = *colorAttachmentInfo->image,
.subresourceRange =
{
.aspectMask = vkhpp::ImageAspectFlagBits::eColor,
.levelCount = 1,
.layerCount = 1,
},
};
commandBuffer->pipelineBarrier(
vkhpp::PipelineStageFlagBits::eTopOfPipe | vkhpp::PipelineStageFlagBits::eTransfer,
vkhpp::PipelineStageFlagBits::eColorAttachmentOutput, vkhpp::DependencyFlags(), nullptr,
nullptr, colorAttachmentBarrier);
commandBuffer->beginRenderPass(renderPassBeginInfo, vkhpp::SubpassContents::eInline);
commandBuffer->bindPipeline(vkhpp::PipelineBindPoint::eGraphics, *pipelineInfo->pipeline);
commandBuffer->bindDescriptorSets(vkhpp::PipelineBindPoint::eGraphics,
*pipelineInfo->pipelineLayout, 0, descriptorSet, nullptr);
commandBuffer->bindVertexBuffers(0, {*vertexBufferInfo->buffer}, {0});
commandBuffer->setViewport(0, vkhpp::Viewport(0.0f, 0.0f, static_cast<float>(kFbWidth),
static_cast<float>(kFbHeight), 0.0f, 1.0f));
commandBuffer->setScissor(
0, vkhpp::Rect2D(vkhpp::Offset2D(0, 0), vkhpp::Extent2D(kFbWidth, kFbHeight)));
commandBuffer->draw(6, 1, 0, 0);
commandBuffer->endRenderPass();
commandBuffer->end();
std::vector<vkhpp::CommandBuffer> commandBufferHandles;
commandBufferHandles.push_back(*commandBuffer);
const vkhpp::SubmitInfo submitInfo = {
.commandBufferCount = static_cast<uint32_t>(commandBufferHandles.size()),
.pCommandBuffers = commandBufferHandles.data(),
};
queue.submit(submitInfo, *fence);
auto waitResult = device->waitForFences(*fence, VK_TRUE, 3000000000L);
ASSERT_THAT(waitResult, IsVkSuccess());
commandBuffer->reset();
vertexBufferInfo->buffer.reset();
// Descriptor snapshot should not crash after underlying buffer is deleted.
SnapshotSaveAndLoad();
}
INSTANTIATE_TEST_CASE_P(GfxstreamEnd2EndTests, GfxstreamEnd2EndVkSnapshotPipelineTest,
::testing::ValuesIn({
TestParams{
.with_gl = false,
.with_vk = true,
.with_features = {"VulkanSnapshots", "VulkanBatchedDescriptorSetUpdate"},
},
}),
&GetTestName);
INSTANTIATE_TEST_CASE_P(GfxstreamEnd2EndTests,
GfxstreamEnd2EndVkSnapshotPipelineWithMultiSamplingTest,
::testing::ValuesIn({
TestParams{
.with_gl = false,
.with_vk = true,
.samples = 1,
.with_features = {"VulkanSnapshots"},
},
TestParams{
.with_gl = false,
.with_vk = true,
.samples = 4,
.with_features = {"VulkanSnapshots"},
},
}),
&GetTestName);
} // namespace
} // namespace tests
} // namespace gfxstream