[cb01aff] | 1 | #ifndef _VULKAN_UTILS_H
|
---|
| 2 | #define _VULKAN_UTILS_H
|
---|
| 3 |
|
---|
[90a424f] | 4 | #include <optional>
|
---|
[8b823e7] | 5 | #include <sstream>
|
---|
| 6 | #include <stdexcept>
|
---|
[b794178] | 7 | #include <string>
|
---|
[cb01aff] | 8 | #include <vector>
|
---|
| 9 |
|
---|
| 10 | #include <vulkan/vulkan.h>
|
---|
| 11 |
|
---|
[7f60b28] | 12 | // TODO: Ideally, vulkan-utils should not have things speciic to windowing apis (glfw, sdl, sfml, etc.).
|
---|
| 13 | // Check what these inclydes are for and if that functionality can be moved
|
---|
[b794178] | 14 | #include <SDL2/SDL.h>
|
---|
| 15 | #include <SDL2/SDL_vulkan.h>
|
---|
| 16 |
|
---|
[cb01aff] | 17 | using namespace std;
|
---|
| 18 |
|
---|
[90a424f] | 19 | struct QueueFamilyIndices {
|
---|
| 20 | optional<uint32_t> graphicsFamily;
|
---|
| 21 | optional<uint32_t> presentFamily;
|
---|
| 22 |
|
---|
| 23 | bool isComplete() {
|
---|
| 24 | return graphicsFamily.has_value() && presentFamily.has_value();
|
---|
| 25 | }
|
---|
| 26 | };
|
---|
| 27 |
|
---|
[b794178] | 28 | struct VulkanImage {
|
---|
| 29 | VkImage image;
|
---|
| 30 | VkDeviceMemory imageMemory;
|
---|
| 31 | VkImageView imageView;
|
---|
| 32 | };
|
---|
| 33 |
|
---|
[cb01aff] | 34 | class VulkanUtils {
|
---|
| 35 | public:
|
---|
[8b823e7] | 36 | static string resultString(VkResult result);
|
---|
| 37 |
|
---|
[cb01aff] | 38 | static bool checkValidationLayerSupport(const vector<const char*> &validationLayers);
|
---|
| 39 |
|
---|
| 40 | static VkResult createDebugUtilsMessengerEXT(VkInstance instance,
|
---|
[e8445f0] | 41 | const VkDebugUtilsMessengerCreateInfoEXT* pCreateInfo,
|
---|
| 42 | const VkAllocationCallbacks* pAllocator,
|
---|
| 43 | VkDebugUtilsMessengerEXT* pDebugMessenger);
|
---|
[cb01aff] | 44 |
|
---|
| 45 | static void destroyDebugUtilsMessengerEXT(VkInstance instance,
|
---|
[e8445f0] | 46 | VkDebugUtilsMessengerEXT debugMessenger,
|
---|
| 47 | const VkAllocationCallbacks* pAllocator);
|
---|
[90a424f] | 48 |
|
---|
[fe5c3ba] | 49 | static QueueFamilyIndices findQueueFamilies(VkPhysicalDevice physicalDevice, VkSurfaceKHR surface);
|
---|
[b794178] | 50 | static bool checkDeviceExtensionSupport(VkPhysicalDevice physicalDevice,
|
---|
[e8445f0] | 51 | const vector<const char*>& deviceExtensions);
|
---|
[7f60b28] | 52 | static VkSurfaceCapabilitiesKHR querySwapChainCapabilities(VkPhysicalDevice physicalDevice,
|
---|
[e8445f0] | 53 | VkSurfaceKHR surface);
|
---|
[7f60b28] | 54 | static vector<VkSurfaceFormatKHR> querySwapChainFormats(VkPhysicalDevice physicalDevice, VkSurfaceKHR surface);
|
---|
| 55 | static vector<VkPresentModeKHR> querySwapChainPresentModes(VkPhysicalDevice physicalDevice,
|
---|
[e8445f0] | 56 | VkSurfaceKHR surface);
|
---|
[7f60b28] | 57 | static VkSurfaceFormatKHR chooseSwapSurfaceFormat(const vector<VkSurfaceFormatKHR>& availableFormats,
|
---|
[e8445f0] | 58 | const vector<VkFormat>& requestedFormats, VkColorSpaceKHR requestedColorSpace);
|
---|
[7f60b28] | 59 | static VkPresentModeKHR chooseSwapPresentMode(const vector<VkPresentModeKHR>& availablePresentModes,
|
---|
[e8445f0] | 60 | const vector<VkPresentModeKHR>& requestedPresentModes);
|
---|
[502bd0b] | 61 | static VkExtent2D chooseSwapExtent(const VkSurfaceCapabilitiesKHR& capabilities, int width, int height);
|
---|
[b794178] | 62 | static VkImageView createImageView(VkDevice device, VkImage image, VkFormat format,
|
---|
[e8445f0] | 63 | VkImageAspectFlags aspectFlags);
|
---|
[6fc24c7] | 64 | static VkFormat findSupportedFormat(VkPhysicalDevice physicalDevice, const vector<VkFormat>& candidates,
|
---|
[e8445f0] | 65 | VkImageTiling tiling, VkFormatFeatureFlags features);
|
---|
[b794178] | 66 | static void createBuffer(VkDevice device, VkPhysicalDevice physicalDevice, VkDeviceSize size,
|
---|
[e8445f0] | 67 | VkBufferUsageFlags usage, VkMemoryPropertyFlags properties, VkBuffer& buffer,
|
---|
| 68 | VkDeviceMemory& bufferMemory);
|
---|
[b794178] | 69 | static uint32_t findMemoryType(VkPhysicalDevice physicalDevice, uint32_t typeFilter,
|
---|
[e8445f0] | 70 | VkMemoryPropertyFlags properties);
|
---|
[b794178] | 71 |
|
---|
| 72 | static void createVulkanImageFromFile(VkDevice device, VkPhysicalDevice physicalDevice,
|
---|
[e8445f0] | 73 | VkCommandPool commandPool, string filename, VulkanImage& image,
|
---|
| 74 | VkQueue graphicsQueue);
|
---|
[b794178] | 75 | static void createVulkanImageFromSDLTexture(VkDevice device, VkPhysicalDevice physicalDevice,
|
---|
[e8445f0] | 76 | SDL_Texture* texture, VulkanImage& image);
|
---|
[d2d9286] | 77 | static void populateVulkanImageFromSDLTexture(VkDevice device, VkPhysicalDevice physicalDevice,
|
---|
[e8445f0] | 78 | VkCommandPool commandPool, SDL_Texture* texture,
|
---|
| 79 | SDL_Renderer* renderer, VulkanImage& image, VkQueue graphicsQueue);
|
---|
[603b5bc] | 80 | static void createDepthImage(VkDevice device, VkPhysicalDevice physicalDevice, VkCommandPool commandPool,
|
---|
[e8445f0] | 81 | VkFormat depthFormat, VkExtent2D extent, VulkanImage& image, VkQueue graphicsQueue);
|
---|
[b794178] | 82 | static void createImage(VkDevice device, VkPhysicalDevice physicalDevice, uint32_t width, uint32_t height,
|
---|
[e8445f0] | 83 | VkFormat format, VkImageTiling tiling, VkImageUsageFlags usage,
|
---|
| 84 | VkMemoryPropertyFlags properties, VulkanImage& image);
|
---|
[b794178] | 85 |
|
---|
| 86 | static void transitionImageLayout(VkDevice device, VkCommandPool commandPool, VkImage image,
|
---|
[e8445f0] | 87 | VkFormat format, VkImageLayout oldLayout, VkImageLayout newLayout,
|
---|
| 88 | VkQueue graphicsQueue);
|
---|
[b794178] | 89 | static VkCommandBuffer beginSingleTimeCommands(VkDevice device, VkCommandPool commandPool);
|
---|
| 90 | static void endSingleTimeCommands(VkDevice device, VkCommandPool commandPool,
|
---|
[e8445f0] | 91 | VkCommandBuffer commandBuffer, VkQueue graphicsQueue);
|
---|
[b794178] | 92 | static void copyBufferToImage(VkDevice device, VkCommandPool commandPool, VkBuffer buffer, VkImage image,
|
---|
[e8445f0] | 93 | uint32_t width, uint32_t height, VkQueue graphicsQueue);
|
---|
[b794178] | 94 |
|
---|
[e3bef3a] | 95 | template<class DataType>
|
---|
| 96 | static void copyDataToBuffer(VkDevice device, VkPhysicalDevice physicalDevice, VkCommandPool commandPool,
|
---|
[e8445f0] | 97 | const vector<DataType>& srcData, VkBuffer dstBuffer, size_t dstVertexOffset,
|
---|
| 98 | VkQueue graphicsQueue);
|
---|
[e3bef3a] | 99 |
|
---|
[87c8f1a] | 100 | static void copyBuffer(VkDevice device, VkCommandPool commandPool, VkBuffer srcBuffer, VkBuffer dstBuffer,
|
---|
[e8445f0] | 101 | VkDeviceSize srcOffset, VkDeviceSize dstOffset, VkDeviceSize size,
|
---|
| 102 | VkQueue graphicsQueue);
|
---|
[87c8f1a] | 103 |
|
---|
[8e02b6b] | 104 | template<class DataType>
|
---|
[e8445f0] | 105 | static void copyDataToMemory(VkDevice device, const DataType& srcData, VkDeviceMemory bufferMemory,
|
---|
| 106 | VkDeviceSize offset);
|
---|
| 107 |
|
---|
| 108 | template<class DataType>
|
---|
| 109 | static void copyDataToMemory(VkDevice device, const DataType& srcData, VkDeviceMemory bufferMemory,
|
---|
| 110 | VkDeviceSize offset, VkDeviceSize size);
|
---|
[8e02b6b] | 111 |
|
---|
[b794178] | 112 | static bool hasStencilComponent(VkFormat format);
|
---|
[e83b155] | 113 |
|
---|
| 114 | static void destroyVulkanImage(VkDevice& device, VulkanImage& image);
|
---|
[cb01aff] | 115 | };
|
---|
| 116 |
|
---|
[e3bef3a] | 117 | template<class DataType>
|
---|
| 118 | void VulkanUtils::copyDataToBuffer(VkDevice device, VkPhysicalDevice physicalDevice, VkCommandPool commandPool,
|
---|
[996dd3e] | 119 | const vector<DataType>& srcData, VkBuffer dstBuffer, size_t dstVertexOffset,
|
---|
| 120 | VkQueue graphicsQueue) {
|
---|
[e3bef3a] | 121 | VkDeviceSize srcDataSize = srcData.size() * sizeof(DataType);
|
---|
| 122 |
|
---|
| 123 | VkBuffer stagingBuffer;
|
---|
| 124 | VkDeviceMemory stagingBufferMemory;
|
---|
| 125 | createBuffer(device, physicalDevice, srcDataSize, VK_BUFFER_USAGE_TRANSFER_SRC_BIT,
|
---|
| 126 | VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT,
|
---|
| 127 | stagingBuffer, stagingBufferMemory);
|
---|
| 128 |
|
---|
| 129 | void* data;
|
---|
| 130 | vkMapMemory(device, stagingBufferMemory, 0, srcDataSize, 0, &data);
|
---|
| 131 | memcpy(data, srcData.data(), (size_t)srcDataSize);
|
---|
| 132 | vkUnmapMemory(device, stagingBufferMemory);
|
---|
| 133 |
|
---|
| 134 | copyBuffer(device, commandPool, stagingBuffer, dstBuffer, 0, dstVertexOffset * sizeof(DataType), srcDataSize,
|
---|
| 135 | graphicsQueue);
|
---|
| 136 |
|
---|
| 137 | vkDestroyBuffer(device, stagingBuffer, nullptr);
|
---|
| 138 | vkFreeMemory(device, stagingBufferMemory, nullptr);
|
---|
| 139 | }
|
---|
| 140 |
|
---|
[8e02b6b] | 141 | template<class DataType>
|
---|
[e8445f0] | 142 | void VulkanUtils::copyDataToMemory(VkDevice device, const DataType& srcData, VkDeviceMemory bufferMemory,
|
---|
[996dd3e] | 143 | VkDeviceSize offset) {
|
---|
[e8445f0] | 144 | copyDataToMemory(device, srcData, bufferMemory, offset, sizeof(DataType));
|
---|
| 145 | }
|
---|
| 146 |
|
---|
[996dd3e] | 147 | // TODO: This would be used when the GPU memory is host-coherent. If it it not, I also need to use vkFlushMappedMemoryRanges
|
---|
| 148 | // I should create a variant that supports non-coherent memory
|
---|
[e8445f0] | 149 | template<class DataType>
|
---|
| 150 | void VulkanUtils::copyDataToMemory(VkDevice device, const DataType& srcData, VkDeviceMemory bufferMemory,
|
---|
| 151 | VkDeviceSize offset, VkDeviceSize size) {
|
---|
[8e02b6b] | 152 | void* data;
|
---|
[5a1ace0] | 153 |
|
---|
[e8445f0] | 154 | vkMapMemory(device, bufferMemory, offset, size, 0, &data);
|
---|
| 155 | memcpy(data, &srcData, size);
|
---|
[8e02b6b] | 156 | vkUnmapMemory(device, bufferMemory);
|
---|
| 157 | }
|
---|
| 158 |
|
---|
[e8445f0] | 159 | // TODO: Use this in vulkan-utils itself as well
|
---|
[8b823e7] | 160 | #define VKUTIL_CHECK_RESULT(f, msg) { \
|
---|
| 161 | VkResult res = (f); \
|
---|
| 162 | \
|
---|
| 163 | if (res != VK_SUCCESS) { \
|
---|
| 164 | ostringstream oss; \
|
---|
| 165 | oss << msg << " VkResult is \"" << VulkanUtils::resultString(res) << "\" in " << __FILE__ << " at line " << __LINE__;\
|
---|
| 166 | \
|
---|
| 167 | if (res < 0) { \
|
---|
| 168 | throw runtime_error("Fatal: " + oss.str()); \
|
---|
| 169 | } else { \
|
---|
| 170 | cerr << oss.str(); \
|
---|
| 171 | } \
|
---|
| 172 | } \
|
---|
| 173 | }
|
---|
| 174 |
|
---|
[cb01aff] | 175 | #endif // _VULKAN_UTILS_H |
---|