// Copyright (C) 2022 Jérôme "Lynix" Leclercq (lynix680@gmail.com) // This file is part of the "Nazara Engine - Vulkan renderer" // For conditions of distribution and use, see copyright notice in Config.hpp #include #include #include #include #include namespace Nz { namespace Vk { inline bool DeviceMemory::Create(Device& device, VkDeviceSize size, UInt32 memoryType, const VkAllocationCallbacks* allocator) { VkMemoryAllocateInfo allocInfo = { VK_STRUCTURE_TYPE_MEMORY_ALLOCATE_INFO, // VkStructureType sType; nullptr, // const void* pNext; size, // VkDeviceSize allocationSize; memoryType // uint32_t memoryTypeIndex; }; return Create(device, allocInfo, allocator); } inline bool DeviceMemory::Create(Device& device, VkDeviceSize size, UInt32 typeBits, VkFlags properties, const VkAllocationCallbacks* allocator) { const Vk::PhysicalDevice& deviceInfo = Vulkan::GetPhysicalDeviceInfo(device.GetPhysicalDevice()); UInt32 typeMask = 1; for (UInt32 i = 0; i < VK_MAX_MEMORY_TYPES; ++i) { if (typeBits & typeMask) { if ((deviceInfo.memoryProperties.memoryTypes[i].propertyFlags & properties) == properties) return Create(device, size, i, allocator); } typeMask <<= 1; } NazaraError("Failed to find a memory type suitable for typeBits: " + NumberToString(typeBits) + " and properties: 0x" + NumberToString(properties, 16)); return false; } inline bool DeviceMemory::FlushMemory() { return FlushMemory(0, VK_WHOLE_SIZE); } inline bool DeviceMemory::FlushMemory(UInt64 offset, UInt64 size) { VkMappedMemoryRange memoryRange = { VK_STRUCTURE_TYPE_MAPPED_MEMORY_RANGE, nullptr, m_handle, offset, size }; m_lastErrorCode = m_device->vkFlushMappedMemoryRanges(*m_device, 1, &memoryRange); if (m_lastErrorCode != VK_SUCCESS) { NazaraError("Failed to flush memory: " + TranslateVulkanError(m_lastErrorCode)); return false; } return true; } inline void* DeviceMemory::GetMappedPointer() { return m_mappedPtr; } inline bool DeviceMemory::Map(VkMemoryMapFlags flags) { return Map(0, VK_WHOLE_SIZE, flags); } inline bool DeviceMemory::Map(VkDeviceSize offset, VkDeviceSize size, VkMemoryMapFlags flags) { void* mappedPtr; m_lastErrorCode = m_device->vkMapMemory(*m_device, m_handle, offset, size, flags, &mappedPtr); if (m_lastErrorCode != VK_SUCCESS) { NazaraError("Failed to map device memory: " + TranslateVulkanError(m_lastErrorCode)); return false; } m_mappedPtr = mappedPtr; return true; } inline void DeviceMemory::Unmap() { NazaraAssert(m_mappedPtr != nullptr, "Memory is not mapped"); m_device->vkUnmapMemory(*m_device, m_handle); m_mappedPtr = nullptr; } inline VkResult DeviceMemory::CreateHelper(Device& device, const VkMemoryAllocateInfo* allocInfo, const VkAllocationCallbacks* allocator, VkDeviceMemory* handle) { return device.vkAllocateMemory(device, allocInfo, allocator, handle); } inline void DeviceMemory::DestroyHelper(Device& device, VkDeviceMemory handle, const VkAllocationCallbacks* allocator) { return device.vkFreeMemory(device, handle, allocator); } } } #include