-#include "vulkan/vk.h"
-extern GLFWwindow win;
+#define GLFW_INCLUDE_VULKAN
+#include <GLFW/glfw3.h>
+
+#include <alloca.h>
+#include <stdio.h>
+#include <string.h>
+#include <string>
+#include <vector>
+
+#include "gfxapi.h"
+#include "vkutil.h"
+
+/* global variables */
+extern GLFWwindow *win;
+extern int win_w;
+extern int win_h;
+
+/* static functions */
+static void error_callback(int error, const char *descr);
+static void clear(float r, float g, float b);
+static void viewport(int x, int y, int width, int height);
+static void zbuffer(bool enable);
+static void cull_face(Gfx_cull_face cf);
+static void reshape(int width, int height);
bool init_vulkan()
{
+ if(!glfwInit()) {
+ fprintf(stderr, "Failed to initialize GLFW.\n");
+ return false;
+ }
+
+ if(!glfwVulkanSupported()) {
+ fprintf(stderr, "No Vulkan support on the device.\n");
+ return false;
+ }
+
+ glfwSetErrorCallback(error_callback);
+
+ if(!vku_create_device()) {
+ fprintf(stderr, "Failed to initialize vulkan.\n");
+ return false;
+ }
+
+ if(!glfwGetPhysicalDevicePresentationSupport(vkinst, vkpdev, vkqfamily)) {
+ fprintf(stderr, "Presentation support not found.\n");
+ return false;
+ }
+
+ glfwWindowHint(GLFW_CLIENT_API, GLFW_NO_API);
+ if(!(win = glfwCreateWindow(win_w, win_h, "vkcow", 0, 0))) {
+ fprintf(stderr, "Failed to create window.\n");
+ return false;
+ }
+
+ if(VkResult err = glfwCreateWindowSurface(vkinst, win, 0, &vksurface)) {
+ fprintf(stderr, "Failed to create KHR surface: %s\n", vku_get_vulkan_error_str(err));
+ return false;
+ }
+
+ if(!(vkswapchain = vku_create_swapchain(vksurface, win_w, win_h, 2,
+ VK_PRESENT_MODE_FIFO_KHR, 0))) {
+ fprintf(stderr, "Failed to create swapchain.\n");
+ return false;
+ }
+
+ vkswapchain_images = vku_get_swapchain_images(vkswapchain, &vknum_swapchain_images);
+ if(!vkswapchain_images) {
+ fprintf(stderr, "Failed to get swapchain images.\n");
+ return false;
+ }
+
+ vkswapchain_views = vku_create_image_views(vkswapchain_images, vknum_swapchain_images);
+ if(!vkswapchain_views) {
+ fprintf(stderr, "Failed to create swapchain image views.\n");
+ delete [] vkswapchain_images;
+ return false;
+ }
+
+ if(!vku_create_renderpass()) {
+ fprintf(stderr, "Failed to create renderpass'\n");
+ return false;
+ }
+
+ if(!vku_create_framebuffers(vkswapchain_views, vknum_swapchain_images, win_w, win_h)) {
+ fprintf(stderr, "Failed to create framebuffers.\n");
+ return false;
+ }
+
+ gfx_clear = clear;
+ gfx_viewport = viewport;
+ gfx_zbuffer = zbuffer;
+ gfx_cull_face = cull_face;
+ gfx_reshape = reshape;
+
return true;
}
void cleanup_vulkan()
{
+ if(win) {
+ glfwDestroyWindow(win);
+ }
+ glfwTerminate();
+
+ //TODOs according to the book:
+ // 1- make sure all threads have been terminated (when I add threads)
+
+ // 2- destroy objects in *reverse* order
+ vkDestroyRenderPass(vkdev, vkrpass, 0);
+ vkDestroySurfaceKHR(vkinst, vksurface, 0);
+
+ vku_cleanup();
}
-GLFWwindow *create_vulkan_window()
+static void error_callback(int error, const char *description)
{
- return 0;
+ fprintf(stderr, "GLFW error %d: %s.\n", error, description);
}
-void clbk_clear_vk()
+static void reshape(int width, int height)
{
-}
\ No newline at end of file
+ VkSwapchainKHR sc;
+ if(!(sc = vku_create_swapchain(vksurface, width, height, 2, VK_PRESENT_MODE_FIFO_KHR,
+ vkswapchain))) {
+ fprintf(stderr, "Failed to create %dx%d double-buffered swapchain\n", width, height);
+ return;
+ }
+ vkswapchain = sc;
+
+ delete [] vkswapchain_images;
+ vkswapchain_images = vku_get_swapchain_images(sc, 0);
+ vknext_swapchain_image = vku_get_next_image(vkswapchain);
+}
+
+static void clear(float r, float g, float b)
+{
+}
+
+static void viewport(int x, int y, int width, int height)
+{
+}
+
+static void zbuffer(bool enable)
+{
+}
+
+static void cull_face(Gfx_cull_face cf)
+{
+}