119 lines
3.5 KiB
C++
119 lines
3.5 KiB
C++
// 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 <Nazara/VulkanRenderer/Wrapper/DeviceMemory.hpp>
|
|
#include <Nazara/VulkanRenderer/Utils.hpp>
|
|
#include <Nazara/VulkanRenderer/Vulkan.hpp>
|
|
#include <Nazara/VulkanRenderer/Wrapper/PhysicalDevice.hpp>
|
|
#include <Nazara/VulkanRenderer/Debug.hpp>
|
|
|
|
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 <Nazara/VulkanRenderer/DebugOff.hpp>
|