/*------------------------------------------------------------------------ * Vulkan Conformance Tests * ------------------------ * * Copyright (c) 2015 The Khronos Group Inc. * Copyright (c) 2015 Imagination Technologies Ltd. * * 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 Utilities for images. *//*--------------------------------------------------------------------*/ #include "vktPipelineImageUtil.hpp" #include "vkImageUtil.hpp" #include "vkMemUtil.hpp" #include "vkQueryUtil.hpp" #include "vkRefUtil.hpp" #include "vkCmdUtil.hpp" #include "vkTypeUtil.hpp" #include "tcuTextureUtil.hpp" #include "tcuAstcUtil.hpp" #include "deRandom.hpp" #include "deSharedPtr.hpp" namespace vkt { namespace pipeline { using namespace vk; /*! Gets the next multiple of a given divisor */ static deUint32 getNextMultiple (deUint32 divisor, deUint32 value) { if (value % divisor == 0) { return value; } return value + divisor - (value % divisor); } /*! Gets the next value that is multiple of all given divisors */ static deUint32 getNextMultiple (const std::vector& divisors, deUint32 value) { deUint32 nextMultiple = value; bool nextMultipleFound = false; while (true) { nextMultipleFound = true; for (size_t divNdx = 0; divNdx < divisors.size(); divNdx++) nextMultipleFound = nextMultipleFound && (nextMultiple % divisors[divNdx] == 0); if (nextMultipleFound) break; DE_ASSERT(nextMultiple < ~((deUint32)0u)); nextMultiple = getNextMultiple(divisors[0], nextMultiple + 1); } return nextMultiple; } bool isSupportedSamplableFormat (const InstanceInterface& instanceInterface, VkPhysicalDevice device, VkFormat format) { if (isCompressedFormat(format)) { VkPhysicalDeviceFeatures physicalFeatures; const tcu::CompressedTexFormat compressedFormat = mapVkCompressedFormat(format); instanceInterface.getPhysicalDeviceFeatures(device, &physicalFeatures); if (tcu::isAstcFormat(compressedFormat)) { if (!physicalFeatures.textureCompressionASTC_LDR) return false; } else if (tcu::isEtcFormat(compressedFormat)) { if (!physicalFeatures.textureCompressionETC2) return false; } else { DE_FATAL("Unsupported compressed format"); } } VkFormatProperties formatProps; instanceInterface.getPhysicalDeviceFormatProperties(device, format, &formatProps); return (formatProps.optimalTilingFeatures & VK_FORMAT_FEATURE_SAMPLED_IMAGE_BIT) != 0u; } bool isLinearFilteringSupported (const InstanceInterface& vki, VkPhysicalDevice physicalDevice, VkFormat format, VkImageTiling tiling) { const VkFormatProperties formatProperties = getPhysicalDeviceFormatProperties(vki, physicalDevice, format); const VkFormatFeatureFlags formatFeatures = tiling == VK_IMAGE_TILING_LINEAR ? formatProperties.linearTilingFeatures : formatProperties.optimalTilingFeatures; return (formatFeatures & VK_FORMAT_FEATURE_SAMPLED_IMAGE_FILTER_LINEAR_BIT) != 0; } bool isMinMaxFilteringSupported (const InstanceInterface& vki, VkPhysicalDevice physicalDevice, VkFormat format, VkImageTiling tiling) { const VkFormatProperties formatProperties = getPhysicalDeviceFormatProperties(vki, physicalDevice, format); const VkFormatFeatureFlags formatFeatures = tiling == VK_IMAGE_TILING_LINEAR ? formatProperties.linearTilingFeatures : formatProperties.optimalTilingFeatures; return (formatFeatures & VK_FORMAT_FEATURE_SAMPLED_IMAGE_FILTER_MINMAX_BIT) != 0; } static bool isBorderColorInt (VkFormat format, bool useStencilAspect) { return (!isCompressedFormat(format) && (isIntFormat(format) || isUintFormat(format) || (isDepthStencilFormat(format) && useStencilAspect))); } VkBorderColor getFormatBorderColor (BorderColor color, VkFormat format, bool useStencilAspect) { if (isBorderColorInt(format, useStencilAspect)) { switch (color) { case BORDER_COLOR_OPAQUE_BLACK: return VK_BORDER_COLOR_INT_OPAQUE_BLACK; case BORDER_COLOR_OPAQUE_WHITE: return VK_BORDER_COLOR_INT_OPAQUE_WHITE; case BORDER_COLOR_TRANSPARENT_BLACK: return VK_BORDER_COLOR_INT_TRANSPARENT_BLACK; case BORDER_COLOR_CUSTOM: return VK_BORDER_COLOR_INT_CUSTOM_EXT; default: break; } } else { switch (color) { case BORDER_COLOR_OPAQUE_BLACK: return VK_BORDER_COLOR_FLOAT_OPAQUE_BLACK; case BORDER_COLOR_OPAQUE_WHITE: return VK_BORDER_COLOR_FLOAT_OPAQUE_WHITE; case BORDER_COLOR_TRANSPARENT_BLACK: return VK_BORDER_COLOR_FLOAT_TRANSPARENT_BLACK; case BORDER_COLOR_CUSTOM: return VK_BORDER_COLOR_FLOAT_CUSTOM_EXT; default: break; } } DE_ASSERT(false); return VK_BORDER_COLOR_FLOAT_TRANSPARENT_BLACK; } rr::GenericVec4 getFormatCustomBorderColor (tcu::Vec4 floatValue, tcu::IVec4 intValue, vk::VkFormat format, bool useStencilAspect) { if (isBorderColorInt(format, useStencilAspect)) { return rr::GenericVec4(intValue); } else { return rr::GenericVec4(floatValue); } } void getLookupScaleBias (vk::VkFormat format, tcu::Vec4& lookupScale, tcu::Vec4& lookupBias, bool useStencilAspect) { if (!isCompressedFormat(format)) { const auto tcuFormat = mapVkFormat(format); if (useStencilAspect) { DE_ASSERT(tcu::hasStencilComponent(tcuFormat.order)); lookupScale = tcu::Vec4(1.0f / 255.0f, 1.0f, 1.0f, 1.0f); lookupBias = tcu::Vec4(0.0f, 0.0f, 0.0f, 0.0f); } else { const tcu::TextureFormatInfo fmtInfo = tcu::getTextureFormatInfo(tcuFormat); // Needed to normalize various formats to 0..1 range for writing into RT lookupScale = fmtInfo.lookupScale; lookupBias = fmtInfo.lookupBias; } } else { switch (format) { case VK_FORMAT_EAC_R11_SNORM_BLOCK: lookupScale = tcu::Vec4(0.5f, 1.0f, 1.0f, 1.0f); lookupBias = tcu::Vec4(0.5f, 0.0f, 0.0f, 0.0f); break; case VK_FORMAT_EAC_R11G11_SNORM_BLOCK: lookupScale = tcu::Vec4(0.5f, 0.5f, 1.0f, 1.0f); lookupBias = tcu::Vec4(0.5f, 0.5f, 0.0f, 0.0f); break; default: // else: All supported compressed formats are fine with no normalization. // ASTC LDR blocks decompress to f16 so querying normalization parameters // based on uncompressed formats would actually lead to massive precision loss // and complete lack of coverage in case of R8G8B8A8_UNORM RT. lookupScale = tcu::Vec4(1.0f); lookupBias = tcu::Vec4(0.0f); break; } } } de::MovePtr readColorAttachment (const vk::DeviceInterface& vk, vk::VkDevice device, vk::VkQueue queue, deUint32 queueFamilyIndex, vk::Allocator& allocator, vk::VkImage image, vk::VkFormat format, const tcu::UVec2& renderSize, vk::VkImageLayout oldLayout) { Move buffer; de::MovePtr bufferAlloc; Move cmdPool; Move cmdBuffer; Move fence; const tcu::TextureFormat tcuFormat = mapVkFormat(format); const VkDeviceSize pixelDataSize = renderSize.x() * renderSize.y() * tcuFormat.getPixelSize(); de::MovePtr resultLevel (new tcu::TextureLevel(tcuFormat, renderSize.x(), renderSize.y())); // Create destination buffer { const VkBufferCreateInfo bufferParams = { VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO, // VkStructureType sType; DE_NULL, // const void* pNext; 0u, // VkBufferCreateFlags flags; pixelDataSize, // VkDeviceSize size; VK_BUFFER_USAGE_TRANSFER_DST_BIT, // VkBufferUsageFlags usage; VK_SHARING_MODE_EXCLUSIVE, // VkSharingMode sharingMode; 0u, // deUint32 queueFamilyIndexCount; DE_NULL // const deUint32* pQueueFamilyIndices; }; buffer = createBuffer(vk, device, &bufferParams); bufferAlloc = allocator.allocate(getBufferMemoryRequirements(vk, device, *buffer), MemoryRequirement::HostVisible); VK_CHECK(vk.bindBufferMemory(device, *buffer, bufferAlloc->getMemory(), bufferAlloc->getOffset())); } // Create command pool and buffer cmdPool = createCommandPool(vk, device, VK_COMMAND_POOL_CREATE_TRANSIENT_BIT, queueFamilyIndex); cmdBuffer = allocateCommandBuffer(vk, device, *cmdPool, VK_COMMAND_BUFFER_LEVEL_PRIMARY); // Create fence fence = createFence(vk, device); beginCommandBuffer(vk, *cmdBuffer); copyImageToBuffer(vk, *cmdBuffer, image, *buffer, tcu::IVec2(renderSize.x(), renderSize.y()), VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT, oldLayout); endCommandBuffer(vk, *cmdBuffer); submitCommandsAndWait(vk, device, queue, cmdBuffer.get()); // Read buffer data invalidateAlloc(vk, device, *bufferAlloc); tcu::copy(*resultLevel, tcu::ConstPixelBufferAccess(resultLevel->getFormat(), resultLevel->getSize(), bufferAlloc->getHostPtr())); return resultLevel; } de::MovePtr readDepthAttachment (const vk::DeviceInterface& vk, vk::VkDevice device, vk::VkQueue queue, deUint32 queueFamilyIndex, vk::Allocator& allocator, vk::VkImage image, vk::VkFormat format, const tcu::UVec2& renderSize, vk::VkImageLayout currentLayout) { Move buffer; de::MovePtr bufferAlloc; Move cmdPool; Move cmdBuffer; tcu::TextureFormat retFormat (tcu::TextureFormat::D, tcu::TextureFormat::CHANNELTYPE_LAST); tcu::TextureFormat bufferFormat (tcu::TextureFormat::D, tcu::TextureFormat::CHANNELTYPE_LAST); const VkImageAspectFlags barrierAspect = VK_IMAGE_ASPECT_DEPTH_BIT | (mapVkFormat(format).order == tcu::TextureFormat::DS ? VK_IMAGE_ASPECT_STENCIL_BIT : (VkImageAspectFlagBits)0); switch (format) { case vk::VK_FORMAT_D16_UNORM: case vk::VK_FORMAT_D16_UNORM_S8_UINT: bufferFormat.type = retFormat.type = tcu::TextureFormat::UNORM_INT16; break; case vk::VK_FORMAT_D24_UNORM_S8_UINT: case vk::VK_FORMAT_X8_D24_UNORM_PACK32: retFormat.type = tcu::TextureFormat::UNORM_INT24; // vkCmdCopyBufferToImage copies D24 data to 32-bit pixels. bufferFormat.type = tcu::TextureFormat::UNSIGNED_INT_24_8_REV; break; case vk::VK_FORMAT_D32_SFLOAT: case vk::VK_FORMAT_D32_SFLOAT_S8_UINT: bufferFormat.type = retFormat.type = tcu::TextureFormat::FLOAT; break; default: TCU_FAIL("unrecognized format"); } const VkDeviceSize pixelDataSize = renderSize.x() * renderSize.y() * bufferFormat.getPixelSize(); de::MovePtr resultLevel (new tcu::TextureLevel(retFormat, renderSize.x(), renderSize.y())); // Create destination buffer { const VkBufferCreateInfo bufferParams = { VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO, // VkStructureType sType; DE_NULL, // const void* pNext; 0u, // VkBufferCreateFlags flags; pixelDataSize, // VkDeviceSize size; VK_BUFFER_USAGE_TRANSFER_DST_BIT, // VkBufferUsageFlags usage; VK_SHARING_MODE_EXCLUSIVE, // VkSharingMode sharingMode; 0u, // deUint32 queueFamilyIndexCount; DE_NULL // const deUint32* pQueueFamilyIndices; }; buffer = createBuffer(vk, device, &bufferParams); bufferAlloc = allocator.allocate(getBufferMemoryRequirements(vk, device, *buffer), MemoryRequirement::HostVisible); VK_CHECK(vk.bindBufferMemory(device, *buffer, bufferAlloc->getMemory(), bufferAlloc->getOffset())); } // Create command pool and buffer cmdPool = createCommandPool(vk, device, VK_COMMAND_POOL_CREATE_TRANSIENT_BIT, queueFamilyIndex); cmdBuffer = allocateCommandBuffer(vk, device, *cmdPool, VK_COMMAND_BUFFER_LEVEL_PRIMARY); beginCommandBuffer(vk, *cmdBuffer); copyImageToBuffer(vk, *cmdBuffer, image, *buffer, tcu::IVec2(renderSize.x(), renderSize.y()), VK_ACCESS_DEPTH_STENCIL_ATTACHMENT_WRITE_BIT, currentLayout, 1u, barrierAspect, VK_IMAGE_ASPECT_DEPTH_BIT); endCommandBuffer(vk, *cmdBuffer); submitCommandsAndWait(vk, device, queue, cmdBuffer.get()); // Read buffer data invalidateAlloc(vk, device, *bufferAlloc); tcu::copy(*resultLevel, tcu::ConstPixelBufferAccess(bufferFormat, resultLevel->getSize(), bufferAlloc->getHostPtr())); return resultLevel; } de::MovePtr readStencilAttachment (const vk::DeviceInterface& vk, vk::VkDevice device, vk::VkQueue queue, deUint32 queueFamilyIndex, vk::Allocator& allocator, vk::VkImage image, vk::VkFormat format, const tcu::UVec2& renderSize, vk::VkImageLayout currentLayout) { Move buffer; de::MovePtr bufferAlloc; Move cmdPool; Move cmdBuffer; tcu::TextureFormat retFormat (tcu::TextureFormat::S, tcu::TextureFormat::UNSIGNED_INT8); tcu::TextureFormat bufferFormat (tcu::TextureFormat::S, tcu::TextureFormat::UNSIGNED_INT8); const VkImageAspectFlags barrierAspect = VK_IMAGE_ASPECT_STENCIL_BIT | (mapVkFormat(format).order == tcu::TextureFormat::DS ? VK_IMAGE_ASPECT_DEPTH_BIT : (VkImageAspectFlagBits)0); const VkDeviceSize pixelDataSize = renderSize.x() * renderSize.y() * bufferFormat.getPixelSize(); de::MovePtr resultLevel (new tcu::TextureLevel(retFormat, renderSize.x(), renderSize.y())); // Create destination buffer { const VkBufferCreateInfo bufferParams = { VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO, // VkStructureType sType; DE_NULL, // const void* pNext; 0u, // VkBufferCreateFlags flags; pixelDataSize, // VkDeviceSize size; VK_BUFFER_USAGE_TRANSFER_DST_BIT, // VkBufferUsageFlags usage; VK_SHARING_MODE_EXCLUSIVE, // VkSharingMode sharingMode; 0u, // deUint32 queueFamilyIndexCount; DE_NULL // const deUint32* pQueueFamilyIndices; }; buffer = createBuffer(vk, device, &bufferParams); bufferAlloc = allocator.allocate(getBufferMemoryRequirements(vk, device, *buffer), MemoryRequirement::HostVisible); VK_CHECK(vk.bindBufferMemory(device, *buffer, bufferAlloc->getMemory(), bufferAlloc->getOffset())); } // Create command pool and buffer cmdPool = createCommandPool(vk, device, VK_COMMAND_POOL_CREATE_TRANSIENT_BIT, queueFamilyIndex); cmdBuffer = allocateCommandBuffer(vk, device, *cmdPool, VK_COMMAND_BUFFER_LEVEL_PRIMARY); beginCommandBuffer(vk, *cmdBuffer); copyImageToBuffer(vk, *cmdBuffer, image, *buffer, tcu::IVec2(renderSize.x(), renderSize.y()), VK_ACCESS_DEPTH_STENCIL_ATTACHMENT_WRITE_BIT, currentLayout, 1u, barrierAspect, VK_IMAGE_ASPECT_STENCIL_BIT); endCommandBuffer(vk, *cmdBuffer); submitCommandsAndWait(vk, device, queue, cmdBuffer.get()); // Read buffer data invalidateAlloc(vk, device, *bufferAlloc); tcu::copy(*resultLevel, tcu::ConstPixelBufferAccess(bufferFormat, resultLevel->getSize(), bufferAlloc->getHostPtr())); return resultLevel; } void uploadTestTextureInternal (const DeviceInterface& vk, VkDevice device, VkQueue queue, deUint32 queueFamilyIndex, Allocator& allocator, const TestTexture& srcTexture, const TestTexture* srcStencilTexture, tcu::TextureFormat format, VkImage destImage, VkImageLayout destImageLayout) { Move buffer; de::MovePtr bufferAlloc; Move cmdPool; Move cmdBuffer; const VkImageAspectFlags imageAspectFlags = getImageAspectFlags(format); deUint32 stencilOffset = 0u; std::vector copyRegions = srcTexture.getBufferCopyRegions(); deUint32 bufferSize = (srcTexture.isCompressed())? srcTexture.getCompressedSize(): srcTexture.getSize(); // Stencil-only texture should be provided if (and only if) the image has a combined DS format DE_ASSERT((tcu::hasDepthComponent(format.order) && tcu::hasStencilComponent(format.order)) == (srcStencilTexture != DE_NULL)); if (srcStencilTexture != DE_NULL) { stencilOffset = static_cast(deAlign32(static_cast(bufferSize), 4)); bufferSize = stencilOffset + srcStencilTexture->getSize(); } // Create source buffer { const VkBufferCreateInfo bufferParams = { VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO, // VkStructureType sType; DE_NULL, // const void* pNext; 0u, // VkBufferCreateFlags flags; bufferSize, // VkDeviceSize size; VK_BUFFER_USAGE_TRANSFER_SRC_BIT, // VkBufferUsageFlags usage; VK_SHARING_MODE_EXCLUSIVE, // VkSharingMode sharingMode; 0u, // deUint32 queueFamilyIndexCount; DE_NULL, // const deUint32* pQueueFamilyIndices; }; buffer = createBuffer(vk, device, &bufferParams); bufferAlloc = allocator.allocate(getBufferMemoryRequirements(vk, device, *buffer), MemoryRequirement::HostVisible); VK_CHECK(vk.bindBufferMemory(device, *buffer, bufferAlloc->getMemory(), bufferAlloc->getOffset())); } // Write buffer data { srcTexture.write(reinterpret_cast(bufferAlloc->getHostPtr())); if (srcStencilTexture != DE_NULL) { DE_ASSERT(stencilOffset != 0u); srcStencilTexture->write(reinterpret_cast(bufferAlloc->getHostPtr()) + stencilOffset); std::vector stencilCopyRegions = srcStencilTexture->getBufferCopyRegions(); for (size_t regionIdx = 0; regionIdx < stencilCopyRegions.size(); regionIdx++) { VkBufferImageCopy region = stencilCopyRegions[regionIdx]; region.bufferOffset += stencilOffset; copyRegions.push_back(region); } } flushAlloc(vk, device, *bufferAlloc); } copyBufferToImage(vk, device, queue, queueFamilyIndex, *buffer, bufferSize, copyRegions, DE_NULL, imageAspectFlags, srcTexture.getNumLevels(), srcTexture.getArraySize(), destImage, destImageLayout); } bool checkSparseImageFormatSupport (const VkPhysicalDevice physicalDevice, const InstanceInterface& instance, const VkImageCreateInfo& imageCreateInfo) { #ifndef CTS_USES_VULKANSC const std::vector sparseImageFormatPropVec = getPhysicalDeviceSparseImageFormatProperties(instance, physicalDevice, imageCreateInfo.format, imageCreateInfo.imageType, imageCreateInfo.samples, imageCreateInfo.usage, imageCreateInfo.tiling); return (sparseImageFormatPropVec.size() != 0); #else DE_UNREF(physicalDevice); DE_UNREF(instance); DE_UNREF(imageCreateInfo); return false; #endif // CTS_USES_VULKANSC } void uploadTestTextureInternalSparse (const DeviceInterface& vk, VkDevice device, const VkPhysicalDevice physicalDevice, const InstanceInterface& instance, const VkImageCreateInfo& imageCreateInfo, VkQueue universalQueue, deUint32 universalQueueFamilyIndex, VkQueue sparseQueue, Allocator& allocator, std::vector >& allocations, const TestTexture& srcTexture, const TestTexture* srcStencilTexture, tcu::TextureFormat format, VkImage destImage) { deUint32 bufferSize = (srcTexture.isCompressed()) ? srcTexture.getCompressedSize(): srcTexture.getSize(); const VkImageAspectFlags imageAspectFlags = getImageAspectFlags(format); deUint32 stencilOffset = 0u; const Unique imageMemoryBindSemaphore(createSemaphore(vk, device)); Move cmdPool = createCommandPool(vk, device, VK_COMMAND_POOL_CREATE_TRANSIENT_BIT, universalQueueFamilyIndex); Move cmdBuffer = allocateCommandBuffer(vk, device, *cmdPool, VK_COMMAND_BUFFER_LEVEL_PRIMARY); Move fence = createFence(vk, device); std::vector copyRegions = srcTexture.getBufferCopyRegions(); Move buffer; de::MovePtr bufferAlloc; // Stencil-only texture should be provided if (and only if) the image has a combined DS format DE_ASSERT((tcu::hasDepthComponent(format.order) && tcu::hasStencilComponent(format.order)) == (srcStencilTexture != DE_NULL)); if (srcStencilTexture != DE_NULL) { stencilOffset = static_cast(deAlign32(static_cast(bufferSize), 4)); bufferSize = stencilOffset + srcStencilTexture->getSize(); } #ifndef CTS_USES_VULKANSC allocateAndBindSparseImage (vk, device, physicalDevice, instance, imageCreateInfo, imageMemoryBindSemaphore.get(), sparseQueue, allocator, allocations, format, destImage); #else DE_UNREF(physicalDevice); DE_UNREF(instance); DE_UNREF(sparseQueue); DE_UNREF(allocations); #endif // CTS_USES_VULKANSC { // Create source buffer const VkBufferCreateInfo bufferParams = { VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO, // VkStructureType sType; DE_NULL, // const void* pNext; 0u, // VkBufferCreateFlags flags; bufferSize, // VkDeviceSize size; VK_BUFFER_USAGE_TRANSFER_SRC_BIT, // VkBufferUsageFlags usage; VK_SHARING_MODE_EXCLUSIVE, // VkSharingMode sharingMode; 0u, // deUint32 queueFamilyIndexCount; DE_NULL, // const deUint32* pQueueFamilyIndices; }; buffer = createBuffer(vk, device, &bufferParams); bufferAlloc = allocator.allocate(getBufferMemoryRequirements(vk, device, *buffer), MemoryRequirement::HostVisible); VK_CHECK(vk.bindBufferMemory(device, *buffer, bufferAlloc->getMemory(), bufferAlloc->getOffset())); } { // Write buffer data srcTexture.write(reinterpret_cast(bufferAlloc->getHostPtr())); if (srcStencilTexture != DE_NULL) { DE_ASSERT(stencilOffset != 0u); srcStencilTexture->write(reinterpret_cast(bufferAlloc->getHostPtr()) + stencilOffset); std::vector stencilCopyRegions = srcStencilTexture->getBufferCopyRegions(); for (size_t regionIdx = 0; regionIdx < stencilCopyRegions.size(); regionIdx++) { VkBufferImageCopy region = stencilCopyRegions[regionIdx]; region.bufferOffset += stencilOffset; copyRegions.push_back(region); } } flushAlloc(vk, device, *bufferAlloc); } copyBufferToImage(vk, device, universalQueue, universalQueueFamilyIndex, *buffer, bufferSize, copyRegions, &(*imageMemoryBindSemaphore), imageAspectFlags, imageCreateInfo.mipLevels, imageCreateInfo.arrayLayers, destImage); } void uploadTestTexture (const DeviceInterface& vk, VkDevice device, VkQueue queue, deUint32 queueFamilyIndex, Allocator& allocator, const TestTexture& srcTexture, VkImage destImage, VkImageLayout destImageLayout) { if (tcu::isCombinedDepthStencilType(srcTexture.getTextureFormat().type)) { de::MovePtr srcDepthTexture; de::MovePtr srcStencilTexture; if (tcu::hasDepthComponent(srcTexture.getTextureFormat().order)) { tcu::TextureFormat format; switch (srcTexture.getTextureFormat().type) { case tcu::TextureFormat::UNSIGNED_INT_16_8_8: format = tcu::TextureFormat(tcu::TextureFormat::D, tcu::TextureFormat::UNORM_INT16); break; case tcu::TextureFormat::UNSIGNED_INT_24_8_REV: format = tcu::TextureFormat(tcu::TextureFormat::D, tcu::TextureFormat::UNSIGNED_INT_24_8_REV); break; case tcu::TextureFormat::FLOAT_UNSIGNED_INT_24_8_REV: format = tcu::TextureFormat(tcu::TextureFormat::D, tcu::TextureFormat::FLOAT); break; default: DE_FATAL("Unexpected source texture format."); break; } srcDepthTexture = srcTexture.copy(format); } if (tcu::hasStencilComponent(srcTexture.getTextureFormat().order)) srcStencilTexture = srcTexture.copy(tcu::getEffectiveDepthStencilTextureFormat(srcTexture.getTextureFormat(), tcu::Sampler::MODE_STENCIL)); uploadTestTextureInternal(vk, device, queue, queueFamilyIndex, allocator, *srcDepthTexture, srcStencilTexture.get(), srcTexture.getTextureFormat(), destImage, destImageLayout); } else uploadTestTextureInternal(vk, device, queue, queueFamilyIndex, allocator, srcTexture, DE_NULL, srcTexture.getTextureFormat(), destImage, destImageLayout); } void uploadTestTextureSparse (const DeviceInterface& vk, VkDevice device, const VkPhysicalDevice physicalDevice, const InstanceInterface& instance, const VkImageCreateInfo& imageCreateInfo, VkQueue universalQueue, deUint32 universalQueueFamilyIndex, VkQueue sparseQueue, Allocator& allocator, std::vector >& allocations, const TestTexture& srcTexture, VkImage destImage) { if (tcu::isCombinedDepthStencilType(srcTexture.getTextureFormat().type)) { de::MovePtr srcDepthTexture; de::MovePtr srcStencilTexture; if (tcu::hasDepthComponent(srcTexture.getTextureFormat().order)) { tcu::TextureFormat format; switch (srcTexture.getTextureFormat().type) { case tcu::TextureFormat::UNSIGNED_INT_16_8_8: format = tcu::TextureFormat(tcu::TextureFormat::D, tcu::TextureFormat::UNORM_INT16); break; case tcu::TextureFormat::UNSIGNED_INT_24_8_REV: format = tcu::TextureFormat(tcu::TextureFormat::D, tcu::TextureFormat::UNSIGNED_INT_24_8_REV); break; case tcu::TextureFormat::FLOAT_UNSIGNED_INT_24_8_REV: format = tcu::TextureFormat(tcu::TextureFormat::D, tcu::TextureFormat::FLOAT); break; default: DE_FATAL("Unexpected source texture format."); break; } srcDepthTexture = srcTexture.copy(format); } if (tcu::hasStencilComponent(srcTexture.getTextureFormat().order)) srcStencilTexture = srcTexture.copy(tcu::getEffectiveDepthStencilTextureFormat(srcTexture.getTextureFormat(), tcu::Sampler::MODE_STENCIL)); uploadTestTextureInternalSparse (vk, device, physicalDevice, instance, imageCreateInfo, universalQueue, universalQueueFamilyIndex, sparseQueue, allocator, allocations, *srcDepthTexture, srcStencilTexture.get(), srcTexture.getTextureFormat(), destImage); } else { uploadTestTextureInternalSparse (vk, device, physicalDevice, instance, imageCreateInfo, universalQueue, universalQueueFamilyIndex, sparseQueue, allocator, allocations, srcTexture, DE_NULL, srcTexture.getTextureFormat(), destImage); } } // Utilities for test textures template void allocateLevels (TcuTextureType& texture) { for (int levelNdx = 0; levelNdx < texture.getNumLevels(); levelNdx++) texture.allocLevel(levelNdx); } template std::vector getLevelsVector (const TcuTextureType& texture) { std::vector levels(texture.getNumLevels()); for (int levelNdx = 0; levelNdx < texture.getNumLevels(); levelNdx++) levels[levelNdx] = *reinterpret_cast(&texture.getLevel(levelNdx)); return levels; } // TestTexture TestTexture::TestTexture (const tcu::TextureFormat& format, int width, int height, int depth) { DE_ASSERT(width >= 1); DE_ASSERT(height >= 1); DE_ASSERT(depth >= 1); DE_UNREF(format); DE_UNREF(width); DE_UNREF(height); DE_UNREF(depth); } TestTexture::TestTexture (const tcu::CompressedTexFormat& format, int width, int height, int depth) { DE_ASSERT(width >= 1); DE_ASSERT(height >= 1); DE_ASSERT(depth >= 1); DE_UNREF(format); DE_UNREF(width); DE_UNREF(height); DE_UNREF(depth); } TestTexture::~TestTexture (void) { for (size_t levelNdx = 0; levelNdx < m_compressedLevels.size(); levelNdx++) delete m_compressedLevels[levelNdx]; } deUint32 TestTexture::getSize (void) const { std::vector offsetMultiples; deUint32 textureSize = 0; offsetMultiples.push_back(4); offsetMultiples.push_back(getLevel(0, 0).getFormat().getPixelSize()); for (int levelNdx = 0; levelNdx < getNumLevels(); levelNdx++) { for (int layerNdx = 0; layerNdx < getArraySize(); layerNdx++) { const tcu::ConstPixelBufferAccess level = getLevel(levelNdx, layerNdx); textureSize = getNextMultiple(offsetMultiples, textureSize); textureSize += level.getWidth() * level.getHeight() * level.getDepth() * level.getFormat().getPixelSize(); } } return textureSize; } deUint32 TestTexture::getCompressedSize (void) const { if (!isCompressed()) throw tcu::InternalError("Texture is not compressed"); std::vector offsetMultiples; deUint32 textureSize = 0; offsetMultiples.push_back(4); offsetMultiples.push_back(tcu::getBlockSize(getCompressedLevel(0, 0).getFormat())); for (int levelNdx = 0; levelNdx < getNumLevels(); levelNdx++) { for (int layerNdx = 0; layerNdx < getArraySize(); layerNdx++) { textureSize = getNextMultiple(offsetMultiples, textureSize); textureSize += getCompressedLevel(levelNdx, layerNdx).getDataSize(); } } return textureSize; } tcu::CompressedTexture& TestTexture::getCompressedLevel (int level, int layer) { DE_ASSERT(level >= 0 && level < getNumLevels()); DE_ASSERT(layer >= 0 && layer < getArraySize()); return *m_compressedLevels[level * getArraySize() + layer]; } const tcu::CompressedTexture& TestTexture::getCompressedLevel (int level, int layer) const { DE_ASSERT(level >= 0 && level < getNumLevels()); DE_ASSERT(layer >= 0 && layer < getArraySize()); return *m_compressedLevels[level * getArraySize() + layer]; } std::vector TestTexture::getBufferCopyRegions (void) const { std::vector offsetMultiples; std::vector regions; deUint32 layerDataOffset = 0; offsetMultiples.push_back(4); if (isCompressed()) { offsetMultiples.push_back(tcu::getBlockSize(getCompressedLevel(0, 0).getFormat())); for (int levelNdx = 0; levelNdx < getNumLevels(); levelNdx++) { for (int layerNdx = 0; layerNdx < getArraySize(); layerNdx++) { const tcu::CompressedTexture& level = getCompressedLevel(levelNdx, layerNdx); tcu::IVec3 blockPixelSize = getBlockPixelSize(level.getFormat()); layerDataOffset = getNextMultiple(offsetMultiples, layerDataOffset); const VkBufferImageCopy layerRegion = { layerDataOffset, // VkDeviceSize bufferOffset; (deUint32)getNextMultiple(blockPixelSize.x(), level.getWidth()), // deUint32 bufferRowLength; (deUint32)getNextMultiple(blockPixelSize.y(), level.getHeight()), // deUint32 bufferImageHeight; { // VkImageSubresourceLayers imageSubresource; VK_IMAGE_ASPECT_COLOR_BIT, (deUint32)levelNdx, (deUint32)layerNdx, 1u }, { 0u, 0u, 0u }, // VkOffset3D imageOffset; { // VkExtent3D imageExtent; (deUint32)level.getWidth(), (deUint32)level.getHeight(), (deUint32)level.getDepth() } }; regions.push_back(layerRegion); layerDataOffset += level.getDataSize(); } } } else { std::vector imageAspects; tcu::TextureFormat textureFormat = getTextureFormat(); if (tcu::hasDepthComponent(textureFormat.order)) imageAspects.push_back(VK_IMAGE_ASPECT_DEPTH_BIT); if (tcu::hasStencilComponent(textureFormat.order)) imageAspects.push_back(VK_IMAGE_ASPECT_STENCIL_BIT); if (imageAspects.empty()) imageAspects.push_back(VK_IMAGE_ASPECT_COLOR_BIT); offsetMultiples.push_back(getLevel(0, 0).getFormat().getPixelSize()); for (int levelNdx = 0; levelNdx < getNumLevels(); levelNdx++) { for (int layerNdx = 0; layerNdx < getArraySize(); layerNdx++) { const tcu::ConstPixelBufferAccess level = getLevel(levelNdx, layerNdx); layerDataOffset = getNextMultiple(offsetMultiples, layerDataOffset); for (size_t aspectIndex = 0; aspectIndex < imageAspects.size(); ++aspectIndex) { const VkBufferImageCopy layerRegion = { layerDataOffset, // VkDeviceSize bufferOffset; (deUint32)level.getWidth(), // deUint32 bufferRowLength; (deUint32)level.getHeight(), // deUint32 bufferImageHeight; { // VkImageSubresourceLayers imageSubresource; imageAspects[aspectIndex], (deUint32)levelNdx, (deUint32)layerNdx, 1u }, { 0u, 0u, 0u }, // VkOffset3D imageOffset; { // VkExtent3D imageExtent; (deUint32)level.getWidth(), (deUint32)level.getHeight(), (deUint32)level.getDepth() } }; regions.push_back(layerRegion); } layerDataOffset += level.getWidth() * level.getHeight() * level.getDepth() * level.getFormat().getPixelSize(); } } } return regions; } void TestTexture::write (deUint8* destPtr) const { std::vector offsetMultiples; deUint32 levelOffset = 0; offsetMultiples.push_back(4); if (isCompressed()) { offsetMultiples.push_back(tcu::getBlockSize(getCompressedLevel(0, 0).getFormat())); for (int levelNdx = 0; levelNdx < getNumLevels(); levelNdx++) { for (int layerNdx = 0; layerNdx < getArraySize(); layerNdx++) { levelOffset = getNextMultiple(offsetMultiples, levelOffset); const tcu::CompressedTexture& compressedTex = getCompressedLevel(levelNdx, layerNdx); deMemcpy(destPtr + levelOffset, compressedTex.getData(), compressedTex.getDataSize()); levelOffset += compressedTex.getDataSize(); } } } else { offsetMultiples.push_back(getLevel(0, 0).getFormat().getPixelSize()); for (int levelNdx = 0; levelNdx < getNumLevels(); levelNdx++) { for (int layerNdx = 0; layerNdx < getArraySize(); layerNdx++) { levelOffset = getNextMultiple(offsetMultiples, levelOffset); const tcu::ConstPixelBufferAccess srcAccess = getLevel(levelNdx, layerNdx); const tcu::PixelBufferAccess destAccess (srcAccess.getFormat(), srcAccess.getSize(), srcAccess.getPitch(), destPtr + levelOffset); tcu::copy(destAccess, srcAccess); levelOffset += srcAccess.getWidth() * srcAccess.getHeight() * srcAccess.getDepth() * srcAccess.getFormat().getPixelSize(); } } } } void TestTexture::copyToTexture (TestTexture& destTexture) const { for (int levelNdx = 0; levelNdx < getNumLevels(); levelNdx++) for (int layerNdx = 0; layerNdx < getArraySize(); layerNdx++) tcu::copy(destTexture.getLevel(levelNdx, layerNdx), getLevel(levelNdx, layerNdx)); } void TestTexture::populateLevels (const std::vector& levels) { for (size_t levelNdx = 0; levelNdx < levels.size(); levelNdx++) TestTexture::fillWithGradient(levels[levelNdx]); } void TestTexture::populateCompressedLevels (tcu::CompressedTexFormat format, const std::vector& decompressedLevels) { // Generate random compressed data and update decompressed data de::Random random(123); for (size_t levelNdx = 0; levelNdx < decompressedLevels.size(); levelNdx++) { const tcu::PixelBufferAccess level = decompressedLevels[levelNdx]; tcu::CompressedTexture* compressedLevel = new tcu::CompressedTexture(format, level.getWidth(), level.getHeight(), level.getDepth()); deUint8* const compressedData = (deUint8*)compressedLevel->getData(); if (tcu::isAstcFormat(format)) { // \todo [2016-01-20 pyry] Comparison doesn't currently handle invalid blocks correctly so we use only valid blocks tcu::astc::generateRandomValidBlocks(compressedData, compressedLevel->getDataSize()/tcu::astc::BLOCK_SIZE_BYTES, format, tcu::TexDecompressionParams::ASTCMODE_LDR, random.getUint32()); } else { // Generate random compressed data // Random initial values cause assertion during the decompression in case of COMPRESSEDTEXFORMAT_ETC1_RGB8 format if (format != tcu::COMPRESSEDTEXFORMAT_ETC1_RGB8) for (int byteNdx = 0; byteNdx < compressedLevel->getDataSize(); byteNdx++) compressedData[byteNdx] = 0xFF & random.getUint32(); // BC7 mode 8 (LSB==0x00) should not be tested as it is underspecified if (format == tcu::COMPRESSEDTEXFORMAT_BC7_UNORM_BLOCK || format == tcu::COMPRESSEDTEXFORMAT_BC7_SRGB_BLOCK) { const int blockSize = tcu::getBlockSize(format); for (int byteNdx = 0; byteNdx < compressedLevel->getDataSize(); byteNdx += blockSize) while (compressedData[byteNdx] == 0x00) compressedData[byteNdx] = 0xFF & random.getUint32(); } } m_compressedLevels.push_back(compressedLevel); // Store decompressed data compressedLevel->decompress(level, tcu::TexDecompressionParams(tcu::TexDecompressionParams::ASTCMODE_LDR)); } } void TestTexture::fillWithGradient (const tcu::PixelBufferAccess& levelAccess) { const tcu::TextureFormatInfo formatInfo = tcu::getTextureFormatInfo(levelAccess.getFormat()); tcu::fillWithComponentGradients2(levelAccess, formatInfo.valueMin, formatInfo.valueMax); } // TestTexture1D TestTexture1D::TestTexture1D (const tcu::TextureFormat& format, int width) : TestTexture (format, width, 1, 1) , m_texture (format, width) { allocateLevels(m_texture); TestTexture::populateLevels(getLevelsVector(m_texture)); } TestTexture1D::TestTexture1D (const tcu::CompressedTexFormat& format, int width) : TestTexture (format, width, 1, 1) , m_texture (tcu::getUncompressedFormat(format), width) { allocateLevels(m_texture); TestTexture::populateCompressedLevels(format, getLevelsVector(m_texture)); } TestTexture1D::~TestTexture1D (void) { } int TestTexture1D::getNumLevels (void) const { return m_texture.getNumLevels(); } tcu::PixelBufferAccess TestTexture1D::getLevel (int level, int layer) { DE_ASSERT(layer == 0); DE_UNREF(layer); return m_texture.getLevel(level); } const tcu::ConstPixelBufferAccess TestTexture1D::getLevel (int level, int layer) const { DE_ASSERT(layer == 0); DE_UNREF(layer); return m_texture.getLevel(level); } const tcu::Texture1D& TestTexture1D::getTexture (void) const { return m_texture; } tcu::Texture1D& TestTexture1D::getTexture (void) { return m_texture; } de::MovePtr TestTexture1D::copy(const tcu::TextureFormat format) const { DE_ASSERT(!isCompressed()); de::MovePtr texture (new TestTexture1D(format, m_texture.getWidth())); copyToTexture(*texture); return texture; } // TestTexture1DArray TestTexture1DArray::TestTexture1DArray (const tcu::TextureFormat& format, int width, int arraySize) : TestTexture (format, width, arraySize, 1) , m_texture (format, width, arraySize) { allocateLevels(m_texture); TestTexture::populateLevels(getLevelsVector(m_texture)); } TestTexture1DArray::TestTexture1DArray (const tcu::CompressedTexFormat& format, int width, int arraySize) : TestTexture (format, width, arraySize, 1) , m_texture (tcu::getUncompressedFormat(format), width, arraySize) { allocateLevels(m_texture); std::vector layers; for (int levelNdx = 0; levelNdx < m_texture.getNumLevels(); levelNdx++) for (int layerNdx = 0; layerNdx < m_texture.getNumLayers(); layerNdx++) layers.push_back(getLevel(levelNdx, layerNdx)); TestTexture::populateCompressedLevels(format, layers); } TestTexture1DArray::~TestTexture1DArray (void) { } int TestTexture1DArray::getNumLevels (void) const { return m_texture.getNumLevels(); } tcu::PixelBufferAccess TestTexture1DArray::getLevel (int level, int layer) { const tcu::PixelBufferAccess levelLayers = m_texture.getLevel(level); const deUint32 layerSize = levelLayers.getWidth() * levelLayers.getFormat().getPixelSize(); const deUint32 layerOffset = layerSize * layer; return tcu::PixelBufferAccess(levelLayers.getFormat(), levelLayers.getWidth(), 1, 1, (deUint8*)levelLayers.getDataPtr() + layerOffset); } const tcu::ConstPixelBufferAccess TestTexture1DArray::getLevel (int level, int layer) const { const tcu::ConstPixelBufferAccess levelLayers = m_texture.getLevel(level); const deUint32 layerSize = levelLayers.getWidth() * levelLayers.getFormat().getPixelSize(); const deUint32 layerOffset = layerSize * layer; return tcu::ConstPixelBufferAccess(levelLayers.getFormat(), levelLayers.getWidth(), 1, 1, (deUint8*)levelLayers.getDataPtr() + layerOffset); } const tcu::Texture1DArray& TestTexture1DArray::getTexture (void) const { return m_texture; } tcu::Texture1DArray& TestTexture1DArray::getTexture (void) { return m_texture; } int TestTexture1DArray::getArraySize (void) const { return m_texture.getNumLayers(); } de::MovePtr TestTexture1DArray::copy(const tcu::TextureFormat format) const { DE_ASSERT(!isCompressed()); de::MovePtr texture (new TestTexture1DArray(format, m_texture.getWidth(), getArraySize())); copyToTexture(*texture); return texture; } // TestTexture2D TestTexture2D::TestTexture2D (const tcu::TextureFormat& format, int width, int height) : TestTexture (format, width, height, 1) , m_texture (format, width, height) { allocateLevels(m_texture); TestTexture::populateLevels(getLevelsVector(m_texture)); } TestTexture2D::TestTexture2D (const tcu::TextureFormat& format, int width, int height, int miplevels) : TestTexture(format, width, height, 1) , m_texture(format, width, height, miplevels) { allocateLevels(m_texture); TestTexture::populateLevels(getLevelsVector(m_texture)); } TestTexture2D::TestTexture2D (const tcu::CompressedTexFormat& format, int width, int height) : TestTexture (format, width, height, 1) , m_texture (tcu::getUncompressedFormat(format), width, height) { allocateLevels(m_texture); TestTexture::populateCompressedLevels(format, getLevelsVector(m_texture)); } TestTexture2D::~TestTexture2D (void) { } int TestTexture2D::getNumLevels (void) const { return m_texture.getNumLevels(); } tcu::PixelBufferAccess TestTexture2D::getLevel (int level, int layer) { DE_ASSERT(layer == 0); DE_UNREF(layer); return m_texture.getLevel(level); } const tcu::ConstPixelBufferAccess TestTexture2D::getLevel (int level, int layer) const { DE_ASSERT(layer == 0); DE_UNREF(layer); return m_texture.getLevel(level); } const tcu::Texture2D& TestTexture2D::getTexture (void) const { return m_texture; } tcu::Texture2D& TestTexture2D::getTexture (void) { return m_texture; } de::MovePtr TestTexture2D::copy(const tcu::TextureFormat format) const { DE_ASSERT(!isCompressed()); de::MovePtr texture (new TestTexture2D(format, m_texture.getWidth(), m_texture.getHeight(), m_texture.getNumLevels())); copyToTexture(*texture); return texture; } // TestTexture2DArray TestTexture2DArray::TestTexture2DArray (const tcu::TextureFormat& format, int width, int height, int arraySize) : TestTexture (format, width, height, arraySize) , m_texture (format, width, height, arraySize) { allocateLevels(m_texture); TestTexture::populateLevels(getLevelsVector(m_texture)); } TestTexture2DArray::TestTexture2DArray (const tcu::CompressedTexFormat& format, int width, int height, int arraySize) : TestTexture (format, width, height, arraySize) , m_texture (tcu::getUncompressedFormat(format), width, height, arraySize) { allocateLevels(m_texture); std::vector layers; for (int levelNdx = 0; levelNdx < m_texture.getNumLevels(); levelNdx++) for (int layerNdx = 0; layerNdx < m_texture.getNumLayers(); layerNdx++) layers.push_back(getLevel(levelNdx, layerNdx)); TestTexture::populateCompressedLevels(format, layers); } TestTexture2DArray::~TestTexture2DArray (void) { } int TestTexture2DArray::getNumLevels (void) const { return m_texture.getNumLevels(); } tcu::PixelBufferAccess TestTexture2DArray::getLevel (int level, int layer) { const tcu::PixelBufferAccess levelLayers = m_texture.getLevel(level); const deUint32 layerSize = levelLayers.getWidth() * levelLayers.getHeight() * levelLayers.getFormat().getPixelSize(); const deUint32 layerOffset = layerSize * layer; return tcu::PixelBufferAccess(levelLayers.getFormat(), levelLayers.getWidth(), levelLayers.getHeight(), 1, (deUint8*)levelLayers.getDataPtr() + layerOffset); } const tcu::ConstPixelBufferAccess TestTexture2DArray::getLevel (int level, int layer) const { const tcu::ConstPixelBufferAccess levelLayers = m_texture.getLevel(level); const deUint32 layerSize = levelLayers.getWidth() * levelLayers.getHeight() * levelLayers.getFormat().getPixelSize(); const deUint32 layerOffset = layerSize * layer; return tcu::ConstPixelBufferAccess(levelLayers.getFormat(), levelLayers.getWidth(), levelLayers.getHeight(), 1, (deUint8*)levelLayers.getDataPtr() + layerOffset); } const tcu::Texture2DArray& TestTexture2DArray::getTexture (void) const { return m_texture; } tcu::Texture2DArray& TestTexture2DArray::getTexture (void) { return m_texture; } int TestTexture2DArray::getArraySize (void) const { return m_texture.getNumLayers(); } de::MovePtr TestTexture2DArray::copy(const tcu::TextureFormat format) const { DE_ASSERT(!isCompressed()); de::MovePtr texture (new TestTexture2DArray(format, m_texture.getWidth(), m_texture.getHeight(), getArraySize())); copyToTexture(*texture); return texture; } // TestTexture3D TestTexture3D::TestTexture3D (const tcu::TextureFormat& format, int width, int height, int depth) : TestTexture (format, width, height, depth) , m_texture (format, width, height, depth) { allocateLevels(m_texture); TestTexture::populateLevels(getLevelsVector(m_texture)); } TestTexture3D::TestTexture3D (const tcu::CompressedTexFormat& format, int width, int height, int depth) : TestTexture (format, width, height, depth) , m_texture (tcu::getUncompressedFormat(format), width, height, depth) { allocateLevels(m_texture); TestTexture::populateCompressedLevels(format, getLevelsVector(m_texture)); } TestTexture3D::~TestTexture3D (void) { } int TestTexture3D::getNumLevels (void) const { return m_texture.getNumLevels(); } tcu::PixelBufferAccess TestTexture3D::getLevel (int level, int layer) { DE_ASSERT(layer == 0); DE_UNREF(layer); return m_texture.getLevel(level); } const tcu::ConstPixelBufferAccess TestTexture3D::getLevel (int level, int layer) const { DE_ASSERT(layer == 0); DE_UNREF(layer); return m_texture.getLevel(level); } const tcu::Texture3D& TestTexture3D::getTexture (void) const { return m_texture; } tcu::Texture3D& TestTexture3D::getTexture (void) { return m_texture; } de::MovePtr TestTexture3D::copy(const tcu::TextureFormat format) const { DE_ASSERT(!isCompressed()); de::MovePtr texture (new TestTexture3D(format, m_texture.getWidth(), m_texture.getHeight(), m_texture.getDepth())); copyToTexture(*texture); return texture; } // TestTextureCube const static tcu::CubeFace tcuFaceMapping[tcu::CUBEFACE_LAST] = { tcu::CUBEFACE_POSITIVE_X, tcu::CUBEFACE_NEGATIVE_X, tcu::CUBEFACE_POSITIVE_Y, tcu::CUBEFACE_NEGATIVE_Y, tcu::CUBEFACE_POSITIVE_Z, tcu::CUBEFACE_NEGATIVE_Z }; TestTextureCube::TestTextureCube (const tcu::TextureFormat& format, int size) : TestTexture (format, size, size, 1) , m_texture (format, size) { for (int levelNdx = 0; levelNdx < getNumLevels(); levelNdx++) { for (int faceNdx = 0; faceNdx < tcu::CUBEFACE_LAST; faceNdx++) { m_texture.allocLevel(tcuFaceMapping[faceNdx], levelNdx); TestTexture::fillWithGradient(m_texture.getLevelFace(levelNdx, tcuFaceMapping[faceNdx])); } } } TestTextureCube::TestTextureCube (const tcu::CompressedTexFormat& format, int size) : TestTexture (format, size, size, 1) , m_texture (tcu::getUncompressedFormat(format), size) { std::vector levels(m_texture.getNumLevels() * tcu::CUBEFACE_LAST); for (int levelNdx = 0; levelNdx < getNumLevels(); levelNdx++) { for (int faceNdx = 0; faceNdx < tcu::CUBEFACE_LAST; faceNdx++) { m_texture.allocLevel(tcuFaceMapping[faceNdx], levelNdx); levels[levelNdx * tcu::CUBEFACE_LAST + faceNdx] = m_texture.getLevelFace(levelNdx, tcuFaceMapping[faceNdx]); } } TestTexture::populateCompressedLevels(format, levels); } TestTextureCube::~TestTextureCube (void) { } int TestTextureCube::getNumLevels (void) const { return m_texture.getNumLevels(); } tcu::PixelBufferAccess TestTextureCube::getLevel (int level, int layer) { return m_texture.getLevelFace(level, tcuFaceMapping[layer]); } const tcu::ConstPixelBufferAccess TestTextureCube::getLevel (int level, int layer) const { return m_texture.getLevelFace(level, tcuFaceMapping[layer]); } int TestTextureCube::getArraySize (void) const { return (int)tcu::CUBEFACE_LAST; } const tcu::TextureCube& TestTextureCube::getTexture (void) const { return m_texture; } tcu::TextureCube& TestTextureCube::getTexture (void) { return m_texture; } de::MovePtr TestTextureCube::copy(const tcu::TextureFormat format) const { DE_ASSERT(!isCompressed()); de::MovePtr texture (new TestTextureCube(format, m_texture.getSize())); copyToTexture(*texture); return texture; } // TestTextureCubeArray TestTextureCubeArray::TestTextureCubeArray (const tcu::TextureFormat& format, int size, int arraySize) : TestTexture (format, size, size, arraySize) , m_texture (format, size, arraySize) { allocateLevels(m_texture); TestTexture::populateLevels(getLevelsVector(m_texture)); } TestTextureCubeArray::TestTextureCubeArray (const tcu::CompressedTexFormat& format, int size, int arraySize) : TestTexture (format, size, size, arraySize) , m_texture (tcu::getUncompressedFormat(format), size, arraySize) { DE_ASSERT(arraySize % 6 == 0); allocateLevels(m_texture); std::vector layers; for (int levelNdx = 0; levelNdx < m_texture.getNumLevels(); levelNdx++) for (int layerNdx = 0; layerNdx < m_texture.getDepth(); layerNdx++) layers.push_back(getLevel(levelNdx, layerNdx)); TestTexture::populateCompressedLevels(format, layers); } TestTextureCubeArray::~TestTextureCubeArray (void) { } int TestTextureCubeArray::getNumLevels (void) const { return m_texture.getNumLevels(); } tcu::PixelBufferAccess TestTextureCubeArray::getLevel (int level, int layer) { const tcu::PixelBufferAccess levelLayers = m_texture.getLevel(level); const deUint32 layerSize = levelLayers.getWidth() * levelLayers.getHeight() * levelLayers.getFormat().getPixelSize(); const deUint32 layerOffset = layerSize * layer; return tcu::PixelBufferAccess(levelLayers.getFormat(), levelLayers.getWidth(), levelLayers.getHeight(), 1, (deUint8*)levelLayers.getDataPtr() + layerOffset); } const tcu::ConstPixelBufferAccess TestTextureCubeArray::getLevel (int level, int layer) const { const tcu::ConstPixelBufferAccess levelLayers = m_texture.getLevel(level); const deUint32 layerSize = levelLayers.getWidth() * levelLayers.getHeight() * levelLayers.getFormat().getPixelSize(); const deUint32 layerOffset = layerSize * layer; return tcu::ConstPixelBufferAccess(levelLayers.getFormat(), levelLayers.getWidth(), levelLayers.getHeight(), 1, (deUint8*)levelLayers.getDataPtr() + layerOffset); } int TestTextureCubeArray::getArraySize (void) const { return m_texture.getDepth(); } const tcu::TextureCubeArray& TestTextureCubeArray::getTexture (void) const { return m_texture; } tcu::TextureCubeArray& TestTextureCubeArray::getTexture (void) { return m_texture; } de::MovePtr TestTextureCubeArray::copy(const tcu::TextureFormat format) const { DE_ASSERT(!isCompressed()); de::MovePtr texture (new TestTextureCubeArray(format, m_texture.getSize(), getArraySize())); copyToTexture(*texture); return texture; } } // pipeline } // vkt