adding manual buffer mapping

This commit is contained in:
2024-12-08 16:43:51 +01:00
parent 52850323fd
commit 848844059c
8 changed files with 76 additions and 21 deletions

View File

@@ -273,7 +273,8 @@ PULSE_API bool PulseDeviceSupportsShaderFormats(PulseDevice device, PulseShaderF
PULSE_API void PulseDestroyDevice(PulseDevice device); PULSE_API void PulseDestroyDevice(PulseDevice device);
PULSE_API PulseBuffer PulseCreateBuffer(PulseDevice device, const PulseBufferCreateInfo* create_infos); PULSE_API PulseBuffer PulseCreateBuffer(PulseDevice device, const PulseBufferCreateInfo* create_infos);
PULSE_API bool PulseGetBufferMap(PulseBuffer buffer, void** data); PULSE_API bool PulseMapBuffer(PulseBuffer buffer, void** data);
PULSE_API void PulseUnmapBuffer(PulseBuffer buffer);
PULSE_API void PulseDestroyBuffer(PulseDevice device, PulseBuffer buffer); PULSE_API void PulseDestroyBuffer(PulseDevice device, PulseBuffer buffer);
PULSE_API PulseImage PulseCreateImage(PulseDevice device, const PulseImageCreateInfo* create_infos); PULSE_API PulseImage PulseCreateImage(PulseDevice device, const PulseImageCreateInfo* create_infos);

View File

@@ -25,23 +25,23 @@ PulseBuffer VulkanCreateBuffer(PulseDevice device, const PulseBufferCreateInfo*
VmaAllocationCreateInfo allocation_create_info = { 0 }; VmaAllocationCreateInfo allocation_create_info = { 0 };
allocation_create_info.usage = VMA_MEMORY_USAGE_AUTO; allocation_create_info.usage = VMA_MEMORY_USAGE_AUTO;
if(buffer->usage & PULSE_BUFFER_USAGE_STORAGE_READ || buffer->usage & PULSE_BUFFER_USAGE_STORAGE_WRITE)
vulkan_buffer->usage |= VK_BUFFER_USAGE_STORAGE_BUFFER_BIT;
if(buffer->usage & PULSE_BUFFER_USAGE_TRANSFER_UPLOAD) if(buffer->usage & PULSE_BUFFER_USAGE_TRANSFER_UPLOAD)
{ {
vulkan_buffer->usage |= VK_BUFFER_USAGE_TRANSFER_SRC_BIT; vulkan_buffer->usage |= VK_BUFFER_USAGE_TRANSFER_SRC_BIT;
allocation_create_info.flags = VMA_ALLOCATION_CREATE_HOST_ACCESS_SEQUENTIAL_WRITE_BIT | VMA_ALLOCATION_CREATE_MAPPED_BIT; allocation_create_info.flags = VMA_ALLOCATION_CREATE_HOST_ACCESS_SEQUENTIAL_WRITE_BIT;
} }
if(buffer->usage & PULSE_BUFFER_USAGE_TRANSFER_DOWNLOAD) if(buffer->usage & PULSE_BUFFER_USAGE_TRANSFER_DOWNLOAD)
{ {
vulkan_buffer->usage |= VK_BUFFER_USAGE_TRANSFER_DST_BIT; vulkan_buffer->usage |= VK_BUFFER_USAGE_TRANSFER_DST_BIT;
allocation_create_info.flags = VMA_ALLOCATION_CREATE_HOST_ACCESS_SEQUENTIAL_WRITE_BIT | VMA_ALLOCATION_CREATE_MAPPED_BIT; allocation_create_info.flags = VMA_ALLOCATION_CREATE_HOST_ACCESS_SEQUENTIAL_WRITE_BIT;
} }
if(buffer->usage & PULSE_BUFFER_USAGE_UNIFORM_ACCESS) if(buffer->usage & PULSE_BUFFER_USAGE_UNIFORM_ACCESS)
{ {
vulkan_buffer->usage |= VK_BUFFER_USAGE_UNIFORM_BUFFER_BIT; vulkan_buffer->usage |= VK_BUFFER_USAGE_UNIFORM_BUFFER_BIT;
allocation_create_info.flags = VMA_ALLOCATION_CREATE_HOST_ACCESS_SEQUENTIAL_WRITE_BIT | VMA_ALLOCATION_CREATE_MAPPED_BIT; allocation_create_info.flags = VMA_ALLOCATION_CREATE_HOST_ACCESS_SEQUENTIAL_WRITE_BIT;
} }
if(buffer->usage & PULSE_BUFFER_USAGE_STORAGE_READ || buffer->usage & PULSE_BUFFER_USAGE_STORAGE_WRITE)
vulkan_buffer->usage |= VK_BUFFER_USAGE_STORAGE_BUFFER_BIT;
VkBufferCreateInfo buffer_create_info = { 0 }; VkBufferCreateInfo buffer_create_info = { 0 };
buffer_create_info.sType = VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO; buffer_create_info.sType = VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO;
@@ -55,20 +55,21 @@ PulseBuffer VulkanCreateBuffer(PulseDevice device, const PulseBufferCreateInfo*
return buffer; return buffer;
} }
bool VulkanGetBufferMap(PulseBuffer buffer, void** data) bool VulkanMapBuffer(PulseBuffer buffer, void** data)
{ {
VulkanBuffer* vulkan_buffer = VULKAN_RETRIEVE_DRIVER_DATA_AS(buffer, VulkanBuffer*); VulkanBuffer* vulkan_buffer = VULKAN_RETRIEVE_DRIVER_DATA_AS(buffer, VulkanBuffer*);
if(vulkan_buffer->allocation_info.pMappedData == PULSE_NULLPTR) VulkanDevice* vulkan_device = VULKAN_RETRIEVE_DRIVER_DATA_AS(buffer->device, VulkanDevice*);
{ CHECK_VK_RETVAL(buffer->device->backend, vmaMapMemory(vulkan_device->allocator, vulkan_buffer->allocation, data), PULSE_ERROR_MAP_FAILED, false);
if(PULSE_IS_BACKEND_LOW_LEVEL_DEBUG(buffer->device->backend))
PulseLogErrorFmt(buffer->device->backend, "(Vulkan) cannot map a buffer (%p); Buffer has to be created with PULSE_BUFFER_USAGE_HOST_ACCESS flag", buffer);
PulseSetInternalError(PULSE_ERROR_MAP_FAILED);
return false;
}
*data = vulkan_buffer->allocation_info.pMappedData;
return true; return true;
} }
void VulkanUnmapBuffer(PulseBuffer buffer)
{
VulkanBuffer* vulkan_buffer = VULKAN_RETRIEVE_DRIVER_DATA_AS(buffer, VulkanBuffer*);
VulkanDevice* vulkan_device = VULKAN_RETRIEVE_DRIVER_DATA_AS(buffer->device, VulkanDevice*);
vmaUnmapMemory(vulkan_device->allocator, vulkan_buffer->allocation);
}
void VulkanDestroyBuffer(PulseDevice device, PulseBuffer buffer) void VulkanDestroyBuffer(PulseDevice device, PulseBuffer buffer)
{ {
VulkanDevice* vulkan_device = VULKAN_RETRIEVE_DRIVER_DATA_AS(device, VulkanDevice*); VulkanDevice* vulkan_device = VULKAN_RETRIEVE_DRIVER_DATA_AS(device, VulkanDevice*);

View File

@@ -23,7 +23,8 @@ typedef struct VulkanBuffer
} VulkanBuffer; } VulkanBuffer;
PulseBuffer VulkanCreateBuffer(PulseDevice device, const PulseBufferCreateInfo* create_infos); PulseBuffer VulkanCreateBuffer(PulseDevice device, const PulseBufferCreateInfo* create_infos);
bool VulkanGetBufferMap(PulseBuffer buffer, void** data); bool VulkanMapBuffer(PulseBuffer buffer, void** data);
void VulkanUnmapBuffer(PulseBuffer buffer);
void VulkanDestroyBuffer(PulseDevice device, PulseBuffer buffer); void VulkanDestroyBuffer(PulseDevice device, PulseBuffer buffer);
#endif // PULSE_VULKAN_BUFFER_H_ #endif // PULSE_VULKAN_BUFFER_H_

View File

@@ -21,11 +21,21 @@ PULSE_API PulseBuffer PulseCreateBuffer(PulseDevice device, const PulseBufferCre
return device->PFN_CreateBuffer(device, create_infos); return device->PFN_CreateBuffer(device, create_infos);
} }
PULSE_API bool PulseGetBufferMap(PulseBuffer buffer, void** data) PULSE_API bool PulseMapBuffer(PulseBuffer buffer, void** data)
{ {
PULSE_CHECK_HANDLE_RETVAL(buffer, false); PULSE_CHECK_HANDLE_RETVAL(buffer, false);
PULSE_CHECK_PTR_RETVAL(data, false); PULSE_CHECK_PTR_RETVAL(data, false);
return buffer->device->PFN_GetBufferMap(buffer, data); bool res = buffer->device->PFN_MapBuffer(buffer, data);
if(res)
buffer->is_mapped = true;
return res;
}
PULSE_API void PulseUnmapBuffer(PulseBuffer buffer)
{
PULSE_CHECK_HANDLE(buffer);
buffer->device->PFN_UnmapBuffer(buffer);
buffer->is_mapped = false;
} }
PULSE_API void PulseDestroyBuffer(PulseDevice device, PulseBuffer buffer) PULSE_API void PulseDestroyBuffer(PulseDevice device, PulseBuffer buffer)

View File

@@ -90,7 +90,8 @@
PULSE_LOAD_DRIVER_DEVICE_FUNCTION(SubmitCommandList, _namespace) \ PULSE_LOAD_DRIVER_DEVICE_FUNCTION(SubmitCommandList, _namespace) \
PULSE_LOAD_DRIVER_DEVICE_FUNCTION(ReleaseCommandList, _namespace) \ PULSE_LOAD_DRIVER_DEVICE_FUNCTION(ReleaseCommandList, _namespace) \
PULSE_LOAD_DRIVER_DEVICE_FUNCTION(CreateBuffer, _namespace) \ PULSE_LOAD_DRIVER_DEVICE_FUNCTION(CreateBuffer, _namespace) \
PULSE_LOAD_DRIVER_DEVICE_FUNCTION(GetBufferMap, _namespace) \ PULSE_LOAD_DRIVER_DEVICE_FUNCTION(MapBuffer, _namespace) \
PULSE_LOAD_DRIVER_DEVICE_FUNCTION(UnmapBuffer, _namespace) \
PULSE_LOAD_DRIVER_DEVICE_FUNCTION(DestroyBuffer, _namespace) \ PULSE_LOAD_DRIVER_DEVICE_FUNCTION(DestroyBuffer, _namespace) \
PULSE_LOAD_DRIVER_DEVICE_FUNCTION(CreateImage, _namespace) \ PULSE_LOAD_DRIVER_DEVICE_FUNCTION(CreateImage, _namespace) \
PULSE_LOAD_DRIVER_DEVICE_FUNCTION(DestroyImage, _namespace) \ PULSE_LOAD_DRIVER_DEVICE_FUNCTION(DestroyImage, _namespace) \

View File

@@ -34,6 +34,7 @@ typedef struct PulseBufferHandler
void* driver_data; void* driver_data;
PulseBufferUsageFlags usage; PulseBufferUsageFlags usage;
PulseDeviceSize size; PulseDeviceSize size;
bool is_mapped;
} PulseBufferHandler; } PulseBufferHandler;
typedef struct PulseCommandListHandler typedef struct PulseCommandListHandler
@@ -70,7 +71,8 @@ typedef struct PulseDeviceHandler
PulseSubmitCommandListPFN PFN_SubmitCommandList; PulseSubmitCommandListPFN PFN_SubmitCommandList;
PulseReleaseCommandListPFN PFN_ReleaseCommandList; PulseReleaseCommandListPFN PFN_ReleaseCommandList;
PulseCreateBufferPFN PFN_CreateBuffer; PulseCreateBufferPFN PFN_CreateBuffer;
PulseGetBufferMapPFN PFN_GetBufferMap; PulseMapBufferPFN PFN_MapBuffer;
PulseUnmapBufferPFN PFN_UnmapBuffer;
PulseDestroyBufferPFN PFN_DestroyBuffer; PulseDestroyBufferPFN PFN_DestroyBuffer;
PulseCreateImagePFN PFN_CreateImage; PulseCreateImagePFN PFN_CreateImage;
PulseDestroyImagePFN PFN_DestroyImage; PulseDestroyImagePFN PFN_DestroyImage;

View File

@@ -25,7 +25,8 @@ typedef PulseCommandList (*PulseRequestCommandListPFN)(PulseDevice, PulseCommand
typedef bool (*PulseSubmitCommandListPFN)(PulseDevice, PulseCommandList, PulseFence); typedef bool (*PulseSubmitCommandListPFN)(PulseDevice, PulseCommandList, PulseFence);
typedef void (*PulseReleaseCommandListPFN)(PulseDevice, PulseCommandList); typedef void (*PulseReleaseCommandListPFN)(PulseDevice, PulseCommandList);
typedef PulseBuffer (*PulseCreateBufferPFN)(PulseDevice, const PulseBufferCreateInfo*); typedef PulseBuffer (*PulseCreateBufferPFN)(PulseDevice, const PulseBufferCreateInfo*);
typedef bool (*PulseGetBufferMapPFN)(PulseBuffer, void**); typedef bool (*PulseMapBufferPFN)(PulseBuffer, void**);
typedef void (*PulseUnmapBufferPFN)(PulseBuffer);
typedef void (*PulseDestroyBufferPFN)(PulseDevice, PulseBuffer); typedef void (*PulseDestroyBufferPFN)(PulseDevice, PulseBuffer);
typedef PulseImage (*PulseCreateImagePFN)(PulseDevice, const PulseImageCreateInfo*); typedef PulseImage (*PulseCreateImagePFN)(PulseDevice, const PulseImageCreateInfo*);
typedef void (*PulseDestroyImagePFN)(PulseDevice, PulseImage); typedef void (*PulseDestroyImagePFN)(PulseDevice, PulseImage);

View File

@@ -1,5 +1,6 @@
#include "Common.h" #include "Common.h"
#include <string.h>
#include <unity/unity.h> #include <unity/unity.h>
#include <Pulse.h> #include <Pulse.h>
@@ -71,7 +72,44 @@ void TestBufferCreation()
CleanupPulse(backend); CleanupPulse(backend);
} }
void TestBufferMapping()
{
PulseBackend backend;
SetupPulse(&backend);
PulseDevice device;
SetupDevice(backend, &device);
const unsigned char data[8] = { 0xA1, 0xFF, 0xDF, 0x17, 0x5B, 0xCC, 0x00, 0x36 };
PulseBufferCreateInfo buffer_create_info = { 0 };
buffer_create_info.size = 8;
buffer_create_info.usage = PULSE_BUFFER_USAGE_TRANSFER_UPLOAD;
PulseBuffer buffer = PulseCreateBuffer(device, &buffer_create_info);
TEST_ASSERT_NOT_EQUAL_MESSAGE(buffer, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
{
void* ptr;
TEST_ASSERT_NOT_EQUAL_MESSAGE(PulseMapBuffer(buffer, &ptr), false, PulseVerbaliseErrorType(PulseGetLastErrorType()));
TEST_ASSERT_NOT_NULL(ptr);
memcpy(ptr, data, 8);
PulseUnmapBuffer(buffer);
}
{
void* ptr;
TEST_ASSERT_NOT_EQUAL_MESSAGE(PulseMapBuffer(buffer, &ptr), false, PulseVerbaliseErrorType(PulseGetLastErrorType()));
TEST_ASSERT_NOT_NULL(ptr);
TEST_ASSERT_EQUAL(memcmp(ptr, data, 8), 0);
PulseUnmapBuffer(buffer);
}
PulseDestroyBuffer(device, buffer);
CleanupDevice(device);
CleanupPulse(backend);
}
void TestBuffer() void TestBuffer()
{ {
RUN_TEST(TestBufferCreation); RUN_TEST(TestBufferCreation);
RUN_TEST(TestBufferMapping);
} }