1487
|
1 import std/tables
|
1485
|
2 import std/strutils
|
|
3 import std/logging
|
1486
|
4 import std/os
|
1485
|
5
|
|
6 include ./vkapi
|
|
7
|
1486
|
8 const VULKAN_VERSION = VK_MAKE_API_VERSION(0, 1, 3, 0)
|
|
9
|
1485
|
10 template checkVkResult*(call: untyped) =
|
|
11 when defined(release):
|
|
12 discard call
|
|
13 else:
|
|
14 # yes, a bit cheap, but this is only for nice debug output
|
|
15 var callstr = astToStr(call).replace("\n", "")
|
|
16 while callstr.find(" ") >= 0:
|
|
17 callstr = callstr.replace(" ", " ")
|
|
18 debug "Calling vulkan: ", callstr
|
|
19 let value = call
|
|
20 if value != VK_SUCCESS:
|
|
21 error "Vulkan error: ", astToStr(call), " returned ", $value
|
|
22 raise newException(
|
|
23 Exception, "Vulkan error: " & astToStr(call) & " returned " & $value
|
|
24 )
|
|
25
|
|
26 type SVkInstance* = object
|
|
27 vkInstance: VkInstance
|
1486
|
28 debugMessenger: VkDebugUtilsMessengerEXT
|
1485
|
29
|
|
30 proc `=copy`(a: var SVkInstance, b: SVkInstance) {.error.}
|
|
31
|
|
32 proc `=destroy`(a: SVkInstance) =
|
|
33 if a.vkInstance.pointer != nil:
|
1486
|
34 if a.debugMessenger.pointer != nil:
|
|
35 vkDestroyDebugUtilsMessengerEXT(a.vkInstance, a.debugMessenger, nil)
|
1485
|
36 a.vkInstance.vkDestroyInstance(nil)
|
|
37
|
1486
|
38 proc debugCallback(
|
|
39 messageSeverity: VkDebugUtilsMessageSeverityFlagBitsEXT,
|
|
40 messageTypes: VkDebugUtilsMessageTypeFlagsEXT,
|
|
41 pCallbackData: ptr VkDebugUtilsMessengerCallbackDataEXT,
|
|
42 userData: pointer,
|
|
43 ): VkBool32 {.cdecl.} =
|
|
44 const LOG_LEVEL_MAPPING = {
|
|
45 VK_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT: lvlDebug,
|
|
46 VK_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT: lvlInfo,
|
|
47 VK_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT: lvlWarn,
|
|
48 VK_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT: lvlError,
|
|
49 }.toTable
|
|
50 log LOG_LEVEL_MAPPING[messageSeverity]
|
|
51 if messageSeverity == VK_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT:
|
|
52 # stderr.write getStackTrace()
|
|
53 # stderr.writeLine LOG_LEVEL_MAPPING[messageSeverity], &"{toEnums messageTypes}: {pCallbackData.pMessage}"
|
|
54 let errorMsg = $pCallbackData.pMessage & ": " & getStackTrace()
|
|
55 raise newException(Exception, errorMsg)
|
|
56 return VK_FALSE
|
|
57
|
1485
|
58 proc svkCreateInstance*(
|
|
59 applicationName: string,
|
|
60 enabledLayers: openArray[string] = [],
|
1486
|
61 enabledExtensions: openArray[string] =
|
|
62 if defined(release):
|
|
63 @["VK_KHR_surface"]
|
|
64 else:
|
|
65 @["VK_KHR_surface", "VK_EXT_debug_utils"],
|
1485
|
66 engineName = "semicongine",
|
1486
|
67 withSwapchain = true,
|
1485
|
68 ): SVkInstance =
|
1486
|
69 putEnv("VK_LOADER_LAYERS_ENABLE", "*validation")
|
|
70 putEnv(
|
|
71 "VK_LAYER_ENABLES",
|
|
72 "VALIDATION_CHECK_ENABLE_VENDOR_SPECIFIC_AMD,VALIDATION_CHECK_ENABLE_VENDOR_SPECIFIC_NVIDIA,VK_VALIDATION_FEATURE_ENABLE_SYNCHRONIZATION_VALIDATION_EXTVK_VALIDATION_FEATURE_ENABLE_GPU_ASSISTED_EXT,VK_VALIDATION_FEATURE_ENABLE_SYNCHRONIZATION_VALIDATION_EXT",
|
|
73 )
|
|
74 initVulkanLoader()
|
|
75
|
1485
|
76 let
|
|
77 appinfo = VkApplicationInfo(
|
|
78 pApplicationName: applicationName,
|
|
79 pEngineName: engineName,
|
1486
|
80 apiVersion: VULKAN_VERSION,
|
1485
|
81 )
|
1486
|
82 enabledLayersC = allocCStringArray(enabledLayers)
|
|
83 enabledExtensionsC = allocCStringArray(enabledExtensions)
|
1485
|
84 createinfo = VkInstanceCreateInfo(
|
|
85 pApplicationInfo: addr appinfo,
|
|
86 enabledLayerCount: enabledLayers.len.uint32,
|
1486
|
87 ppEnabledLayerNames: enabledLayersC,
|
1485
|
88 enabledExtensionCount: enabledExtensions.len.uint32,
|
1486
|
89 ppEnabledExtensionNames: enabledExtensionsC,
|
1485
|
90 )
|
|
91 checkVkResult vkCreateInstance(addr createinfo, nil, addr result.vkInstance)
|
1486
|
92
|
|
93 enabledLayersC.deallocCStringArray()
|
|
94 enabledExtensionsC.deallocCStringArray()
|
|
95
|
|
96 load_VK_VERSION_1_0(result.vkInstance)
|
|
97 load_VK_VERSION_1_1(result.vkInstance)
|
|
98 load_VK_VERSION_1_2(result.vkInstance)
|
|
99 load_VK_VERSION_1_3(result.vkInstance)
|
|
100
|
|
101 for extension in enabledExtensions:
|
|
102 loadExtension(result.vkInstance, extension)
|
|
103 if withSwapchain:
|
|
104 load_VK_KHR_swapchain(result.vkInstance)
|
|
105
|
|
106 when not defined(release):
|
|
107 var debugMessengerCreateInfo = VkDebugUtilsMessengerCreateInfoEXT(
|
|
108 messageSeverity: VkDebugUtilsMessageSeverityFlagBitsEXT.items.toSeq.toBits,
|
|
109 messageType: VkDebugUtilsMessageTypeFlagBitsEXT.items.toSeq.toBits,
|
|
110 pfnUserCallback: debugCallback,
|
|
111 )
|
|
112 checkVkResult vkCreateDebugUtilsMessengerEXT(
|
|
113 result.vkInstance, addr debugMessengerCreateInfo, nil, addr result.debugMessenger
|
|
114 )
|