Integrate STB and abstract Vulkan image creation
This commit is contained in:
parent
c80e574f01
commit
8083104b13
@ -210,5 +210,8 @@ if (NOT BB_IS_CI)
|
||||
target_link_libraries(${COMPILER_NAME} PRIVATE volk)
|
||||
target_sources(${COMPILER_NAME} PRIVATE
|
||||
"bootstrap/bloat-buster/shader_compilation.c"
|
||||
"bootstrap/bloat-buster/image_loader.c"
|
||||
)
|
||||
|
||||
target_include_directories(${COMPILER_NAME} PRIVATE dependencies/stb)
|
||||
endif()
|
||||
|
@ -1,5 +1,5 @@
|
||||
#include <bloat-buster/gui.h>
|
||||
#if BB_CI == 0
|
||||
#include <bloat-buster/gui.h>
|
||||
|
||||
#define GLFW_INCLUDE_NONE
|
||||
#include <GLFW/glfw3.h>
|
||||
@ -7,6 +7,7 @@
|
||||
#include <volk.h>
|
||||
|
||||
#include <bloat-buster/shader_compilation.h>
|
||||
#include <bloat-buster/image_loader.h>
|
||||
|
||||
[[noreturn]] [[gnu::cold]] fn void wrong_vulkan_result(VkResult result, String call_string, String file, int line)
|
||||
{
|
||||
@ -15,8 +16,8 @@
|
||||
}
|
||||
|
||||
#define vkok(call) do {\
|
||||
VkResult result = call; \
|
||||
if (unlikely(result != VK_SUCCESS)) wrong_vulkan_result(result, strlit(#call), strlit(__FILE__), __LINE__); \
|
||||
VkResult _r_e_s_u_l_t_ = call; \
|
||||
if (unlikely(_r_e_s_u_l_t_ != VK_SUCCESS)) wrong_vulkan_result(_r_e_s_u_l_t_, strlit(#call), strlit(__FILE__), __LINE__); \
|
||||
} while(0)
|
||||
|
||||
#define vkok_swapchain(call) do {\
|
||||
@ -78,7 +79,7 @@ fn VKAPI_ATTR VkBool32 VKAPI_CALL debug_callback(VkDebugUtilsMessageSeverityFlag
|
||||
|
||||
#define frame_overlap (2)
|
||||
|
||||
fn void vk_transition_image(VkCommandBuffer command_buffer, VkImage image, VkImageLayout old_layout, VkImageLayout new_layout)
|
||||
fn void vk_image_transition(VkCommandBuffer command_buffer, VkImage image, VkImageLayout old_layout, VkImageLayout new_layout)
|
||||
{
|
||||
VkImageMemoryBarrier2 image_barrier = {
|
||||
.sType = VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER_2,
|
||||
@ -215,6 +216,77 @@ fn VulkanBuffer vk_buffer_create(VkDevice device, const VkAllocationCallbacks* a
|
||||
return result;
|
||||
}
|
||||
|
||||
STRUCT(VulkanImageCreate)
|
||||
{
|
||||
u32 width;
|
||||
u32 height;
|
||||
u32 mip_levels;
|
||||
VkFormat format;
|
||||
VkImageUsageFlags usage;
|
||||
};
|
||||
|
||||
STRUCT(VulkanImage)
|
||||
{
|
||||
VkImage handle;
|
||||
VkImageView view;
|
||||
VkDeviceMemory memory;
|
||||
};
|
||||
|
||||
fn VulkanImage vk_image_create(VkDevice device, const VkAllocationCallbacks* allocation_callbacks, VkPhysicalDeviceMemoryProperties memory_properties, VulkanImageCreate create)
|
||||
{
|
||||
VulkanImage result = {};
|
||||
|
||||
VkImageCreateInfo create_info = {
|
||||
.sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO,
|
||||
.pNext = 0,
|
||||
.flags = 0,
|
||||
.imageType = VK_IMAGE_TYPE_2D,
|
||||
.format = create.format,
|
||||
.extent = {
|
||||
.width = create.width,
|
||||
.height = create.height,
|
||||
.depth = 1,
|
||||
},
|
||||
.mipLevels = create.mip_levels,
|
||||
.arrayLayers = 1,
|
||||
.samples = VK_SAMPLE_COUNT_1_BIT,
|
||||
.tiling = VK_IMAGE_TILING_OPTIMAL,
|
||||
.usage = create.usage,
|
||||
.sharingMode = 0,
|
||||
.queueFamilyIndexCount = 0,
|
||||
.pQueueFamilyIndices = 0,
|
||||
.initialLayout = 0,
|
||||
};
|
||||
vkok(vkCreateImage(device, &create_info, allocation_callbacks, &result.handle));
|
||||
|
||||
VkMemoryRequirements memory_requirements;
|
||||
vkGetImageMemoryRequirements(device, result.handle, &memory_requirements);
|
||||
|
||||
VkMemoryPropertyFlags flags = VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT;
|
||||
u8 use_device_address_bit = 0;
|
||||
result.memory = vk_allocate_memory(device, allocation_callbacks, memory_properties, memory_requirements, flags, use_device_address_bit);
|
||||
|
||||
VkDeviceSize memory_offset = 0;
|
||||
vkok(vkBindImageMemory(device, result.handle, result.memory, memory_offset));
|
||||
|
||||
VkImageViewCreateInfo view_create_info = {
|
||||
.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO,
|
||||
.image = result.handle,
|
||||
.viewType = VK_IMAGE_VIEW_TYPE_2D,
|
||||
.format = create_info.format,
|
||||
.subresourceRange = {
|
||||
.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
|
||||
.baseMipLevel = 0,
|
||||
.levelCount = create.mip_levels,
|
||||
.baseArrayLayer = 0,
|
||||
.layerCount = 1,
|
||||
},
|
||||
};
|
||||
|
||||
vkok(vkCreateImageView(device, &view_create_info, allocation_callbacks, &result.view));
|
||||
return result;
|
||||
}
|
||||
|
||||
STRUCT(ImmediateContext)
|
||||
{
|
||||
VkDevice device;
|
||||
@ -335,7 +407,7 @@ fn void vk_copy_image_to_image(VkCommandBuffer command_buffer, CopyImageArgs arg
|
||||
|
||||
void run_app(Arena* arena)
|
||||
{
|
||||
#if defined(VK_USE_PLATFORM_XLIB_KHR)
|
||||
#if defined(VK_USE_PLATFORM_XLIB_KHR)
|
||||
glfwInitHint(GLFW_PLATFORM, GLFW_PLATFORM_X11);
|
||||
#elif defined(VK_USE_PLATFORM_WAYLAND_KHR)
|
||||
glfwInitHint(GLFW_PLATFORM, GLFW_PLATFORM_WAYLAND);
|
||||
@ -648,58 +720,13 @@ void run_app(Arena* arena)
|
||||
VkPhysicalDeviceMemoryProperties physical_device_memory_properties;
|
||||
vkGetPhysicalDeviceMemoryProperties(physical_device, &physical_device_memory_properties);
|
||||
|
||||
VkImage render_image;
|
||||
VkImageView render_image_view;
|
||||
{
|
||||
VkImageCreateInfo create_info = {
|
||||
.sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO,
|
||||
.pNext = 0,
|
||||
.flags = 0,
|
||||
.imageType = VK_IMAGE_TYPE_2D,
|
||||
.format = common_image_format,
|
||||
.extent = {
|
||||
.width = initial_width,
|
||||
.height = initial_height,
|
||||
.depth = 1,
|
||||
},
|
||||
.mipLevels = 1,
|
||||
.arrayLayers = 1,
|
||||
.samples = VK_SAMPLE_COUNT_1_BIT,
|
||||
.tiling = VK_IMAGE_TILING_OPTIMAL,
|
||||
.usage = VK_IMAGE_USAGE_TRANSFER_SRC_BIT | VK_IMAGE_USAGE_STORAGE_BIT | VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT,
|
||||
.sharingMode = 0,
|
||||
.queueFamilyIndexCount = 0,
|
||||
.pQueueFamilyIndices = 0,
|
||||
.initialLayout = 0,
|
||||
};
|
||||
vkok(vkCreateImage(device, &create_info, allocation_callbacks, &render_image));
|
||||
|
||||
VkMemoryRequirements memory_requirements;
|
||||
vkGetImageMemoryRequirements(device, render_image, &memory_requirements);
|
||||
|
||||
VkMemoryPropertyFlags flags = VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT;
|
||||
u8 use_device_address_bit = 0;
|
||||
VkDeviceMemory memory = vk_allocate_memory(device, allocation_callbacks, physical_device_memory_properties, memory_requirements, flags, use_device_address_bit);
|
||||
|
||||
VkDeviceSize memory_offset = 0;
|
||||
vkok(vkBindImageMemory(device, render_image, memory, memory_offset));
|
||||
|
||||
VkImageViewCreateInfo view_create_info = {
|
||||
.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO,
|
||||
.image = render_image,
|
||||
.viewType = VK_IMAGE_VIEW_TYPE_2D,
|
||||
.format = create_info.format,
|
||||
.subresourceRange = {
|
||||
.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
|
||||
.baseMipLevel = 0,
|
||||
.levelCount = 1,
|
||||
.baseArrayLayer = 0,
|
||||
.layerCount = 1,
|
||||
},
|
||||
};
|
||||
|
||||
vkok(vkCreateImageView(device, &view_create_info, allocation_callbacks, &render_image_view));
|
||||
}
|
||||
VulkanImage render_image = vk_image_create(device, allocation_callbacks, physical_device_memory_properties, (VulkanImageCreate) {
|
||||
.width = initial_width,
|
||||
.height = initial_height,
|
||||
.mip_levels = 1,
|
||||
.format = common_image_format,
|
||||
.usage = VK_IMAGE_USAGE_TRANSFER_SRC_BIT | VK_IMAGE_USAGE_STORAGE_BIT | VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT,
|
||||
});
|
||||
|
||||
VkCommandPool command_pools[frame_overlap];
|
||||
VkCommandBuffer command_buffers[frame_overlap];
|
||||
@ -1089,12 +1116,12 @@ void run_app(Arena* arena)
|
||||
vkok(vkBeginCommandBuffer(command_buffer, &command_buffer_begin_info));
|
||||
}
|
||||
|
||||
vk_transition_image(command_buffer, render_image, VK_IMAGE_LAYOUT_UNDEFINED, VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL);
|
||||
vk_image_transition(command_buffer, render_image.handle, VK_IMAGE_LAYOUT_UNDEFINED, VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL);
|
||||
|
||||
VkRenderingAttachmentInfo color_attachments[] = {
|
||||
{
|
||||
.sType = VK_STRUCTURE_TYPE_RENDERING_ATTACHMENT_INFO,
|
||||
.imageView = render_image_view,
|
||||
.imageView = render_image.view,
|
||||
.imageLayout = VK_IMAGE_LAYOUT_ATTACHMENT_OPTIMAL,
|
||||
.loadOp = VK_ATTACHMENT_LOAD_OP_CLEAR,
|
||||
.storeOp = VK_ATTACHMENT_STORE_OP_STORE,
|
||||
@ -1168,14 +1195,14 @@ void run_app(Arena* arena)
|
||||
|
||||
vkCmdEndRendering(command_buffer);
|
||||
|
||||
vk_transition_image(command_buffer, render_image, VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL);
|
||||
vk_image_transition(command_buffer, render_image.handle, VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL);
|
||||
|
||||
VkImage swapchain_image = swapchain_image_buffer[swapchain_image_index];
|
||||
vk_transition_image(command_buffer, swapchain_image, VK_IMAGE_LAYOUT_UNDEFINED, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL);
|
||||
vk_image_transition(command_buffer, swapchain_image, VK_IMAGE_LAYOUT_UNDEFINED, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL);
|
||||
|
||||
vk_copy_image_to_image(command_buffer, (CopyImageArgs) {
|
||||
.source = {
|
||||
.handle = render_image,
|
||||
.handle = render_image.handle,
|
||||
.extent = {
|
||||
.width = initial_width,
|
||||
.height = initial_height,
|
||||
@ -1190,7 +1217,7 @@ void run_app(Arena* arena)
|
||||
},
|
||||
});
|
||||
|
||||
vk_transition_image(command_buffer, swapchain_image, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, VK_IMAGE_LAYOUT_PRESENT_SRC_KHR);
|
||||
vk_image_transition(command_buffer, swapchain_image, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, VK_IMAGE_LAYOUT_PRESENT_SRC_KHR);
|
||||
|
||||
vkok(vkEndCommandBuffer(command_buffer));
|
||||
}
|
||||
|
20
bootstrap/bloat-buster/image_loader.c
Normal file
20
bootstrap/bloat-buster/image_loader.c
Normal file
@ -0,0 +1,20 @@
|
||||
#include <bloat-buster/image_loader.h>
|
||||
|
||||
#define STB_IMAGE_IMPLEMENTATION
|
||||
#include <stb_image.h>
|
||||
|
||||
EXPORT TextureMemory texture_load_from_file(Arena* arena, String path)
|
||||
{
|
||||
auto file = file_read(arena, path);
|
||||
int width;
|
||||
int height;
|
||||
int channels;
|
||||
u8* buffer = stbi_load_from_memory(file.pointer, file.length, &width, &height, &channels, 0);
|
||||
|
||||
return (TextureMemory) {
|
||||
.pointer = buffer,
|
||||
.width = width,
|
||||
.height = height,
|
||||
.channel_count = channels,
|
||||
};
|
||||
}
|
13
bootstrap/include/bloat-buster/image_loader.h
Normal file
13
bootstrap/include/bloat-buster/image_loader.h
Normal file
@ -0,0 +1,13 @@
|
||||
#include <std/base.h>
|
||||
#include <std/os.h>
|
||||
|
||||
STRUCT(TextureMemory)
|
||||
{
|
||||
u8* pointer;
|
||||
u32 width;
|
||||
u32 height;
|
||||
u32 channel_count;
|
||||
u32 padding[1];
|
||||
};
|
||||
|
||||
EXPORT TextureMemory texture_load_from_file(Arena* arena, String path);
|
3
dependencies/stb/.gitignore
vendored
Normal file
3
dependencies/stb/.gitignore
vendored
Normal file
@ -0,0 +1,3 @@
|
||||
*.o
|
||||
*.obj
|
||||
*.exe
|
37
dependencies/stb/LICENSE
vendored
Normal file
37
dependencies/stb/LICENSE
vendored
Normal file
@ -0,0 +1,37 @@
|
||||
This software is available under 2 licenses -- choose whichever you prefer.
|
||||
------------------------------------------------------------------------------
|
||||
ALTERNATIVE A - MIT License
|
||||
Copyright (c) 2017 Sean Barrett
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||
this software and associated documentation files (the "Software"), to deal in
|
||||
the Software without restriction, including without limitation the rights to
|
||||
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
|
||||
of the Software, and to permit persons to whom the Software is furnished to do
|
||||
so, subject to the following conditions:
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
||||
------------------------------------------------------------------------------
|
||||
ALTERNATIVE B - Public Domain (www.unlicense.org)
|
||||
This is free and unencumbered software released into the public domain.
|
||||
Anyone is free to copy, modify, publish, use, compile, sell, or distribute this
|
||||
software, either in source code form or as a compiled binary, for any purpose,
|
||||
commercial or non-commercial, and by any means.
|
||||
In jurisdictions that recognize copyright laws, the author or authors of this
|
||||
software dedicate any and all copyright interest in the software to the public
|
||||
domain. We make this dedication for the benefit of the public at large and to
|
||||
the detriment of our heirs and successors. We intend this dedication to be an
|
||||
overt act of relinquishment in perpetuity of all present and future rights to
|
||||
this software under copyright law.
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
||||
ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||||
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
7988
dependencies/stb/stb_image.h
vendored
Normal file
7988
dependencies/stb/stb_image.h
vendored
Normal file
File diff suppressed because it is too large
Load Diff
5079
dependencies/stb/stb_truetype.h
vendored
Normal file
5079
dependencies/stb/stb_truetype.h
vendored
Normal file
File diff suppressed because it is too large
Load Diff
Loading…
x
Reference in New Issue
Block a user