/*------------------------------------------------------------------------- * Vulkan CTS Framework * -------------------- * * Copyright (c) 2019 Google Inc. * Copyright (c) 2019 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 Memory management utilities. *//*--------------------------------------------------------------------*/ #include "vkMemUtil.hpp" #include "deDefs.h" #include "vkStrUtil.hpp" #include "vkQueryUtil.hpp" #include "vkRef.hpp" #include "vkRefUtil.hpp" #include "vkImageUtil.hpp" #include "deInt32.h" #include namespace vk { using de::UniquePtr; using de::MovePtr; using std::vector; typedef de::SharedPtr AllocationSp; namespace { class HostPtr { public: HostPtr (const DeviceInterface& vkd, VkDevice device, VkDeviceMemory memory, VkDeviceSize offset, VkDeviceSize size, VkMemoryMapFlags flags); ~HostPtr (void); void* get (void) const { return m_ptr; } private: const DeviceInterface& m_vkd; const VkDevice m_device; const VkDeviceMemory m_memory; void* const m_ptr; }; HostPtr::HostPtr (const DeviceInterface& vkd, VkDevice device, VkDeviceMemory memory, VkDeviceSize offset, VkDeviceSize size, VkMemoryMapFlags flags) : m_vkd (vkd) , m_device (device) , m_memory (memory) , m_ptr (mapMemory(vkd, device, memory, offset, size, flags)) { } HostPtr::~HostPtr (void) { m_vkd.unmapMemory(m_device, m_memory); } bool isHostVisibleMemory (const VkPhysicalDeviceMemoryProperties& deviceMemProps, deUint32 memoryTypeNdx) { DE_ASSERT(memoryTypeNdx < deviceMemProps.memoryTypeCount); return (deviceMemProps.memoryTypes[memoryTypeNdx].propertyFlags & VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT) != 0u; } } // anonymous // Allocation Allocation::Allocation (VkDeviceMemory memory, VkDeviceSize offset, void* hostPtr) : m_memory (memory) , m_offset (offset) , m_hostPtr (hostPtr) { } Allocation::~Allocation (void) { } void flushAlloc (const DeviceInterface& vkd, VkDevice device, const Allocation& alloc) { flushMappedMemoryRange(vkd, device, alloc.getMemory(), alloc.getOffset(), VK_WHOLE_SIZE); } void invalidateAlloc (const DeviceInterface& vkd, VkDevice device, const Allocation& alloc) { invalidateMappedMemoryRange(vkd, device, alloc.getMemory(), alloc.getOffset(), VK_WHOLE_SIZE); } // MemoryRequirement const MemoryRequirement MemoryRequirement::Any = MemoryRequirement(0x0u); const MemoryRequirement MemoryRequirement::HostVisible = MemoryRequirement(MemoryRequirement::FLAG_HOST_VISIBLE); const MemoryRequirement MemoryRequirement::Coherent = MemoryRequirement(MemoryRequirement::FLAG_COHERENT); const MemoryRequirement MemoryRequirement::LazilyAllocated = MemoryRequirement(MemoryRequirement::FLAG_LAZY_ALLOCATION); const MemoryRequirement MemoryRequirement::Protected = MemoryRequirement(MemoryRequirement::FLAG_PROTECTED); const MemoryRequirement MemoryRequirement::Local = MemoryRequirement(MemoryRequirement::FLAG_LOCAL); const MemoryRequirement MemoryRequirement::Cached = MemoryRequirement(MemoryRequirement::FLAG_CACHED); const MemoryRequirement MemoryRequirement::NonLocal = MemoryRequirement(MemoryRequirement::FLAG_NON_LOCAL); const MemoryRequirement MemoryRequirement::DeviceAddress = MemoryRequirement(MemoryRequirement::FLAG_DEVICE_ADDRESS); const MemoryRequirement MemoryRequirement::DeviceAddressCaptureReplay = MemoryRequirement(MemoryRequirement::FLAG_DEVICE_ADDRESS_CAPTURE_REPLAY); bool MemoryRequirement::matchesHeap (VkMemoryPropertyFlags heapFlags) const { // sanity check if ((m_flags & FLAG_COHERENT) && !(m_flags & FLAG_HOST_VISIBLE)) DE_FATAL("Coherent memory must be host-visible"); if ((m_flags & FLAG_HOST_VISIBLE) && (m_flags & FLAG_LAZY_ALLOCATION)) DE_FATAL("Lazily allocated memory cannot be mappable"); if ((m_flags & FLAG_PROTECTED) && (m_flags & FLAG_HOST_VISIBLE)) DE_FATAL("Protected memory cannot be mappable"); // host-visible if ((m_flags & FLAG_HOST_VISIBLE) && !(heapFlags & VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT)) return false; // coherent if ((m_flags & FLAG_COHERENT) && !(heapFlags & VK_MEMORY_PROPERTY_HOST_COHERENT_BIT)) return false; // lazy if ((m_flags & FLAG_LAZY_ALLOCATION) && !(heapFlags & VK_MEMORY_PROPERTY_LAZILY_ALLOCATED_BIT)) return false; // protected if ((m_flags & FLAG_PROTECTED) && !(heapFlags & VK_MEMORY_PROPERTY_PROTECTED_BIT)) return false; // local if ((m_flags & FLAG_LOCAL) && !(heapFlags & VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT)) return false; // cached if ((m_flags & FLAG_CACHED) && !(heapFlags & VK_MEMORY_PROPERTY_HOST_CACHED_BIT)) return false; // non-local if ((m_flags & FLAG_NON_LOCAL) && (heapFlags & VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT)) return false; return true; } MemoryRequirement::MemoryRequirement (deUint32 flags) : m_flags(flags) { } // SimpleAllocator class SimpleAllocation : public Allocation { public: SimpleAllocation (Move mem, MovePtr hostPtr, size_t offset); virtual ~SimpleAllocation (void); private: const Unique m_memHolder; const UniquePtr m_hostPtr; }; SimpleAllocation::SimpleAllocation (Move mem, MovePtr hostPtr, size_t offset) : Allocation (*mem, offset, hostPtr ? hostPtr->get() : DE_NULL) , m_memHolder (mem) , m_hostPtr (hostPtr) { } SimpleAllocation::~SimpleAllocation (void) { } SimpleAllocator::SimpleAllocator (const DeviceInterface& vk, VkDevice device, const VkPhysicalDeviceMemoryProperties& deviceMemProps, size_t offset) : m_vk (vk) , m_device (device) , m_memProps(deviceMemProps) , m_offset (offset) { } MovePtr SimpleAllocator::allocate (const VkMemoryAllocateInfo& allocInfo, VkDeviceSize alignment) { // Align the offset to the requirements. size_t offset = deAlignSize(m_offset, static_cast(alignment)); VkMemoryAllocateInfo info = allocInfo; info.allocationSize += offset; Move mem = allocateMemory(m_vk, m_device, &info); MovePtr hostPtr; if (isHostVisibleMemory(m_memProps, info.memoryTypeIndex)) hostPtr = MovePtr(new HostPtr(m_vk, m_device, *mem, offset, info.allocationSize, 0u)); return MovePtr(new SimpleAllocation(mem, hostPtr, offset)); } MovePtr SimpleAllocator::allocate (const VkMemoryRequirements& memReqs, MemoryRequirement requirement) { const deUint32 memoryTypeNdx = selectMatchingMemoryType(m_memProps, memReqs.memoryTypeBits, requirement); // Align the offset to the requirements. size_t offset = deAlignSize(m_offset, static_cast(memReqs.alignment)); VkMemoryAllocateInfo allocInfo = { VK_STRUCTURE_TYPE_MEMORY_ALLOCATE_INFO, // VkStructureType sType; DE_NULL, // const void* pNext; memReqs.size + offset, // VkDeviceSize allocationSize; memoryTypeNdx, // deUint32 memoryTypeIndex; }; VkMemoryAllocateFlagsInfo allocFlagsInfo = { VK_STRUCTURE_TYPE_MEMORY_ALLOCATE_FLAGS_INFO, // VkStructureType sType DE_NULL, // const void* pNext 0, // VkMemoryAllocateFlags flags 0, // uint32_t deviceMask }; if (requirement & MemoryRequirement::DeviceAddress) allocFlagsInfo.flags |= VK_MEMORY_ALLOCATE_DEVICE_ADDRESS_BIT; if (requirement & MemoryRequirement::DeviceAddressCaptureReplay) allocFlagsInfo.flags |= VK_MEMORY_ALLOCATE_DEVICE_ADDRESS_CAPTURE_REPLAY_BIT; if (allocFlagsInfo.flags) allocInfo.pNext = &allocFlagsInfo; Move mem = allocateMemory(m_vk, m_device, &allocInfo); MovePtr hostPtr; if (requirement & MemoryRequirement::HostVisible) { DE_ASSERT(isHostVisibleMemory(m_memProps, allocInfo.memoryTypeIndex)); hostPtr = MovePtr(new HostPtr(m_vk, m_device, *mem, offset, memReqs.size, 0u)); } return MovePtr(new SimpleAllocation(mem, hostPtr, offset)); } MovePtr allocateExtended (const InstanceInterface& vki, const DeviceInterface& vkd, const VkPhysicalDevice& physDevice, const VkDevice device, const VkMemoryRequirements& memReqs, const MemoryRequirement requirement, const void* pNext) { const VkPhysicalDeviceMemoryProperties memoryProperties = getPhysicalDeviceMemoryProperties(vki, physDevice); const deUint32 memoryTypeNdx = selectMatchingMemoryType(memoryProperties, memReqs.memoryTypeBits, requirement); const VkMemoryAllocateInfo allocInfo = { VK_STRUCTURE_TYPE_MEMORY_ALLOCATE_INFO, // VkStructureType sType pNext, // const void* pNext memReqs.size, // VkDeviceSize allocationSize memoryTypeNdx, // deUint32 memoryTypeIndex }; Move mem = allocateMemory(vkd, device, &allocInfo); MovePtr hostPtr; if (requirement & MemoryRequirement::HostVisible) { DE_ASSERT(isHostVisibleMemory(memoryProperties, allocInfo.memoryTypeIndex)); hostPtr = MovePtr(new HostPtr(vkd, device, *mem, 0u, allocInfo.allocationSize, 0u)); } return MovePtr(new SimpleAllocation(mem, hostPtr, 0u)); } de::MovePtr allocateDedicated (const InstanceInterface& vki, const DeviceInterface& vkd, const VkPhysicalDevice& physDevice, const VkDevice device, const VkBuffer buffer, MemoryRequirement requirement) { const VkMemoryRequirements memoryRequirements = getBufferMemoryRequirements(vkd, device, buffer); const VkMemoryDedicatedAllocateInfo dedicatedAllocationInfo = { VK_STRUCTURE_TYPE_MEMORY_DEDICATED_ALLOCATE_INFO, // VkStructureType sType DE_NULL, // const void* pNext DE_NULL, // VkImage image buffer // VkBuffer buffer }; return allocateExtended(vki, vkd, physDevice, device, memoryRequirements, requirement, &dedicatedAllocationInfo); } de::MovePtr allocateDedicated (const InstanceInterface& vki, const DeviceInterface& vkd, const VkPhysicalDevice& physDevice, const VkDevice device, const VkImage image, MemoryRequirement requirement) { const VkMemoryRequirements memoryRequirements = getImageMemoryRequirements(vkd, device, image); const VkMemoryDedicatedAllocateInfo dedicatedAllocationInfo = { VK_STRUCTURE_TYPE_MEMORY_DEDICATED_ALLOCATE_INFO, // VkStructureType sType DE_NULL, // const void* pNext image, // VkImage image DE_NULL // VkBuffer buffer }; return allocateExtended(vki, vkd, physDevice, device, memoryRequirements, requirement, &dedicatedAllocationInfo); } void* mapMemory (const DeviceInterface& vkd, VkDevice device, VkDeviceMemory mem, VkDeviceSize offset, VkDeviceSize size, VkMemoryMapFlags flags) { void* hostPtr = DE_NULL; VK_CHECK(vkd.mapMemory(device, mem, offset, size, flags, &hostPtr)); TCU_CHECK(hostPtr); return hostPtr; } void flushMappedMemoryRange (const DeviceInterface& vkd, VkDevice device, VkDeviceMemory memory, VkDeviceSize offset, VkDeviceSize size) { const VkMappedMemoryRange range = { VK_STRUCTURE_TYPE_MAPPED_MEMORY_RANGE, DE_NULL, memory, offset, size }; VK_CHECK(vkd.flushMappedMemoryRanges(device, 1u, &range)); } void invalidateMappedMemoryRange (const DeviceInterface& vkd, VkDevice device, VkDeviceMemory memory, VkDeviceSize offset, VkDeviceSize size) { const VkMappedMemoryRange range = { VK_STRUCTURE_TYPE_MAPPED_MEMORY_RANGE, DE_NULL, memory, offset, size }; VK_CHECK(vkd.invalidateMappedMemoryRanges(device, 1u, &range)); } deUint32 selectMatchingMemoryType (const VkPhysicalDeviceMemoryProperties& deviceMemProps, deUint32 allowedMemTypeBits, MemoryRequirement requirement) { const deUint32 compatibleTypes = getCompatibleMemoryTypes(deviceMemProps, requirement); deUint32 candidates = allowedMemTypeBits & compatibleTypes; #ifdef CTS_USES_VULKANSC // in case of Vulkan SC: prefer memory types from SEU-safe heaps ( SEU = single event upsets ) const deUint32 seuSafeTypes = getSEUSafeMemoryTypes(deviceMemProps); deUint32 seuSafeCandidates = candidates & seuSafeTypes; if (seuSafeCandidates != 0u) candidates = seuSafeCandidates; #endif // CTS_USES_VULKANSC if (candidates == 0u) TCU_THROW(NotSupportedError, "No compatible memory type found"); return (deUint32)deCtz32(candidates); } deUint32 getCompatibleMemoryTypes (const VkPhysicalDeviceMemoryProperties& deviceMemProps, MemoryRequirement requirement) { deUint32 compatibleTypes = 0u; for (deUint32 memoryTypeNdx = 0; memoryTypeNdx < deviceMemProps.memoryTypeCount; memoryTypeNdx++) { if (requirement.matchesHeap(deviceMemProps.memoryTypes[memoryTypeNdx].propertyFlags)) compatibleTypes |= (1u << memoryTypeNdx); } return compatibleTypes; } #ifdef CTS_USES_VULKANSC deUint32 getSEUSafeMemoryTypes (const VkPhysicalDeviceMemoryProperties& deviceMemProps) { deUint32 seuSafeTypes = 0u; for (deUint32 memoryTypeNdx = 0; memoryTypeNdx < deviceMemProps.memoryTypeCount; memoryTypeNdx++) { if( ( deviceMemProps.memoryHeaps[deviceMemProps.memoryTypes[memoryTypeNdx].heapIndex].flags & VK_MEMORY_HEAP_SEU_SAFE_BIT ) != 0u ) seuSafeTypes |= (1u << memoryTypeNdx); } return seuSafeTypes; } #endif // CTS_USES_VULKANSC void bindImagePlanesMemory (const DeviceInterface& vkd, const VkDevice device, const VkImage image, const deUint32 numPlanes, vector& allocations, vk::Allocator& allocator, const vk::MemoryRequirement requirement) { vector coreInfos; vector planeInfos; coreInfos.reserve(numPlanes); planeInfos.reserve(numPlanes); for (deUint32 planeNdx = 0; planeNdx < numPlanes; ++planeNdx) { const VkImageAspectFlagBits planeAspect = getPlaneAspect(planeNdx); const VkMemoryRequirements reqs = getImagePlaneMemoryRequirements(vkd, device, image, planeAspect); allocations.push_back(AllocationSp(allocator.allocate(reqs, requirement).release())); VkBindImagePlaneMemoryInfo planeInfo = { VK_STRUCTURE_TYPE_BIND_IMAGE_PLANE_MEMORY_INFO, DE_NULL, planeAspect }; planeInfos.push_back(planeInfo); VkBindImageMemoryInfo coreInfo = { VK_STRUCTURE_TYPE_BIND_IMAGE_MEMORY_INFO, &planeInfos.back(), image, allocations.back()->getMemory(), allocations.back()->getOffset(), }; coreInfos.push_back(coreInfo); } VK_CHECK(vkd.bindImageMemory2(device, numPlanes, coreInfos.data())); } MovePtr bindImage (const DeviceInterface& vk, const VkDevice device, Allocator& allocator, const VkImage image, const MemoryRequirement requirement) { MovePtr alloc = allocator.allocate(getImageMemoryRequirements(vk, device, image), requirement); VK_CHECK(vk.bindImageMemory(device, image, alloc->getMemory(), alloc->getOffset())); return alloc; } MovePtr bindBuffer (const DeviceInterface& vk, const VkDevice device, Allocator& allocator, const VkBuffer buffer, const MemoryRequirement requirement) { MovePtr alloc(allocator.allocate(getBufferMemoryRequirements(vk, device, buffer), requirement)); VK_CHECK(vk.bindBufferMemory(device, buffer, alloc->getMemory(), alloc->getOffset())); return alloc; } void zeroBuffer (const DeviceInterface& vk, const VkDevice device, const Allocation& alloc, const VkDeviceSize size) { deMemset(alloc.getHostPtr(), 0, static_cast(size)); flushAlloc(vk, device, alloc); } } // vk