main.c 81 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258
  1. /* nuklear - 1.32.0 - public domain */
  2. #include <limits.h>
  3. #include <stdbool.h>
  4. #include <stddef.h>
  5. #include <stdint.h>
  6. #include <stdio.h>
  7. #include <time.h>
  8. #define GLFW_INCLUDE_VULKAN
  9. #include <GLFW/glfw3.h>
  10. #define NK_INCLUDE_FIXED_TYPES
  11. #define NK_INCLUDE_STANDARD_IO
  12. #define NK_INCLUDE_STANDARD_VARARGS
  13. #define NK_INCLUDE_DEFAULT_ALLOCATOR
  14. #define NK_INCLUDE_VERTEX_BUFFER_OUTPUT
  15. #define NK_INCLUDE_FONT_BAKING
  16. #define NK_INCLUDE_DEFAULT_FONT
  17. #define NK_IMPLEMENTATION
  18. #define NK_GLFW_VULKAN_IMPLEMENTATION
  19. #include "../../nuklear.h"
  20. #include "nuklear_glfw_vulkan.h"
  21. #define WINDOW_WIDTH 1200
  22. #define WINDOW_HEIGHT 800
  23. #define MAX_VERTEX_BUFFER 512 * 1024
  24. #define MAX_ELEMENT_BUFFER 128 * 1024
  25. /* ===============================================================
  26. *
  27. * EXAMPLE
  28. *
  29. * ===============================================================*/
  30. /* This are some code examples to provide a small overview of what can be
  31. * done with this library. To try out an example uncomment the defines */
  32. /*#define INCLUDE_ALL */
  33. /*#define INCLUDE_STYLE */
  34. /*#define INCLUDE_CALCULATOR */
  35. /*#define INCLUDE_CANVAS */
  36. #define INCLUDE_OVERVIEW
  37. /*#define INCLUDE_CONFIGURATOR */
  38. /*#define INCLUDE_NODE_EDITOR */
  39. #ifdef INCLUDE_ALL
  40. #define INCLUDE_STYLE
  41. #define INCLUDE_CALCULATOR
  42. #define INCLUDE_CANVAS
  43. #define INCLUDE_OVERVIEW
  44. #define INCLUDE_CONFIGURATOR
  45. #define INCLUDE_NODE_EDITOR
  46. #endif
  47. #ifdef INCLUDE_STYLE
  48. #include "../../demo/common/style.c"
  49. #endif
  50. #ifdef INCLUDE_CALCULATOR
  51. #include "../../demo/common/calculator.c"
  52. #endif
  53. #ifdef INCLUDE_CANVAS
  54. #include "../../demo/common/canvas.c"
  55. #endif
  56. #ifdef INCLUDE_OVERVIEW
  57. #include "../../demo/common/overview.c"
  58. #endif
  59. #ifdef INCLUDE_CONFIGURATOR
  60. #include "../../demo/common/style_configurator.c"
  61. #endif
  62. #ifdef INCLUDE_NODE_EDITOR
  63. #include "../../demo/common/node_editor.c"
  64. #endif
  65. /* ===============================================================
  66. *
  67. * DEMO
  68. *
  69. * ===============================================================*/
  70. static const char *validation_layer_name = "VK_LAYER_KHRONOS_validation";
  71. struct queue_family_indices {
  72. int graphics;
  73. int present;
  74. };
  75. struct swap_chain_support_details {
  76. VkSurfaceCapabilitiesKHR capabilities;
  77. VkSurfaceFormatKHR *formats;
  78. uint32_t formats_len;
  79. VkPresentModeKHR *present_modes;
  80. uint32_t present_modes_len;
  81. };
  82. void swap_chain_support_details_free(
  83. struct swap_chain_support_details *swap_chain_support) {
  84. if (swap_chain_support->formats_len > 0) {
  85. free(swap_chain_support->formats);
  86. swap_chain_support->formats = NULL;
  87. }
  88. if (swap_chain_support->present_modes_len > 0) {
  89. free(swap_chain_support->present_modes);
  90. swap_chain_support->present_modes = NULL;
  91. }
  92. }
  93. struct vulkan_demo {
  94. GLFWwindow *win;
  95. VkInstance instance;
  96. VkDebugUtilsMessengerEXT debug_messenger;
  97. VkSurfaceKHR surface;
  98. VkPhysicalDevice physical_device;
  99. struct queue_family_indices indices;
  100. VkDevice device;
  101. VkQueue graphics_queue;
  102. VkQueue present_queue;
  103. VkSampler sampler;
  104. VkSwapchainKHR swap_chain;
  105. VkImage *swap_chain_images;
  106. uint32_t swap_chain_images_len;
  107. VkImageView *swap_chain_image_views;
  108. VkFormat swap_chain_image_format;
  109. VkExtent2D swap_chain_image_extent;
  110. VkImage *overlay_images;
  111. VkImageView *overlay_image_views;
  112. VkDeviceMemory *overlay_image_memories;
  113. VkRenderPass render_pass;
  114. VkFramebuffer *framebuffers;
  115. VkDescriptorSetLayout descriptor_set_layout;
  116. VkDescriptorPool descriptor_pool;
  117. VkDescriptorSet *descriptor_sets;
  118. VkPipelineLayout pipeline_layout;
  119. VkPipeline pipeline;
  120. VkCommandPool command_pool;
  121. VkCommandBuffer *command_buffers;
  122. VkSemaphore image_available;
  123. VkSemaphore render_finished;
  124. VkImage demo_texture_image;
  125. VkImageView demo_texture_image_view;
  126. VkDeviceMemory demo_texture_memory;
  127. VkFence render_fence;
  128. bool framebuffer_resized;
  129. };
  130. static void glfw_framebuffer_resize_callback(GLFWwindow* window, int width, int height) {
  131. struct vulkan_demo* demo;
  132. (void)width;
  133. (void)height;
  134. demo = glfwGetWindowUserPointer(window);
  135. demo->framebuffer_resized = true;
  136. }
  137. static void glfw_error_callback(int e, const char *d) {
  138. fprintf(stderr, "Error %d: %s\n", e, d);
  139. }
  140. VKAPI_ATTR VkBool32 VKAPI_CALL
  141. vulkan_debug_callback(VkDebugUtilsMessageSeverityFlagBitsEXT message_severity,
  142. VkDebugUtilsMessageTypeFlagsEXT message_type,
  143. const VkDebugUtilsMessengerCallbackDataEXT *callback_data,
  144. void *user_data) {
  145. (void)message_severity;
  146. (void)message_type;
  147. (void)user_data;
  148. fprintf(stderr, "validation layer: %s\n", callback_data->pMessage);
  149. return VK_FALSE;
  150. }
  151. bool check_validation_layer_support() {
  152. uint32_t layer_count;
  153. bool ret = false;
  154. VkResult result;
  155. uint32_t i;
  156. VkLayerProperties *available_layers = NULL;
  157. result = vkEnumerateInstanceLayerProperties(&layer_count, NULL);
  158. if (result != VK_SUCCESS) {
  159. fprintf(stderr, "vkEnumerateInstanceLayerProperties failed: %d\n",
  160. result);
  161. return ret;
  162. }
  163. available_layers = malloc(layer_count * sizeof(VkLayerProperties));
  164. result = vkEnumerateInstanceLayerProperties(&layer_count, available_layers);
  165. if (result != VK_SUCCESS) {
  166. fprintf(stderr, "vkEnumerateInstanceLayerProperties failed: %d\n",
  167. result);
  168. goto cleanup;
  169. }
  170. printf("Available vulkan layers:\n");
  171. for (i = 0; i < layer_count; i++) {
  172. printf(" %s\n", available_layers[i].layerName);
  173. if (strcmp(validation_layer_name, available_layers[i].layerName) == 0) {
  174. ret = true;
  175. break;
  176. }
  177. }
  178. cleanup:
  179. free(available_layers);
  180. return ret;
  181. }
  182. VkResult create_debug_utils_messenger_ext(
  183. VkInstance instance, const VkDebugUtilsMessengerCreateInfoEXT *pCreateInfo,
  184. const VkAllocationCallbacks *pAllocator,
  185. VkDebugUtilsMessengerEXT *pDebugMessenger) {
  186. PFN_vkCreateDebugUtilsMessengerEXT func =
  187. (PFN_vkCreateDebugUtilsMessengerEXT)vkGetInstanceProcAddr(
  188. instance, "vkCreateDebugUtilsMessengerEXT");
  189. if (func != NULL) {
  190. return func(instance, pCreateInfo, pAllocator, pDebugMessenger);
  191. } else {
  192. return VK_ERROR_EXTENSION_NOT_PRESENT;
  193. }
  194. }
  195. bool create_debug_callback(struct vulkan_demo *demo) {
  196. VkResult result;
  197. VkDebugUtilsMessengerCreateInfoEXT create_info;
  198. memset(&create_info, 0, sizeof(VkDebugUtilsMessengerCreateInfoEXT));
  199. create_info.sType = VK_STRUCTURE_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT;
  200. create_info.messageSeverity =
  201. VK_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT |
  202. VK_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT;
  203. create_info.messageType = VK_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT |
  204. VK_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT |
  205. VK_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT;
  206. create_info.pfnUserCallback = vulkan_debug_callback;
  207. result = create_debug_utils_messenger_ext(demo->instance, &create_info,
  208. NULL, &demo->debug_messenger);
  209. if (result != VK_SUCCESS) {
  210. fprintf(stderr, "create_debug_utils_messenger_ext failed %d\n", result);
  211. return false;
  212. }
  213. return true;
  214. }
  215. bool create_instance(struct vulkan_demo *demo) {
  216. uint32_t i;
  217. uint32_t available_instance_extension_count;
  218. VkResult result;
  219. VkExtensionProperties *available_instance_extensions = NULL;
  220. bool ret = false;
  221. VkApplicationInfo app_info;
  222. VkInstanceCreateInfo create_info;
  223. uint32_t glfw_extension_count;
  224. const char **glfw_extensions;
  225. uint32_t enabled_extension_count;
  226. const char **enabled_extensions = NULL;
  227. bool validation_layers_installed;
  228. validation_layers_installed = check_validation_layer_support();
  229. if (!validation_layers_installed) {
  230. fprintf(stdout,
  231. "Couldn't find validation layer %s. Continuing without "
  232. "validation layers.\n",
  233. validation_layer_name);
  234. }
  235. result = vkEnumerateInstanceExtensionProperties(
  236. NULL, &available_instance_extension_count, NULL);
  237. if (result != VK_SUCCESS) {
  238. fprintf(stderr, "vkEnumerateInstanceExtensionProperties failed %d\n",
  239. result);
  240. return ret;
  241. }
  242. available_instance_extensions = malloc(available_instance_extension_count *
  243. sizeof(VkExtensionProperties));
  244. result = vkEnumerateInstanceExtensionProperties(
  245. NULL, &available_instance_extension_count,
  246. available_instance_extensions);
  247. if (result != VK_SUCCESS) {
  248. fprintf(stderr, "vkEnumerateInstanceExtensionProperties failed %d\n",
  249. result);
  250. goto cleanup;
  251. }
  252. printf("available instance extensions:\n");
  253. for (i = 0; i < available_instance_extension_count; i++) {
  254. printf(" %s\n", available_instance_extensions[i].extensionName);
  255. }
  256. glfw_extensions = glfwGetRequiredInstanceExtensions(&glfw_extension_count);
  257. enabled_extension_count = glfw_extension_count;
  258. if (validation_layers_installed) {
  259. enabled_extension_count += 1;
  260. enabled_extensions = malloc(enabled_extension_count * sizeof(char *));
  261. memcpy(enabled_extensions, glfw_extensions,
  262. glfw_extension_count * sizeof(char *));
  263. enabled_extensions[glfw_extension_count] =
  264. VK_EXT_DEBUG_UTILS_EXTENSION_NAME;
  265. } else {
  266. enabled_extensions = malloc(enabled_extension_count * sizeof(char *));
  267. memcpy(enabled_extensions, glfw_extensions,
  268. glfw_extension_count * sizeof(char *));
  269. }
  270. printf("Trying to enable the following instance extensions: ");
  271. for (i = 0; i < enabled_extension_count; i++) {
  272. if (i > 0) {
  273. printf(", ");
  274. }
  275. printf("%s\n", enabled_extensions[i]);
  276. }
  277. printf("\n");
  278. for (i = 0; i < enabled_extension_count; i++) {
  279. int extension_missing = 1;
  280. uint32_t j;
  281. for (j = 0; j < available_instance_extension_count; j++) {
  282. if (strcmp(enabled_extensions[i],
  283. available_instance_extensions[j].extensionName) == 0) {
  284. extension_missing = 0;
  285. break;
  286. }
  287. }
  288. if (extension_missing) {
  289. fprintf(stderr, "Extension %s is missing\n", enabled_extensions[i]);
  290. return ret;
  291. }
  292. }
  293. memset(&app_info, 0, sizeof(VkApplicationInfo));
  294. app_info.sType = VK_STRUCTURE_TYPE_APPLICATION_INFO;
  295. app_info.pApplicationName = "Demo";
  296. app_info.applicationVersion = VK_MAKE_VERSION(1, 0, 0);
  297. app_info.pEngineName = "No Engine";
  298. app_info.engineVersion = VK_MAKE_VERSION(1, 0, 0);
  299. app_info.apiVersion = VK_API_VERSION_1_0;
  300. memset(&create_info, 0, sizeof(VkInstanceCreateInfo));
  301. create_info.sType = VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO;
  302. create_info.pApplicationInfo = &app_info;
  303. create_info.enabledExtensionCount = enabled_extension_count;
  304. create_info.ppEnabledExtensionNames = enabled_extensions;
  305. if (validation_layers_installed) {
  306. create_info.enabledLayerCount = 1;
  307. create_info.ppEnabledLayerNames = &validation_layer_name;
  308. }
  309. result = vkCreateInstance(&create_info, NULL, &demo->instance);
  310. if (result != VK_SUCCESS) {
  311. fprintf(stderr, "vkCreateInstance result %d\n", result);
  312. return ret;
  313. }
  314. if (validation_layers_installed) {
  315. ret = create_debug_callback(demo);
  316. } else {
  317. ret = true;
  318. }
  319. cleanup:
  320. if (available_instance_extensions) {
  321. free(available_instance_extensions);
  322. }
  323. if (enabled_extensions) {
  324. free(enabled_extensions);
  325. }
  326. return ret;
  327. }
  328. bool create_surface(struct vulkan_demo *demo) {
  329. VkResult result;
  330. result = glfwCreateWindowSurface(demo->instance, demo->win, NULL,
  331. &demo->surface);
  332. if (result != VK_SUCCESS) {
  333. fprintf(stderr, "creating vulkan surface failed: %d\n", result);
  334. return false;
  335. }
  336. return true;
  337. }
  338. bool find_queue_families(VkPhysicalDevice physical_device, VkSurfaceKHR surface,
  339. struct queue_family_indices *indices) {
  340. VkResult result;
  341. uint32_t queue_family_count = 0;
  342. uint32_t i = 0;
  343. bool ret = false;
  344. VkQueueFamilyProperties *queue_family_properties;
  345. VkBool32 present_support;
  346. vkGetPhysicalDeviceQueueFamilyProperties(physical_device,
  347. &queue_family_count, NULL);
  348. queue_family_properties =
  349. malloc(queue_family_count * sizeof(VkQueueFamilyProperties));
  350. vkGetPhysicalDeviceQueueFamilyProperties(
  351. physical_device, &queue_family_count, queue_family_properties);
  352. for (i = 0; i < queue_family_count; i++) {
  353. if (queue_family_properties[i].queueCount == 0) {
  354. continue;
  355. }
  356. if (queue_family_properties[i].queueFlags & VK_QUEUE_GRAPHICS_BIT) {
  357. indices->graphics = i;
  358. }
  359. result = vkGetPhysicalDeviceSurfaceSupportKHR(
  360. physical_device, i, surface, &present_support);
  361. if (result != VK_SUCCESS) {
  362. fprintf(stderr,
  363. "vkGetPhysicalDeviceSurfaceSupportKHR failed with %d\n",
  364. result);
  365. goto cleanup;
  366. }
  367. if (present_support == VK_TRUE) {
  368. indices->present = i;
  369. }
  370. if (indices->graphics >= 0 && indices->present >= 0) {
  371. break;
  372. }
  373. }
  374. ret = true;
  375. cleanup:
  376. free(queue_family_properties);
  377. return ret;
  378. }
  379. bool query_swap_chain_support(
  380. VkPhysicalDevice device, VkSurfaceKHR surface,
  381. struct swap_chain_support_details *swap_chain_support) {
  382. VkResult result;
  383. result = vkGetPhysicalDeviceSurfaceCapabilitiesKHR(
  384. device, surface, &swap_chain_support->capabilities);
  385. if (result != VK_SUCCESS) {
  386. fprintf(stderr,
  387. "vkGetPhysicalDeviceSurfaceCapabilitiesKHR failed: %d\n",
  388. result);
  389. return false;
  390. }
  391. result = vkGetPhysicalDeviceSurfaceFormatsKHR(
  392. device, surface, &swap_chain_support->formats_len, NULL);
  393. if (result != VK_SUCCESS) {
  394. fprintf(stderr, "vkGetPhysicalDeviceSurfaceFormatsKHR failed: %d\n",
  395. result);
  396. return false;
  397. }
  398. if (swap_chain_support->formats_len != 0) {
  399. swap_chain_support->formats = malloc(swap_chain_support->formats_len *
  400. sizeof(VkSurfaceFormatKHR));
  401. result = vkGetPhysicalDeviceSurfaceFormatsKHR(
  402. device, surface, &swap_chain_support->formats_len,
  403. swap_chain_support->formats);
  404. if (result != VK_SUCCESS) {
  405. fprintf(stderr, "vkGetPhysicalDeviceSurfaceFormatsKHR failed: %d\n",
  406. result);
  407. return false;
  408. }
  409. }
  410. result = vkGetPhysicalDeviceSurfacePresentModesKHR(
  411. device, surface, &swap_chain_support->present_modes_len, NULL);
  412. if (result != VK_SUCCESS) {
  413. fprintf(stderr,
  414. "vkGetPhysicalDeviceSurfacePresentModesKHR failed: %d\n",
  415. result);
  416. return false;
  417. }
  418. if (swap_chain_support->present_modes_len != 0) {
  419. swap_chain_support->present_modes = malloc(
  420. swap_chain_support->present_modes_len * sizeof(VkPresentModeKHR));
  421. result = vkGetPhysicalDeviceSurfacePresentModesKHR(
  422. device, surface, &swap_chain_support->present_modes_len,
  423. swap_chain_support->present_modes);
  424. if (result != VK_SUCCESS) {
  425. fprintf(stderr,
  426. "vkGetPhysicalDeviceSurfacePresentModesKHR failed: %d\n",
  427. result);
  428. return false;
  429. }
  430. }
  431. return true;
  432. }
  433. bool is_suitable_physical_device(VkPhysicalDevice physical_device,
  434. VkSurfaceKHR surface,
  435. struct queue_family_indices *indices) {
  436. VkResult result;
  437. uint32_t device_extension_count;
  438. uint32_t i;
  439. VkExtensionProperties *device_extensions;
  440. bool ret = false;
  441. struct swap_chain_support_details swap_chain_support;
  442. int found_khr_surface = 0;
  443. VkPhysicalDeviceProperties device_properties;
  444. vkGetPhysicalDeviceProperties(physical_device, &device_properties);
  445. printf("Probing physical device %s\n", device_properties.deviceName);
  446. result = vkEnumerateDeviceExtensionProperties(
  447. physical_device, NULL, &device_extension_count, NULL);
  448. if (result != VK_SUCCESS) {
  449. fprintf(stderr, "vkEnumerateDeviceExtensionProperties failed: %d\n",
  450. result);
  451. return false;
  452. }
  453. device_extensions =
  454. malloc(device_extension_count * sizeof(VkExtensionProperties));
  455. result = vkEnumerateDeviceExtensionProperties(
  456. physical_device, NULL, &device_extension_count, device_extensions);
  457. if (result != VK_SUCCESS) {
  458. fprintf(stderr, "vkEnumerateDeviceExtensionProperties failed: %d\n",
  459. result);
  460. goto cleanup;
  461. }
  462. printf(" Supported device extensions:\n");
  463. for (i = 0; i < device_extension_count; i++) {
  464. printf(" %s\n", device_extensions[i].extensionName);
  465. if (strcmp(VK_KHR_SWAPCHAIN_EXTENSION_NAME,
  466. device_extensions[i].extensionName) == 0) {
  467. found_khr_surface = 1;
  468. break;
  469. }
  470. }
  471. if (!found_khr_surface) {
  472. printf(" Device doesnt support %s\n", VK_KHR_SWAPCHAIN_EXTENSION_NAME);
  473. goto cleanup;
  474. }
  475. if (!find_queue_families(physical_device, surface, indices)) {
  476. goto cleanup;
  477. }
  478. if (indices->graphics < 0 || indices->present < 0) {
  479. printf(" Device is missing graphics and/or present support. graphics: "
  480. "%d, present: %d\n",
  481. indices->graphics, indices->present);
  482. goto cleanup;
  483. }
  484. if (!query_swap_chain_support(physical_device, surface,
  485. &swap_chain_support)) {
  486. goto cleanup;
  487. }
  488. if (swap_chain_support.formats_len == 0) {
  489. printf(" Device doesn't support any swap chain formats\n");
  490. goto cleanup;
  491. }
  492. if (swap_chain_support.present_modes_len == 0) {
  493. printf(" Device doesn't support any swap chain present modes\n");
  494. goto cleanup;
  495. }
  496. ret = true;
  497. cleanup:
  498. free(device_extensions);
  499. swap_chain_support_details_free(&swap_chain_support);
  500. return ret;
  501. }
  502. bool create_physical_device(struct vulkan_demo *demo) {
  503. uint32_t device_count = 0;
  504. VkPhysicalDevice *physical_devices;
  505. VkResult result;
  506. uint32_t i;
  507. bool ret = false;
  508. result = vkEnumeratePhysicalDevices(demo->instance, &device_count, NULL);
  509. if (result != VK_SUCCESS) {
  510. fprintf(stderr, "vkEnumeratePhysicalDevices failed: %d\n", result);
  511. return ret;
  512. }
  513. if (device_count == 0) {
  514. fprintf(stderr, "no vulkan capable GPU found!");
  515. return ret;
  516. }
  517. physical_devices = malloc(device_count * sizeof(VkPhysicalDevice));
  518. result = vkEnumeratePhysicalDevices(demo->instance, &device_count,
  519. physical_devices);
  520. if (result != VK_SUCCESS) {
  521. fprintf(stderr, "vkEnumeratePhysicalDevices failed: %d\n", result);
  522. goto cleanup;
  523. }
  524. for (i = 0; i < device_count; i++) {
  525. struct queue_family_indices indices = {-1, -1};
  526. if (is_suitable_physical_device(physical_devices[i], demo->surface,
  527. &indices)) {
  528. printf(" Selecting this device for rendering. Queue families: "
  529. "graphics: %d, present: %d!\n",
  530. indices.graphics, indices.present);
  531. demo->physical_device = physical_devices[i];
  532. demo->indices = indices;
  533. break;
  534. }
  535. }
  536. if (demo->physical_device == NULL) {
  537. fprintf(stderr, "failed to find a suitable GPU!\n");
  538. } else {
  539. ret = true;
  540. }
  541. cleanup:
  542. free(physical_devices);
  543. return ret;
  544. }
  545. bool create_logical_device(struct vulkan_demo *demo) {
  546. VkResult result;
  547. bool ret = false;
  548. float queuePriority = 1.0f;
  549. uint32_t num_queues = 1;
  550. VkDeviceQueueCreateInfo *queue_create_infos;
  551. VkDeviceCreateInfo create_info;
  552. const char *swap_chain_extension_name = VK_KHR_SWAPCHAIN_EXTENSION_NAME;
  553. queue_create_infos = calloc(2, sizeof(VkDeviceQueueCreateInfo));
  554. queue_create_infos[0].sType = VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO;
  555. queue_create_infos[0].queueFamilyIndex = demo->indices.graphics;
  556. queue_create_infos[0].queueCount = 1;
  557. queue_create_infos[0].pQueuePriorities = &queuePriority;
  558. if (demo->indices.present != demo->indices.graphics) {
  559. queue_create_infos[1].sType =
  560. VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO;
  561. queue_create_infos[1].queueFamilyIndex = demo->indices.present;
  562. queue_create_infos[1].queueCount = 1;
  563. queue_create_infos[1].pQueuePriorities = &queuePriority;
  564. num_queues = 2;
  565. }
  566. memset(&create_info, 0, sizeof(VkDeviceCreateInfo));
  567. create_info.sType = VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO;
  568. create_info.queueCreateInfoCount = num_queues;
  569. create_info.pQueueCreateInfos = queue_create_infos;
  570. create_info.enabledExtensionCount = 1;
  571. create_info.ppEnabledExtensionNames = &swap_chain_extension_name;
  572. result = vkCreateDevice(demo->physical_device, &create_info, NULL,
  573. &demo->device);
  574. if (result != VK_SUCCESS) {
  575. fprintf(stderr, "vkCreateDevice failed: %d\n", result);
  576. goto cleanup;
  577. }
  578. vkGetDeviceQueue(demo->device, demo->indices.graphics, 0,
  579. &demo->graphics_queue);
  580. vkGetDeviceQueue(demo->device, demo->indices.present, 0,
  581. &demo->present_queue);
  582. ret = true;
  583. cleanup:
  584. free(queue_create_infos);
  585. return ret;
  586. }
  587. bool create_sampler(struct vulkan_demo *demo) {
  588. VkResult result;
  589. VkSamplerCreateInfo sampler_info;
  590. memset(&sampler_info, 0, sizeof(VkSamplerCreateInfo));
  591. sampler_info.sType = VK_STRUCTURE_TYPE_SAMPLER_CREATE_INFO;
  592. sampler_info.pNext = NULL;
  593. sampler_info.maxAnisotropy = 1.0;
  594. sampler_info.magFilter = VK_FILTER_LINEAR;
  595. sampler_info.minFilter = VK_FILTER_LINEAR;
  596. sampler_info.mipmapMode = VK_SAMPLER_MIPMAP_MODE_LINEAR;
  597. sampler_info.addressModeU = VK_SAMPLER_ADDRESS_MODE_REPEAT;
  598. sampler_info.addressModeV = VK_SAMPLER_ADDRESS_MODE_REPEAT;
  599. sampler_info.addressModeW = VK_SAMPLER_ADDRESS_MODE_REPEAT;
  600. sampler_info.mipLodBias = 0.0f;
  601. sampler_info.compareEnable = VK_FALSE;
  602. sampler_info.compareOp = VK_COMPARE_OP_ALWAYS;
  603. sampler_info.minLod = 0.0f;
  604. sampler_info.maxLod = 0.0f;
  605. sampler_info.borderColor = VK_BORDER_COLOR_FLOAT_OPAQUE_BLACK;
  606. result = vkCreateSampler(demo->device, &sampler_info, NULL, &demo->sampler);
  607. if (result != VK_SUCCESS) {
  608. fprintf(stderr, "vkCreateSampler failed: %d\n", result);
  609. return false;
  610. }
  611. return true;
  612. }
  613. VkSurfaceFormatKHR
  614. choose_swap_surface_format(VkSurfaceFormatKHR *available_formats,
  615. uint32_t available_formats_len) {
  616. VkSurfaceFormatKHR undefined_format = {VK_FORMAT_B8G8R8A8_UNORM,
  617. VK_COLOR_SPACE_SRGB_NONLINEAR_KHR};
  618. uint32_t i;
  619. if (available_formats_len == 1 &&
  620. available_formats[0].format == VK_FORMAT_UNDEFINED) {
  621. return undefined_format;
  622. }
  623. for (i = 0; i < available_formats_len; i++) {
  624. if (available_formats[i].format == VK_FORMAT_B8G8R8A8_UNORM &&
  625. available_formats[i].colorSpace ==
  626. VK_COLOR_SPACE_SRGB_NONLINEAR_KHR) {
  627. return available_formats[i];
  628. }
  629. }
  630. return available_formats[0];
  631. }
  632. VkPresentModeKHR
  633. choose_swap_present_mode(VkPresentModeKHR *available_present_modes,
  634. uint32_t available_present_modes_len) {
  635. uint32_t i;
  636. for (i = 0; i < available_present_modes_len; i++) {
  637. /*
  638. best mode to ensure good input latency while ensuring we are not
  639. producing tearing
  640. */
  641. if (available_present_modes[i] == VK_PRESENT_MODE_MAILBOX_KHR) {
  642. return available_present_modes[i];
  643. }
  644. }
  645. /* must be supported */
  646. return VK_PRESENT_MODE_FIFO_KHR;
  647. }
  648. VkExtent2D choose_swap_extent(struct vulkan_demo *demo,
  649. VkSurfaceCapabilitiesKHR *capabilities) {
  650. int width, height;
  651. VkExtent2D actual_extent;
  652. if (capabilities->currentExtent.width != 0xFFFFFFFF) {
  653. return capabilities->currentExtent;
  654. } else {
  655. /* not window size! */
  656. glfwGetFramebufferSize(demo->win, &width, &height);
  657. actual_extent.width = (uint32_t)width;
  658. actual_extent.height = (uint32_t)height;
  659. actual_extent.width = NK_MAX(
  660. capabilities->minImageExtent.width,
  661. NK_MIN(capabilities->maxImageExtent.width, actual_extent.width));
  662. actual_extent.height = NK_MAX(
  663. capabilities->minImageExtent.height,
  664. NK_MIN(capabilities->maxImageExtent.height, actual_extent.height));
  665. return actual_extent;
  666. }
  667. }
  668. bool create_swap_chain(struct vulkan_demo *demo) {
  669. struct swap_chain_support_details swap_chain_support;
  670. VkSurfaceFormatKHR surface_format;
  671. VkPresentModeKHR present_mode;
  672. VkExtent2D extent;
  673. VkResult result;
  674. VkSwapchainCreateInfoKHR create_info;
  675. uint32_t queue_family_indices[2];
  676. bool ret = false;
  677. queue_family_indices[0] = (uint32_t)demo->indices.graphics;
  678. queue_family_indices[1] = (uint32_t)demo->indices.present;
  679. if (!query_swap_chain_support(demo->physical_device, demo->surface,
  680. &swap_chain_support)) {
  681. goto cleanup;
  682. }
  683. surface_format = choose_swap_surface_format(swap_chain_support.formats,
  684. swap_chain_support.formats_len);
  685. present_mode = choose_swap_present_mode(
  686. swap_chain_support.present_modes, swap_chain_support.present_modes_len);
  687. extent = choose_swap_extent(demo, &swap_chain_support.capabilities);
  688. demo->swap_chain_images_len =
  689. swap_chain_support.capabilities.minImageCount + 1;
  690. if (swap_chain_support.capabilities.maxImageCount > 0 &&
  691. demo->swap_chain_images_len >
  692. swap_chain_support.capabilities.maxImageCount) {
  693. demo->swap_chain_images_len =
  694. swap_chain_support.capabilities.maxImageCount;
  695. }
  696. memset(&create_info, 0, sizeof(VkSwapchainCreateInfoKHR));
  697. create_info.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
  698. create_info.surface = demo->surface;
  699. create_info.minImageCount = demo->swap_chain_images_len;
  700. create_info.imageFormat = surface_format.format;
  701. create_info.imageColorSpace = surface_format.colorSpace;
  702. create_info.imageExtent = extent;
  703. create_info.imageArrayLayers = 1;
  704. create_info.imageUsage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT;
  705. if (demo->indices.graphics != demo->indices.present) {
  706. create_info.imageSharingMode = VK_SHARING_MODE_CONCURRENT;
  707. create_info.queueFamilyIndexCount = 2;
  708. create_info.pQueueFamilyIndices = queue_family_indices;
  709. } else {
  710. create_info.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
  711. }
  712. create_info.preTransform = swap_chain_support.capabilities.currentTransform;
  713. create_info.compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
  714. create_info.presentMode = present_mode;
  715. create_info.clipped = VK_TRUE;
  716. result = vkCreateSwapchainKHR(demo->device, &create_info, NULL,
  717. &demo->swap_chain);
  718. if (result != VK_SUCCESS) {
  719. fprintf(stderr, "vkCreateSwapchainKHR failed: %d\n", result);
  720. goto cleanup;
  721. }
  722. result = vkGetSwapchainImagesKHR(demo->device, demo->swap_chain,
  723. &demo->swap_chain_images_len, NULL);
  724. if (result != VK_SUCCESS) {
  725. fprintf(stderr, "vkGetSwapchainImagesKHR failed: %d\n", result);
  726. goto cleanup;
  727. }
  728. if (demo->swap_chain_images == NULL) {
  729. demo->swap_chain_images =
  730. malloc(demo->swap_chain_images_len * sizeof(VkImage));
  731. }
  732. result = vkGetSwapchainImagesKHR(demo->device, demo->swap_chain,
  733. &demo->swap_chain_images_len,
  734. demo->swap_chain_images);
  735. if (result != VK_SUCCESS) {
  736. fprintf(stderr, "vkGetSwapchainImagesKHR failed: %d\n", result);
  737. return false;
  738. }
  739. demo->swap_chain_image_format = surface_format.format;
  740. demo->swap_chain_image_extent = extent;
  741. ret = true;
  742. cleanup:
  743. swap_chain_support_details_free(&swap_chain_support);
  744. return ret;
  745. }
  746. bool create_swap_chain_image_views(struct vulkan_demo *demo) {
  747. uint32_t i;
  748. VkResult result;
  749. VkImageViewCreateInfo create_info;
  750. memset(&create_info, 0, sizeof(VkImageViewCreateInfo));
  751. create_info.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
  752. create_info.viewType = VK_IMAGE_VIEW_TYPE_2D;
  753. create_info.format = demo->swap_chain_image_format;
  754. create_info.components.r = VK_COMPONENT_SWIZZLE_IDENTITY;
  755. create_info.components.g = VK_COMPONENT_SWIZZLE_IDENTITY;
  756. create_info.components.b = VK_COMPONENT_SWIZZLE_IDENTITY;
  757. create_info.components.a = VK_COMPONENT_SWIZZLE_IDENTITY;
  758. create_info.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
  759. create_info.subresourceRange.baseMipLevel = 0;
  760. create_info.subresourceRange.levelCount = 1;
  761. create_info.subresourceRange.baseArrayLayer = 0;
  762. create_info.subresourceRange.layerCount = 1;
  763. if (!demo->swap_chain_image_views) {
  764. demo->swap_chain_image_views =
  765. malloc(demo->swap_chain_images_len * sizeof(VkImageView));
  766. }
  767. for (i = 0; i < demo->swap_chain_images_len; i++) {
  768. create_info.image = demo->swap_chain_images[i];
  769. result = vkCreateImageView(demo->device, &create_info, NULL,
  770. &demo->swap_chain_image_views[i]);
  771. if (result != VK_SUCCESS) {
  772. fprintf(stderr, "vkCreateImageView failed: %d\n", result);
  773. return false;
  774. }
  775. }
  776. return true;
  777. }
  778. bool create_overlay_images(struct vulkan_demo *demo) {
  779. uint32_t i, j;
  780. VkResult result;
  781. VkMemoryRequirements mem_requirements;
  782. VkPhysicalDeviceMemoryProperties mem_properties;
  783. int found;
  784. VkImageCreateInfo image_info;
  785. VkMemoryAllocateInfo alloc_info;
  786. VkImageViewCreateInfo image_view_info;
  787. memset(&image_info, 0, sizeof(VkImageCreateInfo));
  788. image_info.sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO;
  789. image_info.imageType = VK_IMAGE_TYPE_2D;
  790. image_info.extent.width = demo->swap_chain_image_extent.width;
  791. image_info.extent.height = demo->swap_chain_image_extent.height;
  792. image_info.extent.depth = 1;
  793. image_info.mipLevels = 1;
  794. image_info.arrayLayers = 1;
  795. image_info.format = demo->swap_chain_image_format;
  796. image_info.tiling = VK_IMAGE_TILING_OPTIMAL;
  797. image_info.initialLayout = VK_IMAGE_LAYOUT_UNDEFINED;
  798. image_info.usage =
  799. VK_IMAGE_USAGE_SAMPLED_BIT | VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT;
  800. image_info.samples = VK_SAMPLE_COUNT_1_BIT;
  801. image_info.sharingMode = VK_SHARING_MODE_EXCLUSIVE;
  802. memset(&alloc_info, 0, sizeof(VkMemoryAllocateInfo));
  803. alloc_info.sType = VK_STRUCTURE_TYPE_MEMORY_ALLOCATE_INFO;
  804. memset(&image_view_info, 0, sizeof(VkImageViewCreateInfo));
  805. image_view_info.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
  806. image_view_info.viewType = VK_IMAGE_VIEW_TYPE_2D;
  807. image_view_info.format = demo->swap_chain_image_format;
  808. image_view_info.components.r = VK_COMPONENT_SWIZZLE_IDENTITY;
  809. image_view_info.components.g = VK_COMPONENT_SWIZZLE_IDENTITY;
  810. image_view_info.components.b = VK_COMPONENT_SWIZZLE_IDENTITY;
  811. image_view_info.components.a = VK_COMPONENT_SWIZZLE_IDENTITY;
  812. image_view_info.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
  813. image_view_info.subresourceRange.baseMipLevel = 0;
  814. image_view_info.subresourceRange.levelCount = 1;
  815. image_view_info.subresourceRange.baseArrayLayer = 0;
  816. image_view_info.subresourceRange.layerCount = 1;
  817. if (!demo->overlay_images) {
  818. demo->overlay_images =
  819. malloc(demo->swap_chain_images_len * sizeof(VkImage));
  820. }
  821. if (!demo->overlay_image_memories) {
  822. demo->overlay_image_memories =
  823. malloc(demo->swap_chain_images_len * sizeof(VkDeviceMemory));
  824. }
  825. if (!demo->overlay_image_views) {
  826. demo->overlay_image_views =
  827. malloc(demo->swap_chain_images_len * sizeof(VkImageView));
  828. }
  829. for (i = 0; i < demo->swap_chain_images_len; i++) {
  830. result = vkCreateImage(demo->device, &image_info, NULL,
  831. &demo->overlay_images[i]);
  832. if (result != VK_SUCCESS) {
  833. fprintf(stderr, "vkCreateImage failed for index %lu: %d\n",
  834. (unsigned long)i, result);
  835. return false;
  836. }
  837. vkGetImageMemoryRequirements(demo->device, demo->overlay_images[i],
  838. &mem_requirements);
  839. alloc_info.allocationSize = mem_requirements.size;
  840. vkGetPhysicalDeviceMemoryProperties(demo->physical_device,
  841. &mem_properties);
  842. found = 0;
  843. for (j = 0; j < mem_properties.memoryTypeCount; j++) {
  844. if ((mem_requirements.memoryTypeBits & (1 << j)) &&
  845. (mem_properties.memoryTypes[j].propertyFlags &
  846. VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT) ==
  847. VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT) {
  848. found = 1;
  849. break;
  850. }
  851. }
  852. if (!found) {
  853. fprintf(stderr,
  854. "failed to find suitable memory type for index %lu!\n",
  855. (unsigned long)i);
  856. return false;
  857. }
  858. alloc_info.memoryTypeIndex = j;
  859. result = vkAllocateMemory(demo->device, &alloc_info, NULL,
  860. &demo->overlay_image_memories[i]);
  861. if (result != VK_SUCCESS) {
  862. fprintf(stderr,
  863. "failed to allocate vulkan memory for index %lu: %d!\n",
  864. (unsigned long)i, result);
  865. return false;
  866. }
  867. result = vkBindImageMemory(demo->device, demo->overlay_images[i],
  868. demo->overlay_image_memories[i], 0);
  869. if (result != VK_SUCCESS) {
  870. fprintf(stderr, "Couldn't bind image memory for index %lu: %d\n",
  871. (unsigned long)i, result);
  872. return false;
  873. }
  874. image_view_info.image = demo->overlay_images[i];
  875. result = vkCreateImageView(demo->device, &image_view_info, NULL,
  876. &demo->overlay_image_views[i]);
  877. if (result != VK_SUCCESS) {
  878. fprintf(stderr, "vkCreateImageView failed for index %lu: %d\n",
  879. (unsigned long)i, result);
  880. return false;
  881. }
  882. }
  883. return true;
  884. }
  885. bool create_render_pass(struct vulkan_demo *demo) {
  886. VkAttachmentDescription attachment;
  887. VkAttachmentReference color_attachment_ref;
  888. VkSubpassDescription subpass;
  889. VkSubpassDependency dependency;
  890. VkRenderPassCreateInfo render_pass_info;
  891. VkResult result;
  892. memset(&attachment, 0, sizeof(VkAttachmentDescription));
  893. attachment.format = demo->swap_chain_image_format;
  894. attachment.samples = VK_SAMPLE_COUNT_1_BIT;
  895. attachment.loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR;
  896. attachment.storeOp = VK_ATTACHMENT_STORE_OP_STORE;
  897. attachment.stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE;
  898. attachment.stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE;
  899. attachment.initialLayout = VK_IMAGE_LAYOUT_UNDEFINED;
  900. attachment.finalLayout = VK_IMAGE_LAYOUT_PRESENT_SRC_KHR;
  901. memset(&color_attachment_ref, 0, sizeof(VkAttachmentReference));
  902. color_attachment_ref.attachment = 0;
  903. color_attachment_ref.layout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL;
  904. memset(&subpass, 0, sizeof(VkSubpassDescription));
  905. subpass.pipelineBindPoint = VK_PIPELINE_BIND_POINT_GRAPHICS;
  906. subpass.colorAttachmentCount = 1;
  907. subpass.pColorAttachments = &color_attachment_ref;
  908. memset(&dependency, 0, sizeof(VkSubpassDependency));
  909. dependency.srcSubpass = VK_SUBPASS_EXTERNAL;
  910. dependency.dstSubpass = 0;
  911. dependency.srcStageMask = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT;
  912. dependency.srcAccessMask = 0;
  913. dependency.dstStageMask = VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT;
  914. dependency.dstAccessMask = VK_ACCESS_COLOR_ATTACHMENT_READ_BIT |
  915. VK_ACCESS_COLOR_ATTACHMENT_WRITE_BIT;
  916. memset(&render_pass_info, 0, sizeof(VkRenderPassCreateInfo));
  917. render_pass_info.sType = VK_STRUCTURE_TYPE_RENDER_PASS_CREATE_INFO;
  918. render_pass_info.attachmentCount = 1;
  919. render_pass_info.pAttachments = &attachment;
  920. render_pass_info.subpassCount = 1;
  921. render_pass_info.pSubpasses = &subpass;
  922. render_pass_info.dependencyCount = 1;
  923. render_pass_info.pDependencies = &dependency;
  924. result = vkCreateRenderPass(demo->device, &render_pass_info, NULL,
  925. &demo->render_pass);
  926. if (result != VK_SUCCESS) {
  927. fprintf(stderr, "vkCreateRenderPass failed: %d\n", result);
  928. return false;
  929. }
  930. return true;
  931. }
  932. bool create_framebuffers(struct vulkan_demo *demo) {
  933. uint32_t i;
  934. VkResult result;
  935. VkFramebufferCreateInfo framebuffer_info;
  936. if (!demo->framebuffers) {
  937. demo->framebuffers =
  938. malloc(demo->swap_chain_images_len * sizeof(VkFramebuffer));
  939. }
  940. memset(&framebuffer_info, 0, sizeof(VkFramebufferCreateInfo));
  941. framebuffer_info.sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO;
  942. framebuffer_info.renderPass = demo->render_pass;
  943. framebuffer_info.attachmentCount = 1;
  944. framebuffer_info.width = demo->swap_chain_image_extent.width;
  945. framebuffer_info.height = demo->swap_chain_image_extent.height;
  946. framebuffer_info.layers = 1;
  947. for (i = 0; i < demo->swap_chain_images_len; i++) {
  948. framebuffer_info.pAttachments = &demo->swap_chain_image_views[i];
  949. result = vkCreateFramebuffer(demo->device, &framebuffer_info, NULL,
  950. &demo->framebuffers[i]);
  951. if (result != VK_SUCCESS) {
  952. fprintf(stderr, "vkCreateFramebuffer failed from index %lu: %d\n",
  953. (unsigned long)i, result);
  954. return false;
  955. }
  956. }
  957. return true;
  958. }
  959. bool create_descriptor_set_layout(struct vulkan_demo *demo) {
  960. VkDescriptorSetLayoutBinding overlay_layout_binding;
  961. VkDescriptorSetLayoutCreateInfo descriptor_set_layout_create_nfo;
  962. VkResult result;
  963. memset(&overlay_layout_binding, 0, sizeof(VkDescriptorSetLayoutBinding));
  964. overlay_layout_binding.binding = 0;
  965. overlay_layout_binding.descriptorCount = 1;
  966. overlay_layout_binding.descriptorType =
  967. VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER;
  968. overlay_layout_binding.pImmutableSamplers = NULL;
  969. overlay_layout_binding.stageFlags = VK_SHADER_STAGE_FRAGMENT_BIT;
  970. memset(&descriptor_set_layout_create_nfo, 0,
  971. sizeof(VkDescriptorSetLayoutCreateInfo));
  972. descriptor_set_layout_create_nfo.sType =
  973. VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO;
  974. descriptor_set_layout_create_nfo.bindingCount = 1;
  975. descriptor_set_layout_create_nfo.pBindings = &overlay_layout_binding;
  976. result = vkCreateDescriptorSetLayout(demo->device,
  977. &descriptor_set_layout_create_nfo,
  978. NULL, &demo->descriptor_set_layout);
  979. if (result != VK_SUCCESS) {
  980. fprintf(stderr, "vkCreateDescriptorSetLayout failed: %d\n", result);
  981. return false;
  982. }
  983. return true;
  984. }
  985. bool create_descriptor_pool(struct vulkan_demo *demo) {
  986. VkDescriptorPoolSize pool_size;
  987. VkDescriptorPoolCreateInfo pool_info;
  988. VkResult result;
  989. memset(&pool_size, 0, sizeof(VkDescriptorPoolSize));
  990. pool_size.type = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER;
  991. pool_size.descriptorCount = demo->swap_chain_images_len;
  992. memset(&pool_info, 0, sizeof(VkDescriptorPoolCreateInfo));
  993. pool_info.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_POOL_CREATE_INFO;
  994. pool_info.poolSizeCount = 1;
  995. pool_info.pPoolSizes = &pool_size;
  996. pool_info.maxSets = demo->swap_chain_images_len;
  997. result = vkCreateDescriptorPool(demo->device, &pool_info, NULL,
  998. &demo->descriptor_pool);
  999. if (result != VK_SUCCESS) {
  1000. fprintf(stderr, "vkCreateDescriptorPool failed: %d\n", result);
  1001. return false;
  1002. }
  1003. return true;
  1004. }
  1005. void update_descriptor_sets(struct vulkan_demo *demo) {
  1006. uint32_t i;
  1007. VkDescriptorImageInfo descriptor_image_info;
  1008. VkWriteDescriptorSet descriptor_write;
  1009. memset(&descriptor_image_info, 0, sizeof(VkDescriptorImageInfo));
  1010. descriptor_image_info.imageLayout =
  1011. VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL;
  1012. descriptor_image_info.sampler = demo->sampler;
  1013. memset(&descriptor_write, 0, sizeof(VkWriteDescriptorSet));
  1014. descriptor_write.sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET;
  1015. descriptor_write.dstBinding = 0;
  1016. descriptor_write.dstArrayElement = 0;
  1017. descriptor_write.descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER;
  1018. descriptor_write.descriptorCount = 1;
  1019. descriptor_write.pImageInfo = &descriptor_image_info;
  1020. for (i = 0; i < demo->swap_chain_images_len; i++) {
  1021. descriptor_write.dstSet = demo->descriptor_sets[i];
  1022. descriptor_image_info.imageView = demo->overlay_image_views[i];
  1023. vkUpdateDescriptorSets(demo->device, 1, &descriptor_write, 0, NULL);
  1024. }
  1025. }
  1026. bool create_descriptor_sets(struct vulkan_demo *demo) {
  1027. bool ret = false;
  1028. VkDescriptorSetLayout *descriptor_set_layouts;
  1029. VkDescriptorSetAllocateInfo alloc_info;
  1030. uint32_t i;
  1031. VkResult result;
  1032. demo->descriptor_sets =
  1033. malloc(demo->swap_chain_images_len * sizeof(VkDescriptorSet));
  1034. descriptor_set_layouts =
  1035. malloc(demo->swap_chain_images_len * sizeof(VkDescriptorSetLayout));
  1036. for (i = 0; i < demo->swap_chain_images_len; i++) {
  1037. descriptor_set_layouts[i] = demo->descriptor_set_layout;
  1038. }
  1039. memset(&alloc_info, 0, sizeof(VkDescriptorSetAllocateInfo));
  1040. alloc_info.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_ALLOCATE_INFO;
  1041. alloc_info.descriptorPool = demo->descriptor_pool;
  1042. alloc_info.descriptorSetCount = demo->swap_chain_images_len;
  1043. alloc_info.pSetLayouts = descriptor_set_layouts;
  1044. result = vkAllocateDescriptorSets(demo->device, &alloc_info,
  1045. demo->descriptor_sets);
  1046. if (result != VK_SUCCESS) {
  1047. fprintf(stderr, "vkAllocateDescriptorSets failed: %d\n", result);
  1048. goto cleanup;
  1049. }
  1050. update_descriptor_sets(demo);
  1051. ret = true;
  1052. cleanup:
  1053. free(descriptor_set_layouts);
  1054. return ret;
  1055. }
  1056. bool create_shader_module(VkDevice device, char *shader_buffer,
  1057. size_t shader_buffer_len,
  1058. VkShaderModule *shader_module) {
  1059. VkShaderModuleCreateInfo create_info;
  1060. VkResult result;
  1061. memset(&create_info, 0, sizeof(VkShaderModuleCreateInfo));
  1062. create_info.sType = VK_STRUCTURE_TYPE_SHADER_MODULE_CREATE_INFO;
  1063. create_info.codeSize = shader_buffer_len;
  1064. create_info.pCode = (const uint32_t *)shader_buffer;
  1065. result = vkCreateShaderModule(device, &create_info, NULL, shader_module);
  1066. if (result != VK_SUCCESS) {
  1067. fprintf(stderr, "vkCreateShaderModule failed: %d\n", result);
  1068. return false;
  1069. }
  1070. return true;
  1071. }
  1072. bool create_graphics_pipeline(struct vulkan_demo *demo) {
  1073. bool ret = false;
  1074. char *vert_shader_code = NULL;
  1075. char *frag_shader_code = NULL;
  1076. VkShaderModule vert_shader_module = VK_NULL_HANDLE;
  1077. VkShaderModule frag_shader_module = VK_NULL_HANDLE;
  1078. FILE *fp;
  1079. size_t file_len;
  1080. VkPipelineShaderStageCreateInfo vert_shader_stage_info;
  1081. VkPipelineShaderStageCreateInfo frag_shader_stage_info;
  1082. VkPipelineShaderStageCreateInfo shader_stages[2];
  1083. VkPipelineVertexInputStateCreateInfo vertex_input_info;
  1084. VkPipelineInputAssemblyStateCreateInfo input_assembly;
  1085. VkViewport viewport;
  1086. VkRect2D scissor;
  1087. VkPipelineViewportStateCreateInfo viewport_state;
  1088. VkPipelineRasterizationStateCreateInfo rasterizer;
  1089. VkPipelineMultisampleStateCreateInfo multisampling;
  1090. VkPipelineColorBlendAttachmentState color_blend_attachment;
  1091. VkPipelineColorBlendStateCreateInfo color_blending;
  1092. VkPipelineLayoutCreateInfo pipeline_layout_info;
  1093. VkResult result;
  1094. VkGraphicsPipelineCreateInfo pipeline_info;
  1095. size_t read_result;
  1096. fp = fopen("shaders/demo.vert.spv", "rb");
  1097. if (!fp) {
  1098. fprintf(stderr, "Couldn't open shaders/demo.vert.spv\n");
  1099. return false;
  1100. }
  1101. fseek(fp, 0, SEEK_END);
  1102. file_len = ftell(fp);
  1103. vert_shader_code = malloc(file_len);
  1104. fseek(fp, 0, 0);
  1105. read_result = fread(vert_shader_code, file_len, 1, fp);
  1106. fclose(fp);
  1107. if (read_result != 1) {
  1108. fprintf(stderr, "Could not read fragment shader\n");
  1109. goto cleanup;
  1110. }
  1111. if (!create_shader_module(demo->device, vert_shader_code, file_len,
  1112. &vert_shader_module)) {
  1113. goto cleanup;
  1114. }
  1115. fp = fopen("shaders/demo.frag.spv", "rb");
  1116. if (!fp) {
  1117. fprintf(stderr, "Couldn't open shaders/demo.frag.spv\n");
  1118. return false;
  1119. }
  1120. fseek(fp, 0, SEEK_END);
  1121. file_len = ftell(fp);
  1122. frag_shader_code = malloc(file_len);
  1123. fseek(fp, 0, 0);
  1124. read_result = fread(frag_shader_code, file_len, 1, fp);
  1125. fclose(fp);
  1126. if (read_result != 1) {
  1127. fprintf(stderr, "Could not read fragment shader\n");
  1128. goto cleanup;
  1129. }
  1130. if (!create_shader_module(demo->device, frag_shader_code, file_len,
  1131. &frag_shader_module)) {
  1132. goto cleanup;
  1133. }
  1134. memset(&vert_shader_stage_info, 0, sizeof(VkPipelineShaderStageCreateInfo));
  1135. vert_shader_stage_info.sType =
  1136. VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO;
  1137. vert_shader_stage_info.stage = VK_SHADER_STAGE_VERTEX_BIT;
  1138. vert_shader_stage_info.module = vert_shader_module;
  1139. vert_shader_stage_info.pName = "main";
  1140. memset(&frag_shader_stage_info, 0, sizeof(VkPipelineShaderStageCreateInfo));
  1141. frag_shader_stage_info.sType =
  1142. VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO;
  1143. frag_shader_stage_info.stage = VK_SHADER_STAGE_FRAGMENT_BIT;
  1144. frag_shader_stage_info.module = frag_shader_module;
  1145. frag_shader_stage_info.pName = "main";
  1146. shader_stages[0] = vert_shader_stage_info;
  1147. shader_stages[1] = frag_shader_stage_info;
  1148. memset(&vertex_input_info, 0, sizeof(VkPipelineVertexInputStateCreateInfo));
  1149. vertex_input_info.sType =
  1150. VK_STRUCTURE_TYPE_PIPELINE_VERTEX_INPUT_STATE_CREATE_INFO;
  1151. memset(&input_assembly, 0, sizeof(VkPipelineInputAssemblyStateCreateInfo));
  1152. input_assembly.sType =
  1153. VK_STRUCTURE_TYPE_PIPELINE_INPUT_ASSEMBLY_STATE_CREATE_INFO;
  1154. input_assembly.topology = VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST;
  1155. input_assembly.primitiveRestartEnable = VK_FALSE;
  1156. memset(&viewport, 0, sizeof(VkViewport));
  1157. viewport.x = 0.0f;
  1158. viewport.y = 0.0f;
  1159. viewport.width = (float)demo->swap_chain_image_extent.width;
  1160. viewport.height = (float)demo->swap_chain_image_extent.height;
  1161. viewport.minDepth = 0.0f;
  1162. viewport.maxDepth = 1.0f;
  1163. memset(&scissor, 0, sizeof(VkRect2D));
  1164. scissor.extent.width = demo->swap_chain_image_extent.width;
  1165. scissor.extent.height = demo->swap_chain_image_extent.height;
  1166. memset(&viewport_state, 0, sizeof(VkPipelineViewportStateCreateInfo));
  1167. viewport_state.sType =
  1168. VK_STRUCTURE_TYPE_PIPELINE_VIEWPORT_STATE_CREATE_INFO;
  1169. viewport_state.viewportCount = 1;
  1170. viewport_state.pViewports = &viewport;
  1171. viewport_state.scissorCount = 1;
  1172. viewport_state.pScissors = &scissor;
  1173. memset(&rasterizer, 0, sizeof(VkPipelineRasterizationStateCreateInfo));
  1174. rasterizer.sType =
  1175. VK_STRUCTURE_TYPE_PIPELINE_RASTERIZATION_STATE_CREATE_INFO;
  1176. rasterizer.depthClampEnable = VK_FALSE;
  1177. rasterizer.rasterizerDiscardEnable = VK_FALSE;
  1178. rasterizer.polygonMode = VK_POLYGON_MODE_FILL;
  1179. rasterizer.lineWidth = 1.0f;
  1180. rasterizer.cullMode = VK_CULL_MODE_FRONT_BIT;
  1181. rasterizer.frontFace = VK_FRONT_FACE_COUNTER_CLOCKWISE;
  1182. rasterizer.depthBiasEnable = VK_FALSE;
  1183. memset(&multisampling, 0, sizeof(VkPipelineMultisampleStateCreateInfo));
  1184. multisampling.sType =
  1185. VK_STRUCTURE_TYPE_PIPELINE_MULTISAMPLE_STATE_CREATE_INFO;
  1186. multisampling.sampleShadingEnable = VK_FALSE;
  1187. multisampling.rasterizationSamples = VK_SAMPLE_COUNT_1_BIT;
  1188. memset(&color_blend_attachment, 0,
  1189. sizeof(VkPipelineColorBlendAttachmentState));
  1190. color_blend_attachment.colorWriteMask =
  1191. VK_COLOR_COMPONENT_R_BIT | VK_COLOR_COMPONENT_G_BIT |
  1192. VK_COLOR_COMPONENT_B_BIT | VK_COLOR_COMPONENT_A_BIT;
  1193. color_blend_attachment.blendEnable = VK_TRUE;
  1194. color_blend_attachment.srcColorBlendFactor = VK_BLEND_FACTOR_ONE;
  1195. color_blend_attachment.dstColorBlendFactor =
  1196. VK_BLEND_FACTOR_ONE_MINUS_SRC_ALPHA;
  1197. color_blend_attachment.colorBlendOp = VK_BLEND_OP_ADD;
  1198. color_blend_attachment.srcAlphaBlendFactor = VK_BLEND_FACTOR_ONE;
  1199. color_blend_attachment.dstAlphaBlendFactor = VK_BLEND_FACTOR_ZERO;
  1200. color_blend_attachment.alphaBlendOp = VK_BLEND_OP_ADD;
  1201. memset(&color_blending, 0, sizeof(VkPipelineColorBlendStateCreateInfo));
  1202. color_blending.sType =
  1203. VK_STRUCTURE_TYPE_PIPELINE_COLOR_BLEND_STATE_CREATE_INFO;
  1204. color_blending.logicOpEnable = VK_FALSE;
  1205. color_blending.logicOp = VK_LOGIC_OP_COPY;
  1206. color_blending.attachmentCount = 1;
  1207. color_blending.pAttachments = &color_blend_attachment;
  1208. color_blending.blendConstants[0] = 1.0f;
  1209. color_blending.blendConstants[1] = 1.0f;
  1210. color_blending.blendConstants[2] = 1.0f;
  1211. color_blending.blendConstants[3] = 1.0f;
  1212. memset(&pipeline_layout_info, 0, sizeof(VkPipelineLayoutCreateInfo));
  1213. pipeline_layout_info.sType = VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO;
  1214. pipeline_layout_info.setLayoutCount = 0;
  1215. pipeline_layout_info.pushConstantRangeCount = 0;
  1216. pipeline_layout_info.setLayoutCount = 1;
  1217. pipeline_layout_info.pSetLayouts = &demo->descriptor_set_layout;
  1218. result = vkCreatePipelineLayout(demo->device, &pipeline_layout_info, NULL,
  1219. &demo->pipeline_layout);
  1220. if (result != VK_SUCCESS) {
  1221. fprintf(stderr, "vkCreatePipelineLayout failed: %d\n", result);
  1222. goto cleanup;
  1223. }
  1224. memset(&pipeline_info, 0, sizeof(VkGraphicsPipelineCreateInfo));
  1225. pipeline_info.sType = VK_STRUCTURE_TYPE_GRAPHICS_PIPELINE_CREATE_INFO;
  1226. pipeline_info.stageCount = 2;
  1227. pipeline_info.pStages = shader_stages;
  1228. pipeline_info.pVertexInputState = &vertex_input_info;
  1229. pipeline_info.pInputAssemblyState = &input_assembly;
  1230. pipeline_info.pViewportState = &viewport_state;
  1231. pipeline_info.pRasterizationState = &rasterizer;
  1232. pipeline_info.pMultisampleState = &multisampling;
  1233. pipeline_info.pColorBlendState = &color_blending;
  1234. pipeline_info.layout = demo->pipeline_layout;
  1235. pipeline_info.renderPass = demo->render_pass;
  1236. pipeline_info.basePipelineHandle = NULL;
  1237. result = vkCreateGraphicsPipelines(demo->device, NULL, 1, &pipeline_info,
  1238. NULL, &demo->pipeline);
  1239. if (result != VK_SUCCESS) {
  1240. fprintf(stderr, "vkCreateGraphicsPipelines failed: %d\n", result);
  1241. goto cleanup;
  1242. }
  1243. ret = true;
  1244. cleanup:
  1245. if (frag_shader_module) {
  1246. vkDestroyShaderModule(demo->device, frag_shader_module, NULL);
  1247. }
  1248. if (frag_shader_code) {
  1249. free(frag_shader_code);
  1250. }
  1251. if (vert_shader_module) {
  1252. vkDestroyShaderModule(demo->device, vert_shader_module, NULL);
  1253. }
  1254. if (vert_shader_code) {
  1255. free(vert_shader_code);
  1256. }
  1257. return ret;
  1258. }
  1259. bool create_command_pool(struct vulkan_demo *demo) {
  1260. VkCommandPoolCreateInfo pool_info;
  1261. VkResult result;
  1262. memset(&pool_info, 0, sizeof(VkCommandPoolCreateInfo));
  1263. pool_info.sType = VK_STRUCTURE_TYPE_COMMAND_POOL_CREATE_INFO;
  1264. pool_info.flags = VK_COMMAND_POOL_CREATE_RESET_COMMAND_BUFFER_BIT;
  1265. pool_info.queueFamilyIndex = demo->indices.graphics;
  1266. result = vkCreateCommandPool(demo->device, &pool_info, NULL,
  1267. &demo->command_pool);
  1268. if (result != VK_SUCCESS) {
  1269. fprintf(stderr, "vkCreateCommandPool failed: %d\n", result);
  1270. return false;
  1271. }
  1272. return true;
  1273. }
  1274. bool create_command_buffers(struct vulkan_demo *demo) {
  1275. VkCommandBufferAllocateInfo alloc_info;
  1276. VkResult result;
  1277. demo->command_buffers =
  1278. malloc(demo->swap_chain_images_len * sizeof(VkCommandBuffer));
  1279. memset(&alloc_info, 0, sizeof(VkCommandBufferAllocateInfo));
  1280. alloc_info.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO;
  1281. alloc_info.commandPool = demo->command_pool;
  1282. alloc_info.level = VK_COMMAND_BUFFER_LEVEL_PRIMARY;
  1283. alloc_info.commandBufferCount = demo->swap_chain_images_len;
  1284. result = vkAllocateCommandBuffers(demo->device, &alloc_info,
  1285. demo->command_buffers);
  1286. if (result != VK_SUCCESS) {
  1287. fprintf(stderr, "vkAllocateCommandBuffers failed: %d\n", result);
  1288. return false;
  1289. }
  1290. return true;
  1291. }
  1292. bool create_semaphores(struct vulkan_demo *demo) {
  1293. VkSemaphoreCreateInfo semaphore_info;
  1294. VkResult result;
  1295. memset(&semaphore_info, 0, sizeof(VkSemaphoreCreateInfo));
  1296. semaphore_info.sType = VK_STRUCTURE_TYPE_SEMAPHORE_CREATE_INFO;
  1297. result = vkCreateSemaphore(demo->device, &semaphore_info, NULL,
  1298. &demo->image_available);
  1299. if (result != VK_SUCCESS) {
  1300. fprintf(stderr, "vkCreateSemaphore failed: %d\n", result);
  1301. return false;
  1302. }
  1303. result = vkCreateSemaphore(demo->device, &semaphore_info, NULL,
  1304. &demo->render_finished);
  1305. if (result != VK_SUCCESS) {
  1306. fprintf(stderr, "vkCreateSemaphore failed: %d\n", result);
  1307. return false;
  1308. }
  1309. return true;
  1310. }
  1311. bool create_fence(struct vulkan_demo *demo) {
  1312. VkResult result;
  1313. VkFenceCreateInfo fence_create_info;
  1314. memset(&fence_create_info, 0, sizeof(VkFenceCreateInfo));
  1315. fence_create_info.sType = VK_STRUCTURE_TYPE_FENCE_CREATE_INFO;
  1316. fence_create_info.flags = VK_FENCE_CREATE_SIGNALED_BIT;
  1317. result = vkCreateFence(demo->device, &fence_create_info, NULL,
  1318. &demo->render_fence);
  1319. if (result != VK_SUCCESS) {
  1320. fprintf(stderr, "vkCreateFence failed: %d\n", result);
  1321. return false;
  1322. }
  1323. return true;
  1324. }
  1325. bool create_swap_chain_related_resources(struct vulkan_demo *demo) {
  1326. if (!create_swap_chain(demo)) {
  1327. return false;
  1328. }
  1329. if (!create_swap_chain_image_views(demo)) {
  1330. return false;
  1331. }
  1332. if (!create_overlay_images(demo)) {
  1333. return false;
  1334. }
  1335. if (!create_render_pass(demo)) {
  1336. return false;
  1337. }
  1338. if (!create_framebuffers(demo)) {
  1339. return false;
  1340. }
  1341. if (!create_graphics_pipeline(demo)) {
  1342. return false;
  1343. }
  1344. return true;
  1345. }
  1346. bool destroy_swap_chain_related_resources(struct vulkan_demo *demo) {
  1347. uint32_t i;
  1348. VkResult result;
  1349. result = vkQueueWaitIdle(demo->graphics_queue);
  1350. if (result != VK_SUCCESS) {
  1351. fprintf(stderr, "vkQueueWaitIdle failed: %d\n", result);
  1352. return false;
  1353. }
  1354. for (i = 0; i < demo->swap_chain_images_len; i++) {
  1355. vkDestroyFramebuffer(demo->device, demo->framebuffers[i], NULL);
  1356. vkDestroyImageView(demo->device, demo->overlay_image_views[i], NULL);
  1357. vkDestroyImage(demo->device, demo->overlay_images[i], NULL);
  1358. vkFreeMemory(demo->device, demo->overlay_image_memories[i], NULL);
  1359. vkDestroyImageView(demo->device, demo->swap_chain_image_views[i], NULL);
  1360. }
  1361. vkDestroySwapchainKHR(demo->device, demo->swap_chain, NULL);
  1362. vkDestroyRenderPass(demo->device, demo->render_pass, NULL);
  1363. vkDestroyPipeline(demo->device, demo->pipeline, NULL);
  1364. vkDestroyPipelineLayout(demo->device, demo->pipeline_layout, NULL);
  1365. return true;
  1366. }
  1367. bool create_demo_texture(struct vulkan_demo *demo) {
  1368. VkResult result;
  1369. VkMemoryRequirements mem_requirements;
  1370. VkPhysicalDeviceMemoryProperties mem_properties;
  1371. int found;
  1372. uint32_t i;
  1373. VkImageCreateInfo image_info;
  1374. VkMemoryAllocateInfo alloc_info;
  1375. VkImageViewCreateInfo image_view_info;
  1376. VkBufferCreateInfo buffer_info;
  1377. struct {
  1378. VkDeviceMemory memory;
  1379. VkBuffer buffer;
  1380. } staging_buffer;
  1381. void *data;
  1382. VkCommandBuffer command_buffer;
  1383. VkCommandBufferBeginInfo begin_info;
  1384. VkImageMemoryBarrier image_transfer_dst_memory_barrier;
  1385. VkBufferImageCopy buffer_copy_region;
  1386. VkImageMemoryBarrier image_shader_memory_barrier;
  1387. VkFence fence;
  1388. VkFenceCreateInfo fence_create;
  1389. VkSubmitInfo submit_info;
  1390. memset(&image_info, 0, sizeof(VkImageCreateInfo));
  1391. image_info.sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO;
  1392. image_info.imageType = VK_IMAGE_TYPE_2D;
  1393. image_info.extent.width = 2;
  1394. image_info.extent.height = 2;
  1395. image_info.extent.depth = 1;
  1396. image_info.mipLevels = 1;
  1397. image_info.arrayLayers = 1;
  1398. image_info.format = VK_FORMAT_R8_UNORM;
  1399. image_info.tiling = VK_IMAGE_TILING_LINEAR;
  1400. image_info.initialLayout = VK_IMAGE_LAYOUT_UNDEFINED;
  1401. image_info.usage =
  1402. VK_IMAGE_USAGE_TRANSFER_DST_BIT | VK_IMAGE_USAGE_SAMPLED_BIT;
  1403. image_info.samples = VK_SAMPLE_COUNT_1_BIT;
  1404. image_info.sharingMode = VK_SHARING_MODE_EXCLUSIVE;
  1405. memset(&alloc_info, 0, sizeof(VkMemoryAllocateInfo));
  1406. alloc_info.sType = VK_STRUCTURE_TYPE_MEMORY_ALLOCATE_INFO;
  1407. memset(&image_view_info, 0, sizeof(VkImageViewCreateInfo));
  1408. image_view_info.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
  1409. image_view_info.viewType = VK_IMAGE_VIEW_TYPE_2D;
  1410. image_view_info.format = VK_FORMAT_R8_UNORM;
  1411. image_view_info.components.r = VK_COMPONENT_SWIZZLE_IDENTITY;
  1412. image_view_info.components.g = VK_COMPONENT_SWIZZLE_IDENTITY;
  1413. image_view_info.components.b = VK_COMPONENT_SWIZZLE_IDENTITY;
  1414. image_view_info.components.a = VK_COMPONENT_SWIZZLE_IDENTITY;
  1415. image_view_info.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
  1416. image_view_info.subresourceRange.baseMipLevel = 0;
  1417. image_view_info.subresourceRange.levelCount = 1;
  1418. image_view_info.subresourceRange.baseArrayLayer = 0;
  1419. image_view_info.subresourceRange.layerCount = 1;
  1420. result = vkCreateImage(demo->device, &image_info, NULL,
  1421. &demo->demo_texture_image);
  1422. if (result != VK_SUCCESS) {
  1423. fprintf(stderr, "vkCreateImage failed: %d\n", result);
  1424. return false;
  1425. }
  1426. vkGetImageMemoryRequirements(demo->device, demo->demo_texture_image,
  1427. &mem_requirements);
  1428. alloc_info.allocationSize = mem_requirements.size;
  1429. vkGetPhysicalDeviceMemoryProperties(demo->physical_device, &mem_properties);
  1430. found = 0;
  1431. for (i = 0; i < mem_properties.memoryTypeCount; i++) {
  1432. if ((mem_requirements.memoryTypeBits & (1 << i)) &&
  1433. (mem_properties.memoryTypes[i].propertyFlags &
  1434. VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT) ==
  1435. VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT) {
  1436. found = 1;
  1437. break;
  1438. }
  1439. }
  1440. if (!found) {
  1441. fprintf(stderr, "failed to find suitable memory for demo texture!\n");
  1442. return false;
  1443. }
  1444. alloc_info.memoryTypeIndex = i;
  1445. result = vkAllocateMemory(demo->device, &alloc_info, NULL,
  1446. &demo->demo_texture_memory);
  1447. if (result != VK_SUCCESS) {
  1448. fprintf(stderr,
  1449. "failed to allocate vulkan memory for demo texture: %d!\n",
  1450. result);
  1451. return false;
  1452. }
  1453. result = vkBindImageMemory(demo->device, demo->demo_texture_image,
  1454. demo->demo_texture_memory, 0);
  1455. if (result != VK_SUCCESS) {
  1456. fprintf(stderr, "Couldn't bind image memory for demo texture: %d\n",
  1457. result);
  1458. return false;
  1459. }
  1460. image_view_info.image = demo->demo_texture_image;
  1461. result = vkCreateImageView(demo->device, &image_view_info, NULL,
  1462. &demo->demo_texture_image_view);
  1463. if (result != VK_SUCCESS) {
  1464. fprintf(stderr, "vkCreateImageView failed for demo texture: %d\n",
  1465. result);
  1466. return false;
  1467. }
  1468. memset(&buffer_info, 0, sizeof(VkBufferCreateInfo));
  1469. buffer_info.sType = VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO;
  1470. buffer_info.size = alloc_info.allocationSize;
  1471. buffer_info.usage = VK_BUFFER_USAGE_TRANSFER_SRC_BIT;
  1472. buffer_info.sharingMode = VK_SHARING_MODE_EXCLUSIVE;
  1473. result = vkCreateBuffer(demo->device, &buffer_info, NULL,
  1474. &staging_buffer.buffer);
  1475. if (result != VK_SUCCESS) {
  1476. fprintf(stderr, "vkCreateBuffer failed for demo texture: %d\n", result);
  1477. return false;
  1478. }
  1479. vkGetBufferMemoryRequirements(demo->device, staging_buffer.buffer,
  1480. &mem_requirements);
  1481. alloc_info.allocationSize = mem_requirements.size;
  1482. found = 0;
  1483. for (i = 0; i < mem_properties.memoryTypeCount; i++) {
  1484. if ((mem_requirements.memoryTypeBits & (1 << i)) &&
  1485. (mem_properties.memoryTypes[i].propertyFlags &
  1486. (VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT |
  1487. VK_MEMORY_PROPERTY_HOST_COHERENT_BIT)) ==
  1488. (VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT |
  1489. VK_MEMORY_PROPERTY_HOST_COHERENT_BIT)) {
  1490. found = 1;
  1491. break;
  1492. }
  1493. }
  1494. if (!found) {
  1495. fprintf(stderr, "failed to find suitable staging buffer memory for "
  1496. "demo texture!\n");
  1497. return false;
  1498. }
  1499. alloc_info.memoryTypeIndex = i;
  1500. result = vkAllocateMemory(demo->device, &alloc_info, NULL,
  1501. &staging_buffer.memory);
  1502. if (!found) {
  1503. fprintf(stderr, "vkAllocateMemory failed for demo texture: %d\n",
  1504. result);
  1505. return false;
  1506. }
  1507. result = vkBindBufferMemory(demo->device, staging_buffer.buffer,
  1508. staging_buffer.memory, 0);
  1509. if (!found) {
  1510. fprintf(stderr, "vkBindBufferMemory failed for demo texture: %d\n",
  1511. result);
  1512. return false;
  1513. }
  1514. result = vkMapMemory(demo->device, staging_buffer.memory, 0,
  1515. sizeof(uint32_t), 0, &data);
  1516. if (result != VK_SUCCESS) {
  1517. fprintf(stderr, "vkMapMemory failed for demo texture: %d\n", result);
  1518. return false;
  1519. }
  1520. *((uint32_t *)data) = 0x00FFFF00;
  1521. vkUnmapMemory(demo->device, staging_buffer.memory);
  1522. memset(&begin_info, 0, sizeof(VkCommandBufferBeginInfo));
  1523. begin_info.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO;
  1524. command_buffer = demo->command_buffers[0];
  1525. result = vkBeginCommandBuffer(command_buffer, &begin_info);
  1526. memset(&image_transfer_dst_memory_barrier, 0, sizeof(VkImageMemoryBarrier));
  1527. image_transfer_dst_memory_barrier.sType =
  1528. VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER;
  1529. image_transfer_dst_memory_barrier.image = demo->demo_texture_image;
  1530. image_transfer_dst_memory_barrier.srcQueueFamilyIndex =
  1531. VK_QUEUE_FAMILY_IGNORED;
  1532. image_transfer_dst_memory_barrier.dstQueueFamilyIndex =
  1533. VK_QUEUE_FAMILY_IGNORED;
  1534. image_transfer_dst_memory_barrier.oldLayout = VK_IMAGE_LAYOUT_UNDEFINED;
  1535. image_transfer_dst_memory_barrier.newLayout =
  1536. VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL;
  1537. image_transfer_dst_memory_barrier.subresourceRange.aspectMask =
  1538. VK_IMAGE_ASPECT_COLOR_BIT;
  1539. image_transfer_dst_memory_barrier.subresourceRange.levelCount = 1;
  1540. image_transfer_dst_memory_barrier.subresourceRange.layerCount = 1;
  1541. image_transfer_dst_memory_barrier.dstAccessMask =
  1542. VK_ACCESS_TRANSFER_WRITE_BIT;
  1543. vkCmdPipelineBarrier(command_buffer, VK_PIPELINE_STAGE_TOP_OF_PIPE_BIT,
  1544. VK_PIPELINE_STAGE_TRANSFER_BIT, 0, 0, NULL, 0, NULL, 1,
  1545. &image_transfer_dst_memory_barrier);
  1546. memset(&buffer_copy_region, 0, sizeof(VkBufferImageCopy));
  1547. buffer_copy_region.imageSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
  1548. buffer_copy_region.imageSubresource.layerCount = 1;
  1549. buffer_copy_region.imageExtent.width = 2;
  1550. buffer_copy_region.imageExtent.height = 2;
  1551. buffer_copy_region.imageExtent.depth = 1;
  1552. vkCmdCopyBufferToImage(
  1553. command_buffer, staging_buffer.buffer, demo->demo_texture_image,
  1554. VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, 1, &buffer_copy_region);
  1555. memset(&image_shader_memory_barrier, 0, sizeof(VkImageMemoryBarrier));
  1556. image_shader_memory_barrier.sType = VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER;
  1557. image_shader_memory_barrier.image = demo->demo_texture_image;
  1558. image_shader_memory_barrier.srcQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED;
  1559. image_shader_memory_barrier.dstQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED;
  1560. image_shader_memory_barrier.oldLayout =
  1561. VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL;
  1562. image_shader_memory_barrier.newLayout =
  1563. VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL;
  1564. image_shader_memory_barrier.subresourceRange.aspectMask =
  1565. VK_IMAGE_ASPECT_COLOR_BIT;
  1566. image_shader_memory_barrier.subresourceRange.levelCount = 1;
  1567. image_shader_memory_barrier.subresourceRange.layerCount = 1;
  1568. image_shader_memory_barrier.srcAccessMask = VK_ACCESS_TRANSFER_WRITE_BIT,
  1569. image_shader_memory_barrier.dstAccessMask = VK_ACCESS_SHADER_READ_BIT,
  1570. vkCmdPipelineBarrier(command_buffer, VK_PIPELINE_STAGE_TRANSFER_BIT,
  1571. VK_PIPELINE_STAGE_FRAGMENT_SHADER_BIT, 0, 0, NULL, 0,
  1572. NULL, 1, &image_shader_memory_barrier);
  1573. result = vkEndCommandBuffer(command_buffer);
  1574. if (result != VK_SUCCESS) {
  1575. fprintf(stderr, "vkEndCommandBuffer failed for demo texture: %d\n",
  1576. result);
  1577. return false;
  1578. }
  1579. memset(&fence_create, 0, sizeof(VkFenceCreateInfo));
  1580. fence_create.sType = VK_STRUCTURE_TYPE_FENCE_CREATE_INFO;
  1581. result = vkCreateFence(demo->device, &fence_create, NULL, &fence);
  1582. if (result != VK_SUCCESS) {
  1583. fprintf(stderr, "vkCreateFence failed for demo texture: %d\n", result);
  1584. return false;
  1585. }
  1586. memset(&submit_info, 0, sizeof(VkSubmitInfo));
  1587. submit_info.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO;
  1588. submit_info.commandBufferCount = 1;
  1589. submit_info.pCommandBuffers = &command_buffer;
  1590. result = vkQueueSubmit(demo->graphics_queue, 1, &submit_info, fence);
  1591. if (result != VK_SUCCESS) {
  1592. fprintf(stderr, "vkQueueSubmit failed for demo texture: %d\n", result);
  1593. return false;
  1594. }
  1595. result = vkWaitForFences(demo->device, 1, &fence, VK_TRUE, UINT64_MAX);
  1596. if (result != VK_SUCCESS) {
  1597. fprintf(stderr, "vkWaitForFences failed for demo texture: %d\n",
  1598. result);
  1599. return false;
  1600. }
  1601. vkDestroyBuffer(demo->device, staging_buffer.buffer, NULL);
  1602. vkFreeMemory(demo->device, staging_buffer.memory, NULL);
  1603. vkDestroyFence(demo->device, fence, NULL);
  1604. return true;
  1605. }
  1606. bool create_vulkan_demo(struct vulkan_demo *demo) {
  1607. if (!create_instance(demo)) {
  1608. return false;
  1609. }
  1610. if (!create_surface(demo)) {
  1611. return false;
  1612. }
  1613. if (!create_physical_device(demo)) {
  1614. return false;
  1615. }
  1616. if (!create_logical_device(demo)) {
  1617. return false;
  1618. }
  1619. if (!create_sampler(demo)) {
  1620. return false;
  1621. }
  1622. if (!create_descriptor_set_layout(demo)) {
  1623. return false;
  1624. }
  1625. if (!create_swap_chain_related_resources(demo)) {
  1626. return false;
  1627. }
  1628. if (!create_descriptor_pool(demo)) {
  1629. return false;
  1630. }
  1631. if (!create_descriptor_sets(demo)) {
  1632. return false;
  1633. }
  1634. if (!create_command_pool(demo)) {
  1635. return false;
  1636. }
  1637. if (!create_command_buffers(demo)) {
  1638. return false;
  1639. }
  1640. if (!create_semaphores(demo)) {
  1641. return false;
  1642. }
  1643. if (!create_fence(demo)) {
  1644. return false;
  1645. }
  1646. if (!create_demo_texture(demo)) {
  1647. return false;
  1648. }
  1649. demo->framebuffer_resized = false;
  1650. return true;
  1651. }
  1652. bool recreate_swap_chain(struct vulkan_demo *demo) {
  1653. printf("recreating swapchain\n");
  1654. if (!destroy_swap_chain_related_resources(demo)) {
  1655. return false;
  1656. }
  1657. if (!create_swap_chain_related_resources(demo)) {
  1658. return false;
  1659. }
  1660. update_descriptor_sets(demo);
  1661. nk_glfw3_resize(demo->swap_chain_image_extent.width,
  1662. demo->swap_chain_image_extent.height);
  1663. demo->framebuffer_resized = false;
  1664. return true;
  1665. }
  1666. bool render(struct vulkan_demo *demo, struct nk_colorf *bg,
  1667. VkSemaphore wait_semaphore, uint32_t image_index) {
  1668. VkCommandBufferBeginInfo command_buffer_begin_info;
  1669. VkCommandBuffer command_buffer;
  1670. VkRenderPassBeginInfo render_pass_info;
  1671. VkSubmitInfo submit_info;
  1672. VkPipelineStageFlags wait_stage =
  1673. VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT;
  1674. VkResult result;
  1675. VkPresentInfoKHR present_info;
  1676. VkClearValue clear_color;
  1677. memcpy(&clear_color.color, bg, sizeof(VkClearColorValue));
  1678. memset(&command_buffer_begin_info, 0, sizeof(VkCommandBufferBeginInfo));
  1679. command_buffer_begin_info.sType =
  1680. VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO;
  1681. command_buffer = demo->command_buffers[image_index];
  1682. result = vkBeginCommandBuffer(command_buffer, &command_buffer_begin_info);
  1683. if (result != VK_SUCCESS) {
  1684. fprintf(stderr, "vkBeginCommandBuffer failed: %d\n", result);
  1685. return false;
  1686. }
  1687. memset(&render_pass_info, 0, sizeof(VkRenderPassBeginInfo));
  1688. render_pass_info.sType = VK_STRUCTURE_TYPE_RENDER_PASS_BEGIN_INFO;
  1689. render_pass_info.renderPass = demo->render_pass;
  1690. render_pass_info.framebuffer = demo->framebuffers[image_index];
  1691. render_pass_info.renderArea.offset.x = 0;
  1692. render_pass_info.renderArea.offset.y = 0;
  1693. render_pass_info.renderArea.extent = demo->swap_chain_image_extent;
  1694. render_pass_info.clearValueCount = 1;
  1695. render_pass_info.pClearValues = &clear_color;
  1696. vkCmdBeginRenderPass(command_buffer, &render_pass_info,
  1697. VK_SUBPASS_CONTENTS_INLINE);
  1698. vkCmdBindPipeline(command_buffer, VK_PIPELINE_BIND_POINT_GRAPHICS,
  1699. demo->pipeline);
  1700. vkCmdBindDescriptorSets(command_buffer, VK_PIPELINE_BIND_POINT_GRAPHICS,
  1701. demo->pipeline_layout, 0, 1,
  1702. &demo->descriptor_sets[image_index], 0, NULL);
  1703. vkCmdDraw(command_buffer, 3, 1, 0, 0);
  1704. vkCmdEndRenderPass(command_buffer);
  1705. result = vkEndCommandBuffer(command_buffer);
  1706. if (result != VK_SUCCESS) {
  1707. fprintf(stderr, "vkEndCommandBuffer failed: %d\n", result);
  1708. return false;
  1709. }
  1710. memset(&submit_info, 0, sizeof(VkSubmitInfo));
  1711. submit_info.sType = VK_STRUCTURE_TYPE_SUBMIT_INFO;
  1712. submit_info.waitSemaphoreCount = 1;
  1713. submit_info.pWaitSemaphores = &wait_semaphore;
  1714. submit_info.pWaitDstStageMask = &wait_stage;
  1715. submit_info.commandBufferCount = 1;
  1716. submit_info.pCommandBuffers = &demo->command_buffers[image_index];
  1717. submit_info.signalSemaphoreCount = 1;
  1718. submit_info.pSignalSemaphores = &demo->render_finished;
  1719. result = vkQueueSubmit(demo->graphics_queue, 1, &submit_info,
  1720. demo->render_fence);
  1721. if (result != VK_SUCCESS) {
  1722. fprintf(stderr, "vkQueueSubmit failed: %d\n", result);
  1723. return false;
  1724. }
  1725. memset(&present_info, 0, sizeof(VkPresentInfoKHR));
  1726. present_info.sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR;
  1727. present_info.waitSemaphoreCount = 1;
  1728. present_info.pWaitSemaphores = &demo->render_finished;
  1729. present_info.swapchainCount = 1;
  1730. present_info.pSwapchains = &demo->swap_chain;
  1731. present_info.pImageIndices = &image_index;
  1732. result = vkQueuePresentKHR(demo->present_queue, &present_info);
  1733. if (result == VK_ERROR_OUT_OF_DATE_KHR || result == VK_SUBOPTIMAL_KHR || demo->framebuffer_resized) {
  1734. recreate_swap_chain(demo);
  1735. } else if (result != VK_SUCCESS) {
  1736. fprintf(stderr, "vkQueuePresentKHR failed: %d\n", result);
  1737. return false;
  1738. }
  1739. return true;
  1740. }
  1741. VkResult
  1742. destroy_debug_utils_messenger_ext(VkInstance instance,
  1743. VkDebugUtilsMessengerEXT debugMessenger,
  1744. const VkAllocationCallbacks *pAllocator) {
  1745. PFN_vkDestroyDebugUtilsMessengerEXT func =
  1746. (PFN_vkDestroyDebugUtilsMessengerEXT)vkGetInstanceProcAddr(
  1747. instance, "vkDestroyDebugUtilsMessengerEXT");
  1748. if (func != NULL) {
  1749. func(instance, debugMessenger, pAllocator);
  1750. return VK_SUCCESS;
  1751. } else {
  1752. return VK_ERROR_EXTENSION_NOT_PRESENT;
  1753. }
  1754. }
  1755. bool cleanup(struct vulkan_demo *demo) {
  1756. VkResult result;
  1757. printf("cleaning up\n");
  1758. result = vkDeviceWaitIdle(demo->device);
  1759. if (result != VK_SUCCESS) {
  1760. fprintf(stderr, "vkDeviceWaitIdle failed: %d\n", result);
  1761. return false;
  1762. }
  1763. destroy_swap_chain_related_resources(demo);
  1764. vkFreeCommandBuffers(demo->device, demo->command_pool,
  1765. demo->swap_chain_images_len, demo->command_buffers);
  1766. vkDestroyCommandPool(demo->device, demo->command_pool, NULL);
  1767. vkDestroySampler(demo->device, demo->sampler, NULL);
  1768. vkDestroySemaphore(demo->device, demo->render_finished, NULL);
  1769. vkDestroySemaphore(demo->device, demo->image_available, NULL);
  1770. vkDestroyFence(demo->device, demo->render_fence, NULL);
  1771. vkDestroyImage(demo->device, demo->demo_texture_image, NULL);
  1772. vkDestroyImageView(demo->device, demo->demo_texture_image_view, NULL);
  1773. vkFreeMemory(demo->device, demo->demo_texture_memory, NULL);
  1774. vkDestroyDescriptorSetLayout(demo->device, demo->descriptor_set_layout,
  1775. NULL);
  1776. vkDestroyDescriptorPool(demo->device, demo->descriptor_pool, NULL);
  1777. vkDestroyDevice(demo->device, NULL);
  1778. vkDestroySurfaceKHR(demo->instance, demo->surface, NULL);
  1779. if (demo->debug_messenger) {
  1780. result = destroy_debug_utils_messenger_ext(demo->instance,
  1781. demo->debug_messenger, NULL);
  1782. if (result != VK_SUCCESS) {
  1783. fprintf(stderr, "Couldn't destroy debug messenger: %d\n", result);
  1784. return false;
  1785. }
  1786. }
  1787. vkDestroyInstance(demo->instance, NULL);
  1788. if (demo->swap_chain_images) {
  1789. free(demo->swap_chain_images);
  1790. }
  1791. if (demo->swap_chain_image_views) {
  1792. free(demo->swap_chain_image_views);
  1793. }
  1794. if (demo->overlay_images) {
  1795. free(demo->overlay_images);
  1796. }
  1797. if (demo->overlay_image_views) {
  1798. free(demo->overlay_image_views);
  1799. }
  1800. if (demo->overlay_image_memories) {
  1801. free(demo->overlay_image_memories);
  1802. }
  1803. if (demo->descriptor_sets) {
  1804. free(demo->descriptor_sets);
  1805. }
  1806. if (demo->framebuffers) {
  1807. free(demo->framebuffers);
  1808. }
  1809. if (demo->command_buffers) {
  1810. free(demo->command_buffers);
  1811. }
  1812. return true;
  1813. }
  1814. int main(void) {
  1815. struct vulkan_demo demo;
  1816. struct nk_context *ctx;
  1817. struct nk_colorf bg;
  1818. struct nk_image img;
  1819. uint32_t image_index;
  1820. VkResult result;
  1821. VkSemaphore nk_semaphore;
  1822. #ifdef INCLUDE_CONFIGURATOR
  1823. static struct nk_color color_table[NK_COLOR_COUNT];
  1824. memcpy(color_table, nk_default_color_style, sizeof(color_table));
  1825. #endif
  1826. glfwSetErrorCallback(glfw_error_callback);
  1827. if (!glfwInit()) {
  1828. fprintf(stderr, "[GFLW] failed to init!\n");
  1829. exit(1);
  1830. }
  1831. glfwWindowHint(GLFW_CLIENT_API, GLFW_NO_API);
  1832. memset(&demo, 0, sizeof(struct vulkan_demo));
  1833. demo.win =
  1834. glfwCreateWindow(WINDOW_WIDTH, WINDOW_HEIGHT, "Demo", NULL, NULL);
  1835. glfwSetWindowUserPointer(demo.win, &demo);
  1836. glfwSetFramebufferSizeCallback(demo.win, glfw_framebuffer_resize_callback);
  1837. if (!create_vulkan_demo(&demo)) {
  1838. fprintf(stderr, "failed to create vulkan demo!\n");
  1839. exit(1);
  1840. }
  1841. ctx = nk_glfw3_init(
  1842. demo.win, demo.device, demo.physical_device, demo.indices.graphics,
  1843. demo.overlay_image_views, demo.swap_chain_images_len,
  1844. demo.swap_chain_image_format, NK_GLFW3_INSTALL_CALLBACKS,
  1845. MAX_VERTEX_BUFFER, MAX_ELEMENT_BUFFER);
  1846. /* Load Fonts: if none of these are loaded a default font will be used */
  1847. /* Load Cursor: if you uncomment cursor loading please hide the cursor */
  1848. {
  1849. struct nk_font_atlas *atlas;
  1850. nk_glfw3_font_stash_begin(&atlas);
  1851. /*struct nk_font *droid = nk_font_atlas_add_from_file(atlas,
  1852. * "../../../extra_font/DroidSans.ttf", 14, 0);*/
  1853. /*struct nk_font *roboto = nk_font_atlas_add_from_file(atlas,
  1854. * "../../../extra_font/Roboto-Regular.ttf", 14, 0);*/
  1855. /*struct nk_font *future = nk_font_atlas_add_from_file(atlas,
  1856. * "../../../extra_font/kenvector_future_thin.ttf", 13, 0);*/
  1857. /*struct nk_font *clean = nk_font_atlas_add_from_file(atlas,
  1858. * "../../../extra_font/ProggyClean.ttf", 12, 0);*/
  1859. /*struct nk_font *tiny = nk_font_atlas_add_from_file(atlas,
  1860. * "../../../extra_font/ProggyTiny.ttf", 10, 0);*/
  1861. /*struct nk_font *cousine = nk_font_atlas_add_from_file(atlas,
  1862. * "../../../extra_font/Cousine-Regular.ttf", 13, 0);*/
  1863. nk_glfw3_font_stash_end(demo.graphics_queue);
  1864. /*nk_style_load_all_cursors(ctx, atlas->cursors);*/
  1865. /*nk_style_set_font(ctx, &droid->handle);*/}
  1866. img = nk_image_ptr(demo.demo_texture_image_view);
  1867. bg.r = 0.10f, bg.g = 0.18f, bg.b = 0.24f, bg.a = 1.0f;
  1868. while (!glfwWindowShouldClose(demo.win)) {
  1869. /* Input */
  1870. glfwPollEvents();
  1871. nk_glfw3_new_frame();
  1872. /* GUI */
  1873. if (nk_begin(ctx, "Demo", nk_rect(50, 50, 230, 250),
  1874. NK_WINDOW_BORDER | NK_WINDOW_MOVABLE | NK_WINDOW_SCALABLE |
  1875. NK_WINDOW_MINIMIZABLE | NK_WINDOW_TITLE)) {
  1876. enum { EASY, HARD };
  1877. static int op = EASY;
  1878. static int property = 20;
  1879. nk_layout_row_static(ctx, 30, 80, 1);
  1880. if (nk_button_label(ctx, "button"))
  1881. fprintf(stdout, "button pressed\n");
  1882. nk_layout_row_dynamic(ctx, 30, 2);
  1883. if (nk_option_label(ctx, "easy", op == EASY))
  1884. op = EASY;
  1885. if (nk_option_label(ctx, "hard", op == HARD))
  1886. op = HARD;
  1887. nk_layout_row_dynamic(ctx, 25, 1);
  1888. nk_property_int(ctx, "Compression:", 0, &property, 100, 10, 1);
  1889. nk_layout_row_dynamic(ctx, 20, 1);
  1890. nk_label(ctx, "background:", NK_TEXT_LEFT);
  1891. nk_layout_row_dynamic(ctx, 25, 1);
  1892. if (nk_combo_begin_color(ctx, nk_rgb_cf(bg),
  1893. nk_vec2(nk_widget_width(ctx), 400))) {
  1894. nk_layout_row_dynamic(ctx, 120, 1);
  1895. bg = nk_color_picker(ctx, bg, NK_RGBA);
  1896. nk_layout_row_dynamic(ctx, 25, 1);
  1897. bg.r = nk_propertyf(ctx, "#R:", 0, bg.r, 1.0f, 0.01f, 0.005f);
  1898. bg.g = nk_propertyf(ctx, "#G:", 0, bg.g, 1.0f, 0.01f, 0.005f);
  1899. bg.b = nk_propertyf(ctx, "#B:", 0, bg.b, 1.0f, 0.01f, 0.005f);
  1900. bg.a = nk_propertyf(ctx, "#A:", 0, bg.a, 1.0f, 0.01f, 0.005f);
  1901. nk_combo_end(ctx);
  1902. }
  1903. }
  1904. nk_end(ctx);
  1905. /* Bindless Texture */
  1906. if (nk_begin(ctx, "Texture", nk_rect(500, 300, 200, 200),
  1907. NK_WINDOW_BORDER | NK_WINDOW_MOVABLE | NK_WINDOW_SCALABLE |
  1908. NK_WINDOW_MINIMIZABLE | NK_WINDOW_TITLE)) {
  1909. struct nk_command_buffer *canvas = nk_window_get_canvas(ctx);
  1910. struct nk_rect total_space = nk_window_get_content_region(ctx);
  1911. nk_draw_image(canvas, total_space, &img, nk_white);
  1912. }
  1913. nk_end(ctx);
  1914. /* -------------- EXAMPLES ---------------- */
  1915. #ifdef INCLUDE_CALCULATOR
  1916. calculator(ctx);
  1917. #endif
  1918. #ifdef INCLUDE_CANVAS
  1919. canvas(ctx);
  1920. #endif
  1921. #ifdef INCLUDE_OVERVIEW
  1922. overview(ctx);
  1923. #endif
  1924. #ifdef INCLUDE_CONFIGURATOR
  1925. style_configurator(ctx, color_table);
  1926. #endif
  1927. #ifdef INCLUDE_NODE_EDITOR
  1928. node_editor(ctx);
  1929. #endif
  1930. /* ----------------------------------------- */
  1931. result = vkWaitForFences(demo.device, 1, &demo.render_fence, VK_TRUE,
  1932. UINT64_MAX);
  1933. if (result != VK_SUCCESS) {
  1934. fprintf(stderr, "vkWaitForFences failed: %d\n", result);
  1935. return false;
  1936. }
  1937. result = vkResetFences(demo.device, 1, &demo.render_fence);
  1938. if (result != VK_SUCCESS) {
  1939. fprintf(stderr, "vkResetFences failed: %d\n", result);
  1940. return false;
  1941. }
  1942. result =
  1943. vkAcquireNextImageKHR(demo.device, demo.swap_chain, UINT64_MAX,
  1944. demo.image_available, NULL, &image_index);
  1945. if (result == VK_ERROR_OUT_OF_DATE_KHR) {
  1946. recreate_swap_chain(&demo);
  1947. /* If vkAcquireNextImageKHR does not successfully acquire an image,
  1948. * semaphore and fence are unaffected. */
  1949. continue;
  1950. }
  1951. if (result != VK_SUCCESS && result != VK_SUBOPTIMAL_KHR) {
  1952. fprintf(stderr, "vkAcquireNextImageKHR failed: %d\n", result);
  1953. return false;
  1954. }
  1955. /* Draw */
  1956. nk_semaphore =
  1957. nk_glfw3_render(demo.graphics_queue, image_index,
  1958. demo.image_available, NK_ANTI_ALIASING_ON);
  1959. if (!render(&demo, &bg, nk_semaphore, image_index)) {
  1960. fprintf(stderr, "render failed\n");
  1961. return false;
  1962. }
  1963. }
  1964. nk_glfw3_shutdown();
  1965. cleanup(&demo);
  1966. glfwTerminate();
  1967. return 0;
  1968. }