/*------------------------------------------------------------------------ * 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 #include 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 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 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 cmdPool = createCommandPool(vkd, device, vk::VK_COMMAND_POOL_CREATE_RESET_COMMAND_BUFFER_BIT, queueFamilyIndex); vk::Move cmdBuffer = allocateCommandBuffer(vkd, device, *cmdPool, vk::VK_COMMAND_BUFFER_LEVEL_PRIMARY); vk::Move fence = createFence(vkd, device); vk::Move semaphore = createSemaphore(vkd, device); vk::Move 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 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 buffer = createBuffer(vkd, device, &bufferCreateInfo); const vk::VkMemoryRequirements bufferRequirements = getBufferMemoryRequirements(vkd, device, *buffer); de::MovePtr bufferAllocation = allocator.allocate(bufferRequirements, vk::MemoryRequirement::HostVisible); vkd.bindBufferMemory(device, *buffer, bufferAllocation->getMemory(), 0); vk::Move 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 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 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 image = createImage(vkd, device, &imageCreateInfo); const vk::VkMemoryRequirements imageRequirements = getImageMemoryRequirements(vkd, device, *image); de::MovePtr imageAllocation = allocator.allocate(imageRequirements, vk::MemoryRequirement::Any); vkd.bindImageMemory(device, *image, imageAllocation->getMemory(), imageAllocation->getOffset()); vk::Move 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 shaderModule = createShaderModule(vkd, device, context.getBinaryCollection().get("comp"), 0u); vk::Move renderPass = makeRenderPass(vkd, device, vk::VK_FORMAT_R8G8B8A8_UNORM); vk::Move 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 pipelineCache = createPipelineCache(vkd, device, &pipelineCacheCreateInfo); vk::Move pipelineLayout = makePipelineLayout(vkd, device); vk::Move 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 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 descriptorSetLayout = createDescriptorSetLayout(vkd, device, &descriptorSetLayoutInfo); vk::Move descriptorSet = makeDescriptorSet(vkd, device, *descriptorPool, *descriptorSetLayout); std::map 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 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 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