#include #include #include #include #include #include #define GLFW_INCLUDE_VULKAN #include #ifdef NDEBUG const bool enableValidationLayers = false; #else const bool enableValidationLayers = true; #endif static VKAPI_ATTR VkBool32 VKAPI_CALL debugCallback(VkDebugUtilsMessageSeverityFlagBitsEXT messageSeverity, VkDebugUtilsMessageTypeFlagsEXT messageType, const VkDebugUtilsMessengerCallbackDataEXT *pCallbackData, void *pUserData) { // TODO: Put all messages into a log file if (messageSeverity >= VK_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT) { fprintf(stderr, "validation layer: %s\n", pCallbackData->pMessage); } return VK_FALSE; } const char *validationLayers[] = {"VK_LAYER_KHRONOS_validation"}; size_t validationLayerCount = sizeof(validationLayers) / sizeof(validationLayers[0]); typedef struct Application { GLFWwindow *window; VkInstance instance; VkDebugUtilsMessengerEXT debugMessenger; } Application; bool checkValidationLayerSupport() { uint32_t layerCount; vkEnumerateInstanceLayerProperties(&layerCount, NULL); VkLayerProperties availableLayers[layerCount]; vkEnumerateInstanceLayerProperties(&layerCount, availableLayers); for (int i = 0; i < validationLayerCount; i++) { bool layerFound = false; for (int j = 0; j < layerCount; j++) { if (strcmp(availableLayers[j].layerName, validationLayers[i]) == 0) { layerFound = true; break; } } if (!layerFound) { return false; } } return true; } bool verifyExtensionSupport(uint32_t extensionCount, VkExtensionProperties *extentions, uint32_t glfwExtensionCount, const char **glfwExtensions) { for (uint32_t i = 0; i < glfwExtensionCount; i++) { bool layerFound = false; for (uint32_t j = 0; j < extensionCount; j++) { if (strcmp(glfwExtensions[i], extentions[j].extensionName) == 0) { layerFound = true; break; } } if (!layerFound) { fprintf(stderr, "Missing %s vulkan extention\n", glfwExtensions[i]); return false; } } return true; } void populateDebugMessengerCreateInfo( VkDebugUtilsMessengerCreateInfoEXT *createInfo) { createInfo->sType = VK_STRUCTURE_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT; createInfo->messageSeverity = VK_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT | VK_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT | VK_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT; createInfo->messageType = VK_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT | VK_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT | VK_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT; createInfo->pfnUserCallback = debugCallback; createInfo->pUserData = NULL; // Optional } void createInstance(Application *app) { if (enableValidationLayers && !checkValidationLayerSupport()) { fprintf(stderr, "Validation layers requested, but not available!\n"); exit(EXIT_FAILURE); } VkApplicationInfo appInfo = { .sType = VK_STRUCTURE_TYPE_APPLICATION_INFO, .pApplicationName = "Hello Triangle", .applicationVersion = VK_MAKE_VERSION(1, 0, 0), .pEngineName = "No Engine", .engineVersion = VK_MAKE_VERSION(1, 0, 0), .apiVersion = VK_API_VERSION_1_0, }; uint32_t glfwExtensionCount = 0; const char **glfwExtensions; glfwExtensions = glfwGetRequiredInstanceExtensions(&glfwExtensionCount); const char *glfwExtensionsDebug[glfwExtensionCount + 1]; VkInstanceCreateInfo createInfo = { .sType = VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO, .pApplicationInfo = &appInfo, .enabledExtensionCount = glfwExtensionCount, .ppEnabledExtensionNames = glfwExtensions, .enabledLayerCount = 0, }; uint32_t extensionCount = 0; vkEnumerateInstanceExtensionProperties(NULL, &extensionCount, NULL); VkExtensionProperties extensions[extensionCount]; vkEnumerateInstanceExtensionProperties(NULL, &extensionCount, extensions); VkDebugUtilsMessengerCreateInfoEXT debugCreateInfo = {}; if (enableValidationLayers) { populateDebugMessengerCreateInfo(&debugCreateInfo); createInfo.pNext = (VkDebugUtilsMessengerCreateInfoEXT *)&debugCreateInfo; createInfo.enabledLayerCount = validationLayerCount; createInfo.ppEnabledLayerNames = validationLayers; for (size_t i = 0; i < glfwExtensionCount; i++) { glfwExtensionsDebug[i] = glfwExtensions[i]; } glfwExtensionsDebug[glfwExtensionCount] = VK_EXT_DEBUG_UTILS_EXTENSION_NAME; createInfo.enabledExtensionCount = glfwExtensionCount + 1; createInfo.ppEnabledExtensionNames = glfwExtensionsDebug; if (!verifyExtensionSupport(extensionCount, extensions, glfwExtensionCount + 1, glfwExtensionsDebug)) { fprintf( stderr, "Failed to find all required vulkan extentions for glfw and debug\n"); exit(EXIT_FAILURE); } } else { if (!verifyExtensionSupport(extensionCount, extensions, glfwExtensionCount, glfwExtensions)) { fprintf(stderr, "Failed to find all required vulkan extentions for glfw\n"); exit(EXIT_FAILURE); } } VkResult result = vkCreateInstance(&createInfo, NULL, &app->instance); if (result != VK_SUCCESS) { fprintf(stderr, "Failed to create vulkan instance\n"); exit(EXIT_FAILURE); } printf("Instance Extentions Available:\n"); for (uint i = 0; i < extensionCount; i++) { printf("\t%s\n", extensions[i].extensionName); } } void initWindow(Application *app) { glfwInit(); glfwWindowHint(GLFW_CLIENT_API, GLFW_NO_API); glfwWindowHint(GLFW_RESIZABLE, GLFW_FALSE); app->window = glfwCreateWindow(800, 600, "Vulkan", NULL, NULL); } VkResult CreateDebugUtilsMessengerEXT( VkInstance instance, const VkDebugUtilsMessengerCreateInfoEXT *pCreateInfo, const VkAllocationCallbacks *pAllocator, VkDebugUtilsMessengerEXT *pDebugMessenger) { PFN_vkCreateDebugUtilsMessengerEXT func = (PFN_vkCreateDebugUtilsMessengerEXT)vkGetInstanceProcAddr( instance, "vkCreateDebugUtilsMessengerEXT"); if (func != NULL) { return func(instance, pCreateInfo, pAllocator, pDebugMessenger); } else { return VK_ERROR_EXTENSION_NOT_PRESENT; } } void setupDebugMessenger(Application *app) { VkDebugUtilsMessengerCreateInfoEXT createInfo = {}; populateDebugMessengerCreateInfo(&createInfo); if (CreateDebugUtilsMessengerEXT(app->instance, &createInfo, NULL, &app->debugMessenger) != VK_SUCCESS) { fprintf(stderr, "failed to set up debug messenger!"); } } void DestroyDebugUtilsMessengerEXT(VkInstance instance, VkDebugUtilsMessengerEXT debugMessenger, const VkAllocationCallbacks *pAllocator) { PFN_vkDestroyDebugUtilsMessengerEXT func = (PFN_vkDestroyDebugUtilsMessengerEXT)vkGetInstanceProcAddr( instance, "vkDestroyDebugUtilsMessengerEXT"); if (func != NULL) { func(instance, debugMessenger, pAllocator); } } void initVulkan(Application *app) { if (enableValidationLayers) { setupDebugMessenger(app); } createInstance(app); } void mainLoop(Application *app) { while (!glfwWindowShouldClose(app->window)) { glfwPollEvents(); } } void cleanup(Application *app) { if (enableValidationLayers) { DestroyDebugUtilsMessengerEXT(app->instance, app->debugMessenger, NULL); } vkDestroyInstance(app->instance, NULL); glfwDestroyWindow(app->window); glfwTerminate(); } int main(void) { Application app; initWindow(&app); initVulkan(&app); mainLoop(&app); cleanup(&app); return EXIT_SUCCESS; }