mirror of
https://github.com/Kbz-8/Pulse.git
synced 2026-01-11 15:33:34 +00:00
adding pipeline unit tests
This commit is contained in:
@@ -110,6 +110,8 @@ typedef enum PulseErrorType
|
|||||||
PULSE_ERROR_MAP_FAILED,
|
PULSE_ERROR_MAP_FAILED,
|
||||||
PULSE_ERROR_INVALID_DEVICE,
|
PULSE_ERROR_INVALID_DEVICE,
|
||||||
PULSE_ERROR_INVALID_REGION,
|
PULSE_ERROR_INVALID_REGION,
|
||||||
|
PULSE_ERROR_INVALID_BUFFER_USAGE,
|
||||||
|
PULSE_ERROR_INVALID_IMAGE_USAGE,
|
||||||
} PulseErrorType;
|
} PulseErrorType;
|
||||||
|
|
||||||
typedef enum PulseImageType
|
typedef enum PulseImageType
|
||||||
|
|||||||
49
README.md
49
README.md
@@ -6,3 +6,52 @@
|
|||||||
[](https://github.com/ft-grmhd/Pulse/actions/workflows/windows-build.yml)
|
[](https://github.com/ft-grmhd/Pulse/actions/workflows/windows-build.yml)
|
||||||
|
|
||||||
Pulse is a low level GPGPU library designed for highly intensive general GPU computations with high control over the hardware. It is built on top of Vulkan and a Metal support is in discussion.
|
Pulse is a low level GPGPU library designed for highly intensive general GPU computations with high control over the hardware. It is built on top of Vulkan and a Metal support is in discussion.
|
||||||
|
|
||||||
|
```cpp
|
||||||
|
#include <Pulse.h>
|
||||||
|
|
||||||
|
int main(void)
|
||||||
|
{
|
||||||
|
PulseBackend backend = PulseLoadBackend(PULSE_BACKEND_VULKAN, PULSE_SHADER_FORMAT_SPIRV_BIT, PULSE_NO_DEBUG);
|
||||||
|
PulseDevice device = PulseCreateDevice(backend, NULL, 0);
|
||||||
|
|
||||||
|
const uint8_t shader_bytecode[] = {
|
||||||
|
#include "shader.spv.h"
|
||||||
|
};
|
||||||
|
|
||||||
|
PulseComputePipelineCreateInfo info = { 0 };
|
||||||
|
info.code_size = sizeof(shader_bytecode);
|
||||||
|
info.code = shader_bytecode;
|
||||||
|
info.entrypoint = "main";
|
||||||
|
info.format = PULSE_SHADER_FORMAT_SPIRV_BIT;
|
||||||
|
info.num_readwrite_storage_buffers = 1;
|
||||||
|
PulseComputePipeline pipeline = PulseCreateComputePipeline(device, &info);
|
||||||
|
|
||||||
|
PulseBufferCreateInfo buffer_create_info = { 0 };
|
||||||
|
buffer_create_info.size = 1024;
|
||||||
|
buffer_create_info.usage = PULSE_BUFFER_USAGE_STORAGE_WRITE;
|
||||||
|
PulseBuffer buffer = PulseCreateBuffer(device, &buffer_create_info);
|
||||||
|
|
||||||
|
PulseFence fence = PulseCreateFence(device);
|
||||||
|
PulseCommandList cmd = PulseRequestCommandList(device, PULSE_COMMAND_LIST_GENERAL);
|
||||||
|
|
||||||
|
PulseComputePass pass = PulseBeginComputePass(cmd);
|
||||||
|
PulseBindStorageBuffers(pass, 0, &buffer, 1);
|
||||||
|
PulseBindComputePipeline(pass, pipeline);
|
||||||
|
PulseDispatchComputations(pass, 32, 32, 1);
|
||||||
|
PulseEndComputePass(pass);
|
||||||
|
|
||||||
|
PulseSubmitCommandList(device, cmd, fence);
|
||||||
|
PulseWaitForFences(device, &fence, 1, true);
|
||||||
|
|
||||||
|
PulseReleaseCommandList(device, cmd);
|
||||||
|
PulseDestroyFence(device, fence);
|
||||||
|
PulseDestroyComputePipeline(device, pipeline);
|
||||||
|
|
||||||
|
PulseDestroyBuffer(device, buffer);
|
||||||
|
|
||||||
|
PulseDestroyDevice(device);
|
||||||
|
PulseUnloadBackend(backend);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|||||||
@@ -37,16 +37,32 @@ void VulkanDestroyComputePass(PulseDevice device, PulseComputePass pass)
|
|||||||
void VulkanBindStorageBuffers(PulseComputePass pass, uint32_t starting_slot, const PulseBuffer* buffers, uint32_t num_buffers)
|
void VulkanBindStorageBuffers(PulseComputePass pass, uint32_t starting_slot, const PulseBuffer* buffers, uint32_t num_buffers)
|
||||||
{
|
{
|
||||||
PulseBufferUsageFlags usage = buffers[0]->usage;
|
PulseBufferUsageFlags usage = buffers[0]->usage;
|
||||||
PulseBuffer* array = ((usage & PULSE_BUFFER_USAGE_STORAGE_WRITE) != 0) ? pass->readwrite_storage_buffers : pass->readonly_storage_buffers;
|
bool is_readwrite = (usage & PULSE_BUFFER_USAGE_STORAGE_WRITE) != 0;
|
||||||
|
PulseBuffer* array = is_readwrite ? pass->readwrite_storage_buffers : pass->readonly_storage_buffers;
|
||||||
VulkanComputePass* vulkan_pass = VULKAN_RETRIEVE_DRIVER_DATA_AS(pass, VulkanComputePass*);
|
VulkanComputePass* vulkan_pass = VULKAN_RETRIEVE_DRIVER_DATA_AS(pass, VulkanComputePass*);
|
||||||
|
|
||||||
for(uint32_t i = 0; i < num_buffers; i++)
|
for(uint32_t i = 0; i < num_buffers; i++)
|
||||||
{
|
{
|
||||||
|
if(is_readwrite && (buffers[i]->usage & PULSE_BUFFER_USAGE_STORAGE_WRITE) == 0)
|
||||||
|
{
|
||||||
|
if(PULSE_IS_BACKEND_LOW_LEVEL_DEBUG(pass->cmd->device->backend))
|
||||||
|
PulseLogError(pass->cmd->device->backend, "cannot bind a read only buffer with read-write buffers");
|
||||||
|
PulseSetInternalError(PULSE_ERROR_INVALID_BUFFER_USAGE);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
else if(!is_readwrite && (buffers[i]->usage & PULSE_BUFFER_USAGE_STORAGE_WRITE) != 0)
|
||||||
|
{
|
||||||
|
if(PULSE_IS_BACKEND_LOW_LEVEL_DEBUG(pass->cmd->device->backend))
|
||||||
|
PulseLogError(pass->cmd->device->backend, "cannot bind a read-write buffer with read only buffers");
|
||||||
|
PulseSetInternalError(PULSE_ERROR_INVALID_BUFFER_USAGE);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if(array[starting_slot + i] == buffers[i])
|
if(array[starting_slot + i] == buffers[i])
|
||||||
continue;
|
continue;
|
||||||
array[starting_slot + i] = buffers[i];
|
array[starting_slot + i] = buffers[i];
|
||||||
|
|
||||||
if((usage & PULSE_BUFFER_USAGE_STORAGE_WRITE) != 0)
|
if(is_readwrite)
|
||||||
vulkan_pass->should_recreate_write_descriptor_sets = true;
|
vulkan_pass->should_recreate_write_descriptor_sets = true;
|
||||||
else
|
else
|
||||||
vulkan_pass->should_recreate_read_only_descriptor_sets = true;
|
vulkan_pass->should_recreate_read_only_descriptor_sets = true;
|
||||||
@@ -60,11 +76,27 @@ void VulkanBindUniformData(PulseComputePass pass, uint32_t slot, const void* dat
|
|||||||
void VulkanBindStorageImages(PulseComputePass pass, uint32_t starting_slot, const PulseImage* images, uint32_t num_images)
|
void VulkanBindStorageImages(PulseComputePass pass, uint32_t starting_slot, const PulseImage* images, uint32_t num_images)
|
||||||
{
|
{
|
||||||
PulseImageUsageFlags usage = images[0]->usage;
|
PulseImageUsageFlags usage = images[0]->usage;
|
||||||
PulseImage* array = ((usage & PULSE_IMAGE_USAGE_STORAGE_WRITE) != 0) ? pass->readwrite_images : pass->readonly_images;
|
bool is_readwrite = (usage & PULSE_IMAGE_USAGE_STORAGE_WRITE) != 0;
|
||||||
|
PulseImage* array = is_readwrite ? pass->readwrite_images : pass->readonly_images;
|
||||||
VulkanComputePass* vulkan_pass = VULKAN_RETRIEVE_DRIVER_DATA_AS(pass, VulkanComputePass*);
|
VulkanComputePass* vulkan_pass = VULKAN_RETRIEVE_DRIVER_DATA_AS(pass, VulkanComputePass*);
|
||||||
|
|
||||||
for(uint32_t i = 0; i < num_images; i++)
|
for(uint32_t i = 0; i < num_images; i++)
|
||||||
{
|
{
|
||||||
|
if(is_readwrite && (images[i]->usage & PULSE_IMAGE_USAGE_STORAGE_WRITE) == 0)
|
||||||
|
{
|
||||||
|
if(PULSE_IS_BACKEND_LOW_LEVEL_DEBUG(pass->cmd->device->backend))
|
||||||
|
PulseLogError(pass->cmd->device->backend, "cannot bind a read only image with read-write images");
|
||||||
|
PulseSetInternalError(PULSE_ERROR_INVALID_IMAGE_USAGE);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
else if(!is_readwrite && (images[i]->usage & PULSE_IMAGE_USAGE_STORAGE_WRITE) != 0)
|
||||||
|
{
|
||||||
|
if(PULSE_IS_BACKEND_LOW_LEVEL_DEBUG(pass->cmd->device->backend))
|
||||||
|
PulseLogError(pass->cmd->device->backend, "cannot bind a read-write image with read only images");
|
||||||
|
PulseSetInternalError(PULSE_ERROR_INVALID_IMAGE_USAGE);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if(array[starting_slot + i] == images[i])
|
if(array[starting_slot + i] == images[i])
|
||||||
continue;
|
continue;
|
||||||
array[starting_slot + i] = images[i];
|
array[starting_slot + i] = images[i];
|
||||||
|
|||||||
@@ -89,8 +89,9 @@ void VulkanDestroyComputePipeline(PulseDevice device, PulseComputePipeline pipel
|
|||||||
vulkan_device->vkDestroyPipelineLayout(vulkan_device->device, vulkan_pipeline->layout, PULSE_NULLPTR);
|
vulkan_device->vkDestroyPipelineLayout(vulkan_device->device, vulkan_pipeline->layout, PULSE_NULLPTR);
|
||||||
vulkan_device->vkDestroyPipeline(vulkan_device->device, vulkan_pipeline->pipeline, PULSE_NULLPTR);
|
vulkan_device->vkDestroyPipeline(vulkan_device->device, vulkan_pipeline->pipeline, PULSE_NULLPTR);
|
||||||
free(vulkan_pipeline);
|
free(vulkan_pipeline);
|
||||||
free(pipeline);
|
|
||||||
|
|
||||||
if(PULSE_IS_BACKEND_HIGH_LEVEL_DEBUG(device->backend))
|
if(PULSE_IS_BACKEND_HIGH_LEVEL_DEBUG(device->backend))
|
||||||
PulseLogInfoFmt(device->backend, "(Vulkan) destroyed compute pipeline %p", pipeline);
|
PulseLogInfoFmt(device->backend, "(Vulkan) destroyed compute pipeline %p", pipeline);
|
||||||
|
|
||||||
|
free(pipeline);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -275,6 +275,8 @@ void VulkanBindDescriptorSets(PulseComputePass pass)
|
|||||||
|
|
||||||
for(uint32_t i = 0; i < pass->current_pipeline->num_readonly_storage_images; i++)
|
for(uint32_t i = 0; i < pass->current_pipeline->num_readonly_storage_images; i++)
|
||||||
{
|
{
|
||||||
|
if(pass->readonly_images[i] == PULSE_NULL_HANDLE)
|
||||||
|
continue;
|
||||||
VkWriteDescriptorSet* write_descriptor_set = &writes[write_count];
|
VkWriteDescriptorSet* write_descriptor_set = &writes[write_count];
|
||||||
|
|
||||||
write_descriptor_set->sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET;
|
write_descriptor_set->sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET;
|
||||||
@@ -301,6 +303,8 @@ void VulkanBindDescriptorSets(PulseComputePass pass)
|
|||||||
|
|
||||||
for(uint32_t i = 0; i < pass->current_pipeline->num_readonly_storage_buffers; i++)
|
for(uint32_t i = 0; i < pass->current_pipeline->num_readonly_storage_buffers; i++)
|
||||||
{
|
{
|
||||||
|
if(pass->readonly_storage_buffers[i] == PULSE_NULL_HANDLE)
|
||||||
|
continue;
|
||||||
VkWriteDescriptorSet* write_descriptor_set = &writes[write_count];
|
VkWriteDescriptorSet* write_descriptor_set = &writes[write_count];
|
||||||
|
|
||||||
write_descriptor_set->sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET;
|
write_descriptor_set->sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET;
|
||||||
@@ -336,6 +340,8 @@ void VulkanBindDescriptorSets(PulseComputePass pass)
|
|||||||
|
|
||||||
for(uint32_t i = 0; i < pass->current_pipeline->num_readwrite_storage_images; i++)
|
for(uint32_t i = 0; i < pass->current_pipeline->num_readwrite_storage_images; i++)
|
||||||
{
|
{
|
||||||
|
if(pass->readwrite_images[i] == PULSE_NULL_HANDLE)
|
||||||
|
continue;
|
||||||
VkWriteDescriptorSet* write_descriptor_set = &writes[write_count];
|
VkWriteDescriptorSet* write_descriptor_set = &writes[write_count];
|
||||||
|
|
||||||
write_descriptor_set->sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET;
|
write_descriptor_set->sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET;
|
||||||
@@ -362,6 +368,8 @@ void VulkanBindDescriptorSets(PulseComputePass pass)
|
|||||||
|
|
||||||
for(uint32_t i = 0; i < pass->current_pipeline->num_readwrite_storage_buffers; i++)
|
for(uint32_t i = 0; i < pass->current_pipeline->num_readwrite_storage_buffers; i++)
|
||||||
{
|
{
|
||||||
|
if(pass->readwrite_storage_buffers[i] == PULSE_NULL_HANDLE)
|
||||||
|
continue;
|
||||||
VkWriteDescriptorSet* write_descriptor_set = &writes[write_count];
|
VkWriteDescriptorSet* write_descriptor_set = &writes[write_count];
|
||||||
|
|
||||||
write_descriptor_set->sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET;
|
write_descriptor_set->sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET;
|
||||||
|
|||||||
@@ -164,8 +164,10 @@ PULSE_API const char* PulseVerbaliseErrorType(PulseErrorType error)
|
|||||||
case PULSE_ERROR_DEVICE_LOST: return "device has been lost";
|
case PULSE_ERROR_DEVICE_LOST: return "device has been lost";
|
||||||
case PULSE_ERROR_INVALID_INTERNAL_POINTER: return "invalid internal pointer";
|
case PULSE_ERROR_INVALID_INTERNAL_POINTER: return "invalid internal pointer";
|
||||||
case PULSE_ERROR_MAP_FAILED: return "memory mapping failed";
|
case PULSE_ERROR_MAP_FAILED: return "memory mapping failed";
|
||||||
case PULSE_ERROR_INVALID_DEVICE: return "device is invalid";
|
case PULSE_ERROR_INVALID_DEVICE: return "invalid device";
|
||||||
case PULSE_ERROR_INVALID_REGION: return "region is invalid";
|
case PULSE_ERROR_INVALID_REGION: return "invalid region";
|
||||||
|
case PULSE_ERROR_INVALID_BUFFER_USAGE: return "invalid buffer usage";
|
||||||
|
case PULSE_ERROR_INVALID_IMAGE_USAGE: return "invalid image usage";
|
||||||
|
|
||||||
default: return "invalid error type";
|
default: return "invalid error type";
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -325,6 +325,75 @@ void TestBufferComputeWrite()
|
|||||||
CleanupPulse(backend);
|
CleanupPulse(backend);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TestBufferComputeCopy()
|
||||||
|
{
|
||||||
|
PulseBackend backend;
|
||||||
|
SetupPulse(&backend);
|
||||||
|
PulseDevice device;
|
||||||
|
SetupDevice(backend, &device);
|
||||||
|
|
||||||
|
const uint8_t shader_bytecode[] = {
|
||||||
|
#include "Shaders/BufferCopy.spv.h"
|
||||||
|
};
|
||||||
|
|
||||||
|
uint32_t data[256];
|
||||||
|
memset(data, 0xFF, 256 * sizeof(uint32_t));
|
||||||
|
|
||||||
|
PulseBufferCreateInfo buffer_create_info = { 0 };
|
||||||
|
buffer_create_info.size = 256 * sizeof(int32_t);
|
||||||
|
buffer_create_info.usage = PULSE_BUFFER_USAGE_STORAGE_READ | PULSE_BUFFER_USAGE_TRANSFER_UPLOAD;
|
||||||
|
PulseBuffer read_buffer = PulseCreateBuffer(device, &buffer_create_info);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(read_buffer, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
{
|
||||||
|
void* ptr;
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(PulseMapBuffer(read_buffer, &ptr), false, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
TEST_ASSERT_NOT_NULL(ptr);
|
||||||
|
memcpy(ptr, data, 256 * sizeof(uint32_t));
|
||||||
|
PulseUnmapBuffer(read_buffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
buffer_create_info.usage = PULSE_BUFFER_USAGE_STORAGE_WRITE | PULSE_BUFFER_USAGE_TRANSFER_DOWNLOAD;
|
||||||
|
PulseBuffer write_buffer = PulseCreateBuffer(device, &buffer_create_info);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(write_buffer, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseComputePipeline pipeline;
|
||||||
|
LoadComputePipeline(device, &pipeline, shader_bytecode, sizeof(shader_bytecode), 0, 1, 0, 1, 0);
|
||||||
|
|
||||||
|
PulseFence fence = PulseCreateFence(device);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(fence, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
PulseCommandList cmd = PulseRequestCommandList(device, PULSE_COMMAND_LIST_GENERAL);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(cmd, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseComputePass pass = PulseBeginComputePass(cmd);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(pass, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
PulseBindStorageBuffers(pass, 0, &read_buffer, 1);
|
||||||
|
PulseBindStorageBuffers(pass, 0, &write_buffer, 1);
|
||||||
|
PulseBindComputePipeline(pass, pipeline);
|
||||||
|
PulseDispatchComputations(pass, 32, 32, 1);
|
||||||
|
PulseEndComputePass(pass);
|
||||||
|
|
||||||
|
TEST_ASSERT_TRUE_MESSAGE(PulseSubmitCommandList(device, cmd, fence), PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
TEST_ASSERT_TRUE_MESSAGE(PulseWaitForFences(device, &fence, 1, true), PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
{
|
||||||
|
void* ptr;
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(PulseMapBuffer(write_buffer, &ptr), false, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
TEST_ASSERT_NOT_NULL(ptr);
|
||||||
|
TEST_ASSERT_EQUAL(memcmp(ptr, data, 256 * sizeof(uint32_t)), 0);
|
||||||
|
PulseUnmapBuffer(write_buffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
PulseReleaseCommandList(device, cmd);
|
||||||
|
PulseDestroyFence(device, fence);
|
||||||
|
PulseDestroyBuffer(device, read_buffer);
|
||||||
|
PulseDestroyBuffer(device, write_buffer);
|
||||||
|
|
||||||
|
CleanupPipeline(device, pipeline);
|
||||||
|
CleanupDevice(device);
|
||||||
|
CleanupPulse(backend);
|
||||||
|
}
|
||||||
|
|
||||||
void TestBufferDestruction()
|
void TestBufferDestruction()
|
||||||
{
|
{
|
||||||
PulseBackend backend;
|
PulseBackend backend;
|
||||||
@@ -365,5 +434,6 @@ void TestBuffer()
|
|||||||
RUN_TEST(TestBufferCopy);
|
RUN_TEST(TestBufferCopy);
|
||||||
RUN_TEST(TestBufferCopyImage);
|
RUN_TEST(TestBufferCopyImage);
|
||||||
RUN_TEST(TestBufferComputeWrite);
|
RUN_TEST(TestBufferComputeWrite);
|
||||||
|
RUN_TEST(TestBufferComputeCopy);
|
||||||
RUN_TEST(TestBufferDestruction);
|
RUN_TEST(TestBufferDestruction);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,6 +10,211 @@ void TestPipelineSetup()
|
|||||||
PulseDevice device;
|
PulseDevice device;
|
||||||
SetupDevice(backend, &device);
|
SetupDevice(backend, &device);
|
||||||
|
|
||||||
|
const uint8_t shader_bytecode[] = {
|
||||||
|
#include "Shaders/Simple.spv.h"
|
||||||
|
};
|
||||||
|
|
||||||
|
PulseComputePipeline pipeline;
|
||||||
|
LoadComputePipeline(device, &pipeline, shader_bytecode, sizeof(shader_bytecode), 0, 0, 0, 0, 0);
|
||||||
|
|
||||||
|
PulseFence fence = PulseCreateFence(device);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(fence, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
PulseCommandList cmd = PulseRequestCommandList(device, PULSE_COMMAND_LIST_GENERAL);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(cmd, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseComputePass pass = PulseBeginComputePass(cmd);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(pass, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
PulseBindComputePipeline(pass, pipeline);
|
||||||
|
PulseDispatchComputations(pass, 32, 32, 1);
|
||||||
|
PulseEndComputePass(pass);
|
||||||
|
|
||||||
|
TEST_ASSERT_TRUE_MESSAGE(PulseSubmitCommandList(device, cmd, fence), PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
TEST_ASSERT_TRUE_MESSAGE(PulseWaitForFences(device, &fence, 1, true), PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseReleaseCommandList(device, cmd);
|
||||||
|
PulseDestroyFence(device, fence);
|
||||||
|
|
||||||
|
CleanupPipeline(device, pipeline);
|
||||||
|
CleanupDevice(device);
|
||||||
|
CleanupPulse(backend);
|
||||||
|
}
|
||||||
|
|
||||||
|
void TestPipelineReadOnlyBindings()
|
||||||
|
{
|
||||||
|
PulseBackend backend;
|
||||||
|
SetupPulse(&backend);
|
||||||
|
PulseDevice device;
|
||||||
|
SetupDevice(backend, &device);
|
||||||
|
|
||||||
|
const uint8_t shader_bytecode[] = {
|
||||||
|
#include "Shaders/ReadOnlyBindings.spv.h"
|
||||||
|
};
|
||||||
|
|
||||||
|
PulseBufferCreateInfo buffer_create_info = { 0 };
|
||||||
|
buffer_create_info.size = 256 * sizeof(int32_t);
|
||||||
|
buffer_create_info.usage = PULSE_BUFFER_USAGE_STORAGE_READ;
|
||||||
|
PulseBuffer buffer = PulseCreateBuffer(device, &buffer_create_info);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(buffer, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseImageCreateInfo image_create_info = { 0 };
|
||||||
|
image_create_info.type = PULSE_IMAGE_TYPE_2D;
|
||||||
|
image_create_info.format = PULSE_IMAGE_FORMAT_R8G8B8A8_UNORM;
|
||||||
|
image_create_info.usage = PULSE_IMAGE_USAGE_STORAGE_READ;
|
||||||
|
image_create_info.width = 256;
|
||||||
|
image_create_info.height = 256;
|
||||||
|
image_create_info.layer_count_or_depth = 1;
|
||||||
|
PulseImage image = PulseCreateImage(device, &image_create_info);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(image, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseComputePipeline pipeline;
|
||||||
|
LoadComputePipeline(device, &pipeline, shader_bytecode, sizeof(shader_bytecode), 1, 1, 0, 0, 0);
|
||||||
|
|
||||||
|
PulseFence fence = PulseCreateFence(device);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(fence, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
PulseCommandList cmd = PulseRequestCommandList(device, PULSE_COMMAND_LIST_GENERAL);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(cmd, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseComputePass pass = PulseBeginComputePass(cmd);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(pass, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
PulseBindStorageBuffers(pass, 0, &buffer, 1);
|
||||||
|
PulseBindStorageImages(pass, 1, &image, 1);
|
||||||
|
PulseBindComputePipeline(pass, pipeline);
|
||||||
|
PulseDispatchComputations(pass, 32, 32, 1);
|
||||||
|
PulseEndComputePass(pass);
|
||||||
|
|
||||||
|
TEST_ASSERT_TRUE_MESSAGE(PulseSubmitCommandList(device, cmd, fence), PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
TEST_ASSERT_TRUE_MESSAGE(PulseWaitForFences(device, &fence, 1, true), PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseReleaseCommandList(device, cmd);
|
||||||
|
PulseDestroyFence(device, fence);
|
||||||
|
PulseDestroyBuffer(device, buffer);
|
||||||
|
PulseDestroyImage(device, image);
|
||||||
|
|
||||||
|
CleanupPipeline(device, pipeline);
|
||||||
|
CleanupDevice(device);
|
||||||
|
CleanupPulse(backend);
|
||||||
|
}
|
||||||
|
|
||||||
|
void TestPipelineWriteOnlyBindings()
|
||||||
|
{
|
||||||
|
PulseBackend backend;
|
||||||
|
SetupPulse(&backend);
|
||||||
|
PulseDevice device;
|
||||||
|
SetupDevice(backend, &device);
|
||||||
|
|
||||||
|
const uint8_t shader_bytecode[] = {
|
||||||
|
#include "Shaders/ReadOnlyBindings.spv.h"
|
||||||
|
};
|
||||||
|
|
||||||
|
PulseBufferCreateInfo buffer_create_info = { 0 };
|
||||||
|
buffer_create_info.size = 256 * sizeof(int32_t);
|
||||||
|
buffer_create_info.usage = PULSE_BUFFER_USAGE_STORAGE_WRITE;
|
||||||
|
PulseBuffer buffer = PulseCreateBuffer(device, &buffer_create_info);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(buffer, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseImageCreateInfo image_create_info = { 0 };
|
||||||
|
image_create_info.type = PULSE_IMAGE_TYPE_2D;
|
||||||
|
image_create_info.format = PULSE_IMAGE_FORMAT_R8G8B8A8_UNORM;
|
||||||
|
image_create_info.usage = PULSE_IMAGE_USAGE_STORAGE_WRITE;
|
||||||
|
image_create_info.width = 256;
|
||||||
|
image_create_info.height = 256;
|
||||||
|
image_create_info.layer_count_or_depth = 1;
|
||||||
|
PulseImage image = PulseCreateImage(device, &image_create_info);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(image, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseComputePipeline pipeline;
|
||||||
|
LoadComputePipeline(device, &pipeline, shader_bytecode, sizeof(shader_bytecode), 0, 0, 1, 1, 0);
|
||||||
|
|
||||||
|
PulseFence fence = PulseCreateFence(device);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(fence, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
PulseCommandList cmd = PulseRequestCommandList(device, PULSE_COMMAND_LIST_GENERAL);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(cmd, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseComputePass pass = PulseBeginComputePass(cmd);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(pass, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
PulseBindStorageBuffers(pass, 0, &buffer, 1);
|
||||||
|
PulseBindStorageImages(pass, 1, &image, 1);
|
||||||
|
PulseBindComputePipeline(pass, pipeline);
|
||||||
|
PulseDispatchComputations(pass, 32, 32, 1);
|
||||||
|
PulseEndComputePass(pass);
|
||||||
|
|
||||||
|
TEST_ASSERT_TRUE_MESSAGE(PulseSubmitCommandList(device, cmd, fence), PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
TEST_ASSERT_TRUE_MESSAGE(PulseWaitForFences(device, &fence, 1, true), PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseReleaseCommandList(device, cmd);
|
||||||
|
PulseDestroyFence(device, fence);
|
||||||
|
PulseDestroyBuffer(device, buffer);
|
||||||
|
PulseDestroyImage(device, image);
|
||||||
|
|
||||||
|
CleanupPipeline(device, pipeline);
|
||||||
|
CleanupDevice(device);
|
||||||
|
CleanupPulse(backend);
|
||||||
|
}
|
||||||
|
|
||||||
|
void TestPipelineReadWriteBindings()
|
||||||
|
{
|
||||||
|
PulseBackend backend;
|
||||||
|
SetupPulse(&backend);
|
||||||
|
PulseDevice device;
|
||||||
|
SetupDevice(backend, &device);
|
||||||
|
|
||||||
|
const uint8_t shader_bytecode[] = {
|
||||||
|
#include "Shaders/ReadOnlyBindings.spv.h"
|
||||||
|
};
|
||||||
|
|
||||||
|
PulseBufferCreateInfo buffer_create_info = { 0 };
|
||||||
|
buffer_create_info.size = 256 * sizeof(int32_t);
|
||||||
|
buffer_create_info.usage = PULSE_BUFFER_USAGE_STORAGE_READ;
|
||||||
|
PulseBuffer read_buffer = PulseCreateBuffer(device, &buffer_create_info);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(read_buffer, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
buffer_create_info.usage = PULSE_BUFFER_USAGE_STORAGE_WRITE;
|
||||||
|
PulseBuffer write_buffer = PulseCreateBuffer(device, &buffer_create_info);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(write_buffer, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseImageCreateInfo image_create_info = { 0 };
|
||||||
|
image_create_info.type = PULSE_IMAGE_TYPE_2D;
|
||||||
|
image_create_info.format = PULSE_IMAGE_FORMAT_R8G8B8A8_UNORM;
|
||||||
|
image_create_info.usage = PULSE_IMAGE_USAGE_STORAGE_READ;
|
||||||
|
image_create_info.width = 256;
|
||||||
|
image_create_info.height = 256;
|
||||||
|
image_create_info.layer_count_or_depth = 1;
|
||||||
|
PulseImage read_image = PulseCreateImage(device, &image_create_info);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(read_image, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
image_create_info.usage = PULSE_IMAGE_USAGE_STORAGE_WRITE;
|
||||||
|
PulseImage write_image = PulseCreateImage(device, &image_create_info);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(write_image, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseComputePipeline pipeline;
|
||||||
|
LoadComputePipeline(device, &pipeline, shader_bytecode, sizeof(shader_bytecode), 1, 1, 1, 1, 0);
|
||||||
|
|
||||||
|
PulseFence fence = PulseCreateFence(device);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(fence, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
PulseCommandList cmd = PulseRequestCommandList(device, PULSE_COMMAND_LIST_GENERAL);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(cmd, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseComputePass pass = PulseBeginComputePass(cmd);
|
||||||
|
TEST_ASSERT_NOT_EQUAL_MESSAGE(pass, PULSE_NULL_HANDLE, PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
PulseBindStorageBuffers(pass, 0, &read_buffer, 1);
|
||||||
|
PulseBindStorageBuffers(pass, 0, &write_buffer, 1);
|
||||||
|
PulseBindStorageImages(pass, 1, &read_image, 1);
|
||||||
|
PulseBindStorageImages(pass, 1, &write_image, 1);
|
||||||
|
PulseBindComputePipeline(pass, pipeline);
|
||||||
|
PulseDispatchComputations(pass, 32, 32, 1);
|
||||||
|
PulseEndComputePass(pass);
|
||||||
|
|
||||||
|
TEST_ASSERT_TRUE_MESSAGE(PulseSubmitCommandList(device, cmd, fence), PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
TEST_ASSERT_TRUE_MESSAGE(PulseWaitForFences(device, &fence, 1, true), PulseVerbaliseErrorType(PulseGetLastErrorType()));
|
||||||
|
|
||||||
|
PulseReleaseCommandList(device, cmd);
|
||||||
|
PulseDestroyFence(device, fence);
|
||||||
|
PulseDestroyBuffer(device, read_buffer);
|
||||||
|
PulseDestroyBuffer(device, write_buffer);
|
||||||
|
PulseDestroyImage(device, read_image);
|
||||||
|
PulseDestroyImage(device, write_image);
|
||||||
|
|
||||||
|
CleanupPipeline(device, pipeline);
|
||||||
CleanupDevice(device);
|
CleanupDevice(device);
|
||||||
CleanupPulse(backend);
|
CleanupPulse(backend);
|
||||||
}
|
}
|
||||||
@@ -17,4 +222,7 @@ void TestPipelineSetup()
|
|||||||
void TestPipeline()
|
void TestPipeline()
|
||||||
{
|
{
|
||||||
RUN_TEST(TestPipelineSetup);
|
RUN_TEST(TestPipelineSetup);
|
||||||
|
RUN_TEST(TestPipelineReadOnlyBindings);
|
||||||
|
RUN_TEST(TestPipelineWriteOnlyBindings);
|
||||||
|
RUN_TEST(TestPipelineReadWriteBindings);
|
||||||
}
|
}
|
||||||
|
|||||||
26
Tests/Vulkan/Shaders/BufferCopy.nzsl
git.filemode.normal_file
26
Tests/Vulkan/Shaders/BufferCopy.nzsl
git.filemode.normal_file
@@ -0,0 +1,26 @@
|
|||||||
|
[nzsl_version("1.0")]
|
||||||
|
module;
|
||||||
|
|
||||||
|
struct Input
|
||||||
|
{
|
||||||
|
[builtin(global_invocation_indices)] indices: vec3[u32]
|
||||||
|
}
|
||||||
|
|
||||||
|
[layout(std430)]
|
||||||
|
struct SSBO
|
||||||
|
{
|
||||||
|
data: dyn_array[u32]
|
||||||
|
}
|
||||||
|
|
||||||
|
external
|
||||||
|
{
|
||||||
|
[set(0), binding(0)] read_ssbo: storage[SSBO, readonly],
|
||||||
|
[set(1), binding(0)] write_ssbo: storage[SSBO, writeonly],
|
||||||
|
}
|
||||||
|
|
||||||
|
[entry(compute)]
|
||||||
|
[workgroup(32, 32, 1)]
|
||||||
|
fn main(input: Input)
|
||||||
|
{
|
||||||
|
write_ssbo.data[input.indices.x * input.indices.y] = read_ssbo.data[input.indices.x * input.indices.y];
|
||||||
|
}
|
||||||
25
Tests/Vulkan/Shaders/ReadOnlyBindings.nzsl
git.filemode.normal_file
25
Tests/Vulkan/Shaders/ReadOnlyBindings.nzsl
git.filemode.normal_file
@@ -0,0 +1,25 @@
|
|||||||
|
[nzsl_version("1.0")]
|
||||||
|
module;
|
||||||
|
|
||||||
|
struct Input
|
||||||
|
{
|
||||||
|
[builtin(global_invocation_indices)] indices: vec3[u32]
|
||||||
|
}
|
||||||
|
|
||||||
|
[layout(std430)]
|
||||||
|
struct SSBO
|
||||||
|
{
|
||||||
|
data: dyn_array[u32]
|
||||||
|
}
|
||||||
|
|
||||||
|
external
|
||||||
|
{
|
||||||
|
[set(0), binding(0)] read_ssbo: storage[SSBO, readonly],
|
||||||
|
[set(0), binding(1)] read_texture: texture2D[f32, readonly, rgba8],
|
||||||
|
}
|
||||||
|
|
||||||
|
[entry(compute)]
|
||||||
|
[workgroup(32, 32, 1)]
|
||||||
|
fn main(input: Input)
|
||||||
|
{
|
||||||
|
}
|
||||||
27
Tests/Vulkan/Shaders/ReadWriteBindings.nzsl
git.filemode.normal_file
27
Tests/Vulkan/Shaders/ReadWriteBindings.nzsl
git.filemode.normal_file
@@ -0,0 +1,27 @@
|
|||||||
|
[nzsl_version("1.0")]
|
||||||
|
module;
|
||||||
|
|
||||||
|
struct Input
|
||||||
|
{
|
||||||
|
[builtin(global_invocation_indices)] indices: vec3[u32]
|
||||||
|
}
|
||||||
|
|
||||||
|
[layout(std430)]
|
||||||
|
struct SSBO
|
||||||
|
{
|
||||||
|
data: dyn_array[u32]
|
||||||
|
}
|
||||||
|
|
||||||
|
external
|
||||||
|
{
|
||||||
|
[set(0), binding(0)] read_ssbo: storage[SSBO, readonly],
|
||||||
|
[set(0), binding(1)] read_texture: texture2D[f32, readonly, rgba8],
|
||||||
|
[set(1), binding(0)] write_ssbo: storage[SSBO, writeonly],
|
||||||
|
[set(1), binding(1)] write_texture: texture2D[f32, readonly, rgba8],
|
||||||
|
}
|
||||||
|
|
||||||
|
[entry(compute)]
|
||||||
|
[workgroup(32, 32, 1)]
|
||||||
|
fn main(input: Input)
|
||||||
|
{
|
||||||
|
}
|
||||||
13
Tests/Vulkan/Shaders/Simple.nzsl
git.filemode.normal_file
13
Tests/Vulkan/Shaders/Simple.nzsl
git.filemode.normal_file
@@ -0,0 +1,13 @@
|
|||||||
|
[nzsl_version("1.0")]
|
||||||
|
module;
|
||||||
|
|
||||||
|
struct Input
|
||||||
|
{
|
||||||
|
[builtin(global_invocation_indices)] indices: vec3[u32]
|
||||||
|
}
|
||||||
|
|
||||||
|
[entry(compute)]
|
||||||
|
[workgroup(32, 32, 1)]
|
||||||
|
fn main(input: Input)
|
||||||
|
{
|
||||||
|
}
|
||||||
25
Tests/Vulkan/Shaders/WriteOnlyBindings.nzsl
git.filemode.normal_file
25
Tests/Vulkan/Shaders/WriteOnlyBindings.nzsl
git.filemode.normal_file
@@ -0,0 +1,25 @@
|
|||||||
|
[nzsl_version("1.0")]
|
||||||
|
module;
|
||||||
|
|
||||||
|
struct Input
|
||||||
|
{
|
||||||
|
[builtin(global_invocation_indices)] indices: vec3[u32]
|
||||||
|
}
|
||||||
|
|
||||||
|
[layout(std430)]
|
||||||
|
struct SSBO
|
||||||
|
{
|
||||||
|
data: dyn_array[u32]
|
||||||
|
}
|
||||||
|
|
||||||
|
external
|
||||||
|
{
|
||||||
|
[set(1), binding(0)] write_ssbo: storage[SSBO],
|
||||||
|
[set(1), binding(1)] write_texture: texture2D[f32, readonly, rgba8],
|
||||||
|
}
|
||||||
|
|
||||||
|
[entry(compute)]
|
||||||
|
[workgroup(32, 32, 1)]
|
||||||
|
fn main(input: Input)
|
||||||
|
{
|
||||||
|
}
|
||||||
@@ -7,6 +7,7 @@ extern void TestBackend();
|
|||||||
extern void TestDevice();
|
extern void TestDevice();
|
||||||
extern void TestBuffer();
|
extern void TestBuffer();
|
||||||
extern void TestImage();
|
extern void TestImage();
|
||||||
|
extern void TestPipeline();
|
||||||
|
|
||||||
int main(void)
|
int main(void)
|
||||||
{
|
{
|
||||||
@@ -15,5 +16,6 @@ int main(void)
|
|||||||
TestDevice();
|
TestDevice();
|
||||||
TestBuffer();
|
TestBuffer();
|
||||||
TestImage();
|
TestImage();
|
||||||
|
TestPipeline();
|
||||||
return UNITY_END();
|
return UNITY_END();
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user