| // 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"}, |
| }, |
| TestParams{ |
| .with_gl = false, |
| .with_vk = true, |
| .with_features = {"VulkanSnapshots"}, |
| }, |
| }), |
| &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 |