159 lines
7.9 KiB
C
159 lines
7.9 KiB
C
#include <vulkan/vulkan.h>
|
|
#include <output_device/vulkan_physical_device.h>
|
|
#include <output_device/vulkan_output_devices.h>
|
|
#include "vulkan_device_extensions.h"
|
|
#include "instance/gryphn_instance.h"
|
|
#include "commands/command_buffer/vulkan_command_buffer.h"
|
|
#include "instance/vulkan_instance.h"
|
|
#include "vulkan_result_converter.h"
|
|
#include "string.h"
|
|
|
|
#include "stdio.h"
|
|
|
|
gnReturnCode createVulkanOutputDevice(gnInstanceHandle instance, gnOutputDeviceHandle device, gnOutputDeviceInfo deviceInfo) {
|
|
device->outputDevice = malloc(sizeof(gnPlatformOutputDevice));
|
|
device->outputDevice->physicalDevice = deviceInfo.physicalDevice->physicalDevice->device;
|
|
|
|
int createQueueCount = 0;
|
|
VkDeviceQueueCreateInfo* queueCreateInfos = NULL;
|
|
|
|
if (!instance->enabledExtensions[GN_EXT_QUEUES]) {
|
|
queueCreateInfos = malloc(sizeof(VkDeviceQueueCreateInfo) * deviceInfo.physicalDevice->physicalDevice->neededQueueCount);
|
|
createQueueCount = deviceInfo.physicalDevice->physicalDevice->neededQueueCount;
|
|
float queuePriority = 1.0f;
|
|
for (uint32_t i = 0; i < deviceInfo.physicalDevice->physicalDevice->neededQueueCount; i++) {
|
|
queueCreateInfos[i].sType = VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO;
|
|
queueCreateInfos[i].pNext = NULL;
|
|
queueCreateInfos[i].flags = 0;
|
|
queueCreateInfos[i].queueCount = 1;
|
|
queueCreateInfos[i].queueFamilyIndex = deviceInfo.physicalDevice->physicalDevice->neededQueues[i].queueIndex;
|
|
queueCreateInfos[i].pQueuePriorities = &queuePriority;
|
|
}
|
|
} else {
|
|
createQueueCount = deviceInfo.queueInfoCount;
|
|
queueCreateInfos = malloc(sizeof(VkDeviceQueueCreateInfo) * deviceInfo.queueInfoCount);
|
|
for (uint32_t i = 0; i < deviceInfo.queueInfoCount; i++) {
|
|
queueCreateInfos[i].sType = VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO;
|
|
queueCreateInfos[i].flags = 0;
|
|
queueCreateInfos[i].pNext = NULL;
|
|
queueCreateInfos[i].queueCount = deviceInfo.queueInfos[i].queueCount;
|
|
queueCreateInfos[i].queueFamilyIndex = deviceInfo.queueInfos[i].queueFamilyIndex;
|
|
queueCreateInfos[i].pQueuePriorities = deviceInfo.queueInfos[i].queuePrioritys;
|
|
}
|
|
}
|
|
|
|
VkPhysicalDeviceFeatures deviceFeatures = {
|
|
.samplerAnisotropy = VK_TRUE
|
|
};
|
|
|
|
VkDeviceCreateInfo deviceCreateInfo = {
|
|
.sType = VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO,
|
|
.flags = 0,
|
|
.pNext = NULL,
|
|
.queueCreateInfoCount = createQueueCount,
|
|
.pQueueCreateInfos = queueCreateInfos,
|
|
.pEnabledFeatures = &deviceFeatures
|
|
};
|
|
deviceCreateInfo.ppEnabledExtensionNames = vkGetGryphnDeviceExtensions(&deviceCreateInfo.enabledExtensionCount, deviceInfo.physicalDevice->physicalDevice->device);
|
|
|
|
device->outputDevice->enabledOversizedDescriptorPools = GN_FALSE;
|
|
for (uint32_t i = 0; i < deviceCreateInfo.enabledExtensionCount; i++)
|
|
if (strcmp(deviceCreateInfo.ppEnabledExtensionNames[i], VK_NV_DESCRIPTOR_POOL_OVERALLOCATION_EXTENSION_NAME) == 0) device->outputDevice->enabledOversizedDescriptorPools = GN_TRUE;
|
|
|
|
if (instance->hasDebugger)
|
|
deviceCreateInfo.enabledLayerCount = 0;
|
|
else {
|
|
const char* validation_layers[1] = { "VK_LAYER_KHRONOS_validation" };
|
|
deviceCreateInfo.enabledLayerCount = 1;
|
|
deviceCreateInfo.ppEnabledLayerNames = validation_layers;
|
|
}
|
|
|
|
VkResult result = vkCreateDevice(deviceInfo.physicalDevice->physicalDevice->device, &deviceCreateInfo, NULL, &device->outputDevice->device);
|
|
if (result != VK_SUCCESS)
|
|
return VkResultToGnReturnCode(result);
|
|
|
|
device->outputDevice->queues = malloc(sizeof(vulkanQueue) * deviceInfo.physicalDevice->physicalDevice->neededQueueCount);
|
|
uint32_t transferQueue = 0;
|
|
gnBool foundTransferQueue = GN_FALSE, foundGraphicsQueue = GN_FALSE;
|
|
for (uint32_t i = 0; i < deviceInfo.physicalDevice->physicalDevice->neededQueueCount; i++) {
|
|
device->outputDevice->queues[i].queueInfo = deviceInfo.physicalDevice->physicalDevice->neededQueues[i];
|
|
|
|
vkGetDeviceQueue(device->outputDevice->device, deviceInfo.physicalDevice->physicalDevice->neededQueues[i].queueIndex, 0, &device->outputDevice->queues[i].queue);
|
|
if ((device->outputDevice->queues[i].queueInfo.createFlags & VK_QUEUE_TRANSFER_BIT) == VK_QUEUE_TRANSFER_BIT && !foundTransferQueue) {
|
|
device->outputDevice->transferQueueIndex = i;
|
|
transferQueue = device->outputDevice->queues[i].queueInfo.queueIndex;
|
|
foundTransferQueue = GN_TRUE;
|
|
}
|
|
|
|
if ((device->outputDevice->queues[i].queueInfo.createFlags & VK_QUEUE_GRAPHICS_BIT) == VK_QUEUE_GRAPHICS_BIT && !foundGraphicsQueue) {
|
|
device->outputDevice->graphicsQueueIndex = i;
|
|
foundGraphicsQueue = GN_FALSE;
|
|
}
|
|
}
|
|
|
|
VkCommandPoolCreateInfo poolInfo = {
|
|
.sType = VK_STRUCTURE_TYPE_COMMAND_POOL_CREATE_INFO,
|
|
.flags = VK_COMMAND_POOL_CREATE_RESET_COMMAND_BUFFER_BIT,
|
|
.queueFamilyIndex = transferQueue,
|
|
.pNext = NULL
|
|
};
|
|
|
|
VkResult command_pool_result = vkCreateCommandPool(device->outputDevice->device, &poolInfo, NULL, &device->outputDevice->transferCommandPool);
|
|
if (command_pool_result != VK_SUCCESS)
|
|
return VkResultToGnReturnCode(command_pool_result);
|
|
|
|
VkFenceCreateInfo fenceInfo = { .sType = VK_STRUCTURE_TYPE_FENCE_CREATE_INFO };
|
|
VkResult fence_result = vkCreateFence(device->outputDevice->device, &fenceInfo, NULL, &device->outputDevice->barrierFence);
|
|
if (fence_result != VK_SUCCESS) VkResultToGnReturnCode(fence_result);
|
|
|
|
VmaAllocatorCreateInfo allocatorCreateInfo = {
|
|
.device = device->outputDevice->device,
|
|
.physicalDevice = device->outputDevice->physicalDevice,
|
|
.instance = instance->instance->vk_instance,
|
|
};
|
|
if (vmaCreateAllocator(&allocatorCreateInfo, &device->outputDevice->allocator) != VK_SUCCESS)
|
|
return GN_FAILED_CREATE_ALLOCATOR;
|
|
|
|
// create the massive staging buffer
|
|
device->outputDevice->stagingBufferSize = 128 * 1024 * 1024;
|
|
|
|
VkBufferCreateInfo bufferCreateInfo = {
|
|
.sType = VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO,
|
|
.pNext = NULL,
|
|
.flags = 0,
|
|
.queueFamilyIndexCount = 0,
|
|
.size = device->outputDevice->stagingBufferSize,
|
|
.sharingMode = VK_SHARING_MODE_EXCLUSIVE,
|
|
.usage = VK_BUFFER_USAGE_TRANSFER_SRC_BIT
|
|
};
|
|
|
|
VmaAllocationCreateInfo bufferAllocationInfo = {
|
|
.usage = VMA_MEMORY_USAGE_CPU_ONLY,
|
|
.flags = VMA_ALLOCATION_CREATE_HOST_ACCESS_SEQUENTIAL_WRITE_BIT,
|
|
.requiredFlags = VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT,
|
|
};
|
|
|
|
return VkResultToGnReturnCode(vmaCreateBuffer(device->outputDevice->allocator, &bufferCreateInfo, &bufferAllocationInfo, &device->outputDevice->stagingBuffer.buffer, &device->outputDevice->stagingBuffer.allocation, NULL)); // lowkey is a hack
|
|
}
|
|
|
|
void waitForDevice(const gnOutputDeviceHandle device) {
|
|
vkDeviceWaitIdle(device->outputDevice->device);
|
|
}
|
|
|
|
void destroyVulkanOutputDevice(gnOutputDeviceHandle device) {
|
|
vkDestroyFence(device->outputDevice->device, device->outputDevice->barrierFence, NULL);
|
|
vkDestroyCommandPool(device->outputDevice->device, device->outputDevice->transferCommandPool, NULL);
|
|
vmaDestroyBuffer(device->outputDevice->allocator, device->outputDevice->stagingBuffer.buffer, device->outputDevice->stagingBuffer.allocation);
|
|
vmaDestroyAllocator(device->outputDevice->allocator);
|
|
vkDestroyDevice(device->outputDevice->device, NULL);
|
|
free(device->outputDevice);
|
|
}
|
|
|
|
|
|
VkCommandBuffer gnBeginVulkanTransferOperation(gnDevice device) {
|
|
return VkBeginTransferOperation(device->outputDevice->device, device->outputDevice->transferCommandPool);
|
|
}
|
|
void gnEndVulkanTransferOperation(gnDevice device, VkCommandBuffer buffer) {
|
|
VkEndTransferOperation(buffer, device->outputDevice->transferCommandPool, device->outputDevice->queues[device->outputDevice->transferQueueIndex].queue, device->outputDevice->device);
|
|
}
|