From 769b2a871619f1027829e40901d3374a1fe263a8 Mon Sep 17 00:00:00 2001 From: Dominic Masters Date: Tue, 16 Feb 2021 19:22:12 +1100 Subject: [PATCH] Vulkan hard :( --- .gitignore | 70 +++++++++ CMakeLists.txt | 60 ++++++++ assets/hello.txt | 1 + src/main.c | 319 ++++++++++++++++++++++++++++++++++++++++++ src/main.h | 42 ++++++ test/display/render.c | 75 ++++++++++ test/display/render.h | 62 ++++++++ test/display/shader.c | 9 ++ test/display/shader.h | 36 +++++ test/file/asset.c | 62 ++++++++ test/file/asset.h | 59 ++++++++ test/game/game.c | 32 +++++ test/game/game.h | 33 +++++ test/main.c | 17 +++ 14 files changed, 877 insertions(+) create mode 100644 .gitignore create mode 100644 CMakeLists.txt create mode 100644 assets/hello.txt create mode 100644 src/main.c create mode 100644 src/main.h create mode 100644 test/display/render.c create mode 100644 test/display/render.h create mode 100644 test/display/shader.c create mode 100644 test/display/shader.h create mode 100644 test/file/asset.c create mode 100644 test/file/asset.h create mode 100644 test/game/game.c create mode 100644 test/game/game.h create mode 100644 test/main.c diff --git a/.gitignore b/.gitignore new file mode 100644 index 00000000..de160309 --- /dev/null +++ b/.gitignore @@ -0,0 +1,70 @@ +# Prerequisites +*.d + +# Object files +*.o +*.ko +*.obj +*.elf + +# Linker output +*.ilk +*.map +*.exp + +# Precompiled Headers +*.gch +*.pch + +# Libraries +*.lib +*.a +*.la +*.lo + +# Shared objects (inc. Windows DLLs) +*.dll +*.so +*.so.* +*.dylib + +# Executables +*.exe +*.out +*.app +*.i*86 +*.x86_64 +*.hex + +# Debug files +*.dSYM/ +*.su +*.idb +*.pdb + +# Kernel Module Compile Results +*.mod* +*.cmd +.tmp_versions/ +modules.order +Module.symvers +Mkfile.old +dkms.conf + +# CMake +CMakeLists.txt.user +CMakeCache.txt +CMakeFiles +CMakeScripts +Testing +Makefile +cmake_install.cmake +install_manifest.txt +compile_commands.json +CTestTestfile.cmake +_deps + +# Custom +build +.vscode +lib \ No newline at end of file diff --git a/CMakeLists.txt b/CMakeLists.txt new file mode 100644 index 00000000..7bac9935 --- /dev/null +++ b/CMakeLists.txt @@ -0,0 +1,60 @@ +#################################### CMAKE ##################################### +cmake_minimum_required(VERSION 3.15) +set(CMAKE_C_STANDARD 99) +set(CMAKE_C_STANDARD_REQUIRED ON) + +#Include +include(FetchContent) + +#################################### PROJECT ################################### +project(Dawn VERSION 1.0) + +##################################### SRCS ##################################### +file(GLOB_RECURSE SOURCE_FILES ${CMAKE_SOURCE_DIR}/src/*.c) +file(GLOB_RECURSE HEADER_FILES ${CMAKE_SOURCE_DIR}/src/*.h) +file(COPY ${CMAKE_CURRENT_LIST_DIR}/assets DESTINATION ${CMAKE_CURRENT_BINARY_DIR}) + +##################################### LIBS ################ ##################### +include_directories(${CMAKE_SOURCE_DIR}/lib/stb) + +################################## EXECUTABLE ################################## +add_executable(${PROJECT_NAME} ${HEADER_FILES} ${SOURCE_FILES}) + +################################# STATIC LIBS ################################## +# GLFW +# find_package(glfw3 3.3.2) +if(NOT glfw3_FOUND) + FetchContent_Declare( + glfw + GIT_REPOSITORY https://github.com/glfw/glfw + GIT_TAG 3.3.2 + ) + FetchContent_MakeAvailable(glfw) +endif() +target_link_libraries(${PROJECT_NAME} glfw) + + +# CGLM +# find_package(cglm) +if(NOT cglm_FOUND) + FetchContent_Declare( + cglm + GIT_REPOSITORY https://github.com/recp/cglm + GIT_TAG v0.7.9 + ) + FetchContent_MakeAvailable(cglm) +endif() +target_link_libraries(${PROJECT_NAME} cglm) + + +# OpenMP +# find_package(OpenMP) +# if (OPENMP_FOUND) +# set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${OpenMP_C_FLAGS}") +# set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${OpenMP_CXX_FLAGS}") +# set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} ${OpenMP_EXE_LINKER_FLAGS}") +# endif() + +# Vulkan +find_package(Vulkan REQUIRED FATAL_ERROR) +target_link_libraries(${PROJECT_NAME} Vulkan::Vulkan) \ No newline at end of file diff --git a/assets/hello.txt b/assets/hello.txt new file mode 100644 index 00000000..c57eff55 --- /dev/null +++ b/assets/hello.txt @@ -0,0 +1 @@ +Hello World! \ No newline at end of file diff --git a/src/main.c b/src/main.c new file mode 100644 index 00000000..fbdb0c55 --- /dev/null +++ b/src/main.c @@ -0,0 +1,319 @@ +#include "main.h" + +int32_t main() { + // Init variables + int i; + + //Prepare GLFW, get some info now that we pass into vulkan later. + if(!glfwInit()) { + printf("Failed to init glfw\n"); + return 1; + } + glfwWindowHint(GLFW_CLIENT_API, GLFW_NO_API); + glfwWindowHint(GLFW_RESIZABLE, GLFW_FALSE); + GLFWwindow *window = glfwCreateWindow( + WINDOW_WIDTH, WINDOW_HEIGHT, "Vulkan window", NULL, NULL + ); + uint32_t glfwExtensionCount = 0; + const char** glfwExtensions = glfwGetRequiredInstanceExtensions(&glfwExtensionCount); + + //Prepare information about the app for Vulkan + VkApplicationInfo appInfo = { + .sType = VK_STRUCTURE_TYPE_APPLICATION_INFO, + .pNext = NULL, + .pApplicationName = "Step 1", + .applicationVersion = 1, + .pEngineName = NULL, + .engineVersion = VK_MAKE_VERSION(1, 0, 0), + .apiVersion = VK_API_VERSION_1_0 + }; + + // Create the Vulkan instance + VkInstanceCreateInfo createInfo = { + .sType = VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO, + .pNext = NULL, + .flags = 0, + .ppEnabledLayerNames = NULL, + .pApplicationInfo = &appInfo, + .enabledExtensionCount = glfwExtensionCount, + .ppEnabledExtensionNames = glfwExtensions, + .enabledLayerCount = 0 + }; + VkInstance instance; + VkResult result = vkCreateInstance(&createInfo, NULL, &instance); + if(result != VK_SUCCESS) { + printf("Failed to init vulkan\n"); + return 1; + } + + // Create the rendering surface + VkSurfaceKHR surface; + if(glfwCreateWindowSurface(instance, window, NULL, &surface) != VK_SUCCESS) { + printf("Failed to create window surface!\n"); + return 1; + } + + // Get and print the extension support + uint32_t extensionCount = 0; + vkEnumerateInstanceExtensionProperties(NULL, &extensionCount, NULL); + VkExtensionProperties *extensionProperties = malloc(sizeof(VkExtensionProperties) * extensionCount); + vkEnumerateInstanceExtensionProperties(NULL, &extensionCount, extensionProperties); + printf("Available extensions:\n"); + for(i = 0; i < extensionCount; i++) { + printf(" %s\n", extensionProperties[i].extensionName); + } + + // Physical Device finding. + uint32_t deviceCount = 0; + VkPhysicalDevice physicalDevice = VK_NULL_HANDLE; + vkEnumeratePhysicalDevices(instance, &deviceCount, NULL); + if(deviceCount == 0) { + printf("No Vulkan GPU\n"); + return 1; + } + VkPhysicalDevice *devices = malloc(sizeof(VkPhysicalDevice) * deviceCount); + + // Select a suitable device from that list of devices we just made. + QueueFamilyIndices indices; + SwapChainSupportDetails details; + vkEnumeratePhysicalDevices(instance, &deviceCount, devices); + for(i = 0; i < deviceCount; i++) { + VkPhysicalDevice device = devices[i]; + VkPhysicalDeviceProperties deviceProperties; + VkPhysicalDeviceFeatures deviceFeatures; + vkGetPhysicalDeviceProperties(device, &deviceProperties); + vkGetPhysicalDeviceFeatures(device, &deviceFeatures); + + // For this we select a DISCRETE GPU, not an internal. + if(deviceProperties.deviceType != VK_PHYSICAL_DEVICE_TYPE_DISCRETE_GPU) continue; + // Make sure it can run a geom shader + if(!deviceFeatures.geometryShader) continue; + + // This is something I don't really understand but I know we need it. + // Follow the method for info. + indices = findQueueFamilies(device); + if(!indices.graphicsFamilyFound) continue; + + // Supports Presenting? + VkBool32 presentSupport = false; + vkGetPhysicalDeviceSurfaceSupportKHR(device, i, surface, &presentSupport); + if(!presentSupport) continue; + indices.presentFamily = i; + indices.presentFamilyFound = true; + + // Now I'm trying to find out what extensions this device supports + uint32_t extensionCount; + vkEnumerateDeviceExtensionProperties(device, NULL, &extensionCount, NULL); + VkExtensionProperties *availableExtensions = malloc(sizeof(VkExtensionProperties) * extensionCount); + vkEnumerateDeviceExtensionProperties(device, NULL, &extensionCount, availableExtensions); + + // Make sure it supports the things I need + int countHas = 0; + for(int j = 0; j < extensionCount; j++) { + VkExtensionProperties prop = availableExtensions[j]; + if(strcmp(prop.extensionName, VK_KHR_SWAPCHAIN_EXTENSION_NAME) == 0) { + countHas++; + continue; + } + } + if(countHas == 0) continue; + + // Find out the capabilities of the swap chain. + vkGetPhysicalDeviceSurfaceCapabilitiesKHR(device, surface, &details.capabilities); + + uint32_t formatCount, presentModeCount; + vkGetPhysicalDeviceSurfaceFormatsKHR(device, surface, &formatCount, NULL); + if(formatCount == 0) continue; + + vkGetPhysicalDeviceSurfacePresentModesKHR(device, surface, &presentModeCount, NULL); + if(presentModeCount == 0) continue; + + details.formats = malloc(sizeof(VkSurfaceFormatKHR) * formatCount); + details.presentModes = malloc(sizeof(VkPresentModeKHR) * presentModeCount); + vkGetPhysicalDeviceSurfaceFormatsKHR(device, surface, &formatCount, details.formats); + vkGetPhysicalDeviceSurfacePresentModesKHR(device, surface, &presentModeCount, details.presentModes); + + // Determine the surface format + details.surfaceFormat = details.formats[0]; + for(int j = 0; j < formatCount; j++) { + VkSurfaceFormatKHR availableFormat = details.formats[j]; + if(availableFormat.format != VK_FORMAT_B8G8R8A8_SRGB) continue; + if(availableFormat.colorSpace != VK_COLOR_SPACE_SRGB_NONLINEAR_KHR) continue; + details.surfaceFormat = availableFormat; + break; + } + + // Choose the surface mode, prefer triple buffered, fallback to vsync + details.presentMode = VK_PRESENT_MODE_FIFO_KHR; + for(int j = 0; j < presentModeCount; j++) { + VkPresentModeKHR availableMode = details.presentModes[j]; + if(availableMode != VK_PRESENT_MODE_MAILBOX_KHR) continue; + details.presentMode = VK_PRESENT_MODE_MAILBOX_KHR; + break; + } + + physicalDevice = device; + break; + } + + // Did we find it? + if(physicalDevice == VK_NULL_HANDLE) { + printf("No Vulkan Device\n"); + return 1; + } + + // Swap Extent + if (details.capabilities.currentExtent.width != UINT32_MAX) { + details.extent = details.capabilities.currentExtent; + } else { + int width, height; + glfwGetFramebufferSize(window, &width, &height); + + VkExtent2D actualExtent = { + .width = width, + .height = height + }; + + actualExtent.width = MAX( + details.capabilities.minImageExtent.width, + MIN(details.capabilities.maxImageExtent.width, actualExtent.width) + ); + actualExtent.height = MAX( + details.capabilities.minImageExtent.height, + MIN(details.capabilities.maxImageExtent.height, actualExtent.height) + ); + + details.extent = actualExtent; + } + + uint32_t imageCount = details.capabilities.minImageCount + 1; + if (details.capabilities.maxImageCount > 0 && imageCount > details.capabilities.maxImageCount) { + imageCount = details.capabilities.maxImageCount; + } + + VkSwapchainCreateInfoKHR swapChainCreateInfo = { + .sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR, + .surface = surface, + .minImageCount = imageCount, + .imageFormat = details.surfaceFormat.format, + .imageColorSpace = details.surfaceFormat.colorSpace, + .imageExtent = details.extent, + .imageArrayLayers = 1, + .imageUsage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT, + .preTransform = details.capabilities.currentTransform, + .compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR, + .presentMode = details.presentMode, + .clipped = VK_TRUE, + .oldSwapchain = VK_NULL_HANDLE + }; + + uint32_t queueFamilyIndices[2]; + queueFamilyIndices[0] = indices.graphicsFamily; + queueFamilyIndices[1] = indices.presentFamily; + if (indices.graphicsFamily != indices.presentFamily) { + swapChainCreateInfo.imageSharingMode = VK_SHARING_MODE_CONCURRENT; + swapChainCreateInfo.queueFamilyIndexCount = 2; + swapChainCreateInfo.pQueueFamilyIndices = queueFamilyIndices; + } else { + swapChainCreateInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE; + swapChainCreateInfo.queueFamilyIndexCount = 0; // Optional + swapChainCreateInfo.pQueueFamilyIndices = NULL; // Optional + } + + // Now we create a LOGICAL device. I think this is a way of reserving the + // queue from the indices and using it for our own needs. + float queuePriority = 1.0f; + VkDeviceQueueCreateInfo queueCreateInfo[DEV_QUEUE_COUNT]; + for(i = 0; i < DEV_QUEUE_COUNT; i++) { + queueCreateInfo[i].sType = VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO; + queueCreateInfo[i].queueFamilyIndex = i == 0 ? indices.graphicsFamily : indices.presentFamily; + queueCreateInfo[i].queueCount = 1; + queueCreateInfo[i].pNext = NULL; + queueCreateInfo[i].pQueuePriorities = &queuePriority;// Related to threading + } + + VkPhysicalDeviceFeatures deviceFeatures = { + .alphaToOne = VK_FALSE, + }; + + char *logicalEnabledExtensions[1]; + logicalEnabledExtensions[0] = VK_KHR_SWAPCHAIN_EXTENSION_NAME; + VkDeviceCreateInfo logicalCreateInfo = { + .sType = VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO, + .pQueueCreateInfos = queueCreateInfo, + .queueCreateInfoCount = DEV_QUEUE_COUNT, + .pEnabledFeatures = &deviceFeatures, + .enabledExtensionCount = 1, + .ppEnabledExtensionNames = logicalEnabledExtensions, + .enabledLayerCount = 0, + .pNext = NULL + }; + VkDevice device; + if(vkCreateDevice(physicalDevice, &logicalCreateInfo, NULL, &device) != VK_SUCCESS) { + printf("Failed to create logical device.\n"); + return 1; + } + + // Get back our queues we made. + VkQueue graphicsQueue, presentQueue; + vkGetDeviceQueue(device, indices.graphicsFamily, 0, &graphicsQueue); + vkGetDeviceQueue(device, indices.presentFamily, 0, &presentQueue); + + VkSwapchainKHR swapChain; + if(vkCreateSwapchainKHR(device, &swapChainCreateInfo, NULL, &swapChain) != VK_SUCCESS) { + printf("Failed to create a swap chain\n"); + return 1; + } + + //Idk + uint32_t imageCount; + vkGetSwapchainImagesKHR(device, swapChain, &imageCount, NULL); + VkImage* swapChainImages = malloc(sizeof(VkImage) * imageCount); + vkGetSwapchainImagesKHR(device, swapChain, &imageCount, swapChainImages); + + VkImageView> swapChainImageViews; + + // Begin render loop + while(!glfwWindowShouldClose(window)) { + glfwPollEvents(); + } + + // Cleanup + vkDestroySwapchainKHR(device, swapChain, NULL); + vkDestroyDevice(device, NULL); + vkDestroySurfaceKHR(instance, surface, NULL); + vkDestroyInstance(instance, NULL); + glfwDestroyWindow(window); + glfwTerminate(); + return 0; +} + +QueueFamilyIndices findQueueFamilies(VkPhysicalDevice device) { + QueueFamilyIndices indices = { + .graphicsFamily = 0, + .graphicsFamilyFound = false, + .presentFamily = 0, + .presentFamilyFound = false + }; + + // Basically what I understand is we're trying to find a specific kind of + // queue from the capabilities of queues that the device supports. I think + // that I will understand queues more later but for now we're just finding... + uint32_t queueFamilyCount = 0; + vkGetPhysicalDeviceQueueFamilyProperties(device, &queueFamilyCount, NULL); + + VkQueueFamilyProperties *properties = malloc(sizeof(VkQueueFamilyProperties) * queueFamilyCount); + vkGetPhysicalDeviceQueueFamilyProperties(device, &queueFamilyCount, properties); + + for(int i = 0; i < queueFamilyCount; i++) { + // ... A queue that has this particular bit flag. + VkQueueFamilyProperties queueFamily = properties[i]; + + if(queueFamily.queueFlags & VK_QUEUE_GRAPHICS_BIT) { + indices.graphicsFamily = i; + indices.graphicsFamilyFound = true; + } + } + + return indices; +} \ No newline at end of file diff --git a/src/main.h b/src/main.h new file mode 100644 index 00000000..3f0bbf9a --- /dev/null +++ b/src/main.h @@ -0,0 +1,42 @@ +#pragma once +#include +#include +#include +#include + +#define VK_USE_PLATFORM_WIN32_KHR +#define GLFW_INCLUDE_VULKAN +#include +#include + +#define WINDOW_WIDTH 800 +#define WINDOW_HEIGHT 600 +#define MIN(a,b) (((a)<(b))?(a):(b)) +#define MAX(a,b) (((a)>(b))?(a):(b)) +#define DEV_QUEUE_COUNT 2 + +typedef struct { + uint32_t graphicsFamily; + bool graphicsFamilyFound; + + uint32_t presentFamily; + bool presentFamilyFound; +} QueueFamilyIndices; + +QueueFamilyIndices findQueueFamilies(VkPhysicalDevice device); + +typedef struct { + VkSurfaceCapabilitiesKHR capabilities; + VkSurfaceFormatKHR *formats; + VkPresentModeKHR *presentModes; + + VkSurfaceFormatKHR surfaceFormat; + VkPresentModeKHR presentMode; + VkExtent2D extent; +} SwapChainSupportDetails; + +/** + * Entry of the program + * @return 0 if success, anything else for failure. + */ +int32_t main(); diff --git a/test/display/render.c b/test/display/render.c new file mode 100644 index 00000000..c96ad3fd --- /dev/null +++ b/test/display/render.c @@ -0,0 +1,75 @@ +#include "render.h" + +render_t *RENDER_CURRENT = NULL; + +render_t * renderInit(int32_t width, int32_t height, char *name) { + // Can't contextualize twice + if(RENDER_CURRENT != NULL) return NULL; + + // Attempt to init GLFW + if(!glfwInit()) return NULL; + + // Initialize the renderer + render_t *render = malloc(sizeof(render_t)); + if(!render) return NULL; + render->width = width; + render->height = height; + + // Create the window. + render->window = glfwCreateWindow(width, height, name, NULL, NULL); + if(!render->window) { + free(render); + glfwTerminate(); + return NULL; + } + + // Make the window the context current + glfwMakeContextCurrent(render->window); + RENDER_CURRENT = render; + + // Listeners + glfwSetWindowSizeCallback(render->window, &renderOnResize); + + //GLEnable Things + glEnable(GL_BLEND); + glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); + glEnable(GL_DEPTH_TEST); + glDepthMask(GL_TRUE); + glDepthFunc(GL_LEQUAL); + + return render; +} + +void renderFrame(render_t *render) { + glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); + + glBegin(GL_TRIANGLES); + glColor3f(1, 0, 0); + glVertex3f(-1, -1, 0); + + glColor3f(0, 1, 0); + glVertex3f(0, 1, 0); + + glColor3f(0, 0, 1); + glVertex3f(1, -1, 0); + glEnd(); +} + +bool renderDispose(render_t *render) { + // Cleanup the callback + glfwSetWindowSizeCallback(render->window, NULL); + + // Free up the renderer + free(render); + RENDER_CURRENT = NULL; + + // Terminate the GLFW context. + glfwTerminate(); + return true; +} + +void renderOnResize(GLFWwindow *window, int32_t width, int32_t height) { + RENDER_CURRENT->width = width; + RENDER_CURRENT->height = height; + glViewport(0, 0, width, height); +} \ No newline at end of file diff --git a/test/display/render.h b/test/display/render.h new file mode 100644 index 00000000..ea31f884 --- /dev/null +++ b/test/display/render.h @@ -0,0 +1,62 @@ +#pragma once + +#include +#include +#include +#include + +#define GLFW_INCLUDE_VULKAN +#include + +#include +#include + +/** + * Contains information about the current render state, can be used for querying + * how the renderer is currently set up. + */ +typedef struct { + /** The GLFW Window Context Pointer */ + GLFWwindow *window; + + /** Resolution (in pixels) */ + int32_t width, height; +} render_t; + +/** Current active renderer. */ +extern render_t *RENDER_CURRENT; + +/** + * Initialize the renderer. + * + * @param width Width (in pixels) of the window. + * @param height Height (in pixels) of the window. + * @param name String of the windows' name. + * @return Rendering Context information. + */ +render_t * renderInit(int32_t width, int32_t height, char *name); + +/** + * Render a single frame of the render loop. The renderer is not (currently) + * responsible for render looping. + * + * @param render The renderer to loop for. + */ +void renderFrame(render_t *render); + +/** + * Cleanup a render context. + * + * @param render Renderer to cleanup. + * @return True or False if Successful/Not. + */ +bool renderDispose(render_t *render); + +/** + * Resize callbacks. + * + * @param window Window that was resized. + * @param width New window width. + * @param height New window height. + */ +void renderOnResize(GLFWwindow *window, int32_t width, int32_t height); \ No newline at end of file diff --git a/test/display/shader.c b/test/display/shader.c new file mode 100644 index 00000000..dc06de5a --- /dev/null +++ b/test/display/shader.c @@ -0,0 +1,9 @@ +#include "shader.h" + +shader_t * shaderCompile(char *vertexShaderSource, char* fragmentShaderSource) { +} + + +bool shaderDipose(shader_t *shader) { + return true; +} \ No newline at end of file diff --git a/test/display/shader.h b/test/display/shader.h new file mode 100644 index 00000000..1ee5951e --- /dev/null +++ b/test/display/shader.h @@ -0,0 +1,36 @@ +#pragma once +#include +#include +#include + +/** + * Structure containing information about an OpenGL Shader. For simplicity sake + * we demand certain uninforms to be present on the shader target. + */ +typedef struct { + /** Pointer to an uploaded vertex shader program */ + unsigned int shaderVertex; + + /** Pointer to an uploaded fragment shader program */ + unsigned int shaderFrag; + + /** Pointer to an uploaded shader program linked */ + unsigned int shaderProgram; +} shader_t; + +/** + * Create a shader from vertex and fragment shader code. + * + * @param vertexShaderSource The raw vertex shader code. + * @param fragmentShaderSource The raw fragment shader code. + * @return Pointer to the loaded shader. + */ +shader_t * shaderCompile(char *vertexShaderSource, char* fragmentShaderSource); + +/** + * Cleanup and unload a previously loaded shader. + * + * @param shader The shader to unload + * @return True if successfully disposed. + */ +bool shaderDipose(shader_t *shader); \ No newline at end of file diff --git a/test/file/asset.c b/test/file/asset.c new file mode 100644 index 00000000..cffce76f --- /dev/null +++ b/test/file/asset.c @@ -0,0 +1,62 @@ +#include "asset.h" + +char * assetLoadString(char *assetName) { + // Open a buffer. + FILE *fptr = assetBufferOpen(assetName); + if(fptr == NULL) return NULL; + + // Read the count of bytes in the file + fseek(fptr, 0, SEEK_END);// Seek to the end + size_t length = ftell(fptr);// Get our current position (the end) + fseek(fptr, 0, SEEK_SET);// Reset the seek + + // Create the string buffer + char *str = malloc(length + 1);// Add 1 for the null terminator. + if(str == NULL) { + assetBufferClose(fptr); + return NULL; + } + + // Read and seal the string. + fread(str, 1, length, fptr);// Read all the bytes + str[length] = '\0';// Null terminate. + assetBufferClose(fptr); // Close the buffer. + + return str; +} + +FILE * assetBufferOpen(char *assetName) { + // Get the directory based on the raw input by creating a new string. + size_t lenAsset = strlen(assetName);// Get the length of asset + size_t lenPrefix = strlen(ASSET_PREFIX);// Get the length of the prefix + + // Create str to house both the prefix and asset, and null terminator + char *joined = malloc(lenAsset + lenPrefix + 1); + if(joined == NULL) return NULL;// Mem okay? + + joined[0] = '\0';//Start at null + strcat(joined, ASSET_PREFIX);//Add prefix + strcat(joined, assetName);//Add body + + // Open the file pointer now. + FILE *fptr = fopen(joined, "rb"); + free(joined);// Free the string we just created + if(!fptr) return NULL;// File available? + return fptr; +} + +bool assetBufferClose(FILE *buffer) { + return fclose(buffer); +} + +int32_t assetBufferRead(FILE *buffer, char *data, int32_t size) { + return (int32_t)fread(data, 1, size, buffer); +} + +int32_t assetBufferEnd(FILE *buffer) { + return feof(buffer); +} + +void assetBufferSkip(FILE *buffer, int32_t n) { + fseek(buffer, n, SEEK_CUR); +} \ No newline at end of file diff --git a/test/file/asset.h b/test/file/asset.h new file mode 100644 index 00000000..3d885c8b --- /dev/null +++ b/test/file/asset.h @@ -0,0 +1,59 @@ +#pragma once +#include +#include +#include +#include +#include + +/** Prefix of all asset load methods, may be customizable in future. */ +#define ASSET_PREFIX "../assets/" + +/** + * Method to load an asset into memory as a raw string. + * + * @param assetName Path leading to the asset within the root asset directory. + * @return Pointer to char array of data from asset, NULL if unsuccesful. + */ +char * assetLoadString(char *assetName); + +/** + * Platform-centric method to open a file buffer to an asset. + * + * @param assetName The asset name to open a buffer for. + * @return Pointer to a buffer, NULL if unsuccessfuil. + */ +FILE * assetBufferOpen(char *assetName); + +/** + * Closes a previously opened asset buffer. + * + * @param buffer Buffer to close. + * @return True if successful, otherwise false. + */ +bool assetBufferClose(FILE *buffer); + +/** + * Read bytes from buffer. + * + * @param buffer The buffer pointing to an asset. + * @param data Pointer to a ubyte array to buffer data into. + * @param size Length of the data buffer. Represents how many bytes can be read. + * @return The count of bytes read. Complete when less than data array size. + */ +int32_t assetBufferRead(FILE *buffer, char *data, int32_t size); + +/** + * Skip to the end of the buffer, useful to find the length of the buffer. + * + * @param Buffer The buffer pointing to an asset. + * @return How many bytes were skipped + */ +int32_t assetBufferEnd(FILE *buffer); + +/** + * Method to skip n bytes in the buffer + * + * @param buffer The buffer pointing to an asset. + * @param n Count of bytes to skip. + */ +void assetBufferSkip(FILE *buffer, int32_t n); diff --git a/test/game/game.c b/test/game/game.c new file mode 100644 index 00000000..e3f67991 --- /dev/null +++ b/test/game/game.c @@ -0,0 +1,32 @@ +#include "game.h" + +game_t * gameInit(char *gameName) { + // Create the game instance + game_t *game = malloc(sizeof(game_t)); + if(game == NULL) return NULL; + + // Setup the renderer + game->render = renderInit(640, 480, gameName); + if(game->render == NULL) { + free(game); + return NULL; + } + + return game; +} + +void gameStart(game_t *game) { + while(!glfwWindowShouldClose(game->render->window)) { + renderFrame(game->render); + + glfwSwapBuffers(game->render->window); + glfwPollEvents(); + } +} + +bool gameDispose(game_t *game) { + if(!renderDispose(game->render)) return false; + free(game); + + return true; +} \ No newline at end of file diff --git a/test/game/game.h b/test/game/game.h new file mode 100644 index 00000000..476b8847 --- /dev/null +++ b/test/game/game.h @@ -0,0 +1,33 @@ +#pragma once +#include +#include "../display/render.h" + +/** Information about the current game context. */ +typedef struct { + /** Renderer for the game */ + render_t *render; +} game_t; + +/** + * Initialize the game context. + * + * @param gameName Name of the game being initialized. + * @return The game instance context. + */ +game_t * gameInit(char *gameName); + +/** + * Start the main game loop. + * + * @param game The game to start the loop for. + */ +void gameStart(game_t *game); + +/** + * Cleanup a previously constructed. + * + * @param game The game to cleanup. + * @return True if successful or not. + */ +bool gameDispose(game_t *game); + diff --git a/test/main.c b/test/main.c new file mode 100644 index 00000000..2cae98df --- /dev/null +++ b/test/main.c @@ -0,0 +1,17 @@ +#include "main.h" + +int32_t main() { + // Create the game instance + game_t *game = gameInit("Dawn"); + if(game == NULL) return 1; + + // Start running the game instance + gameStart(game); + + // Game has finished running, cleanup. + if(!gameDispose(game)) { + return 1; + } + + return 0; +} \ No newline at end of file