blob: 065f64061f68bc7c95ebe5ae5a292a2001ea9891 [file] [log] [blame]
/*------------------------------------------------------------------------
* Vulkan Conformance Tests
* ------------------------
*
* Copyright (c) 2021 The Khronos Group Inc.
*
* 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.
*
*//*!
* \file
* \brief Vulkan SC VK_KHR_object_refresh Tests
*//*--------------------------------------------------------------------*/
#include "vktObjectRefreshTests.hpp"
#include "vkDefs.hpp"
#include "vkRefUtil.hpp"
#include "vkDefs.hpp"
#include "vkDeviceUtil.hpp"
#include "vkPlatform.hpp"
#include "vkTypeUtil.hpp"
#include "vkPrograms.hpp"
#include "vkCmdUtil.hpp"
#include "vkObjUtil.hpp"
#include "vkMemUtil.hpp"
#include "vktTestCaseUtil.hpp"
#include "vkQueryUtil.hpp"
#include <map>
#include <vector>
namespace vkt
{
namespace sc
{
namespace
{
tcu::TestStatus queryRefreshableObjects(Context& context)
{
deUint32 countReported = 0u;
vk::VkPhysicalDevice physicalDevice = context.getPhysicalDevice();
const vk::InstanceInterface& vki = context.getInstanceInterface();
// get number of refreshable objects
vk::VkResult result = vki.getPhysicalDeviceRefreshableObjectTypesKHR(physicalDevice, &countReported, DE_NULL);
if (result != vk::VK_SUCCESS)
TCU_THROW(NotSupportedError, "vkGetPhysicalDeviceRefreshableObjectTypesKHR returned invalid return code");
if (countReported == 0)
TCU_THROW(NotSupportedError, "No refreshable objects available");
deUint32 refreshableObjectsMaxCount (countReported + 2);
std::vector<vk::VkObjectType> refreshableObjects (refreshableObjectsMaxCount);
for (deUint32 countRequested = 0; countRequested < refreshableObjectsMaxCount; ++countRequested)
{
// get refreshable objects
deUint32 countRetrieved = countRequested;
std::fill(refreshableObjects.begin(), refreshableObjects.end(), vk::VK_OBJECT_TYPE_UNKNOWN);
result = vki.getPhysicalDeviceRefreshableObjectTypesKHR(physicalDevice, &countRetrieved, refreshableObjects.data());
// verify returned code
if ((result != vk::VK_SUCCESS) && (result != vk::VK_INCOMPLETE))
TCU_THROW(NotSupportedError, "vkGetPhysicalDeviceRefreshableObjectTypesKHR returned invalid return code");
// verify number of retrieved objects
if (countRetrieved != std::min(countRequested, countReported))
TCU_THROW(NotSupportedError, "vkGetPhysicalDeviceRefreshableObjectTypesKHR returned invalid number of retrieved objects");
// verify retrieved objects
for (deUint32 i = 0; i < countRetrieved; ++i)
{
if (refreshableObjects[i] == vk::VK_OBJECT_TYPE_UNKNOWN)
TCU_THROW(NotSupportedError, "vkGetPhysicalDeviceRefreshableObjectTypesKHR returned invalid object type");
}
}
return tcu::TestStatus::pass("pass");
}
tcu::TestStatus refreshObjects(Context& context, bool individualRefresh)
{
deUint32 countReported = 0u;
vk::VkPhysicalDevice physicalDevice = context.getPhysicalDevice();
const vk::InstanceInterface& vki = context.getInstanceInterface();
const vk::DeviceInterface& vkd = context.getDeviceInterface();
vk::VkResult result = vki.getPhysicalDeviceRefreshableObjectTypesKHR(physicalDevice, &countReported, DE_NULL);
if ((result != vk::VK_SUCCESS) || (countReported == 0))
TCU_THROW(NotSupportedError, "vkGetPhysicalDeviceRefreshableObjectTypesKHR failed");
std::vector<vk::VkObjectType> refreshableObjectTypes(countReported);
result = vki.getPhysicalDeviceRefreshableObjectTypesKHR(physicalDevice, &countReported, refreshableObjectTypes.data());
if (result != vk::VK_SUCCESS)
TCU_THROW(NotSupportedError, "vkGetPhysicalDeviceRefreshableObjectTypesKHR failed");
// create all possible objects
const deUint32 queueFamilyIndex = context.getUniversalQueueFamilyIndex();
vk::VkDevice device = context.getDevice();
vk::VkQueue queue = context.getUniversalQueue();
vk::Allocator& allocator = context.getDefaultAllocator();
vk::Move<vk::VkCommandPool> cmdPool = createCommandPool(vkd, device, vk::VK_COMMAND_POOL_CREATE_RESET_COMMAND_BUFFER_BIT, queueFamilyIndex);
vk::Move<vk::VkCommandBuffer> cmdBuffer = allocateCommandBuffer(vkd, device, *cmdPool, vk::VK_COMMAND_BUFFER_LEVEL_PRIMARY);
vk::Move<vk::VkFence> fence = createFence(vkd, device);
vk::Move<vk::VkSemaphore> semaphore = createSemaphore(vkd, device);
vk::Move<vk::VkEvent> event = createEvent(vkd, device);
const vk::VkQueryPoolCreateInfo queryPoolCreateInfo
{
vk::VK_STRUCTURE_TYPE_QUERY_POOL_CREATE_INFO, // VkStructureType sType;
DE_NULL, // const void* pNext;
0u, // VkQueryPoolCreateFlags flags;
vk::VK_QUERY_TYPE_OCCLUSION, // VkQueryType queryType;
1u, // deUint32 queryCount;
0u, // VkQueryPipelineStatisticFlags pipelineStatistics;
};
vk::Move<vk::VkQueryPool> queryPool = createQueryPool(vkd, device, &queryPoolCreateInfo);
const vk::VkBufferCreateInfo bufferCreateInfo
{
vk::VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO, // VkStructureType sType
DE_NULL, // const void* pNext
0u, // VkBufferCreateFlags flags
(vk::VkDeviceSize)64, // VkDeviceSize size
vk::VK_BUFFER_USAGE_TRANSFER_SRC_BIT | vk::VK_BUFFER_USAGE_STORAGE_TEXEL_BUFFER_BIT, // VkBufferUsageFlags usage
vk::VK_SHARING_MODE_EXCLUSIVE, // VkSharingMode sharingMode
0u, // deUint32 queueFamilyIndexCount
DE_NULL // const deUint32* pQueueFamilyIndices
};
vk::Move<vk::VkBuffer> buffer = createBuffer(vkd, device, &bufferCreateInfo);
const vk::VkMemoryRequirements bufferRequirements = getBufferMemoryRequirements(vkd, device, *buffer);
de::MovePtr<vk::Allocation> bufferAllocation = allocator.allocate(bufferRequirements, vk::MemoryRequirement::HostVisible);
vkd.bindBufferMemory(device, *buffer, bufferAllocation->getMemory(), 0);
vk::Move<vk::VkBufferView> bufferView = makeBufferView(vkd, device, *buffer, vk::VK_FORMAT_R32G32B32A32_SFLOAT, 0ull, VK_WHOLE_SIZE);
const vk::VkSamplerCreateInfo samplerCreateInfo
{
vk::VK_STRUCTURE_TYPE_SAMPLER_CREATE_INFO, // VkStructureType sType;
DE_NULL, // const void* pNext;
(vk::VkSamplerCreateFlags)0, // VkSamplerCreateFlags flags;
vk::VK_FILTER_NEAREST, // VkFilter magFilter;
vk::VK_FILTER_NEAREST, // VkFilter minFilter;
vk::VK_SAMPLER_MIPMAP_MODE_NEAREST, // VkSamplerMipmapMode mipmapMode;
vk::VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE, // VkSamplerAddressMode addressModeU;
vk::VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE, // VkSamplerAddressMode addressModeV;
vk::VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE, // VkSamplerAddressMode addressModeW;
0.0f, // float mipLodBias;
VK_FALSE, // VkBool32 anisotropyEnable;
1.0f, // float maxAnisotropy;
VK_FALSE, // VkBool32 compareEnable;
vk::VK_COMPARE_OP_ALWAYS, // VkCompareOp compareOp;
0.0f, // float minLod;
0.0f, // float maxLod;
vk::VK_BORDER_COLOR_FLOAT_TRANSPARENT_BLACK, // VkBorderColor borderColor;
VK_FALSE, // VkBool32 unnormalizedCoordinates;
};
vk::Move<vk::VkSampler> sampler = createSampler(vkd, device, &samplerCreateInfo);
const vk::VkSamplerYcbcrConversionCreateInfo conversionInfo
{
vk::VK_STRUCTURE_TYPE_SAMPLER_YCBCR_CONVERSION_CREATE_INFO, // VkStructureType sType;
DE_NULL, // const void* pNext;
vk::VK_FORMAT_G8_B8_R8_3PLANE_420_UNORM, // VkFormat format;
vk::VK_SAMPLER_YCBCR_MODEL_CONVERSION_RGB_IDENTITY, // VkSamplerYcbcrModelConversion ycbcrModel;
vk::VK_SAMPLER_YCBCR_RANGE_ITU_FULL, // VkSamplerYcbcrRange ycbcrRange;
{ // VkComponentMapping components;
vk::VK_COMPONENT_SWIZZLE_IDENTITY,
vk::VK_COMPONENT_SWIZZLE_IDENTITY,
vk::VK_COMPONENT_SWIZZLE_IDENTITY,
vk::VK_COMPONENT_SWIZZLE_IDENTITY
},
vk::VK_CHROMA_LOCATION_MIDPOINT, // VkChromaLocation xChromaOffset;
vk::VK_CHROMA_LOCATION_MIDPOINT, // VkChromaLocation yChromaOffset;
vk::VK_FILTER_NEAREST, // VkFilter chromaFilter;
DE_FALSE // VkBool32 forceExplicitReconstruction;
};
vk::Move<vk::VkSamplerYcbcrConversion> ycbcrConversion = vk::createSamplerYcbcrConversion(vkd, device, &conversionInfo);
const vk::VkImageCreateInfo imageCreateInfo
{
vk::VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO, // VkStructureType sType;
DE_NULL, // const void* pNext;
0u, // VkImageCreateFlags flags;
vk::VK_IMAGE_TYPE_2D, // VkImageType imageType;
vk::VK_FORMAT_R8G8B8A8_UNORM, // VkFormat format;
{ 64, 64, 1 }, // VkExtent3D extent;
1u, // deUint32 mipLevels;
1u, // deUint32 arrayLayers;
vk::VK_SAMPLE_COUNT_1_BIT, // VkSampleCountFlagBits samples;
vk::VK_IMAGE_TILING_OPTIMAL, // VkImageTiling tiling;
vk::VK_IMAGE_USAGE_TRANSFER_SRC_BIT | vk::VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT, // VkImageUsageFlags usage;
vk::VK_SHARING_MODE_EXCLUSIVE, // VkSharingMode sharingMode;
1u, // deUint32 queueFamilyIndexCount;
&queueFamilyIndex, // const deUint32* pQueueFamilyIndices;
vk::VK_IMAGE_LAYOUT_UNDEFINED, // VkImageLayout initialLayout;
};
vk::Move<vk::VkImage> image = createImage(vkd, device, &imageCreateInfo);
const vk::VkMemoryRequirements imageRequirements = getImageMemoryRequirements(vkd, device, *image);
de::MovePtr<vk::Allocation> imageAllocation = allocator.allocate(imageRequirements, vk::MemoryRequirement::Any);
vkd.bindImageMemory(device, *image, imageAllocation->getMemory(), imageAllocation->getOffset());
vk::Move<vk::VkImageView> imageView = makeImageView(vkd, device, *image, vk::VK_IMAGE_VIEW_TYPE_2D, vk::VK_FORMAT_R8G8B8A8_UNORM, makeImageSubresourceRange(vk::VK_IMAGE_ASPECT_COLOR_BIT, 0u, 1u, 0u, 1u));
vk::Move<vk::VkShaderModule> shaderModule = createShaderModule(vkd, device, context.getBinaryCollection().get("comp"), 0u);
vk::Move<vk::VkRenderPass> renderPass = makeRenderPass(vkd, device, vk::VK_FORMAT_R8G8B8A8_UNORM);
vk::Move<vk::VkFramebuffer> framebuffer = makeFramebuffer(vkd, device, *renderPass, *imageView, 64, 64);
const vk::VkPipelineCacheCreateInfo pipelineCacheCreateInfo
{
vk::VK_STRUCTURE_TYPE_PIPELINE_CACHE_CREATE_INFO, // VkStructureType sType;
DE_NULL, // const void* pNext;
vk::VK_PIPELINE_CACHE_CREATE_READ_ONLY_BIT |
vk::VK_PIPELINE_CACHE_CREATE_USE_APPLICATION_STORAGE_BIT, // VkPipelineCacheCreateFlags flags;
context.getResourceInterface()->getCacheDataSize(), // deUintptr initialDataSize;
context.getResourceInterface()->getCacheData() // const void* pInitialData;
};
vk::Move<vk::VkPipelineCache> pipelineCache = createPipelineCache(vkd, device, &pipelineCacheCreateInfo);
vk::Move<vk::VkPipelineLayout> pipelineLayout = makePipelineLayout(vkd, device);
vk::Move<vk::VkPipeline> pipeline = makeComputePipeline(vkd, device, *pipelineLayout, 0u, *shaderModule, 0u, DE_NULL);
const vk::VkDescriptorPoolSize descriptorPoolSize
{
vk::VK_DESCRIPTOR_TYPE_SAMPLER, // VkDescriptorType type;
8 // deUint32 descriptorCount;
};
const vk::VkDescriptorPoolCreateInfo descriptorPoolCreateInfo
{
vk::VK_STRUCTURE_TYPE_DESCRIPTOR_POOL_CREATE_INFO, // VkStructureType sType;
NULL, // const void* pNext;
0u, // VkDescriptorPoolCreateFlags flags;
8, // deUint32 maxSets;
1, // deUint32 poolSizeCount;
&descriptorPoolSize // const VkDescriptorPoolSize* pPoolSizes;
};
vk::Move<vk::VkDescriptorPool> descriptorPool = createDescriptorPool(vkd, device, &descriptorPoolCreateInfo);
const vk::VkDescriptorSetLayoutBinding descriptorSetLayoutBinding
{
0, // deUint32 binding;
vk::VK_DESCRIPTOR_TYPE_SAMPLER, // VkDescriptorType descriptorType;
1, // deUint32 descriptorCount;
vk::VK_SHADER_STAGE_ALL, // VkShaderStageFlags stageFlags;
NULL // const VkSampler* pImmutableSamplers;
};
const vk::VkDescriptorSetLayoutCreateInfo descriptorSetLayoutInfo
{
vk::VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO, // VkStructureType sType;
NULL, // const void* pNext;
0, // VkDescriptorSetLayoutCreateFlags flags;
1, // deUint32 bindingCount;
&descriptorSetLayoutBinding // const VkDescriptorSetLayoutBinding* pBindings;
};
vk::Move<vk::VkDescriptorSetLayout> descriptorSetLayout = createDescriptorSetLayout(vkd, device, &descriptorSetLayoutInfo);
vk::Move<vk::VkDescriptorSet> descriptorSet = makeDescriptorSet(vkd, device, *descriptorPool, *descriptorSetLayout);
std::map<vk::VkObjectType, deUint64> objectHandlesMap
{
{ vk::VK_OBJECT_TYPE_INSTANCE, 0 },
{ vk::VK_OBJECT_TYPE_PHYSICAL_DEVICE, 0 },
{ vk::VK_OBJECT_TYPE_DEVICE, 0 },
{ vk::VK_OBJECT_TYPE_QUEUE, 0 },
{ vk::VK_OBJECT_TYPE_SEMAPHORE, semaphore.get().getInternal() },
{ vk::VK_OBJECT_TYPE_COMMAND_BUFFER, 0 },
{ vk::VK_OBJECT_TYPE_FENCE, fence.get().getInternal() },
{ vk::VK_OBJECT_TYPE_DEVICE_MEMORY, bufferAllocation->getMemory().getInternal() },
{ vk::VK_OBJECT_TYPE_BUFFER, buffer.get().getInternal() },
{ vk::VK_OBJECT_TYPE_IMAGE, image.get().getInternal() },
{ vk::VK_OBJECT_TYPE_EVENT, event.get().getInternal() },
{ vk::VK_OBJECT_TYPE_QUERY_POOL, queryPool.get().getInternal() },
{ vk::VK_OBJECT_TYPE_BUFFER_VIEW, bufferView.get().getInternal() },
{ vk::VK_OBJECT_TYPE_IMAGE_VIEW, imageView.get().getInternal() },
{ vk::VK_OBJECT_TYPE_SHADER_MODULE, shaderModule.get().getInternal() },
{ vk::VK_OBJECT_TYPE_PIPELINE_CACHE, pipelineCache.get().getInternal() },
{ vk::VK_OBJECT_TYPE_PIPELINE_LAYOUT, pipelineLayout.get().getInternal() },
{ vk::VK_OBJECT_TYPE_RENDER_PASS, renderPass.get().getInternal() },
{ vk::VK_OBJECT_TYPE_PIPELINE, pipeline.get().getInternal() },
{ vk::VK_OBJECT_TYPE_DESCRIPTOR_SET_LAYOUT, descriptorSetLayout.get().getInternal() },
{ vk::VK_OBJECT_TYPE_SAMPLER, sampler.get().getInternal() },
{ vk::VK_OBJECT_TYPE_DESCRIPTOR_POOL, descriptorPool.get().getInternal() },
{ vk::VK_OBJECT_TYPE_DESCRIPTOR_SET, descriptorSet.get().getInternal() },
{ vk::VK_OBJECT_TYPE_FRAMEBUFFER, framebuffer.get().getInternal() },
{ vk::VK_OBJECT_TYPE_COMMAND_POOL, cmdPool.get().getInternal() },
{ vk::VK_OBJECT_TYPE_SAMPLER_YCBCR_CONVERSION, ycbcrConversion.get().getInternal() },
{ vk::VK_OBJECT_TYPE_SURFACE_KHR, 0 },
{ vk::VK_OBJECT_TYPE_SWAPCHAIN_KHR, 0 },
{ vk::VK_OBJECT_TYPE_DISPLAY_KHR, 0 },
{ vk::VK_OBJECT_TYPE_DISPLAY_MODE_KHR, 0 },
{ vk::VK_OBJECT_TYPE_DEBUG_UTILS_MESSENGER_EXT, 0 }
};
const vk::VkMemoryBarrier objRefreshBarrier
{
vk::VK_STRUCTURE_TYPE_MEMORY_BARRIER,
DE_NULL,
vk::VK_ACCESS_TRANSFER_WRITE_BIT,
vk::VK_ACCESS_MEMORY_READ_BIT
};
// record command buffer
vk::beginCommandBuffer(vkd, *cmdBuffer);
if (individualRefresh)
{
for (const auto& object : objectHandlesMap)
{
vk::VkObjectType objectType = object.first;
deUint64 objectHandle = object.second;
// skip object type if it can't be refreshed on current implementation
if (std::find(refreshableObjectTypes.begin(), refreshableObjectTypes.end(), objectType) == refreshableObjectTypes.end())
continue;
if (!objectHandle)
continue;
vk::VkRefreshObjectKHR refreshObject = { objectType, objectHandle, 0 };
vk::VkRefreshObjectListKHR refreshList = { vk::VK_STRUCTURE_TYPE_REFRESH_OBJECT_LIST_KHR, DE_NULL, 1, &refreshObject };
vkd.cmdRefreshObjectsKHR(*cmdBuffer, &refreshList);
vkd.cmdPipelineBarrier(*cmdBuffer, vk::VK_PIPELINE_STAGE_TRANSFER_BIT, vk::VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, 0, 1, &objRefreshBarrier, 0, DE_NULL, 0, DE_NULL);
}
}
else
{
deUint32 countUsed = 0;
std::vector<vk::VkRefreshObjectKHR> objectsToRefreshList(countReported);
for (deUint32 i = 0 ; i < countReported ; ++i)
{
vk::VkObjectType objectType = refreshableObjectTypes[i];
deUint64 objectHandle = objectHandlesMap.at(objectType);
if (!objectHandle)
continue;
objectsToRefreshList[countUsed++] =
{
objectType,
objectHandle,
0
};
}
vk::VkRefreshObjectListKHR refreshList = { vk::VK_STRUCTURE_TYPE_REFRESH_OBJECT_LIST_KHR, DE_NULL, countUsed, objectsToRefreshList.data() };
vkd.cmdRefreshObjectsKHR(*cmdBuffer, &refreshList);
vkd.cmdPipelineBarrier(*cmdBuffer, vk::VK_PIPELINE_STAGE_TRANSFER_BIT, vk::VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT, 0, 1, &objRefreshBarrier, 0, DE_NULL, 0, DE_NULL);
}
vk::endCommandBuffer(vkd, *cmdBuffer);
vk::submitCommandsAndWait(vkd, device, queue, *cmdBuffer);
return tcu::TestStatus::pass("Pass");
}
void createComputeSource(vk::SourceCollections& dst)
{
dst.glslSources.add("comp") << glu::ComputeSource(
"#version 450\n"
"layout(local_size_x = 1, local_size_y = 1, local_size_z = 1) in;\n"
"void main (void)\n"
"{\n"
" vec4 dummy = vec4(1.0);\n"
"}\n");
}
tcu::TestStatus refreshIndividualObjects(Context& context)
{
return refreshObjects(context, true);
}
tcu::TestStatus refreshAllObjects(Context& context)
{
return refreshObjects(context, false);
}
void checkRefreshSupport(Context& context)
{
context.requireDeviceFunctionality("VK_KHR_object_refresh");
}
} // anonymous
tcu::TestCaseGroup* createObjectRefreshTests (tcu::TestContext& testCtx)
{
de::MovePtr<tcu::TestCaseGroup> group(new tcu::TestCaseGroup(testCtx, "object_refresh", "Tests VK_KHR_object_refresh"));
addFunctionCase(group.get(), "query_refreshable_objects", "Test VK_KHR_object_refresh extension", checkRefreshSupport, queryRefreshableObjects);
addFunctionCaseWithPrograms(group.get(), "refresh_individual_objects", "Test VK_KHR_object_refresh extension", checkRefreshSupport, createComputeSource, refreshIndividualObjects);
addFunctionCaseWithPrograms(group.get(), "refresh_all_objects", "Test VK_KHR_object_refresh extension", checkRefreshSupport, createComputeSource, refreshAllObjects);
return group.release();
}
} // sc
} // vkt