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