18#define GLFW_INCLUDE_VULKAN
19#include <GLFW/glfw3.h>
21#include <ext/vulkan/glslang/Public/ShaderLang.h>
22#include <ext/vulkan/spirv/GlslangToSpv.h>
23#include <ext/vulkan/vma/src/VmaUsage.h>
24#include <ext/vulkan/OGLCompilersDLL/InitializeDll.h>
26#define THSVS_SIMPLER_VULKAN_SYNCHRONIZATION_IMPLEMENTATION
27#include <ext/vulkan/svs/thsvs_simpler_vulkan_synchronization.h>
36#include <unordered_map>
37#include <unordered_set>
75#define ARRAY_SIZE(a) (sizeof(a) / sizeof(a[0]))
76#define ERR_EXIT(err_msg, err_class) \
78 Console::println(err_msg); \
79 Application::exit(1); \
82#define GET_INSTANCE_PROC_ADDR(inst, entrypoint) \
84 fp##entrypoint = (PFN_vk##entrypoint)vkGetInstanceProcAddr(inst, "vk" #entrypoint); \
85 if (fp##entrypoint == nullptr) { \
86 ERR_EXIT("vkGetInstanceProcAddr failed to find vk" #entrypoint, "vkGetInstanceProcAddr Failure"); \
90#define GET_DEVICE_PROC_ADDR(dev, entrypoint) \
92 fp##entrypoint = (PFN_vk##entrypoint)vkGetDeviceProcAddr(dev, "vk" #entrypoint); \
93 if (fp##entrypoint == nullptr) { \
94 ERR_EXIT("vkGetDeviceProcAddr failed to find vk" #entrypoint, "vkGetDeviceProcAddr Failure"); \
105using std::unordered_map;
106using std::unordered_set;
139VKRenderer::VKRenderer():
141 queueSpinlock(
"queue_spinlock"),
142 buffersMutex(
"buffers_mutex"),
143 texturesMutex(
"textures_mutex"),
144 deleteMutex(
"delete_mutex"),
145 disposeMutex(
"dispose_mutex"),
146 pipelinesSpinLock(
"pipelines_spinlock"),
147 vmaSpinlock(
"vma_spinlock")
181inline VkBool32
VKRenderer::checkLayers(uint32_t checkCount,
const char **checkNames,
const vector<VkLayerProperties>& instanceLayers) {
183 for (i = 0; i < checkCount; i++) {
185 for (j = 0; j < instanceLayers.size(); j++) {
186 if (!strcmp(checkNames[i], instanceLayers[j].layerName)) {
192 fprintf(stderr,
"Cannot find layer: %s\n", checkNames[i]);
200 auto& currentContext =
contexts[contextIdx];
201 if (currentContext.setupCommandInUse == VK_NULL_HANDLE) {
202 currentContext.setupCommandInUse = currentContext.setupCommand;
205 const VkCommandBufferBeginInfo cmdBufInfo = {
206 .sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO,
208 .flags = VK_COMMAND_BUFFER_USAGE_ONE_TIME_SUBMIT_BIT,
209 .pInheritanceInfo =
nullptr
213 err = vkBeginCommandBuffer(currentContext.setupCommandInUse, &cmdBufInfo);
222 auto& currentContext =
contexts[contextIdx];
225 if (currentContext.setupCommandInUse != VK_NULL_HANDLE) {
228 err = vkEndCommandBuffer(currentContext.setupCommandInUse);
231 VkFence nullFence = { VK_NULL_HANDLE };
232 VkSubmitInfo submitInfo = {
233 .sType = VK_STRUCTURE_TYPE_SUBMIT_INFO,
235 .waitSemaphoreCount = 0,
236 .pWaitSemaphores =
nullptr,
237 .pWaitDstStageMask =
nullptr,
238 .commandBufferCount = 1,
239 .pCommandBuffers = ¤tContext.setupCommandInUse,
240 .signalSemaphoreCount = 0,
241 .pSignalSemaphores =
nullptr
246 err = vkQueueSubmit(
queue, 1, &submitInfo, currentContext.setupFence);
256 VkResult fenceResult;
258 fenceResult = vkWaitForFences(
device, 1, ¤tContext.setupFence, VK_TRUE, 100000000);
259 }
while (fenceResult == VK_TIMEOUT);
260 vkResetFences(
device, 1, ¤tContext.setupFence);
263 currentContext.setupCommandInUse = VK_NULL_HANDLE;
268 auto& currentContext =
contexts[contextIdx];
271 if (bufferId == -1) bufferId = currentContext.currentCommandBuffer;
274 auto& commandBuffer = currentContext.commandBuffers[bufferId];
275 if (commandBuffer.drawCmdStarted ==
true)
return false;
278 VkResult fenceResult;
280 fenceResult = vkWaitForFences(
device, 1, &commandBuffer.drawFence, VK_TRUE, 100000000);
281 }
while (fenceResult == VK_TIMEOUT);
282 vkResetFences(
device, 1, &commandBuffer.drawFence);
285 const VkCommandBufferBeginInfo cmdBufInfo = {
286 .sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO,
288 .flags = VK_COMMAND_BUFFER_USAGE_ONE_TIME_SUBMIT_BIT,
289 .pInheritanceInfo =
nullptr
294 err = vkBeginCommandBuffer(commandBuffer.drawCommand, &cmdBufInfo);
297 array<ThsvsAccessType, 2> nextAccessTypes { THSVS_ACCESS_COLOR_ATTACHMENT_WRITE, THSVS_ACCESS_NONE };
298 ThsvsImageLayout nextLayout { THSVS_IMAGE_LAYOUT_OPTIMAL };
303 ThsvsImageBarrier svsImageBarrier = {
306 .nextAccessCount =
static_cast<uint32_t
>(nextAccessTypes[1] != THSVS_ACCESS_NONE?2:1),
307 .pNextAccesses = nextAccessTypes.data(),
309 .nextLayout = nextLayout,
310 .discardContents =
true,
311 .srcQueueFamilyIndex = 0,
312 .dstQueueFamilyIndex = 0,
314 .subresourceRange = {
315 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
322 VkImageMemoryBarrier vkImageMemoryBarrier;
323 VkPipelineStageFlags srcStages;
324 VkPipelineStageFlags dstStages;
325 thsvsGetVulkanImageMemoryBarrier(
329 &vkImageMemoryBarrier
336 vkCmdPipelineBarrier(commandBuffer.drawCommand, srcStages, dstStages, 0, 0,
nullptr, 0,
nullptr, 1, &vkImageMemoryBarrier);
344 commandBuffer.drawCmdStarted =
true;
354 auto& currentContext =
contexts[contextIdx];
357 if (bufferId == -1) bufferId = currentContext.currentCommandBuffer;
358 auto& commandBuffer = currentContext.commandBuffers[bufferId];
361 currentContext.pipeline = VK_NULL_HANDLE;
364 if (commandBuffer.drawCmdStarted ==
false)
return VK_NULL_HANDLE;
368 err = vkEndCommandBuffer(commandBuffer.drawCommand);
372 auto endedCommandBuffer = commandBuffer.drawCommand;
375 commandBuffer.drawCmdStarted =
false;
378 if (cycleBuffers ==
true) currentContext.currentCommandBuffer = (currentContext.currentCommandBuffer + 1) %
DRAW_COMMANDBUFFER_MAX;
381 return endedCommandBuffer;
387 VkPipelineStageFlags pipeStageFlags = VK_PIPELINE_STAGE_BOTTOM_OF_PIPE_BIT;
388 VkSubmitInfo submitInfo = {
389 .sType = VK_STRUCTURE_TYPE_SUBMIT_INFO,
391 .waitSemaphoreCount = 0,
392 .pWaitSemaphores =
nullptr,
393 .pWaitDstStageMask = &pipeStageFlags,
394 .commandBufferCount =
static_cast<uint32_t
>(commandBufferCount),
395 .pCommandBuffers = commandBuffers,
396 .signalSemaphoreCount = 0,
397 .pSignalSemaphores =
nullptr
403 err = vkQueueSubmit(
queue, 1, &submitInfo, fence);
416inline void VKRenderer::setImageLayout(
int contextIdx,
texture_type* textureObject,
const array<ThsvsAccessType,2>& nextAccessTypes, ThsvsImageLayout nextLayout,
bool discardContent, uint32_t baseMipLevel, uint32_t levelCount,
bool submit) {
417 auto& currentContext =
contexts[contextIdx];
426 if (_textureObject->accessTypes[baseArrayLayer] == nextAccessTypes && _textureObject->svsLayout == nextLayout)
return;
428 ThsvsImageBarrier svsImageBarrier = {
429 .prevAccessCount =
static_cast<uint32_t
>(_textureObject->accessTypes[baseArrayLayer][1] != THSVS_ACCESS_NONE?2:1),
430 .pPrevAccesses = _textureObject->
accessTypes[baseArrayLayer].data(),
431 .nextAccessCount =
static_cast<uint32_t
>(nextAccessTypes[1] != THSVS_ACCESS_NONE?2:1),
432 .pNextAccesses = nextAccessTypes.data(),
433 .prevLayout = _textureObject->svsLayout,
434 .nextLayout = nextLayout,
435 .discardContents = discardContent,
436 .srcQueueFamilyIndex = 0,
437 .dstQueueFamilyIndex = 0,
438 .image = _textureObject->image,
439 .subresourceRange = {
440 .aspectMask = _textureObject->aspectMask,
441 .baseMipLevel = baseMipLevel,
442 .levelCount = levelCount,
443 .baseArrayLayer = baseArrayLayer,
447 VkImageMemoryBarrier vkImageMemoryBarrier;
448 VkPipelineStageFlags srcStages;
449 VkPipelineStageFlags dstStages;
450 thsvsGetVulkanImageMemoryBarrier(
454 &vkImageMemoryBarrier
459 vkCmdPipelineBarrier(currentContext.setupCommandInUse, srcStages, dstStages, 0, 0,
nullptr, 0,
nullptr, 1, &vkImageMemoryBarrier);
463 _textureObject->accessTypes[baseArrayLayer] = nextAccessTypes;
464 _textureObject->svsLayout = nextLayout;
465 _textureObject->vkLayout = vkImageMemoryBarrier.newLayout;
471 const array<ThsvsAccessType,2>& prevAccessTypes,
472 const array<ThsvsAccessType,2>& nextAccessTypes,
473 ThsvsImageLayout prevLayout,
474 ThsvsImageLayout nextLayout,
476 uint32_t baseMipLevel,
486 ThsvsImageBarrier svsImageBarrier = {
487 .prevAccessCount =
static_cast<uint32_t
>(prevAccessTypes[1] != THSVS_ACCESS_NONE?2:1),
488 .pPrevAccesses = prevAccessTypes.data(),
489 .nextAccessCount =
static_cast<uint32_t
>(nextAccessTypes[1] != THSVS_ACCESS_NONE?2:1),
490 .pNextAccesses = nextAccessTypes.data(),
491 .prevLayout = prevLayout,
492 .nextLayout = nextLayout,
493 .discardContents = discardContent,
494 .srcQueueFamilyIndex = 0,
495 .dstQueueFamilyIndex = 0,
496 .
image = _textureObject->image,
497 .subresourceRange = {
498 .aspectMask = _textureObject->aspectMask,
499 .baseMipLevel = baseMipLevel,
500 .levelCount = levelCount,
501 .baseArrayLayer = baseArrayLayer,
505 thsvsGetVulkanImageMemoryBarrier(
514 imageLayoutChange.
svsLayout = nextLayout;
516 imageLayoutChange.
valid =
true;
520 auto& currentContext =
contexts[contextIdx];
534 _textureObject->accessTypes[baseArrayLayer] = imageLayoutChange.
accessTypes;
535 _textureObject->svsLayout = imageLayoutChange.
svsLayout;
536 _textureObject->vkLayout = imageLayoutChange.
vkLayout;
540 auto& currentContext =
contexts[contextIdx];
543 array<VkImageMemoryBarrier, 8> vkImageMemoryBarriers = {
544 imageLayoutChanges[0].vkImageMemoryBarrier,
545 imageLayoutChanges[1].vkImageMemoryBarrier,
546 imageLayoutChanges[2].vkImageMemoryBarrier,
547 imageLayoutChanges[3].vkImageMemoryBarrier,
548 imageLayoutChanges[4].vkImageMemoryBarrier,
549 imageLayoutChanges[5].vkImageMemoryBarrier,
550 imageLayoutChanges[6].vkImageMemoryBarrier,
551 imageLayoutChanges[7].vkImageMemoryBarrier
556 vkCmdPipelineBarrier(currentContext.setupCommandInUse, imageLayoutChanges[0].srcStages, imageLayoutChanges[0].dstStages, 0, 0,
nullptr, 0,
nullptr, vkImageMemoryBarriers.size(), vkImageMemoryBarriers.data());
561 for (
auto textureObject: textureObjects) {
563 auto _textureObject = textureObject->cubemapBufferTexture !=
nullptr?textureObject->cubemapBufferTexture:textureObject;
566 auto baseArrayLayer =
static_cast<uint32_t
>(textureObject->cubemapBufferTexture !=
nullptr?textureObject->cubemapTextureIndex -
CUBEMAPTEXTUREINDEX_MIN:0);
569 _textureObject->accessTypes[baseArrayLayer] = imageLayoutChanges[i].accessTypes;
570 _textureObject->svsLayout = imageLayoutChanges[i].svsLayout;
571 _textureObject->vkLayout = imageLayoutChanges[i].vkLayout;
576inline void VKRenderer::setImageLayout2(
int contextIdx,
texture_type* textureObject,
const array<ThsvsAccessType,2>& accessTypes,
const array<ThsvsAccessType,2>& nextAccessTypes, ThsvsImageLayout layout, ThsvsImageLayout nextLayout,
bool discardContent, uint32_t baseMipLevel, uint32_t levelCount, uint32_t baseArrayLayer, uint32_t layerCount,
bool updateTextureObject) {
577 auto& currentContext =
contexts[contextIdx];
580 ThsvsImageBarrier svsImageBarrier = {
581 .prevAccessCount =
static_cast<uint32_t
>(accessTypes[1] != THSVS_ACCESS_NONE?2:1),
582 .pPrevAccesses = accessTypes.data(),
583 .nextAccessCount =
static_cast<uint32_t
>(nextAccessTypes[1] != THSVS_ACCESS_NONE?2:1),
584 .pNextAccesses = nextAccessTypes.data(),
585 .prevLayout = layout,
586 .nextLayout = nextLayout,
587 .discardContents = discardContent,
588 .srcQueueFamilyIndex = 0,
589 .dstQueueFamilyIndex = 0,
590 .image = textureObject->
image,
591 .subresourceRange = {
593 .baseMipLevel = baseMipLevel,
594 .levelCount = levelCount,
595 .baseArrayLayer = baseArrayLayer,
596 .layerCount = layerCount
599 VkImageMemoryBarrier vkImageMemoryBarrier;
600 VkPipelineStageFlags srcStages;
601 VkPipelineStageFlags dstStages;
602 thsvsGetVulkanImageMemoryBarrier(
606 &vkImageMemoryBarrier
611 vkCmdPipelineBarrier(currentContext.setupCommandInUse, srcStages, dstStages, 0, 0,
nullptr, 0,
nullptr, 1, &vkImageMemoryBarrier);
615 textureObject->
accessTypes[baseArrayLayer] = nextAccessTypes;
617 textureObject->
vkLayout = vkImageMemoryBarrier.newLayout;
620inline void VKRenderer::setImageLayout3(
int contextIdx, VkImage image, VkImageAspectFlags aspectMask,
const array<ThsvsAccessType,2>& accessTypes,
const array<ThsvsAccessType,2>& nextAccessTypes, ThsvsImageLayout layout, ThsvsImageLayout nextLayout) {
621 auto& currentContext =
contexts[contextIdx];
624 ThsvsImageBarrier svsImageBarrier = {
625 .prevAccessCount =
static_cast<uint32_t
>(accessTypes[1] != THSVS_ACCESS_NONE?2:1),
626 .pPrevAccesses = accessTypes.data(),
627 .nextAccessCount =
static_cast<uint32_t
>(nextAccessTypes[1] != THSVS_ACCESS_NONE?2:1),
628 .pNextAccesses = nextAccessTypes.data(),
629 .prevLayout = layout,
630 .nextLayout = nextLayout,
631 .discardContents =
false,
632 .srcQueueFamilyIndex = 0,
633 .dstQueueFamilyIndex = 0,
635 .subresourceRange = {
636 .aspectMask = aspectMask,
643 VkImageMemoryBarrier vkImageMemoryBarrier;
644 VkPipelineStageFlags srcStages;
645 VkPipelineStageFlags dstStages;
646 thsvsGetVulkanImageMemoryBarrier(
650 &vkImageMemoryBarrier
655 vkCmdPipelineBarrier(currentContext.setupCommandInUse, srcStages, dstStages, 0, 0,
nullptr, 0,
nullptr, 1, &vkImageMemoryBarrier);
662 auto borderSize = 32;
664 while (borderSize > 4) {
674inline void VKRenderer::prepareTextureImage(
int contextIdx,
struct texture_type* textureObject, VkImageTiling tiling, VkImageUsageFlags usage, VkFlags requiredFlags,
Texture* texture,
const array<ThsvsAccessType,2>& nextAccesses, ThsvsImageLayout imageLayout,
bool disableMipMaps, uint32_t baseLevel, uint32_t levelCount) {
681 const VkImageCreateInfo imageCreateInfo = {
682 .sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO,
685 .imageType = VK_IMAGE_TYPE_2D,
686 .format = texture->
getDepth() == 32?VK_FORMAT_R8G8B8A8_UNORM:VK_FORMAT_R8G8B8A8_UNORM,
688 .width = textureWidth,
689 .height = textureHeight,
694 .samples = VK_SAMPLE_COUNT_1_BIT,
697 .sharingMode = VK_SHARING_MODE_EXCLUSIVE,
698 .queueFamilyIndexCount = 0,
699 .pQueueFamilyIndices = 0,
700 .initialLayout = VK_IMAGE_LAYOUT_PREINITIALIZED
703 VmaAllocationCreateInfo imageAllocCreateInfo = {};
704 imageAllocCreateInfo.usage = VMA_MEMORY_USAGE_UNKNOWN;
705 imageAllocCreateInfo.requiredFlags = requiredFlags;
707 VmaAllocationInfo allocationInfo = {};
708 err = vmaCreateImage(
vmaAllocator, &imageCreateInfo, &imageAllocCreateInfo, &textureObject->
image, &textureObject->
allocation, &allocationInfo);
711 if ((requiredFlags & VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT) == VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT) {
712 const VkImageSubresource imageSubResource = {
713 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
717 VkSubresourceLayout subResourceLayout;
718 vkGetImageSubresourceLayout(
device, textureObject->
image, &imageSubResource, &subResourceLayout);
727 auto bytesPerPixel = texture->
getDepth() / 8;
729 for (
auto y = 0; y < textureHeight; y++) {
730 uint8_t* row = (uint8_t*)((uint8_t*)data + subResourceLayout.offset + subResourceLayout.rowPitch * y);
731 for (
auto x = 0; x < textureWidth; x++) {
732 row[x * 4 + 0] = textureBuffer->
get((y * textureWidth * bytesPerPixel) + (x * bytesPerPixel) + 0);
733 row[x * 4 + 1] = textureBuffer->get((y * textureWidth * bytesPerPixel) + (x * bytesPerPixel) + 1);
734 row[x * 4 + 2] = textureBuffer->get((y * textureWidth * bytesPerPixel) + (x * bytesPerPixel) + 2);
735 row[x * 4 + 3] = bytesPerPixel == 4?textureBuffer->get((y * textureWidth * bytesPerPixel) + (x * bytesPerPixel) + 3):0xff;
745 textureObject->
accessTypes = { THSVS_ACCESS_HOST_PREINITIALIZED, THSVS_ACCESS_NONE };
750 THSVS_IMAGE_LAYOUT_OPTIMAL,
759 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
765 VkSurfaceCapabilitiesKHR surfCapabilities;
767 assert(err == VK_SUCCESS);
769 uint32_t presentModeCount;
771 assert(err == VK_SUCCESS);
772 vector<VkPresentModeKHR> presentModes(presentModeCount);
774 assert(err == VK_SUCCESS);
776 VkExtent2D swapchainExtent;
778 if (surfCapabilities.currentExtent.width == 0xFFFFFFFF) {
785 if (swapchainExtent.width < surfCapabilities.minImageExtent.width) {
786 swapchainExtent.width = surfCapabilities.minImageExtent.width;
787 }
else if (swapchainExtent.width > surfCapabilities.maxImageExtent.width) {
788 swapchainExtent.width = surfCapabilities.maxImageExtent.width;
791 if (swapchainExtent.height < surfCapabilities.minImageExtent.height) {
792 swapchainExtent.height = surfCapabilities.minImageExtent.height;
793 }
else if (swapchainExtent.height > surfCapabilities.maxImageExtent.height) {
794 swapchainExtent.height = surfCapabilities.maxImageExtent.height;
798 swapchainExtent = surfCapabilities.currentExtent;
799 windowWidth = surfCapabilities.currentExtent.width;
806 uint32_t desiredNumOfSwapchainImages = surfCapabilities.minImageCount;
809 if ((surfCapabilities.maxImageCount > 0) && (desiredNumOfSwapchainImages > surfCapabilities.maxImageCount)) {
811 desiredNumOfSwapchainImages = surfCapabilities.maxImageCount;
815 VkSurfaceTransformFlagsKHR preTransform {};
816 if (surfCapabilities.supportedTransforms & VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR) {
817 preTransform = VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR;
819 preTransform = surfCapabilities.currentTransform;
822 const VkSwapchainCreateInfoKHR swapchainCreateInfo = {
823 .sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR,
827 .minImageCount = desiredNumOfSwapchainImages,
831 .width = swapchainExtent.width,
832 .height = swapchainExtent.height
834 .imageArrayLayers = 1,
835 .imageUsage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT | VK_IMAGE_USAGE_TRANSFER_SRC_BIT,
836 .imageSharingMode = VK_SHARING_MODE_EXCLUSIVE,
837 .queueFamilyIndexCount = 0,
838 .pQueueFamilyIndices = 0,
839 .preTransform = (VkSurfaceTransformFlagBitsKHR)preTransform,
840 .compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR,
843 .oldSwapchain = oldSwapchain,
853 if (oldSwapchain != VK_NULL_HANDLE) {
862 assert(err == VK_SUCCESS);
866 assert(err == VK_SUCCESS);
876 VkImageViewCreateInfo colorAttachmentView = {
877 .sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO,
880 .image = swapchainImages[i],
881 .viewType = VK_IMAGE_VIEW_TYPE_2D,
884 .r = VK_COMPONENT_SWIZZLE_R,
885 .g = VK_COMPONENT_SWIZZLE_G,
886 .b = VK_COMPONENT_SWIZZLE_B,
887 .a = VK_COMPONENT_SWIZZLE_A
889 .subresourceRange = {
890 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
899 assert(err == VK_SUCCESS);
906 return StringTools::tokenize(
deviceName,
" \t\n")[0];
924 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
928 if (FileSystem::getInstance()->fileExists(
"shader/vk") ==
false) {
929 FileSystem::getInstance()->createPath(
"shader/vk");
932 Console::println(
string() +
"An error occurred: " + exception.what());
939 glslang::InitProcess();
940 glslang::InitThread();
945 uint32_t requiredExtensionCount = 0;
946 uint32_t instanceExtensionCount = 0;
947 uint32_t instanceLayerCount = 0;
948 uint32_t validationLayerCount = 0;
949 const char** requiredExtensions =
nullptr;
950 const char** instanceValidationLayers =
nullptr;
952 uint32_t enabledExtensionCount = 0;
953 uint32_t enabledLayerCount = 0;
954 array<const char*, 64> extensionNames {};
955 array<const char*, 64>enabledLayers {};
957 const char* instanceValidationLayersAlt1[] = {
958 "VK_LAYER_KHRONOS_validation"
960 const char* instanceValidationLayersAlt2[] = {
961 "VK_LAYER_LUNARG_standard_validation"
965 auto enableValidationLayers = Application::getApplication()->isDebuggingEnabled();
966 if (enableValidationLayers ==
true) {
967 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): \"--debug\" mode enabled: Enabling validation layers");
969 VkBool32 validationFound = 0;
970 err = vkEnumerateInstanceLayerProperties(&instanceLayerCount,
nullptr);
973 instanceValidationLayers = (
const char**)instanceValidationLayersAlt1;
974 if (instanceLayerCount > 0) {
975 vector<VkLayerProperties> instanceLayers(instanceLayerCount);
976 err = vkEnumerateInstanceLayerProperties(&instanceLayerCount, instanceLayers.data());
981 instanceValidationLayers,
984 if (validationFound) {
985 enabledLayerCount =
ARRAY_SIZE(instanceValidationLayersAlt1);
986 enabledLayers[0] =
"VK_LAYER_LUNARG_standard_validation";
987 validationLayerCount = 1;
990 instanceValidationLayers = (
const char**) instanceValidationLayersAlt2;
991 enabledLayerCount =
ARRAY_SIZE(instanceValidationLayersAlt2);
994 instanceValidationLayers,
997 validationLayerCount =
ARRAY_SIZE(instanceValidationLayersAlt2);
998 for (i = 0; i < validationLayerCount; i++) {
999 enabledLayers[i] = instanceValidationLayers[i];
1004 if (!validationFound) {
1005 ERR_EXIT(
"vkEnumerateInstanceLayerProperties failed to find "
1006 "required validation layer.\n\n"
1007 "Please look at the Getting Started guide for additional "
1008 "information.\n",
"vkCreateInstance Failure");
1013 requiredExtensions = glfwGetRequiredInstanceExtensions(&requiredExtensionCount);
1014 if (!requiredExtensions) {
1015 ERR_EXIT(
"glfwGetRequiredInstanceExtensions failed to find the "
1016 "platform surface extensions.\n\nDo you have a compatible "
1017 "Vulkan installable client driver (ICD) installed?\nPlease "
1018 "look at the Getting Started guide for additional "
1019 "information.\n",
"vkCreateInstance Failure"
1023 for (i = 0; i < requiredExtensionCount; i++) {
1024 extensionNames[enabledExtensionCount++] = requiredExtensions[i];
1025 assert(enabledExtensionCount < 64);
1028 err = vkEnumerateInstanceExtensionProperties(
1029 nullptr, &instanceExtensionCount,
nullptr);
1032 if (instanceExtensionCount > 0) {
1033 vector<VkExtensionProperties> instanceExtensions(instanceExtensionCount);
1034 err = vkEnumerateInstanceExtensionProperties(
nullptr, &instanceExtensionCount, instanceExtensions.data());
1036 for (i = 0; i < instanceExtensionCount; i++) {
1037 if (!strcmp(VK_EXT_DEBUG_REPORT_EXTENSION_NAME, instanceExtensions[i].extensionName)) {
1038 if (enableValidationLayers ==
true) {
1039 extensionNames[enabledExtensionCount++] = VK_EXT_DEBUG_REPORT_EXTENSION_NAME;
1042 assert(enabledExtensionCount < 64);
1046 const VkApplicationInfo applicationInfo = {
1047 .sType = VK_STRUCTURE_TYPE_APPLICATION_INFO,
1049 .pApplicationName = Application::getApplication()->getTitle().c_str(),
1050 .applicationVersion = 0,
1051 .pEngineName =
"TDME2",
1052 .engineVersion = 200,
1053 .apiVersion = VK_API_VERSION_1_0,
1055 VkInstanceCreateInfo instanceCreateInfo = {
1056 .sType = VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO,
1059 .pApplicationInfo = &applicationInfo,
1060 .enabledLayerCount = enabledLayerCount,
1061 .ppEnabledLayerNames = instanceValidationLayers,
1062 .enabledExtensionCount = enabledExtensionCount,
1063 .ppEnabledExtensionNames = extensionNames.data(),
1065 err = vkCreateInstance(&instanceCreateInfo,
nullptr, &
instance);
1066 if (err == VK_ERROR_INCOMPATIBLE_DRIVER) {
1067 ERR_EXIT(
"Cannot find a compatible Vulkan installable client driver "
1068 "(ICD).\n\nPlease look at the Getting Started guide for "
1069 "additional information.\n",
"vkCreateInstance Failure");
1071 if (err == VK_ERROR_EXTENSION_NOT_PRESENT) {
1072 ERR_EXIT(
"Cannot find a specified extension library"
1073 ".\nMake sure your layers path is set appropriately\n",
1074 "vkCreateInstance Failure");
1077 ERR_EXIT(
"vkCreateInstance failed.\n\nDo you have a compatible Vulkan "
1078 "installable client driver (ICD) installed?\nPlease look at "
1079 "the Getting Started guide for additional information.\n",
1080 "vkCreateInstance Failure");
1084 uint32_t gpuCount = 0;
1085 err = vkEnumeratePhysicalDevices(
instance, &gpuCount,
nullptr);
1086 assert(!err && gpuCount > 0);
1090 vector<VkPhysicalDevice> physicalDevices(gpuCount);
1091 err = vkEnumeratePhysicalDevices(
instance, &gpuCount, physicalDevices.data());
1096 "vkEnumeratePhysicalDevices reported zero accessible devices."
1097 "\n\nDo you have a compatible Vulkan installable client"
1098 " driver (ICD) installed?\nPlease look at the Getting Started"
1099 " guide for additional information.\n",
1100 "vkEnumeratePhysicalDevices Failure"
1105 uint32_t deviceExtensionCount = 0;
1106 VkBool32 swapchainExtFound = 0;
1107 enabledExtensionCount = 0;
1109 err = vkEnumerateDeviceExtensionProperties(
physicalDevice,
nullptr, &deviceExtensionCount,
nullptr);
1112 if (deviceExtensionCount > 0) {
1113 vector<VkExtensionProperties> deviceExtensions(deviceExtensionCount);
1114 err = vkEnumerateDeviceExtensionProperties(
physicalDevice,
nullptr, &deviceExtensionCount, deviceExtensions.data());
1116 for (i = 0; i < deviceExtensionCount; i++) {
1117 if (!strcmp(VK_KHR_SWAPCHAIN_EXTENSION_NAME, deviceExtensions[i].extensionName)) {
1118 swapchainExtFound = 1;
1119 extensionNames[enabledExtensionCount++] = VK_KHR_SWAPCHAIN_EXTENSION_NAME;
1121 assert(enabledExtensionCount < 64);
1125 if (!swapchainExtFound) {
1127 "vkEnumerateDeviceExtensionProperties failed to find "
1128 "the " VK_KHR_SWAPCHAIN_EXTENSION_NAME
1129 " extension.\n\nDo you have a compatible "
1130 "Vulkan installable client driver (ICD) installed?\nPlease "
1131 "look at the Getting Started guide for additional "
1132 "information.\n",
"vkCreateInstance Failure"
1158 err = glfwCreateWindowSurface(
instance, Application::glfwWindow,
nullptr, &
surface);
1162 vector<VkBool32> supportsPresent(
queueCount);
1170 uint32_t presentQueueNodeIndex = UINT32_MAX;
1176 if (supportsPresent[i] == VK_TRUE) {
1178 presentQueueNodeIndex = i;
1183 if (presentQueueNodeIndex == UINT32_MAX) {
1187 if (supportsPresent[i] == VK_TRUE) {
1188 presentQueueNodeIndex = i;
1197 "Could not find a graphics and a present queue\n",
1198 "Swapchain Initialization Failure"
1209 "Could not find a common graphics and a present queue\n",
1210 "Swapchain Initialization Failure"
1215 array<float, 1> queuePriorities { 0.0f };
1216 const VkDeviceQueueCreateInfo queueCreateInfo = {
1217 .sType = VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO,
1221 .queueCount = queuePriorities.size(),
1222 .pQueuePriorities = queuePriorities.data()
1225 VkPhysicalDeviceFeatures features {};
1226 if (
gpuFeatures.shaderClipDistance) features.shaderClipDistance = VK_TRUE;
1227 if (
gpuFeatures.wideLines) features.wideLines = VK_TRUE;
1229 VkDeviceCreateInfo deviceCreateInfo = {
1230 .sType = VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO,
1233 .queueCreateInfoCount = 1,
1234 .pQueueCreateInfos = &queueCreateInfo,
1235 .enabledLayerCount = 0,
1236 .ppEnabledLayerNames =
nullptr,
1237 .enabledExtensionCount = enabledExtensionCount,
1238 .ppEnabledExtensionNames = extensionNames.data(),
1239 .pEnabledFeatures = &features
1253 uint32_t surfaceFormatCount;
1256 vector<VkSurfaceFormatKHR> surfaceFormats(surfaceFormatCount);
1264 if (surfaceFormatCount == 1 && surfaceFormats[0].format == VK_FORMAT_UNDEFINED) {
1268 for (
auto i = 0; i < surfaceFormatCount; i++) {
1269 if (surfaceFormats[i].format == VK_FORMAT_B8G8R8A8_UNORM) {
1277 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): No format given");
1279 "Could not use VK_FORMAT_R8G8B8A8_UNORM as format\n",
1288 VmaAllocatorCreateInfo allocatorCreateInfo = {};
1290 allocatorCreateInfo.device =
device;
1291 allocatorCreateInfo.instance =
instance;
1292 allocatorCreateInfo.vulkanApiVersion = VK_API_VERSION_1_0;
1295 vmaCreateAllocator(&allocatorCreateInfo, &
vmaAllocator);
1302 array<VkDescriptorPoolSize, 2> desc1TypesCount = {{
1304 .type = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER,
1309 .type = VK_DESCRIPTOR_TYPE_STORAGE_BUFFER,
1314 const VkDescriptorPoolCreateInfo descriptorPoolCreateInfo = {
1315 .sType = VK_STRUCTURE_TYPE_DESCRIPTOR_POOL_CREATE_INFO,
1320 .poolSizeCount = desc1TypesCount.size(),
1321 .pPoolSizes = desc1TypesCount.data(),
1330 const VkDescriptorPoolSize desc2TypesCount = {
1331 .type = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER,
1335 const VkDescriptorPoolCreateInfo descriptorPoolCreateInfo = {
1336 .sType = VK_STRUCTURE_TYPE_DESCRIPTOR_POOL_CREATE_INFO,
1342 .pPoolSizes = &desc2TypesCount,
1353 auto& context =
contexts[contextIdx];
1355 context.idx = contextIdx;
1356 context.setupCommandInUse = VK_NULL_HANDLE;
1357 context.currentCommandBuffer = 0;
1358 context.pipelineIdx =
ID_NONE;
1359 context.pipeline = VK_NULL_HANDLE;
1360 context.renderPassStarted =
false;
1364 context.commandBuffers[i].drawCmdStarted =
false;
1365 VkFenceCreateInfo fenceCreateInfoSignaled = {
1366 .sType = VK_STRUCTURE_TYPE_FENCE_CREATE_INFO,
1368 .flags = VK_FENCE_CREATE_SIGNALED_BIT
1370 vkCreateFence(
device, &fenceCreateInfoSignaled,
nullptr, &context.commandBuffers[i].drawFence);
1377 const VkCommandPoolCreateInfo commandPoolCreateInfo = {
1378 .sType = VK_STRUCTURE_TYPE_COMMAND_POOL_CREATE_INFO,
1380 .flags = VK_COMMAND_POOL_CREATE_RESET_COMMAND_BUFFER_BIT,
1383 err = vkCreateCommandPool(
device, &commandPoolCreateInfo,
nullptr, &context.setupCommandPool);
1387 const VkCommandBufferAllocateInfo commandBufferAllocationInfo = {
1388 .sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO,
1390 .commandPool = context.setupCommandPool,
1391 .level = VK_COMMAND_BUFFER_LEVEL_PRIMARY,
1392 .commandBufferCount = 1
1394 err = vkAllocateCommandBuffers(
device, &commandBufferAllocationInfo, &context.setupCommand);
1400 const VkCommandPoolCreateInfo commandPoolCreateInfo = {
1401 .sType = VK_STRUCTURE_TYPE_COMMAND_POOL_CREATE_INFO,
1403 .flags = VK_COMMAND_POOL_CREATE_RESET_COMMAND_BUFFER_BIT,
1406 err = vkCreateCommandPool(
device, &commandPoolCreateInfo,
nullptr, &context.drawCommandPool);
1410 const VkCommandBufferAllocateInfo commandBufferAllocationInfo = {
1411 .sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO,
1413 .commandPool = context.drawCommandPool,
1414 .level = VK_COMMAND_BUFFER_LEVEL_PRIMARY,
1415 .commandBufferCount = 1
1418 err = vkAllocateCommandBuffers(
device, &commandBufferAllocationInfo, &context.commandBuffers[i].drawCommand);
1424 VkFenceCreateInfo fenceCreateInfo = {
1425 .sType = VK_STRUCTURE_TYPE_FENCE_CREATE_INFO,
1429 vkCreateFence(
device, &fenceCreateInfo,
nullptr, &context.setupFence);
1440 array<float, 16> bogusVertexBuffer = {{
1441 0.0f, 0.0f, 0.0f, 0.0f,
1442 0.0f, 0.0f, 0.0f, 0.0f,
1443 0.0f, 0.0f, 0.0f, 0.0f,
1444 0.0f, 0.0f, 0.0f, 0.0f,
1446 uploadBufferObjectInternal(0,
emptyVertexBuffer, bogusVertexBuffer.size() *
sizeof(
float), (uint8_t*)bogusVertexBuffer.data(), (VkBufferUsageFlagBits)(VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT));
1455 "cubemap-default-white",
1456 TextureReader::read(
"resources/engine/textures",
"transparent_pixel.png"),
1457 TextureReader::read(
"resources/engine/textures",
"transparent_pixel.png"),
1458 TextureReader::read(
"resources/engine/textures",
"transparent_pixel.png"),
1459 TextureReader::read(
"resources/engine/textures",
"transparent_pixel.png"),
1460 TextureReader::read(
"resources/engine/textures",
"transparent_pixel.png"),
1461 TextureReader::read(
"resources/engine/textures",
"transparent_pixel.png"),
1476 for (
auto i = 0; i < rendererContext.lights.size(); i++) {
1477 rendererContext.lights[i].spotCosCutoff =
static_cast<float>(Math::cos(Math::PI / 180.0f * 180.0f));
1479 rendererContext.textureMatrix.identity();
1498 { THSVS_ACCESS_DEPTH_STENCIL_ATTACHMENT_READ, THSVS_ACCESS_DEPTH_STENCIL_ATTACHMENT_WRITE },
1499 THSVS_IMAGE_LAYOUT_OPTIMAL,
1504 array<VkAttachmentDescription, 2> attachments = {{
1508 .samples = VK_SAMPLE_COUNT_1_BIT,
1509 .loadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE,
1510 .storeOp = VK_ATTACHMENT_STORE_OP_STORE,
1511 .stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE,
1512 .stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE,
1513 .initialLayout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL,
1514 .finalLayout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL
1518 .format = VK_FORMAT_D32_SFLOAT,
1519 .samples = VK_SAMPLE_COUNT_1_BIT,
1520 .loadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE,
1521 .storeOp = VK_ATTACHMENT_STORE_OP_DONT_CARE,
1522 .stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE,
1523 .stencilStoreOp = VK_ATTACHMENT_STORE_OP_STORE,
1524 .initialLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL,
1525 .finalLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL
1528 const VkAttachmentReference colorReference = {
1530 .layout = VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL,
1532 const VkAttachmentReference depthReference = {
1534 .layout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL,
1536 const VkSubpassDescription subPass = {
1538 .pipelineBindPoint = VK_PIPELINE_BIND_POINT_GRAPHICS,
1539 .inputAttachmentCount = 0,
1540 .pInputAttachments =
nullptr,
1541 .colorAttachmentCount = 1,
1542 .pColorAttachments = &colorReference,
1543 .pResolveAttachments =
nullptr,
1544 .pDepthStencilAttachment = &depthReference,
1545 .preserveAttachmentCount = 0,
1546 .pPreserveAttachments =
nullptr
1548 const VkRenderPassCreateInfo rp_info = {
1549 .sType = VK_STRUCTURE_TYPE_RENDER_PASS_CREATE_INFO,
1552 .attachmentCount = 2,
1553 .pAttachments = attachments.data(),
1555 .pSubpasses = &subPass,
1556 .dependencyCount = 0,
1557 .pDependencies =
nullptr
1564 auto& currentContext =
contexts[contextIdx];
1566 if (currentContext.renderPassStarted ==
true)
return;
1567 currentContext.renderPassStarted =
true;
1573 if (frameBuffer ==
nullptr) {
1574 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): framebuffer not found: " + to_string(
boundFrameBufferId));
1576 usedFrameBuffer = frameBuffer->frameBuffer;
1577 vkRenderPass = frameBuffer->renderPass;
1582 const VkRenderPassBeginInfo renderPassBegin = {
1583 .sType = VK_STRUCTURE_TYPE_RENDER_PASS_BEGIN_INFO,
1585 .renderPass = vkRenderPass,
1586 .framebuffer = usedFrameBuffer,
1588 .clearValueCount = 0,
1589 .pClearValues =
nullptr
1591 vkCmdBeginRenderPass(currentContext.commandBuffers[currentContext.currentCommandBuffer].drawCommand, &renderPassBegin, VK_SUBPASS_CONTENTS_INLINE);
1598 auto& currentContext =
contexts[contextIdx];
1599 if (currentContext.renderPassStarted ==
false)
return;
1600 currentContext.renderPassStarted =
false;
1601 vkCmdEndRenderPass(currentContext.commandBuffers[currentContext.currentCommandBuffer].drawCommand);
1605 array<VkImageView, 2> attachments;
1607 attachments[1] = depthBufferTexture->view;
1609 const VkFramebufferCreateInfo frameBufferCreateInfo = {
1610 .sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO,
1614 .attachmentCount = 2,
1615 .pAttachments = attachments.data(),
1629 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
1635 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(
windowWidth) +
" x " + to_string(
windowHeight));
1661 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
1665 int32_t currentWidth;
1666 int32_t currentHeight;
1667 glfwGetWindowSize(Application::glfwWindow, ¤tWidth, ¤tHeight);
1671 if (needsReshape ==
true)
reshape();
1676 VkSemaphoreCreateInfo semaphoreCreateInfo = {
1677 .sType = VK_STRUCTURE_TYPE_SEMAPHORE_CREATE_INFO,
1695 if (err == VK_ERROR_OUT_OF_DATE_KHR) {
1713 if (err == VK_SUBOPTIMAL_KHR) {
1721 for (
auto i = 0; i <
contexts.size(); i++) {
1730 if (program ==
nullptr || program->contexts[context.idx].texturesDescriptorSetsCacheTextureIds.empty() ==
true)
continue;
1731 auto& programContext = program->contexts[context.idx];
1732 auto& descriptorSets2CacheTextureIds = programContext.texturesDescriptorSetsCacheTextureIds;
1733 auto descriptorSets2CacheTextureIdsIt = descriptorSets2CacheTextureIds.find(textureId);
1734 if (descriptorSets2CacheTextureIdsIt != descriptorSets2CacheTextureIds.end()) {
1735 auto& descriptorSets2Cache = programContext.texturesDescriptorSetsCache;
1736 for (
auto& descriptorSets2CacheHash: descriptorSets2CacheTextureIdsIt->second) {
1737 auto descriptorSets2CacheHashIt = descriptorSets2Cache.find(descriptorSets2CacheHash);
1738 if (descriptorSets2CacheHashIt != descriptorSets2Cache.end()) {
1739 auto desc_sets2_idx = descriptorSets2CacheHashIt->second;
1740 programContext.freeTextureDescriptorSetsIds.push_back(desc_sets2_idx);
1741 descriptorSets2Cache.erase(descriptorSets2CacheHashIt);
1744 descriptorSets2CacheTextureIds.erase(descriptorSets2CacheTextureIdsIt);
1752 auto clearedPipelinesParents = 0;
1753 auto clearedPipelines = 0;
1763 if (framebufferPipelines->id ==
ID_NONE)
continue;
1765 for (
auto pipeline: framebufferPipelines->pipelines) {
1766 if (pipeline != VK_NULL_HANDLE) {
1771 delete framebufferPipelines;
1774 clearedPipelinesParents++;
1780 "VKRenderer::" +
string(__FUNCTION__) +
"(): " +
1781 "cleared pipelines parents: " + to_string(clearedPipelinesParents) +
", " +
1782 "cleared pipelines: " + to_string(clearedPipelines)
1788 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
1796 context.program =
nullptr;
1803 array<ThsvsAccessType, 2> nextAccessTypes { THSVS_ACCESS_PRESENT, THSVS_ACCESS_NONE };
1804 ThsvsImageLayout nextLayout { THSVS_IMAGE_LAYOUT_OPTIMAL };
1808 ThsvsImageBarrier svsImageBarrier = {
1811 .nextAccessCount =
static_cast<uint32_t
>(nextAccessTypes[1] != THSVS_ACCESS_NONE?2:1),
1812 .pNextAccesses = nextAccessTypes.data(),
1814 .nextLayout = nextLayout,
1815 .discardContents =
false,
1816 .srcQueueFamilyIndex = 0,
1817 .dstQueueFamilyIndex = 0,
1819 .subresourceRange = {
1820 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
1823 .baseArrayLayer = 0,
1827 VkImageMemoryBarrier vkImageMemoryBarrier;
1828 VkPipelineStageFlags srcStages;
1829 VkPipelineStageFlags dstStages;
1830 thsvsGetVulkanImageMemoryBarrier(
1834 &vkImageMemoryBarrier
1842 vkCmdPipelineBarrier(
contexts[0].setupCommandInUse, srcStages, dstStages, 0, 0,
nullptr, 0,
nullptr, 1, &vkImageMemoryBarrier);
1851 VkResult presentResult = VK_SUCCESS;
1852 VkPresentInfoKHR presentInfoKHR = {
1853 .sType = VK_STRUCTURE_TYPE_PRESENT_INFO_KHR,
1855 .waitSemaphoreCount = 0,
1856 .pWaitSemaphores =
nullptr,
1857 .swapchainCount = 1,
1860 .pResults = &presentResult
1866 auto needsReshape =
false;
1867 if (err == VK_ERROR_OUT_OF_DATE_KHR) {
1868 needsReshape =
true;
1870 if (err == VK_SUBOPTIMAL_KHR) {
1873 needsReshape =
true;
1893 if (texture ==
nullptr) {
1894 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): disposing texture: texture not found: " + to_string(textureId));
1900 .image = texture->image,
1901 .allocation = texture->allocation,
1902 .imageView = texture->view,
1903 .sampler = texture->sampler
1906 if (texture->cubemapColorBuffer !=
nullptr) {
1909 .image = texture->cubemapColorBuffer->image,
1910 .allocation = texture->cubemapColorBuffer->allocation,
1911 .imageView = texture->cubemapColorBuffer->view,
1912 .sampler = texture->cubemapColorBuffer->sampler
1916 if (texture->cubemapDepthBuffer !=
nullptr) {
1919 .image = texture->cubemapDepthBuffer->image,
1920 .allocation = texture->cubemapDepthBuffer->allocation,
1921 .imageView = texture->cubemapDepthBuffer->view,
1922 .sampler = texture->cubemapDepthBuffer->sampler
1938 if (buffer ==
nullptr) {
1939 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): disposing buffer object: buffer with id " + to_string(bufferObjectId) +
" does not exist");
1942 for (
auto& reusableBufferIt: buffer->buffers) {
1943 auto& reusableBuffer = reusableBufferIt;
1944 if (reusableBuffer.size == 0)
continue;
1948 .buffer = reusableBuffer.buf,
1949 .allocation = reusableBuffer.allocation
1953 buffers[bufferObjectId] =
nullptr;
1961 vkDestroyPipeline(
device, pipeline,
nullptr);
1970 vmaDestroyBuffer(
vmaAllocator, deleteBuffer.buffer, deleteBuffer.allocation);
1976 if (deleteImage.imageView != VK_NULL_HANDLE) vkDestroyImageView(
device, deleteImage.imageView,
nullptr);
1977 if (deleteImage.sampler != VK_NULL_HANDLE) vkDestroySampler(
device, deleteImage.sampler,
nullptr);
1978 if (deleteImage.image != VK_NULL_HANDLE) vmaDestroyImage(
vmaAllocator, deleteImage.image, deleteImage.allocation);
1988 uint32_t bufferSize = 0;
1990 context.boundIndicesBuffer = VK_NULL_HANDLE;
1992 context.boundBufferSizes.fill(bufferSize);
1995 context.uploadedTextureIds.clear();
2063int32_t
VKRenderer::loadShader(int32_t type,
const string& pathName,
const string& fileName,
const string& definitions,
const string& functions)
2065 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): INIT: " + pathName +
"/" + fileName +
": " + definitions);
2069 auto& shader = *shaderPtr;
2080 auto& currentContext =
contexts[contextIdx];
2083 currentContext.pipelineIdx =
ID_NONE;
2084 currentContext.pipeline = VK_NULL_HANDLE;
2088 rasterizationStateCreateInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_RASTERIZATION_STATE_CREATE_INFO;
2089 rasterizationStateCreateInfo.polygonMode = VK_POLYGON_MODE_FILL;
2090 rasterizationStateCreateInfo.cullMode =
contexts[contextIdx].cullingEnabled ==
true?
cullMode:VK_CULL_MODE_NONE;
2091 rasterizationStateCreateInfo.frontFace = (VkFrontFace)(
contexts[contextIdx].frontFace - 1);
2092 rasterizationStateCreateInfo.depthClampEnable = VK_FALSE;
2093 rasterizationStateCreateInfo.rasterizerDiscardEnable = VK_FALSE;
2094 rasterizationStateCreateInfo.depthBiasEnable = VK_FALSE;
2095 rasterizationStateCreateInfo.lineWidth = 1.0f;
2099 blendAttachmentState.colorWriteMask = VK_COLOR_COMPONENT_R_BIT | VK_COLOR_COMPONENT_G_BIT | VK_COLOR_COMPONENT_B_BIT | VK_COLOR_COMPONENT_A_BIT;
2102 blendAttachmentState.dstColorBlendFactor =
blendingMode ==
BLENDING_NORMAL?VK_BLEND_FACTOR_ONE_MINUS_SRC_ALPHA:VK_BLEND_FACTOR_ONE;
2103 blendAttachmentState.colorBlendOp = VK_BLEND_OP_ADD;
2105 blendAttachmentState.dstAlphaBlendFactor =
blendingMode ==
BLENDING_NORMAL?VK_BLEND_FACTOR_ONE_MINUS_SRC_ALPHA:VK_BLEND_FACTOR_ONE;
2106 blendAttachmentState.alphaBlendOp = VK_BLEND_OP_ADD;
2110 depthStencilStateCreateInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_DEPTH_STENCIL_STATE_CREATE_INFO;
2111 depthStencilStateCreateInfo.depthTestEnable =
depthBufferTesting ==
true?VK_TRUE:VK_FALSE;
2112 depthStencilStateCreateInfo.depthWriteEnable =
depthBufferWriting ==
true?VK_TRUE:VK_FALSE;
2113 depthStencilStateCreateInfo.depthCompareOp = (VkCompareOp)
depthFunction;
2114 depthStencilStateCreateInfo.back.failOp = VK_STENCIL_OP_KEEP;
2115 depthStencilStateCreateInfo.back.passOp = VK_STENCIL_OP_KEEP;
2116 depthStencilStateCreateInfo.back.compareOp = VK_COMPARE_OP_ALWAYS;
2117 depthStencilStateCreateInfo.stencilTestEnable = VK_FALSE;
2118 depthStencilStateCreateInfo.front = depthStencilStateCreateInfo.back;
2119 depthStencilStateCreateInfo.depthBoundsTestEnable = VK_FALSE;
2120 depthStencilStateCreateInfo.minDepthBounds = 0.0f;
2121 depthStencilStateCreateInfo.maxDepthBounds = 1.0f;
2133 (program->
id & 0x7f) +
2134 ((
contexts[contextIdx].cullingEnabled ==
true?
cullMode:VK_CULL_MODE_NONE & 0x3) << 7) +
2135 ((
contexts[contextIdx].frontFaceIndex & 0x3) << 9) +
2144 vector<VkDescriptorSetLayoutBinding> layoutBindings1(program->
layoutBindings);
2145 vector<VkDescriptorSetLayoutBinding> layoutBindings2(program->
layoutBindings);
2148 auto samplerIdx = 0;
2150 for (
auto shader: program->
shaders) {
2151 if (shader->uboBindingIdx != -1) {
2152 layoutBindings1[uboIdx++] = {
2153 .binding =
static_cast<uint32_t
>(shader->uboBindingIdx),
2154 .descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER,
2155 .descriptorCount = 1,
2156 .stageFlags =
static_cast<VkShaderStageFlags
>(shader->type),
2157 .pImmutableSamplers =
nullptr
2161 for (
auto uniform: shader->samplerUniformList) {
2162 layoutBindings2[samplerIdx++] = {
2163 .binding =
static_cast<uint32_t
>(uniform->position),
2164 .descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER,
2165 .descriptorCount = 1,
2166 .stageFlags =
static_cast<VkShaderStageFlags
>(shader->type),
2167 .pImmutableSamplers =
nullptr
2173 const VkDescriptorSetLayoutCreateInfo descriptorSetLayoutCreateInfo = {
2174 .sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO,
2177 .bindingCount =
static_cast<uint32_t
>(uboIdx),
2178 .pBindings = layoutBindings1.data(),
2184 const VkDescriptorSetLayoutCreateInfo descriptorSetLayoutCreateInfo = {
2185 .sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO,
2188 .bindingCount =
static_cast<uint32_t
>(samplerIdx),
2189 .pBindings = layoutBindings2.data(),
2197 const VkPipelineLayoutCreateInfo pipelineLayoutCreateInfo = {
2198 .sType = VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO,
2201 .setLayoutCount = descriptorSetLayouts.size(),
2202 .pSetLayouts = descriptorSetLayouts.data()
2207 array<VkDescriptorSetLayout, DESC_MAX_UNCACHED> descriptorSetLayouts1;
2210 VkDescriptorSetAllocateInfo descriptorSetAllocateInfo = {
2211 .sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_ALLOCATE_INFO,
2215 .pSetLayouts = descriptorSetLayouts1.data()
2218 err = vkAllocateDescriptorSets(
device, &descriptorSetAllocateInfo, program->
contexts[context.idx].commandBuffers[i].uboDescriptorSets.data());
2225 array<VkDescriptorSetLayout, DESC_MAX_CACHED> descriptorSetLayouts2;
2228 VkDescriptorSetAllocateInfo descriptorSetAllocateInfo = {
2229 .sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_ALLOCATE_INFO,
2233 .pSetLayouts = descriptorSetLayouts2.data()
2236 err = vkAllocateDescriptorSets(
device, &descriptorSetAllocateInfo, program->
contexts[context.idx].descriptorSets2.data());
2243 array<VkDescriptorSetLayout, DESC_MAX_UNCACHED> descriptorSetLayouts2;
2246 VkDescriptorSetAllocateInfo descriptorSetAllocateInfo = {
2247 .sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_ALLOCATE_INFO,
2251 .pSetLayouts = descriptorSetLayouts2.data()
2254 err = vkAllocateDescriptorSets(
device, &descriptorSetAllocateInfo, program->
contexts[context.idx].commandBuffers[i].texturesDescriptorSetsUncached.data());
2260 err = vkCreatePipelineLayout(
device, &pipelineLayoutCreateInfo,
nullptr, &program->
pipelineLayout);
2265 auto& currentContext =
contexts[contextIdx];
2269 if (framebufferPipelines ==
nullptr) {
2275 auto haveDepthBuffer =
true;
2276 auto haveColorBuffer =
true;
2277 auto haveGeometryBuffer =
false;
2280 if (frameBuffer !=
nullptr) {
2281 haveDepthBuffer = frameBuffer->depthTextureId !=
ID_NONE;
2282 haveColorBuffer = frameBuffer->colorTextureId !=
ID_NONE;
2284 usedRenderPass = frameBuffer->renderPass;
2286 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): framebuffer with id: " + to_string(
boundFrameBufferId) +
" not found!");
2294 VkGraphicsPipelineCreateInfo pipeline {};
2297 VkPipelineCacheCreateInfo pipelineCacheCreateInfo {};
2298 VkPipelineCache pipelineCache = VK_NULL_HANDLE;
2300 VkPipelineVertexInputStateCreateInfo vi {};
2301 VkPipelineInputAssemblyStateCreateInfo ia {};
2302 VkPipelineRasterizationStateCreateInfo rs {};
2303 VkPipelineColorBlendStateCreateInfo cb {};
2304 VkPipelineDepthStencilStateCreateInfo ds {};
2305 VkPipelineViewportStateCreateInfo vp {};
2306 VkPipelineMultisampleStateCreateInfo ms {};
2311 array<VkPipelineShaderStageCreateInfo, 2> shaderStages {};
2315 for (
auto shader: program->
shaders) {
2316 shaderStages[
shaderIdx].sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO;
2317 shaderStages[
shaderIdx].stage = shader->type;
2318 shaderStages[
shaderIdx].module = shader->module;
2323 pipeline.sType = VK_STRUCTURE_TYPE_GRAPHICS_PIPELINE_CREATE_INFO;
2327 ia.sType = VK_STRUCTURE_TYPE_PIPELINE_INPUT_ASSEMBLY_STATE_CREATE_INFO;
2328 ia.topology = VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST;
2330 array<VkPipelineColorBlendAttachmentState, 8> bas;
2331 if (haveColorBuffer ==
true) {
2334 if (haveGeometryBuffer ==
true) {
2335 for (
auto i = 0; i < 8; i++) {
2340 cb.sType = VK_STRUCTURE_TYPE_PIPELINE_COLOR_BLEND_STATE_CREATE_INFO;
2341 cb.logicOpEnable = VK_FALSE;
2342 cb.attachmentCount = haveColorBuffer ==
true?1:(haveGeometryBuffer ==
true?8:0);
2343 cb.pAttachments = haveColorBuffer ==
true || haveGeometryBuffer ==
true?bas.data():
nullptr;
2345 vp.sType = VK_STRUCTURE_TYPE_PIPELINE_VIEWPORT_STATE_CREATE_INFO;
2346 vp.viewportCount = 1;
2348 vp.scissorCount = 1;
2351 ms.sType = VK_STRUCTURE_TYPE_PIPELINE_MULTISAMPLE_STATE_CREATE_INFO;
2352 ms.pSampleMask =
nullptr;
2353 ms.rasterizationSamples = VK_SAMPLE_COUNT_1_BIT;
2355 array<VkVertexInputBindingDescription, 10> vb {};
2356 array<VkVertexInputAttributeDescription, 13> va {};
2360 vb[0].stride =
sizeof(float) * 3;
2361 vb[0].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2364 va[0].format = VK_FORMAT_R32G32B32_SFLOAT;
2369 vb[1].stride =
sizeof(float) * 3;
2370 vb[1].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2373 va[1].format = VK_FORMAT_R32G32B32_SFLOAT;
2378 vb[2].stride =
sizeof(float) * 2;
2379 vb[2].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2382 va[2].format = VK_FORMAT_R32G32_SFLOAT;
2387 vb[3].stride =
sizeof(float) * 4;
2388 vb[3].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2391 va[3].format = VK_FORMAT_R32G32B32A32_SFLOAT;
2396 vb[4].stride =
sizeof(float) * 3;
2397 vb[4].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2400 va[4].format = VK_FORMAT_R32G32B32_SFLOAT;
2405 vb[5].stride =
sizeof(float) * 3;
2406 vb[5].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2409 va[5].format = VK_FORMAT_R32G32B32_SFLOAT;
2414 vb[6].stride =
sizeof(float) * 4 * 4;
2415 vb[6].inputRate = VK_VERTEX_INPUT_RATE_INSTANCE;
2418 va[6].format = VK_FORMAT_R32G32B32A32_SFLOAT;
2419 va[6].offset =
sizeof(float) * 4 * 0;
2424 va[7].format = VK_FORMAT_R32G32B32A32_SFLOAT;
2425 va[7].offset =
sizeof(float) * 4 * 1;
2430 va[8].format = VK_FORMAT_R32G32B32A32_SFLOAT;
2431 va[8].offset =
sizeof(float) * 4 * 2;
2436 va[9].format = VK_FORMAT_R32G32B32A32_SFLOAT;
2437 va[9].offset =
sizeof(float) * 4 * 3;
2441 vb[7].stride =
sizeof(float) * 4;
2442 vb[7].inputRate = VK_VERTEX_INPUT_RATE_INSTANCE;
2444 va[10].location = 10;
2445 va[10].format = VK_FORMAT_R32G32B32A32_SFLOAT;
2450 vb[8].stride =
sizeof(float) * 4;
2451 vb[8].inputRate = VK_VERTEX_INPUT_RATE_INSTANCE;
2453 va[11].location = 11;
2454 va[11].format = VK_FORMAT_R32G32B32A32_SFLOAT;
2459 vb[9].stride =
sizeof(float) * 3;
2460 vb[9].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2462 va[12].location = 12;
2463 va[12].format = VK_FORMAT_R32G32B32_SFLOAT;
2466 vi.sType = VK_STRUCTURE_TYPE_PIPELINE_VERTEX_INPUT_STATE_CREATE_INFO;
2468 vi.vertexBindingDescriptionCount = vb.size();
2469 vi.pVertexBindingDescriptions = vb.data();
2470 vi.vertexAttributeDescriptionCount = va.size();
2471 vi.pVertexAttributeDescriptions = va.data();
2473 pipeline.pVertexInputState = &vi;
2474 pipeline.pInputAssemblyState = &ia;
2475 pipeline.pRasterizationState = &rs;
2476 pipeline.pColorBlendState = haveColorBuffer ==
true || haveGeometryBuffer ==
true?&cb:
nullptr;
2477 pipeline.pMultisampleState = &ms;
2478 pipeline.pViewportState = &vp;
2479 pipeline.pDepthStencilState = haveDepthBuffer ==
true?&ds:
nullptr;
2480 pipeline.pStages = shaderStages.data();
2481 pipeline.renderPass = usedRenderPass;
2482 pipeline.pDynamicState =
nullptr;
2484 pipelineCacheCreateInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_CACHE_CREATE_INFO;
2485 err = vkCreatePipelineCache(
device, &pipelineCacheCreateInfo,
nullptr, &pipelineCache);
2488 err = vkCreateGraphicsPipelines(
device, pipelineCache, 1, &pipeline,
nullptr, &framebufferPipelines->pipelines[currentContext.pipelineIdx]);
2492 vkDestroyPipelineCache(
device, pipelineCache,
nullptr);
2496 auto& currentContext =
contexts[contextIdx];
2497 if (currentContext.pipelineIdx ==
ID_NONE || currentContext.pipeline == VK_NULL_HANDLE) {
2501 if (pipeline == VK_NULL_HANDLE) {
2507 auto& commandBuffer = currentContext.commandBuffers[currentContext.currentCommandBuffer];
2508 vkCmdBindPipeline(commandBuffer.drawCommand, VK_PIPELINE_BIND_POINT_GRAPHICS, pipeline);
2509 currentContext.pipeline = pipeline;
2514 auto& currentContext =
contexts[contextIdx];
2518 if (framebufferPipelines ==
nullptr) {
2524 auto haveDepthBuffer =
true;
2525 auto haveColorBuffer =
true;
2526 auto haveGeometryBuffer =
false;
2529 if (frameBuffer !=
nullptr) {
2530 haveDepthBuffer = frameBuffer->depthTextureId !=
ID_NONE;
2531 haveColorBuffer = frameBuffer->colorTextureId !=
ID_NONE;
2533 usedRenderPass = frameBuffer->renderPass;
2541 VkGraphicsPipelineCreateInfo pipeline {};
2544 array<VkPipelineShaderStageCreateInfo, 2> shaderStages {};
2548 for (
auto shader: program->
shaders) {
2549 shaderStages[
shaderIdx].sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO;
2550 shaderStages[
shaderIdx].stage = shader->type;
2551 shaderStages[
shaderIdx].module = shader->module;
2557 VkPipelineCacheCreateInfo pipelineCacheCreateInfo {};
2558 VkPipelineCache pipelineCache = VK_NULL_HANDLE;
2560 VkPipelineVertexInputStateCreateInfo vi {};
2561 VkPipelineInputAssemblyStateCreateInfo ia {};
2562 VkPipelineRasterizationStateCreateInfo rs {};
2563 VkPipelineColorBlendStateCreateInfo cb {};
2564 VkPipelineDepthStencilStateCreateInfo ds {};
2565 VkPipelineViewportStateCreateInfo vp {};
2566 VkPipelineMultisampleStateCreateInfo ms {};
2571 pipeline.sType = VK_STRUCTURE_TYPE_GRAPHICS_PIPELINE_CREATE_INFO;
2575 ia.sType = VK_STRUCTURE_TYPE_PIPELINE_INPUT_ASSEMBLY_STATE_CREATE_INFO;
2576 ia.topology = VK_PRIMITIVE_TOPOLOGY_POINT_LIST;
2578 array<VkPipelineColorBlendAttachmentState, 8> bas;
2579 if (haveColorBuffer ==
true) {
2582 if (haveGeometryBuffer ==
true) {
2583 for (
auto i = 0; i < 8; i++) {
2587 cb.sType = VK_STRUCTURE_TYPE_PIPELINE_COLOR_BLEND_STATE_CREATE_INFO;
2588 cb.logicOpEnable = VK_FALSE;
2589 cb.attachmentCount = haveColorBuffer ==
true?1:(haveGeometryBuffer ==
true?8:0);
2590 cb.pAttachments = haveColorBuffer ==
true || haveGeometryBuffer ==
true?bas.data():
nullptr;
2592 vp.sType = VK_STRUCTURE_TYPE_PIPELINE_VIEWPORT_STATE_CREATE_INFO;
2593 vp.viewportCount = 1;
2595 vp.scissorCount = 1;
2598 ms.sType = VK_STRUCTURE_TYPE_PIPELINE_MULTISAMPLE_STATE_CREATE_INFO;
2599 ms.pSampleMask =
nullptr;
2600 ms.rasterizationSamples = VK_SAMPLE_COUNT_1_BIT;
2602 array<VkVertexInputBindingDescription, 9> vb {};
2603 array<VkVertexInputAttributeDescription, 9> va {};
2607 vb[0].stride =
sizeof(float) * 3;
2608 vb[0].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2611 va[0].format = VK_FORMAT_R32G32B32_SFLOAT;
2616 vb[1].stride =
sizeof(uint16_t) * 2;
2617 vb[1].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2620 va[1].format = VK_FORMAT_R16G16_UINT;
2625 vb[2].stride =
sizeof(float);
2626 vb[2].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2629 va[2].format = VK_FORMAT_R32_SFLOAT;
2634 vb[3].stride =
sizeof(float) * 4;
2635 vb[3].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2638 va[3].format = VK_FORMAT_R32G32B32A32_SFLOAT;
2643 vb[4].stride =
sizeof(float);
2644 vb[4].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2647 va[4].format = VK_FORMAT_R32_SFLOAT;
2652 vb[5].stride =
sizeof(float);
2653 vb[5].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2656 va[5].format = VK_FORMAT_R32_SFLOAT;
2661 vb[6].stride =
sizeof(uint16_t) * 2;
2662 vb[6].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2665 va[6].format = VK_FORMAT_R16G16_UINT;
2670 vb[7].stride =
sizeof(float) * 4;
2671 vb[7].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2673 va[7].location = 10;
2674 va[7].format = VK_FORMAT_R32G32B32A32_SFLOAT;
2679 vb[8].stride =
sizeof(float) * 4;
2680 vb[8].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2682 va[8].location = 11;
2683 va[8].format = VK_FORMAT_R32G32B32A32_SFLOAT;
2687 vi.sType = VK_STRUCTURE_TYPE_PIPELINE_VERTEX_INPUT_STATE_CREATE_INFO;
2689 vi.vertexBindingDescriptionCount = vb.size();
2690 vi.pVertexBindingDescriptions = vb.data();
2691 vi.vertexAttributeDescriptionCount = va.size();
2692 vi.pVertexAttributeDescriptions = va.data();
2694 pipeline.pVertexInputState = &vi;
2695 pipeline.pInputAssemblyState = &ia;
2696 pipeline.pRasterizationState = &rs;
2697 pipeline.pColorBlendState = haveColorBuffer ==
true || haveGeometryBuffer ==
true?&cb:
nullptr;
2698 pipeline.pMultisampleState = &ms;
2699 pipeline.pViewportState = &vp;
2700 pipeline.pDepthStencilState = haveDepthBuffer ==
true?&ds:
nullptr;
2701 pipeline.pStages = shaderStages.data();
2702 pipeline.renderPass = usedRenderPass;
2703 pipeline.pDynamicState =
nullptr;
2705 pipelineCacheCreateInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_CACHE_CREATE_INFO;
2706 err = vkCreatePipelineCache(
device, &pipelineCacheCreateInfo,
nullptr, &pipelineCache);
2709 err = vkCreateGraphicsPipelines(
device, pipelineCache, 1, &pipeline,
nullptr, &framebufferPipelines->pipelines[currentContext.pipelineIdx]);
2713 vkDestroyPipelineCache(
device, pipelineCache,
nullptr);
2717 auto& currentContext =
contexts[contextIdx];
2718 if (currentContext.pipelineIdx ==
ID_NONE || currentContext.pipeline == VK_NULL_HANDLE) {
2722 if (pipeline == VK_NULL_HANDLE) {
2729 auto& commandBuffer = currentContext.commandBuffers[currentContext.currentCommandBuffer];
2730 vkCmdBindPipeline(commandBuffer.drawCommand, VK_PIPELINE_BIND_POINT_GRAPHICS, pipeline);
2731 currentContext.pipeline = pipeline;
2736 auto& currentContext =
contexts[contextIdx];
2740 if (framebufferPipelines ==
nullptr) {
2746 auto haveDepthBuffer =
true;
2747 auto haveColorBuffer =
true;
2748 auto haveGeometryBuffer =
false;
2751 if (frameBuffer !=
nullptr) {
2752 haveDepthBuffer = frameBuffer->depthTextureId !=
ID_NONE;
2753 haveColorBuffer = frameBuffer->colorTextureId !=
ID_NONE;
2755 usedRenderPass = frameBuffer->renderPass;
2763 VkGraphicsPipelineCreateInfo pipeline {};
2764 VkPipelineCache pipelineCache = VK_NULL_HANDLE;
2767 array<VkPipelineShaderStageCreateInfo, 2> shaderStages {};
2771 for (
auto shader: program->
shaders) {
2772 shaderStages[
shaderIdx].sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO;
2773 shaderStages[
shaderIdx].stage = shader->type;
2774 shaderStages[
shaderIdx].module = shader->module;
2780 VkPipelineCacheCreateInfo pipelineCacheCreateInfo {};
2782 VkPipelineVertexInputStateCreateInfo vi {};
2783 VkPipelineInputAssemblyStateCreateInfo ia {};
2784 VkPipelineRasterizationStateCreateInfo rs {};
2785 VkPipelineColorBlendStateCreateInfo cb {};
2786 VkPipelineDepthStencilStateCreateInfo ds {};
2787 VkPipelineViewportStateCreateInfo vp {};
2788 VkPipelineMultisampleStateCreateInfo ms {};
2789 array<VkDynamicState, 1> dse {};
2790 VkPipelineDynamicStateCreateInfo dsc {};
2795 dse[dsc.dynamicStateCount++] = VK_DYNAMIC_STATE_LINE_WIDTH;
2796 dsc.sType = VK_STRUCTURE_TYPE_PIPELINE_DYNAMIC_STATE_CREATE_INFO;
2797 dsc.pDynamicStates = dse.data();
2799 pipeline.sType = VK_STRUCTURE_TYPE_GRAPHICS_PIPELINE_CREATE_INFO;
2803 ia.sType = VK_STRUCTURE_TYPE_PIPELINE_INPUT_ASSEMBLY_STATE_CREATE_INFO;
2804 ia.topology = VK_PRIMITIVE_TOPOLOGY_LINE_LIST;
2806 array<VkPipelineColorBlendAttachmentState, 8> bas;
2807 if (haveColorBuffer ==
true) {
2810 if (haveGeometryBuffer ==
true) {
2811 for (
auto i = 0; i < 8; i++) {
2816 cb.sType = VK_STRUCTURE_TYPE_PIPELINE_COLOR_BLEND_STATE_CREATE_INFO;
2817 cb.logicOpEnable = VK_FALSE;
2818 cb.attachmentCount = haveColorBuffer ==
true?1:(haveGeometryBuffer ==
true?8:0);
2819 cb.pAttachments = haveColorBuffer ==
true || haveGeometryBuffer ==
true?bas.data():
nullptr;
2821 vp.sType = VK_STRUCTURE_TYPE_PIPELINE_VIEWPORT_STATE_CREATE_INFO;
2822 vp.viewportCount = 1;
2824 vp.scissorCount = 1;
2827 ms.sType = VK_STRUCTURE_TYPE_PIPELINE_MULTISAMPLE_STATE_CREATE_INFO;
2828 ms.pSampleMask =
nullptr;
2829 ms.rasterizationSamples = VK_SAMPLE_COUNT_1_BIT;
2831 array<VkVertexInputBindingDescription, 4> vb {};
2832 array<VkVertexInputAttributeDescription, 4> va {};
2836 vb[0].stride =
sizeof(float) * 3;
2837 vb[0].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2840 va[0].format = VK_FORMAT_R32G32B32_SFLOAT;
2845 vb[1].stride =
sizeof(float) * 3;
2846 vb[1].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2849 va[1].format = VK_FORMAT_R32G32B32_SFLOAT;
2854 vb[2].stride =
sizeof(float) * 2;
2855 vb[2].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2858 va[2].format = VK_FORMAT_R32G32_SFLOAT;
2863 vb[3].stride =
sizeof(float) * 4;
2864 vb[3].inputRate = VK_VERTEX_INPUT_RATE_VERTEX;
2867 va[3].format = VK_FORMAT_R32G32B32A32_SFLOAT;
2870 vi.sType = VK_STRUCTURE_TYPE_PIPELINE_VERTEX_INPUT_STATE_CREATE_INFO;
2872 vi.vertexBindingDescriptionCount = vb.size();
2873 vi.pVertexBindingDescriptions = vb.data();
2874 vi.vertexAttributeDescriptionCount = va.size();
2875 vi.pVertexAttributeDescriptions = va.data();
2877 pipeline.pVertexInputState = &vi;
2878 pipeline.pInputAssemblyState = &ia;
2879 pipeline.pRasterizationState = &rs;
2880 pipeline.pColorBlendState = haveColorBuffer ==
true || haveGeometryBuffer ==
true?&cb:
nullptr;
2881 pipeline.pMultisampleState = &ms;
2882 pipeline.pViewportState = &vp;
2883 pipeline.pDepthStencilState = haveDepthBuffer ==
true?&ds:
nullptr;
2884 pipeline.pStages = shaderStages.data();
2885 pipeline.renderPass = usedRenderPass;
2886 pipeline.pDynamicState = &dsc;
2888 pipelineCacheCreateInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_CACHE_CREATE_INFO;
2889 err = vkCreatePipelineCache(
device, &pipelineCacheCreateInfo,
nullptr, &pipelineCache);
2892 err = vkCreateGraphicsPipelines(
device, pipelineCache, 1, &pipeline,
nullptr, &framebufferPipelines->pipelines[currentContext.pipelineIdx]);
2896 vkDestroyPipelineCache(
device, pipelineCache,
nullptr);
2900 auto& currentContext =
contexts[contextIdx];
2901 if (currentContext.pipelineIdx ==
ID_NONE || currentContext.pipeline == VK_NULL_HANDLE) {
2905 if (pipeline == VK_NULL_HANDLE) {
2912 auto& commandBuffer = currentContext.commandBuffers[currentContext.currentCommandBuffer];
2913 vkCmdBindPipeline(commandBuffer.drawCommand, VK_PIPELINE_BIND_POINT_GRAPHICS, pipeline);
2914 currentContext.pipeline = pipeline;
2924 if (framebufferPipelines ==
nullptr) {
2932 vector<VkDescriptorSetLayoutBinding> layoutBindings1(program->
layoutBindings);
2935 vector<VkPipelineShaderStageCreateInfo> shaderStages(program->
shaders.size());
2938 for (
auto shader: program->
shaders) {
2939 shaderStages[
shaderIdx].sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO;
2940 shaderStages[
shaderIdx].stage = shader->type;
2941 shaderStages[
shaderIdx].module = shader->module;
2944 for (
int i = 0; i <= shader->maxBindings; i++) {
2945 layoutBindings1[i] = {
2946 .binding =
static_cast<uint32_t
>(i),
2947 .descriptorType = VK_DESCRIPTOR_TYPE_STORAGE_BUFFER,
2948 .descriptorCount = 1,
2949 .stageFlags =
static_cast<VkShaderStageFlags
>(shader->type),
2950 .pImmutableSamplers =
nullptr
2954 if (shader->uboBindingIdx != -1) {
2955 layoutBindings1[shader->uboBindingIdx] = {
2956 .binding =
static_cast<uint32_t
>(shader->uboBindingIdx),
2957 .descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER,
2958 .descriptorCount = 1,
2959 .stageFlags =
static_cast<VkShaderStageFlags
>(shader->type),
2960 .pImmutableSamplers =
nullptr
2965 const VkDescriptorSetLayoutCreateInfo descriptorSetlayoutCreateInfo = {
2966 .sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO,
2970 .pBindings = layoutBindings1.data(),
2977 const VkPipelineLayoutCreateInfo pPipelineLayoutCreateInfo = {
2978 .sType = VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO,
2981 .setLayoutCount = 1,
2987 array<VkDescriptorSetLayout, DESC_MAX_UNCACHED> descriptorSetLayouts1;
2990 VkDescriptorSetAllocateInfo descriptorSetAllocateInfo = {
2991 .sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_ALLOCATE_INFO,
2995 .pSetLayouts = descriptorSetLayouts1.data()
2998 err = vkAllocateDescriptorSets(
device, &descriptorSetAllocateInfo, program->
contexts[context.idx].commandBuffers[i].uboDescriptorSets.data());
3004 err = vkCreatePipelineLayout(
device, &pPipelineLayoutCreateInfo,
nullptr, &program->
pipelineLayout);
3008 VkPipelineCacheCreateInfo pipelineCacheCreateInfo = {
3009 .sType = VK_STRUCTURE_TYPE_PIPELINE_CACHE_CREATE_INFO,
3012 .initialDataSize = 0,
3013 .pInitialData =
nullptr
3015 VkPipelineCache pipelineCache = VK_NULL_HANDLE;
3017 err = vkCreatePipelineCache(
device, &pipelineCacheCreateInfo,
nullptr, &pipelineCache);
3021 VkComputePipelineCreateInfo pipeline = {
3022 .sType = VK_STRUCTURE_TYPE_COMPUTE_PIPELINE_CREATE_INFO,
3025 .stage = shaderStages[0],
3027 .basePipelineHandle =
nullptr,
3028 .basePipelineIndex = 0
3032 err = vkCreateComputePipelines(
device, pipelineCache, 1, &pipeline,
nullptr, &framebufferPipelines->pipelines[
ID_NONE]);
3036 vkDestroyPipelineCache(
device, pipelineCache,
nullptr);
3043 auto& currentContext =
contexts[contextIdx];
3044 if (currentContext.pipelineIdx ==
ID_NONE || currentContext.pipeline == VK_NULL_HANDLE) {
3050 vkCmdBindPipeline(currentContext.commandBuffers[currentContext.currentCommandBuffer].drawCommand, VK_PIPELINE_BIND_POINT_COMPUTE, pipeline);
3051 currentContext.pipeline = pipeline;
3057 auto& currentContext =
contexts[contextIdx];
3060 if (currentContext.program !=
nullptr && currentContext.program->id == programId)
return;
3066 currentContext.program =
nullptr;
3069 if (programId ==
ID_NONE)
return;
3072 if (programId < ID_NONE || programId >=
programVector.size()) {
3073 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): program does not exist: " + to_string(programId));
3079 currentContext.program = program;
3084 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
3086 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): could not create program, maximum is " + to_string(
PROGRAMS_MAX));
3090 auto& program = *programPtr;
3091 program.type = type;
3094 for (
auto& programContext: program.contexts) {
3095 programContext.descriptorSets2Idx = 0;
3096 programContext.descriptorSets2.fill(VK_NULL_HANDLE);
3097 for (
auto& programContextCommandBuffer: programContext.commandBuffers) {
3098 programContextCommandBuffer.uboDescriptorSetsIdx = 0;
3099 programContextCommandBuffer.texturesDescriptorSetsIdxUncached = 0;
3100 programContextCommandBuffer.uboDescriptorSets.fill(VK_NULL_HANDLE);
3101 programContextCommandBuffer.texturesDescriptorSetsUncached.fill(VK_NULL_HANDLE);
3105 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): program id: " + to_string(program.id));
3111 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
3112 auto shaderIt =
shaders.find(shaderId);
3113 if (shaderIt ==
shaders.end()) {
3114 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): shader does not exist");
3118 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): program does not exist");
3122 program->shaderIds.push_back(shaderId);
3123 program->shaders.push_back(shaderIt->second);
3128 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(programId));
3130 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): program does not exist");
3141 for (
auto shader: program.shaders) {
3143 if (shader->uboSize > 0) {
3146 auto& uniformBuffer = shader->uniformBuffers[context.idx];
3147 uniformBuffer.size = shader->uboSize;
3148 uniformBuffer.uniformBufferData.resize(shader->uboSize);
3149 for (
auto& uniformBufferBuffer: uniformBuffer.buffers) {
3150 VmaAllocationInfo allocationInfo = {};
3153 VK_BUFFER_USAGE_UNIFORM_BUFFER_BIT,
3154 VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT,
3155 uniformBufferBuffer.buffer,
3156 uniformBufferBuffer.allocation,
3159 VkMemoryPropertyFlags memoryFlags;
3160 vmaGetMemoryTypeProperties(
vmaAllocator, allocationInfo.memoryType, &memoryFlags);
3161 auto memoryMapped = (memoryFlags & VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT) == VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT;
3162 if (memoryMapped ==
true) {
3164 vmaMapMemory(
vmaAllocator, uniformBufferBuffer.allocation, &mmData);
3166 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): Could not create memory mappable uniform buffer");
3175 for (
auto shader: program.shaders) {
3179 VkShaderModuleCreateInfo shaderModuleCreateInfo;
3180 shaderModuleCreateInfo.sType = VK_STRUCTURE_TYPE_SHADER_MODULE_CREATE_INFO;
3181 shaderModuleCreateInfo.pNext =
nullptr;
3182 shaderModuleCreateInfo.codeSize = shader->
spirv.size() *
sizeof(uint32_t);
3183 shaderModuleCreateInfo.pCode = shader->spirv.data();
3184 shaderModuleCreateInfo.flags = 0;
3185 err = vkCreateShaderModule(
device, &shaderModuleCreateInfo,
nullptr, &shader->module);
3186 if (err == VK_SUCCESS) {
3190 string(
"VKRenderer::") +
3191 string(__FUNCTION__) +
3193 to_string(shader->id) +
3202 string(
"VKRenderer::") +
3203 string(__FUNCTION__) +
3205 to_string(shader->id) +
3210 Console::println(shader->source);
3216 shaderLast = shader;
3227 string(
"VKRenderer::") +
3228 string(__FUNCTION__) +
3230 to_string(programId) +
3232 string(
": unknown program: ") +
3233 to_string(program.type)
3243 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + name);
3245 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): program does not exist");
3249 for (
auto& uniformIt: program->uniforms) {
3250 if (uniformIt.second == name) {
3251 return uniformIt.first;
3254 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): uniform not found: '" + name +
"'");
3259 auto& currentContext =
contexts[contextIdx];
3263 for (
auto shader: currentContext.program->shaders) {
3265 if (uniformId < 0 || uniformId >= shader->uniformList.size()) {
3268 string(__FUNCTION__) +
3269 "(): program: uniform id out of uniform list bounds: " +
3270 to_string(currentContext.idx) +
": " +
3271 to_string(currentContext.program->id) +
": " +
3272 to_string(uniformId) +
" / " +
3273 to_string(shader->uniformList.size())
3277 to_string(currentContext.idx) +
": " +
3278 to_string(currentContext.program->id) +
": " +
3279 to_string(uniformId) +
" / " +
3280 currentContext.program->uniforms[uniformId]
3284 auto shaderUniformPtr = uniformId != -1?shader->uniformList[uniformId]:
nullptr;
3285 if (shaderUniformPtr ==
nullptr) {
3289 auto& shaderUniform = *shaderUniformPtr;
3343 auto& uniformBuffer = shader->uniformBuffers[contextIdx];
3344 auto remainingSize = size;
3347 auto dst =
static_cast<uint8_t*
>(&uniformBuffer.uniformBufferData[shaderUniform.position]);
3348 while (remainingSize >= 8) {
3349 *(uint64_t*)dst = *(uint64_t*)src;
3354 while (remainingSize >= 4) {
3355 *(uint32_t*)dst = *(uint32_t*)src;
3362 shaderUniform.textureUnit = *((int32_t*)data);
3365 shaderUniform.textureUnit = *((int32_t*)data);
3383 array<float, 12> _data = {
3461 auto& currentContext =
contexts[contextIdx];
3462 if (currentContext.cullingEnabled ==
true)
return;
3464 currentContext.cullingEnabled =
true;
3465 currentContext.frontFaceIndex = currentContext.frontFace;
3470 auto& currentContext =
contexts[contextIdx];
3471 if (currentContext.cullingEnabled ==
false)
return;
3473 currentContext.cullingEnabled =
false;
3474 currentContext.frontFaceIndex = 0;
3479 auto& currentContext =
contexts[contextIdx];
3480 if (currentContext.frontFace == frontFace)
return;
3482 currentContext.frontFace = frontFace;
3483 currentContext.frontFaceIndex = currentContext.cullingEnabled ==
true?frontFace:0;
3490 cullMode = (VkCullModeFlagBits)cullFace;
3554 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
3560 auto attachmentIdx = 0;
3561 array<VkClearAttachment, 9> attachments;
3564 for (
auto i = 0; i < 8; i++) {
3565 attachments[attachmentIdx].aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
3566 attachments[attachmentIdx].colorAttachment = attachmentIdx;
3572 attachments[attachmentIdx].aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
3573 attachments[attachmentIdx].colorAttachment = attachmentIdx;
3580 attachments[attachmentIdx].aspectMask = VK_IMAGE_ASPECT_DEPTH_BIT;
3581 attachments[attachmentIdx].colorAttachment = 0;
3582 attachments[attachmentIdx].clearValue.depthStencil = { 1.0f, 0 };
3585 VkClearRect clearRect = {
3587 .baseArrayLayer = 0,
3590 vkCmdClearAttachments(
3598 auto currentBufferIdx =
contexts[0].currentCommandBuffer;
3600 if (commandBuffer != VK_NULL_HANDLE) {
3608 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
3611 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): could not allocate texture, maximum is " + to_string(
TEXTURES_MAX));
3615 auto reuseTextureId = -1;
3622 auto& texture = *texturePtr;
3623 texture.id = reuseTextureId != -1?reuseTextureId:
textureIdx++;
3631 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(width) +
"x" + to_string(height));
3635 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): width: " + to_string(width) +
" <= 0, using 1");
3639 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): height: " + to_string(height) +
" <= 0, using 1");
3646 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): could not allocate texture, maximum is " + to_string(
TEXTURES_MAX));
3650 auto reuseTextureId = -1;
3657 auto& texture = *texturePtr;
3658 texture.id = reuseTextureId != -1?reuseTextureId:
textureIdx++;
3659 texture.bindTexture = texturePtr;
3668 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(textureId) +
" / " + to_string(width) +
"x" + to_string(height));
3672 depthBufferTexture.format = VK_FORMAT_D32_SFLOAT;
3673 depthBufferTexture.width = width;
3674 depthBufferTexture.height = height;
3675 depthBufferTexture.cubemapTextureIndex = cubeMapTextureId ==
ID_NONE?0:cubeMapTextureIndex;
3679 depthBufferTexture.cubemapBufferTexture = cubeMapTexture !=
nullptr?cubeMapTexture->cubemapDepthBuffer:
nullptr;
3685 if (cubeMapTexture ==
nullptr) {
3690 .image = depthBufferTexture.image,
3691 .allocation = depthBufferTexture.allocation,
3692 .imageView = depthBufferTexture.view,
3693 .sampler = depthBufferTexture.sampler
3699 const VkImageCreateInfo imageCreateInfo = {
3700 .sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO,
3703 .imageType = VK_IMAGE_TYPE_2D,
3704 .format = depthBufferTexture.format,
3706 .width = depthBufferTexture.width,
3707 .height = depthBufferTexture.height,
3712 .samples = VK_SAMPLE_COUNT_1_BIT,
3713 .tiling = VK_IMAGE_TILING_OPTIMAL,
3714 .usage = VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | VK_IMAGE_USAGE_SAMPLED_BIT | VK_IMAGE_USAGE_TRANSFER_SRC_BIT,
3715 .sharingMode = VK_SHARING_MODE_EXCLUSIVE,
3716 .queueFamilyIndexCount = 0,
3717 .pQueueFamilyIndices = 0,
3718 .initialLayout = VK_IMAGE_LAYOUT_UNDEFINED,
3721 VmaAllocationCreateInfo allocationCreateInfo = {};
3722 allocationCreateInfo.usage = VMA_MEMORY_USAGE_GPU_ONLY;
3724 VmaAllocationInfo allocationInfo = {};
3725 err = vmaCreateImage(
vmaAllocator, &imageCreateInfo, &allocationCreateInfo, &depthBufferTexture.image, &depthBufferTexture.allocation, &allocationInfo);
3730 depthBufferTexture.aspectMask = VK_IMAGE_ASPECT_DEPTH_BIT;
3731 depthBufferTexture.accessTypes = { THSVS_ACCESS_NONE, THSVS_ACCESS_NONE };
3732 depthBufferTexture.svsLayout = THSVS_IMAGE_LAYOUT_OPTIMAL;
3733 depthBufferTexture.vkLayout = VK_IMAGE_LAYOUT_UNDEFINED;
3737 const VkSamplerCreateInfo samplerCreateInfo = {
3738 .sType = VK_STRUCTURE_TYPE_SAMPLER_CREATE_INFO,
3741 .magFilter = VK_FILTER_NEAREST,
3742 .minFilter = VK_FILTER_NEAREST,
3743 .mipmapMode = VK_SAMPLER_MIPMAP_MODE_NEAREST,
3744 .addressModeU = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE,
3745 .addressModeV = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE,
3746 .addressModeW = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE,
3748 .anisotropyEnable = VK_FALSE,
3750 .compareEnable = VK_FALSE,
3751 .compareOp = VK_COMPARE_OP_NEVER,
3754 .borderColor = VK_BORDER_COLOR_FLOAT_OPAQUE_WHITE,
3755 .unnormalizedCoordinates = VK_FALSE,
3757 err = vkCreateSampler(
device, &samplerCreateInfo,
nullptr, &depthBufferTexture.sampler);
3763 VkImageViewCreateInfo viewCreateInfo = {
3764 .sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO,
3767 .image = cubeMapTexture !=
nullptr?cubeMapTexture->cubemapDepthBuffer->image:depthBufferTexture.image,
3768 .viewType = VK_IMAGE_VIEW_TYPE_2D,
3769 .format = depthBufferTexture.format,
3770 .components = VkComponentMapping(),
3771 .subresourceRange = {
3772 .aspectMask = VK_IMAGE_ASPECT_DEPTH_BIT,
3775 .baseArrayLayer = cubeMapTexture !=
nullptr?
static_cast<uint32_t
>(cubeMapTextureIndex -
CUBEMAPTEXTUREINDEX_MIN):0,
3779 err = vkCreateImageView(
device, &viewCreateInfo,
nullptr, &depthBufferTexture.view);
3785 &depthBufferTexture,
3786 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
3787 THSVS_IMAGE_LAYOUT_OPTIMAL,
3796 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(width) +
"x" + to_string(height));
3800 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): width: " + to_string(width) +
" <= 0, using 1");
3804 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): height: " + to_string(height) +
" <= 0, using 1");
3811 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): could not allocate texture, maximum is " + to_string(
TEXTURES_MAX));
3815 auto reuseTextureId = -1;
3822 auto& texture = *texturePtr;
3823 texture.id = reuseTextureId != -1?reuseTextureId:
textureIdx++;
3824 texture.bindTexture = texturePtr;
3833 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(textureId) +
" / " + to_string(width) +
"x" + to_string(height) +
"(" + to_string(cubeMapTextureId) +
" / " + to_string(cubeMapTextureIndex) +
")");
3835 colorBufferTexture.format = format;
3836 colorBufferTexture.width = width;
3837 colorBufferTexture.height = height;
3838 colorBufferTexture.cubemapTextureIndex = cubeMapTextureId ==
ID_NONE?0:cubeMapTextureIndex;
3843 colorBufferTexture.cubemapBufferTexture = cubeMapTexture !=
nullptr?cubeMapTexture->cubemapColorBuffer:
nullptr;
3849 if (cubeMapTexture ==
nullptr) {
3854 .image = colorBufferTexture.image,
3855 .allocation = colorBufferTexture.allocation,
3856 .imageView = colorBufferTexture.view,
3857 .sampler = colorBufferTexture.sampler
3862 const VkImageCreateInfo imageCreateInfo = {
3863 .sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO,
3866 .imageType = VK_IMAGE_TYPE_2D,
3867 .format = colorBufferTexture.format,
3869 .width = colorBufferTexture.width,
3870 .height = colorBufferTexture.height,
3875 .samples = VK_SAMPLE_COUNT_1_BIT,
3876 .tiling = VK_IMAGE_TILING_OPTIMAL,
3877 .usage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT | VK_IMAGE_USAGE_SAMPLED_BIT | VK_IMAGE_USAGE_TRANSFER_SRC_BIT,
3878 .sharingMode = VK_SHARING_MODE_EXCLUSIVE,
3879 .queueFamilyIndexCount = 0,
3880 .pQueueFamilyIndices = 0,
3881 .initialLayout = VK_IMAGE_LAYOUT_UNDEFINED,
3884 VmaAllocationCreateInfo allocationCreateInfo = {};
3885 allocationCreateInfo.usage = VMA_MEMORY_USAGE_GPU_ONLY;
3887 VmaAllocationInfo allocationInfo = {};
3888 err = vmaCreateImage(
vmaAllocator, &imageCreateInfo, &allocationCreateInfo, &colorBufferTexture.image, &colorBufferTexture.allocation, &allocationInfo);
3893 colorBufferTexture.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
3894 colorBufferTexture.accessTypes = { THSVS_ACCESS_NONE, THSVS_ACCESS_NONE };
3895 colorBufferTexture.svsLayout = THSVS_IMAGE_LAYOUT_OPTIMAL;
3896 colorBufferTexture.vkLayout = VK_IMAGE_LAYOUT_UNDEFINED;
3900 const VkSamplerCreateInfo samplerCreateInfo = {
3901 .sType = VK_STRUCTURE_TYPE_SAMPLER_CREATE_INFO,
3904 .magFilter = VK_FILTER_LINEAR,
3905 .minFilter = VK_FILTER_LINEAR,
3906 .mipmapMode = VK_SAMPLER_MIPMAP_MODE_NEAREST,
3907 .addressModeU = VK_SAMPLER_ADDRESS_MODE_REPEAT,
3908 .addressModeV = VK_SAMPLER_ADDRESS_MODE_REPEAT,
3909 .addressModeW = VK_SAMPLER_ADDRESS_MODE_REPEAT,
3911 .anisotropyEnable = VK_FALSE,
3913 .compareEnable = VK_FALSE,
3914 .compareOp = VK_COMPARE_OP_NEVER,
3917 .borderColor = VK_BORDER_COLOR_FLOAT_OPAQUE_WHITE,
3918 .unnormalizedCoordinates = VK_FALSE,
3920 err = vkCreateSampler(
device, &samplerCreateInfo,
nullptr, &colorBufferTexture.sampler);
3926 VkImageViewCreateInfo viewCreateInfo = {
3927 .sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO,
3930 .image = cubeMapTexture !=
nullptr?cubeMapTexture->cubemapColorBuffer->image:colorBufferTexture.image,
3931 .viewType = VK_IMAGE_VIEW_TYPE_2D,
3932 .format = colorBufferTexture.format,
3934 .r = VK_COMPONENT_SWIZZLE_R,
3935 .g = VK_COMPONENT_SWIZZLE_G,
3936 .b = VK_COMPONENT_SWIZZLE_B,
3937 .a = VK_COMPONENT_SWIZZLE_A,
3939 .subresourceRange = {
3940 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
3943 .baseArrayLayer = cubeMapTexture !=
nullptr?
static_cast<uint32_t
>(cubeMapTextureIndex -
CUBEMAPTEXTUREINDEX_MIN):0,
3947 err = vkCreateImageView(
device, &viewCreateInfo,
nullptr, &colorBufferTexture.view);
3953 &colorBufferTexture,
3954 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
3955 THSVS_IMAGE_LAYOUT_OPTIMAL,
3964 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(width) +
"x" + to_string(height));
3968 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): width: " + to_string(width) +
" <= 0, using 1");
3972 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): height: " + to_string(height) +
" <= 0, using 1");
3979 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): could not allocate texture, maximum is " + to_string(
TEXTURES_MAX));
3983 auto reuseTextureId = -1;
3990 auto& texture = *texturePtr;
3991 texture.id = reuseTextureId != -1?reuseTextureId:
textureIdx++;
3992 texture.bindTexture = texturePtr;
4000 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(width) +
"x" + to_string(height));
4004 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): width: " + to_string(width) +
" <= 0, using 1");
4008 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): height: " + to_string(height) +
" <= 0, using 1");
4015 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): could not allocate texture, maximum is " + to_string(
TEXTURES_MAX));
4019 auto reuseTextureId = -1;
4026 auto& texture = *texturePtr;
4027 texture.id = reuseTextureId != -1?reuseTextureId:
textureIdx++;
4028 texture.bindTexture = texturePtr;
4038 "VKRenderer::" +
string(__FUNCTION__) +
"(): " +
4039 textureLeft->
getId() +
" / " +
4040 textureRight->
getId() +
" / " +
4041 textureTop->
getId() +
" / " +
4042 textureBottom->
getId() +
" / " +
4043 textureFront->
getId() +
" / " +
4044 textureBack->
getId()
4049 auto& currentContext =
contexts[contextIdx];
4050 auto& boundTexture = currentContext.boundTextures[currentContext.activeTextureUnit];
4054 if (textureObjectPtr ==
nullptr) {
4055 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): texture not found: " + to_string(boundTexture.id));
4058 auto& texture = *textureObjectPtr;
4061 if (texture.uploaded ==
true) {
4062 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): texture already uploaded: " + to_string(boundTexture.id));
4069 texture.format = VK_FORMAT_R8G8B8A8_UNORM;
4070 texture.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
4071 texture.accessTypes = { THSVS_ACCESS_HOST_PREINITIALIZED, THSVS_ACCESS_NONE };
4072 texture.vkLayout = VK_IMAGE_LAYOUT_PREINITIALIZED;
4075 const VkImageCreateInfo imageCreateInfo = {
4076 .sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO,
4078 .flags = VK_IMAGE_CREATE_CUBE_COMPATIBLE_BIT,
4079 .imageType = VK_IMAGE_TYPE_2D,
4080 .format = texture.format,
4082 .width = texture.width,
4083 .height = texture.height,
4088 .samples = VK_SAMPLE_COUNT_1_BIT,
4089 .tiling = VK_IMAGE_TILING_OPTIMAL,
4090 .usage = VK_IMAGE_USAGE_SAMPLED_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT,
4091 .sharingMode = VK_SHARING_MODE_EXCLUSIVE,
4092 .queueFamilyIndexCount = 0,
4093 .pQueueFamilyIndices = 0,
4094 .initialLayout = VK_IMAGE_LAYOUT_PREINITIALIZED,
4100 VmaAllocationCreateInfo allocationCreateInfo = {};
4101 allocationCreateInfo.usage = VMA_MEMORY_USAGE_GPU_ONLY;
4103 VmaAllocationInfo allocationInfo = {};
4104 err = vmaCreateImage(
vmaAllocator, &imageCreateInfo, &allocationCreateInfo, &texture.image, &texture.allocation, &allocationInfo);
4119 { THSVS_ACCESS_TRANSFER_WRITE, THSVS_ACCESS_NONE },
4120 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
4121 THSVS_IMAGE_LAYOUT_OPTIMAL,
4122 THSVS_IMAGE_LAYOUT_OPTIMAL,
4130 texture.accessTypes =
4132 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
4133 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
4134 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
4135 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
4136 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
4137 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE }
4141 const VkSamplerCreateInfo samplerCreateInfo = {
4142 .sType = VK_STRUCTURE_TYPE_SAMPLER_CREATE_INFO,
4145 .magFilter = VK_FILTER_LINEAR,
4146 .minFilter = VK_FILTER_LINEAR,
4147 .mipmapMode = VK_SAMPLER_MIPMAP_MODE_NEAREST,
4148 .addressModeU = VK_SAMPLER_ADDRESS_MODE_REPEAT,
4149 .addressModeV = VK_SAMPLER_ADDRESS_MODE_REPEAT,
4150 .addressModeW = VK_SAMPLER_ADDRESS_MODE_REPEAT,
4152 .anisotropyEnable = VK_FALSE,
4154 .compareEnable = VK_FALSE,
4155 .compareOp = VK_COMPARE_OP_NEVER,
4158 .borderColor = VK_BORDER_COLOR_FLOAT_OPAQUE_WHITE,
4159 .unnormalizedCoordinates = VK_FALSE,
4161 err = vkCreateSampler(
device, &samplerCreateInfo,
nullptr, &texture.sampler);
4165 VkImageViewCreateInfo viewCreateInfo = {
4166 .sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO,
4169 .image = texture.image,
4170 .viewType = VK_IMAGE_VIEW_TYPE_CUBE,
4171 .format = texture.format,
4173 .r = VK_COMPONENT_SWIZZLE_R,
4174 .g = VK_COMPONENT_SWIZZLE_G,
4175 .b = VK_COMPONENT_SWIZZLE_B,
4176 .a = VK_COMPONENT_SWIZZLE_A,
4178 .subresourceRange = {
4179 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
4182 .baseArrayLayer = 0,
4186 err = vkCreateImageView(
device, &viewCreateInfo,
nullptr, &texture.view);
4190 boundTexture.sampler = texture.sampler;
4191 boundTexture.view = texture.view;
4192 boundTexture.layout = texture.vkLayout;
4195 texture.uploaded =
true;
4196 texture.bindTexture = &texture;
4199 currentContext.uploadedTextureIds.insert(texture.id);
4203 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(width) +
"x" + to_string(height));
4206 auto& currentContext =
contexts[contextIdx];
4210 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): width: " + to_string(width) +
" <= 0, using 1");
4214 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): height: " + to_string(height) +
" <= 0, using 1");
4221 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): could not allocate texture, maximum is " + to_string(
TEXTURES_MAX));
4225 auto reuseTextureId = -1;
4232 auto& texture = *texturePtr;
4233 texture.id = reuseTextureId != -1?reuseTextureId:
textureIdx++;
4236 texture.width = width;
4237 texture.height = height;
4238 texture.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
4239 texture.vkLayout = VK_IMAGE_LAYOUT_UNDEFINED;
4246 texture.cubemapColorBuffer->id = -1;
4249 texture.cubemapColorBuffer->width = width;
4250 texture.cubemapColorBuffer->height = height;
4251 texture.cubemapColorBuffer->aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
4252 texture.cubemapColorBuffer->vkLayout = VK_IMAGE_LAYOUT_UNDEFINED;
4253 const VkImageCreateInfo imageCreateInfo = {
4254 .sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO,
4256 .flags = VK_IMAGE_CREATE_CUBE_COMPATIBLE_BIT,
4257 .imageType = VK_IMAGE_TYPE_2D,
4258 .format = texture.cubemapColorBuffer->format,
4260 .width = texture.cubemapColorBuffer->width,
4261 .height = texture.cubemapColorBuffer->height,
4266 .samples = VK_SAMPLE_COUNT_1_BIT,
4267 .tiling = VK_IMAGE_TILING_OPTIMAL,
4268 .usage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT | VK_IMAGE_USAGE_SAMPLED_BIT,
4269 .sharingMode = VK_SHARING_MODE_EXCLUSIVE,
4270 .queueFamilyIndexCount = 0,
4271 .pQueueFamilyIndices = 0,
4272 .initialLayout = VK_IMAGE_LAYOUT_UNDEFINED
4278 VmaAllocationCreateInfo allocationCreateInfo = {};
4279 allocationCreateInfo.usage = VMA_MEMORY_USAGE_GPU_ONLY;
4281 VmaAllocationInfo allocationInfo = {};
4282 err = vmaCreateImage(
vmaAllocator, &imageCreateInfo, &allocationCreateInfo, &texture.cubemapColorBuffer->image, &texture.cubemapColorBuffer->allocation, &allocationInfo);
4286 const VkSamplerCreateInfo samplerCreateInfo = {
4287 .sType = VK_STRUCTURE_TYPE_SAMPLER_CREATE_INFO,
4290 .magFilter = VK_FILTER_LINEAR,
4291 .minFilter = VK_FILTER_LINEAR,
4292 .mipmapMode = VK_SAMPLER_MIPMAP_MODE_NEAREST,
4293 .addressModeU = VK_SAMPLER_ADDRESS_MODE_REPEAT,
4294 .addressModeV = VK_SAMPLER_ADDRESS_MODE_REPEAT,
4295 .addressModeW = VK_SAMPLER_ADDRESS_MODE_REPEAT,
4297 .anisotropyEnable = VK_FALSE,
4299 .compareEnable = VK_FALSE,
4300 .compareOp = VK_COMPARE_OP_NEVER,
4303 .borderColor = VK_BORDER_COLOR_FLOAT_OPAQUE_WHITE,
4304 .unnormalizedCoordinates = VK_FALSE,
4306 err = vkCreateSampler(
device, &samplerCreateInfo,
nullptr, &texture.sampler);
4310 VkImageViewCreateInfo viewCreateInfo = {
4311 .sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO,
4314 .image = texture.cubemapColorBuffer->image,
4315 .viewType = VK_IMAGE_VIEW_TYPE_CUBE,
4316 .format = texture.cubemapColorBuffer->format,
4318 .r = VK_COMPONENT_SWIZZLE_R,
4319 .g = VK_COMPONENT_SWIZZLE_G,
4320 .b = VK_COMPONENT_SWIZZLE_B,
4321 .a = VK_COMPONENT_SWIZZLE_A,
4323 .subresourceRange = {
4324 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
4327 .baseArrayLayer = 0,
4331 err = vkCreateImageView(
device, &viewCreateInfo,
nullptr, &texture.view);
4338 texture.cubemapColorBuffer,
4339 { THSVS_ACCESS_NONE, THSVS_ACCESS_NONE },
4340 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
4341 THSVS_IMAGE_LAYOUT_OPTIMAL,
4342 THSVS_IMAGE_LAYOUT_OPTIMAL,
4354 texture.cubemapDepthBuffer->id = -1;
4355 texture.cubemapDepthBuffer->format = VK_FORMAT_D32_SFLOAT;
4356 texture.cubemapDepthBuffer->width = width;
4357 texture.cubemapDepthBuffer->height = height;
4359 texture.cubemapDepthBuffer->aspectMask = VK_IMAGE_ASPECT_DEPTH_BIT;
4360 const VkImageCreateInfo imageCreateInfo = {
4361 .sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO,
4363 .flags = VK_IMAGE_CREATE_CUBE_COMPATIBLE_BIT,
4364 .imageType = VK_IMAGE_TYPE_2D,
4365 .format = texture.cubemapDepthBuffer->format,
4367 .width = texture.cubemapDepthBuffer->width,
4368 .height = texture.cubemapDepthBuffer->height,
4373 .samples = VK_SAMPLE_COUNT_1_BIT,
4374 .tiling = VK_IMAGE_TILING_OPTIMAL,
4375 .usage = VK_IMAGE_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | VK_IMAGE_USAGE_SAMPLED_BIT,
4376 .sharingMode = VK_SHARING_MODE_EXCLUSIVE,
4377 .queueFamilyIndexCount = 0,
4378 .pQueueFamilyIndices = 0,
4379 .initialLayout = VK_IMAGE_LAYOUT_UNDEFINED,
4385 VmaAllocationCreateInfo allocationCreateInfo = {};
4386 allocationCreateInfo.usage = VMA_MEMORY_USAGE_GPU_ONLY;
4388 VmaAllocationInfo allocation_info = {};
4389 err = vmaCreateImage(
vmaAllocator, &imageCreateInfo, &allocationCreateInfo, &texture.cubemapDepthBuffer->image, &texture.cubemapDepthBuffer->allocation, &allocation_info);
4395 texture.cubemapDepthBuffer,
4396 { THSVS_ACCESS_NONE, THSVS_ACCESS_NONE },
4397 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
4398 THSVS_IMAGE_LAYOUT_OPTIMAL,
4399 THSVS_IMAGE_LAYOUT_OPTIMAL,
4410 texture.vkLayout = texture.cubemapColorBuffer->vkLayout;
4422 auto& currentContext =
contexts[contextIdx];
4423 auto& boundTexture = currentContext.boundTextures[currentContext.activeTextureUnit];
4427 if (textureObjectPtr ==
nullptr) {
4429 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): texture not found: " + to_string(boundTexture.id));
4432 auto& textureType = *textureObjectPtr;
4435 if (textureType.uploaded ==
true) {
4436 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): texture already uploaded: " + to_string(boundTexture.id));
4442 uint32_t mipLevels = 1;
4446 textureType.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
4449 const VkFormat textureFormat = texture->
getDepth() == 32?VK_FORMAT_R8G8B8A8_UNORM:VK_FORMAT_R8G8B8A8_UNORM;
4450 VkFormatProperties textureFormatProperties;
4454 vkGetPhysicalDeviceFormatProperties(
physicalDevice, textureFormat, &textureFormatProperties);
4455 if ((textureFormatProperties.optimalTilingFeatures & VK_FORMAT_FEATURE_SAMPLED_IMAGE_BIT) == VK_FORMAT_FEATURE_SAMPLED_IMAGE_BIT) {
4461 stagingTexture.
aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
4470 VK_IMAGE_TILING_LINEAR,
4471 VK_IMAGE_USAGE_TRANSFER_SRC_BIT,
4472 VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT,
4474 { THSVS_ACCESS_TRANSFER_READ, THSVS_ACCESS_NONE },
4475 THSVS_IMAGE_LAYOUT_OPTIMAL
4480 VK_IMAGE_TILING_OPTIMAL,
4481 VK_IMAGE_USAGE_TRANSFER_DST_BIT | VK_IMAGE_USAGE_SAMPLED_BIT,
4482 VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT,
4484 { THSVS_ACCESS_TRANSFER_WRITE, THSVS_ACCESS_NONE },
4485 THSVS_IMAGE_LAYOUT_OPTIMAL,
4490 VkImageCopy imageCopy = {
4492 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
4494 .baseArrayLayer = 0,
4503 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
4505 .baseArrayLayer = 0,
4514 .width = textureType.width,
4515 .height = textureType.height,
4521 currentContext.setupCommandInUse,
4522 stagingTexture.
image,
4523 VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
4525 VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
4534 for (uint32_t i = 1; i < mipLevels; i++) {
4535 const VkImageBlit imageBlit = {
4537 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
4539 .baseArrayLayer = 0,
4555 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
4557 .baseArrayLayer = 0,
4567 .x = int32_t(textureWidth >> i),
4568 .y = int32_t(textureHeight >> i),
4575 currentContext.setupCommandInUse,
4576 stagingTexture.
image,
4577 VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
4579 VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
4591 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
4592 THSVS_IMAGE_LAYOUT_OPTIMAL,
4603 .image = stagingTexture.
image,
4605 .imageView = VK_NULL_HANDLE,
4606 .sampler = VK_NULL_HANDLE
4611 if ((textureFormatProperties.linearTilingFeatures & VK_FORMAT_FEATURE_SAMPLED_IMAGE_BIT) == VK_FORMAT_FEATURE_SAMPLED_IMAGE_BIT) {
4617 VK_IMAGE_TILING_LINEAR,
4618 VK_IMAGE_USAGE_SAMPLED_BIT,
4619 VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT,
4621 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
4622 THSVS_IMAGE_LAYOUT_OPTIMAL
4626 const VkSamplerCreateInfo samplerCreateInfo = {
4627 .sType = VK_STRUCTURE_TYPE_SAMPLER_CREATE_INFO,
4630 .magFilter = VK_FILTER_LINEAR,
4631 .minFilter = VK_FILTER_LINEAR,
4632 .mipmapMode = VK_SAMPLER_MIPMAP_MODE_LINEAR,
4633 .addressModeU = texture->
isRepeat() ==
true?VK_SAMPLER_ADDRESS_MODE_REPEAT:(texture->
getClampMode() == Texture::CLAMPMODE_EDGE?VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE:VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_BORDER),
4634 .addressModeV = texture->
isRepeat() ==
true?VK_SAMPLER_ADDRESS_MODE_REPEAT:(texture->
getClampMode() == Texture::CLAMPMODE_EDGE?VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE:VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_BORDER),
4635 .addressModeW = texture->
isRepeat() ==
true?VK_SAMPLER_ADDRESS_MODE_REPEAT:(texture->
getClampMode() == Texture::CLAMPMODE_EDGE?VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE:VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_BORDER),
4637 .anisotropyEnable = VK_FALSE,
4639 .compareEnable = VK_FALSE,
4640 .compareOp = VK_COMPARE_OP_NEVER,
4642 .maxLod = texture->
isUseMipMap() ==
true?
static_cast<float>(mipLevels):0.0f,
4643 .borderColor = VK_BORDER_COLOR_FLOAT_TRANSPARENT_BLACK,
4644 .unnormalizedCoordinates = VK_FALSE,
4646 VkImageViewCreateInfo viewCreateInfo = {
4647 .sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO,
4650 .image = textureType.image,
4651 .viewType = VK_IMAGE_VIEW_TYPE_2D,
4652 .format = textureFormat,
4654 .r = VK_COMPONENT_SWIZZLE_R,
4655 .g = VK_COMPONENT_SWIZZLE_G,
4656 .b = VK_COMPONENT_SWIZZLE_B,
4657 .a = VK_COMPONENT_SWIZZLE_A,
4659 .subresourceRange = {
4660 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
4662 .levelCount = mipLevels,
4663 .baseArrayLayer = 0,
4669 err = vkCreateSampler(
device, &samplerCreateInfo,
nullptr, &textureType.sampler);
4673 err = vkCreateImageView(
device, &viewCreateInfo,
nullptr, &textureType.view);
4677 boundTexture.sampler = textureType.sampler;
4678 boundTexture.view = textureType.view;
4679 boundTexture.layout = textureType.vkLayout;
4682 textureType.uploaded =
true;
4683 textureType.bindTexture = textureObjectPtr;
4686 currentContext.uploadedTextureIds.insert(textureType.id);
4694 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + texture->
getId());
4697 auto& currentContext =
contexts[contextIdx];
4698 auto& cubemapTextureTypeRef = *cubemapTextureType;
4701 const VkFormat textureFormat = texture->
getDepth() == 32?VK_FORMAT_R8G8B8A8_UNORM:VK_FORMAT_R8G8B8A8_UNORM;
4702 VkFormatProperties textureFormatProperties;
4704 vkGetPhysicalDeviceFormatProperties(
physicalDevice, textureFormat, &textureFormatProperties);
4705 if ((textureFormatProperties.optimalTilingFeatures & VK_FORMAT_FEATURE_SAMPLED_IMAGE_BIT) == VK_FORMAT_FEATURE_SAMPLED_IMAGE_BIT) {
4711 staging_texture.
aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
4717 VK_IMAGE_TILING_LINEAR,
4718 VK_IMAGE_USAGE_TRANSFER_SRC_BIT,
4719 VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT,
4721 { THSVS_ACCESS_TRANSFER_READ, THSVS_ACCESS_NONE },
4722 THSVS_IMAGE_LAYOUT_OPTIMAL
4728 &cubemapTextureTypeRef,
4729 { THSVS_ACCESS_HOST_PREINITIALIZED, THSVS_ACCESS_NONE },
4730 { THSVS_ACCESS_TRANSFER_WRITE, THSVS_ACCESS_NONE },
4731 THSVS_IMAGE_LAYOUT_OPTIMAL,
4732 THSVS_IMAGE_LAYOUT_OPTIMAL,
4742 VkImageCopy imageCopy = {
4744 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
4746 .baseArrayLayer = 0,
4755 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
4757 .baseArrayLayer = baseArrayLayer,
4766 .width = cubemapTextureTypeRef.width,
4767 .height = cubemapTextureTypeRef.height,
4775 currentContext.setupCommandInUse,
4776 staging_texture.
image,
4777 VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
4778 cubemapTextureTypeRef.image,
4779 VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
4790 .image = staging_texture.
image,
4792 .imageView = VK_NULL_HANDLE,
4793 .sampler = VK_NULL_HANDLE
4798 if ((textureFormatProperties.linearTilingFeatures & VK_FORMAT_FEATURE_SAMPLED_IMAGE_BIT) == VK_FORMAT_FEATURE_SAMPLED_IMAGE_BIT) {
4808 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(textureId) +
" / " + to_string(width) +
"x" + to_string(height));
4817 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): width: " + to_string(width) +
" <= 0, using 1");
4821 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): height: " + to_string(height) +
" <= 0, using 1");
4827 if (texturePtr ==
nullptr) {
4828 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): texture not found: " + to_string(textureId));
4833 auto& texture = *texturePtr;
4834 if (texture.width == width && texture.height == height)
return;
4849 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(textureId) +
" / " + to_string(width) +
"x" + to_string(height));
4858 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): width: " + to_string(width) +
" <= 0, using 1");
4862 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): height: " + to_string(height) +
" <= 0, using 1");
4868 if (texturePtr ==
nullptr) {
4869 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): texture not found: " + to_string(textureId));
4874 auto& texture = *texturePtr;
4875 if (texture.width == width && texture.height == height)
return;
4889 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(textureId) +
" / " + to_string(width) +
"x" + to_string(height));
4898 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): width: " + to_string(width) +
" <= 0, using 1");
4902 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): height: " + to_string(height) +
" <= 0, using 1");
4908 if (texturePtr ==
nullptr) {
4909 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): texture not found: " + to_string(textureId));
4914 auto& texture = *texturePtr;
4915 if (texture.width == width && texture.height == height)
return;
4929 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(textureId) +
" / " + to_string(width) +
"x" + to_string(height));
4938 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): width: " + to_string(width) +
" <= 0, using 1");
4942 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): height: " + to_string(height) +
" <= 0, using 1");
4948 if (texturePtr ==
nullptr) {
4949 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): texture not found: " + to_string(textureId));
4954 auto& texture = *texturePtr;
4955 if (texture.width == width && texture.height == height)
return;
4976 auto& currentContext =
contexts[contextIdx];
4977 auto& boundTexture = currentContext.boundTextures[currentContext.activeTextureUnit];
4983 boundTexture.id = textureId;
4984 if (textureObject !=
nullptr) {
4985 boundTexture.sampler = textureObject->sampler;
4986 boundTexture.view = textureObject->view;
4987 boundTexture.layout = textureObject->vkLayout;
4989 boundTexture.sampler = VK_NULL_HANDLE;
4990 boundTexture.view = VK_NULL_HANDLE;
4991 boundTexture.layout = VK_IMAGE_LAYOUT_UNDEFINED;
5007 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(frameBufferId));
5009 if (frameBuffer ==
nullptr) {
5010 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): frame buffer not found: " + to_string(frameBufferId));
5013 auto& frameBufferStruct = *frameBuffer;
5020 if (depthBufferTexture ==
nullptr) {
5021 if (frameBufferStruct.depthTextureId !=
ID_NONE) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): color buffer: depth buffer texture not found: " + to_string(frameBufferStruct.depthTextureId));
5023 if (colorBufferTexture ==
nullptr) {
5024 if (frameBufferStruct.colorTextureId !=
ID_NONE) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): color buffer: color buffer texture not found: " + to_string(frameBufferStruct.colorTextureId));
5028 if (depthBufferTexture !=
nullptr) {
5033 if (colorBufferTexture !=
nullptr) {
5040 if (depthBufferTexture !=
nullptr && colorBufferTexture !=
nullptr &&
5041 (depthBufferTexture->
width != colorBufferTexture->
width ||
5042 depthBufferTexture->
height != colorBufferTexture->
height)) {
5043 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): color buffer: attachments with different dimension found: Not creating!");
5052 if (frameBufferStruct.renderPass != VK_NULL_HANDLE) vkDestroyRenderPass(
device, frameBufferStruct.renderPass,
nullptr);
5054 auto attachmentIdx = 0;
5055 array<VkAttachmentDescription, 2> attachments;
5056 if (colorBufferTexture !=
nullptr) {
5057 attachments[attachmentIdx++] = {
5059 .format = colorBufferTexture->
format,
5060 .samples = VK_SAMPLE_COUNT_1_BIT,
5061 .loadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE,
5062 .storeOp = VK_ATTACHMENT_STORE_OP_STORE,
5063 .stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE,
5064 .stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE,
5065 .initialLayout = colorBufferTexture->
vkLayout == VK_IMAGE_LAYOUT_GENERAL?VK_IMAGE_LAYOUT_GENERAL:VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL,
5066 .finalLayout = colorBufferTexture->
vkLayout == VK_IMAGE_LAYOUT_GENERAL?VK_IMAGE_LAYOUT_GENERAL:VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL,
5069 if (depthBufferTexture !=
nullptr) {
5070 attachments[attachmentIdx++] = {
5072 .format = depthBufferTexture->
format,
5073 .samples = VK_SAMPLE_COUNT_1_BIT,
5074 .loadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE,
5075 .storeOp = VK_ATTACHMENT_STORE_OP_DONT_CARE,
5076 .stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE,
5077 .stencilStoreOp = VK_ATTACHMENT_STORE_OP_STORE,
5078 .initialLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL,
5079 .finalLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL
5082 const VkAttachmentReference colorReference = {
5084 .layout = colorBufferTexture !=
nullptr?(colorBufferTexture->
vkLayout == VK_IMAGE_LAYOUT_GENERAL?VK_IMAGE_LAYOUT_GENERAL:VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL):VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL,
5086 const VkAttachmentReference depthReference = {
5087 .attachment =
static_cast<uint32_t
>(colorBufferTexture !=
nullptr?1:0),
5088 .layout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL,
5090 const VkSubpassDescription subpassDescription = {
5092 .pipelineBindPoint = VK_PIPELINE_BIND_POINT_GRAPHICS,
5093 .inputAttachmentCount = 0,
5094 .pInputAttachments =
nullptr,
5095 .colorAttachmentCount =
static_cast<uint32_t
>(colorBufferTexture !=
nullptr?1:0),
5096 .pColorAttachments = colorBufferTexture !=
nullptr?&colorReference:
nullptr,
5097 .pResolveAttachments =
nullptr,
5098 .pDepthStencilAttachment = depthBufferTexture !=
nullptr?&depthReference:
nullptr,
5099 .preserveAttachmentCount = 0,
5100 .pPreserveAttachments =
nullptr
5102 const VkRenderPassCreateInfo renderPassCreateInfo = {
5103 .sType = VK_STRUCTURE_TYPE_RENDER_PASS_CREATE_INFO,
5106 .attachmentCount =
static_cast<uint32_t
>(attachmentIdx),
5107 .pAttachments = attachments.data(),
5109 .pSubpasses = &subpassDescription,
5110 .dependencyCount = 0,
5111 .pDependencies =
nullptr
5113 err = vkCreateRenderPass(
device, &renderPassCreateInfo,
nullptr, &frameBufferStruct.renderPass);
5119 if (frameBufferStruct.frameBuffer != VK_NULL_HANDLE) vkDestroyFramebuffer(
device, frameBufferStruct.frameBuffer,
nullptr);
5120 auto attachmentIdx = 0;
5121 array<VkImageView, 2> attachments;
5122 if (colorBufferTexture !=
nullptr) attachments[attachmentIdx++] = colorBufferTexture->
view;
5123 if (depthBufferTexture !=
nullptr) attachments[attachmentIdx++] = depthBufferTexture->
view;
5124 const VkFramebufferCreateInfo fb_info = {
5125 .sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO,
5128 .renderPass = frameBufferStruct.renderPass,
5129 .attachmentCount =
static_cast<uint32_t
>(attachmentIdx),
5130 .pAttachments = attachments.data(),
5131 .width = colorBufferTexture !=
nullptr?colorBufferTexture->
width:depthBufferTexture->
width,
5132 .height = colorBufferTexture !=
nullptr?colorBufferTexture->
height:depthBufferTexture->
height,
5135 err = vkCreateFramebuffer(
device, &fb_info,
nullptr, &frameBufferStruct.frameBuffer);
5141 auto geometryBufferTexture1 =
getTextureInternal(frameBufferStruct.gbufferGeometryBufferTextureId1);
5142 auto geometryBufferTexture2 =
getTextureInternal(frameBufferStruct.gbufferGeometryBufferTextureId2);
5143 auto geometryBufferTexture3 =
getTextureInternal(frameBufferStruct.gbufferGeometryBufferTextureId3);
5144 auto colorBufferTexture1 =
getTextureInternal(frameBufferStruct.gbufferColorBufferTextureId1);
5145 auto colorBufferTexture2 =
getTextureInternal(frameBufferStruct.gbufferColorBufferTextureId2);
5146 auto colorBufferTexture3 =
getTextureInternal(frameBufferStruct.gbufferColorBufferTextureId3);
5147 auto colorBufferTexture4 =
getTextureInternal(frameBufferStruct.gbufferColorBufferTextureId4);
5148 auto colorBufferTexture5 =
getTextureInternal(frameBufferStruct.gbufferColorBufferTextureId5);
5150 depthBufferTexture->frameBufferBindImageLayoutChange.valid =
false;
5151 depthBufferTexture->frameBufferUnbindImageLayoutChange.valid =
false;
5152 geometryBufferTexture1->frameBufferBindImageLayoutChange.valid =
false;
5153 geometryBufferTexture1->frameBufferUnbindImageLayoutChange.valid =
false;
5154 geometryBufferTexture2->frameBufferBindImageLayoutChange.valid =
false;
5155 geometryBufferTexture2->frameBufferUnbindImageLayoutChange.valid =
false;
5156 geometryBufferTexture3->frameBufferBindImageLayoutChange.valid =
false;
5157 geometryBufferTexture3->frameBufferUnbindImageLayoutChange.valid =
false;
5158 colorBufferTexture1->frameBufferBindImageLayoutChange.valid =
false;
5159 colorBufferTexture1->frameBufferUnbindImageLayoutChange.valid =
false;
5160 colorBufferTexture2->frameBufferBindImageLayoutChange.valid =
false;
5161 colorBufferTexture2->frameBufferUnbindImageLayoutChange.valid =
false;
5162 colorBufferTexture3->frameBufferBindImageLayoutChange.valid =
false;
5163 colorBufferTexture3->frameBufferUnbindImageLayoutChange.valid =
false;
5164 colorBufferTexture4->frameBufferBindImageLayoutChange.valid =
false;
5165 colorBufferTexture4->frameBufferUnbindImageLayoutChange.valid =
false;
5166 colorBufferTexture5->frameBufferBindImageLayoutChange.valid =
false;
5167 colorBufferTexture5->frameBufferUnbindImageLayoutChange.valid =
false;
5170 depthBufferTexture->frameBufferObjectId = frameBufferStruct.id;
5171 geometryBufferTexture1->frameBufferObjectId = frameBufferStruct.id;
5172 geometryBufferTexture2->frameBufferObjectId = frameBufferStruct.id;
5173 geometryBufferTexture3->frameBufferObjectId = frameBufferStruct.id;
5174 colorBufferTexture1->frameBufferObjectId = frameBufferStruct.id;
5175 colorBufferTexture2->frameBufferObjectId = frameBufferStruct.id;
5176 colorBufferTexture3->frameBufferObjectId = frameBufferStruct.id;
5177 colorBufferTexture4->frameBufferObjectId = frameBufferStruct.id;
5178 colorBufferTexture5->frameBufferObjectId = frameBufferStruct.id;
5181 if (depthBufferTexture->width == 0 || depthBufferTexture->height == 0 ||
5182 depthBufferTexture->width != geometryBufferTexture1->width || depthBufferTexture->height != geometryBufferTexture1->height ||
5183 depthBufferTexture->width != geometryBufferTexture2->width || depthBufferTexture->height != geometryBufferTexture2->height ||
5184 depthBufferTexture->width != geometryBufferTexture3->width || depthBufferTexture->height != geometryBufferTexture3->height ||
5185 depthBufferTexture->width != colorBufferTexture1->width || depthBufferTexture->height != colorBufferTexture1->height ||
5186 depthBufferTexture->width != colorBufferTexture2->width || depthBufferTexture->height != colorBufferTexture2->height ||
5187 depthBufferTexture->width != colorBufferTexture3->width || depthBufferTexture->height != colorBufferTexture3->height ||
5188 depthBufferTexture->width != colorBufferTexture4->width || depthBufferTexture->height != colorBufferTexture4->height ||
5189 depthBufferTexture->width != colorBufferTexture5->width || depthBufferTexture->height != colorBufferTexture5->height) {
5190 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): geometry buffer: attachments with different dimension found: Not creating!");
5195 array<texture_type*, 8> colorBufferTextures = {
5196 geometryBufferTexture1,
5197 geometryBufferTexture2,
5198 geometryBufferTexture3,
5199 colorBufferTexture1,
5200 colorBufferTexture2,
5201 colorBufferTexture3,
5202 colorBufferTexture4,
5211 if (frameBufferStruct.renderPass != VK_NULL_HANDLE) vkDestroyRenderPass(
device, frameBufferStruct.renderPass,
nullptr);
5214 auto attachmentIdx = 0;
5215 array<VkAttachmentDescription, 9> attachments;
5216 for (
auto colorBufferTexture: colorBufferTextures) {
5217 attachments[attachmentIdx++] = {
5219 .format = colorBufferTexture->format,
5220 .samples = VK_SAMPLE_COUNT_1_BIT,
5221 .loadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE,
5222 .storeOp = VK_ATTACHMENT_STORE_OP_STORE,
5223 .stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE,
5224 .stencilStoreOp = VK_ATTACHMENT_STORE_OP_DONT_CARE,
5225 .initialLayout = colorBufferTexture->vkLayout == VK_IMAGE_LAYOUT_GENERAL?VK_IMAGE_LAYOUT_GENERAL:VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL,
5226 .finalLayout = colorBufferTexture->vkLayout == VK_IMAGE_LAYOUT_GENERAL?VK_IMAGE_LAYOUT_GENERAL:VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL,
5229 attachments[attachmentIdx++] = {
5231 .format = depthBufferTexture->format,
5232 .samples = VK_SAMPLE_COUNT_1_BIT,
5233 .loadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE,
5234 .storeOp = VK_ATTACHMENT_STORE_OP_DONT_CARE,
5235 .stencilLoadOp = VK_ATTACHMENT_LOAD_OP_DONT_CARE,
5236 .stencilStoreOp = VK_ATTACHMENT_STORE_OP_STORE,
5237 .initialLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL,
5238 .finalLayout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL
5240 array<VkAttachmentReference, 8> colorReferences;
5243 for (
auto colorBufferTexture: colorBufferTextures) {
5244 colorReferences[i] = {
5245 .attachment =
static_cast<uint32_t
>(i),
5246 .layout = colorBufferTexture->vkLayout == VK_IMAGE_LAYOUT_GENERAL?VK_IMAGE_LAYOUT_GENERAL:VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL
5251 const VkAttachmentReference depthReference = {
5253 .layout = VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL,
5255 const VkSubpassDescription subpassDescription = {
5257 .pipelineBindPoint = VK_PIPELINE_BIND_POINT_GRAPHICS,
5258 .inputAttachmentCount = 0,
5259 .pInputAttachments =
nullptr,
5260 .colorAttachmentCount = 8,
5261 .pColorAttachments = colorReferences.data(),
5262 .pResolveAttachments =
nullptr,
5263 .pDepthStencilAttachment = &depthReference,
5264 .preserveAttachmentCount = 0,
5265 .pPreserveAttachments =
nullptr
5267 const VkRenderPassCreateInfo renderPassCreateInfo = {
5268 .sType = VK_STRUCTURE_TYPE_RENDER_PASS_CREATE_INFO,
5271 .attachmentCount =
static_cast<uint32_t
>(attachmentIdx),
5272 .pAttachments = attachments.data(),
5274 .pSubpasses = &subpassDescription,
5275 .dependencyCount = 0,
5276 .pDependencies =
nullptr
5278 err = vkCreateRenderPass(
device, &renderPassCreateInfo,
nullptr, &frameBufferStruct.renderPass);
5284 if (frameBufferStruct.frameBuffer != VK_NULL_HANDLE) vkDestroyFramebuffer(
device, frameBufferStruct.frameBuffer,
nullptr);
5285 auto attachmentIdx = 0;
5286 array<VkImageView, 9> attachments;
5287 for (
auto colorBufferTexture: colorBufferTextures) {
5288 attachments[attachmentIdx++] = colorBufferTexture->view;
5290 attachments[attachmentIdx++] = depthBufferTexture->view;
5291 const VkFramebufferCreateInfo frameBufferCreateInfo = {
5292 .sType = VK_STRUCTURE_TYPE_FRAMEBUFFER_CREATE_INFO,
5295 .renderPass = frameBufferStruct.renderPass,
5296 .attachmentCount =
static_cast<uint32_t
>(attachmentIdx),
5297 .pAttachments = attachments.data(),
5298 .width = depthBufferTexture->width,
5299 .height = depthBufferTexture->height,
5302 err = vkCreateFramebuffer(
device, &frameBufferCreateInfo,
nullptr, &frameBufferStruct.frameBuffer);
5310 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(depthBufferTextureId) +
", " + to_string(colorBufferTextureId) +
" / " + to_string(cubeMapTextureId) +
" / " + to_string(cubeMapTextureIndex));
5313 auto reuseIndex = -1;
5323 auto& frameBuffer = *frameBufferPtr;
5324 frameBuffer.id = reuseIndex != -1?reuseIndex:
framebuffers.size();
5326 frameBuffer.depthTextureId = depthBufferTextureId;
5327 frameBuffer.colorTextureId = colorBufferTextureId;
5328 frameBuffer.cubemapTextureId = cubeMapTextureId;
5329 frameBuffer.cubemapTextureIndex = cubeMapTextureIndex;
5330 if (cubeMapTextureId !=
ID_NONE) {
5332 if (cubeMapTexture ==
nullptr) {
5333 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): missing cube map texture with id: " + to_string(cubeMapTextureId));
5335 cubeMapTexture->bindTexture = cubeMapTexture;
5340 if (reuseIndex != -1) {
5348 return frameBuffer.id;
5352 int32_t depthBufferTextureId,
5353 int32_t geometryBufferTextureId1,
5354 int32_t geometryBufferTextureId2,
5355 int32_t geometryBufferTextureId3,
5356 int32_t colorBufferTextureId1,
5357 int32_t colorBufferTextureId2,
5358 int32_t colorBufferTextureId3,
5359 int32_t colorBufferTextureId4,
5360 int32_t colorBufferTextureId5
5364 "VKRenderer::" +
string(__FUNCTION__) +
"(): " +
5365 to_string(depthBufferTextureId) +
", " +
5366 to_string(geometryBufferTextureId1) +
", " +
5367 to_string(geometryBufferTextureId2) +
", " +
5368 to_string(geometryBufferTextureId3) +
", " +
5369 to_string(colorBufferTextureId1) +
", " +
5370 to_string(colorBufferTextureId2) +
", " +
5371 to_string(colorBufferTextureId3) +
", " +
5372 to_string(colorBufferTextureId4) +
", " +
5373 to_string(colorBufferTextureId5)
5378 auto reuseIndex = -1;
5388 auto& frameBuffer = *frameBufferPtr;
5389 frameBuffer.id = reuseIndex != -1?reuseIndex:
framebuffers.size();
5391 frameBuffer.depthTextureId = depthBufferTextureId;
5392 frameBuffer.gbufferGeometryBufferTextureId1 = geometryBufferTextureId1;
5393 frameBuffer.gbufferGeometryBufferTextureId2 = geometryBufferTextureId2;
5394 frameBuffer.gbufferGeometryBufferTextureId3 = geometryBufferTextureId3;
5395 frameBuffer.gbufferColorBufferTextureId1 = colorBufferTextureId1;
5396 frameBuffer.gbufferColorBufferTextureId2 = colorBufferTextureId2;
5397 frameBuffer.gbufferColorBufferTextureId3 = colorBufferTextureId3;
5398 frameBuffer.gbufferColorBufferTextureId4 = colorBufferTextureId4;
5399 frameBuffer.gbufferColorBufferTextureId5 = colorBufferTextureId5;
5402 if (reuseIndex != -1) {
5410 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): new geometry frame buffer: " + to_string(frameBuffer.id));
5411 return frameBuffer.id;
5417 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(
boundFrameBufferId) +
" --> " + to_string(frameBufferId));
5431 if (frameBuffer ==
nullptr) {
5432 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): framebuffer not found: " + to_string(
boundFrameBufferId));
5435 auto depthBufferTextureId = frameBuffer->depthTextureId;
5436 if (depthBufferTextureId !=
ID_NONE) {
5437 auto& depthBufferTexture = *
textures[depthBufferTextureId];
5438 if (depthBufferTexture.frameBufferUnbindImageLayoutChange.valid ==
false) {
5440 depthBufferTexture.frameBufferUnbindImageLayoutChange,
5441 &depthBufferTexture,
5442 { THSVS_ACCESS_DEPTH_STENCIL_ATTACHMENT_READ, THSVS_ACCESS_DEPTH_STENCIL_ATTACHMENT_WRITE },
5443 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
5444 THSVS_IMAGE_LAYOUT_OPTIMAL,
5445 THSVS_IMAGE_LAYOUT_OPTIMAL,
5451 applyImageLayoutChange(0, depthBufferTexture.frameBufferUnbindImageLayoutChange, &depthBufferTexture,
false);
5453 auto colorBufferTextureId = frameBuffer->colorTextureId;
5454 if (colorBufferTextureId !=
ID_NONE) {
5455 auto& colorBufferTexture = *
textures[colorBufferTextureId];
5456 if (colorBufferTexture.frameBufferUnbindImageLayoutChange.valid ==
false) {
5458 colorBufferTexture.frameBufferUnbindImageLayoutChange,
5459 &colorBufferTexture,
5460 { THSVS_ACCESS_COLOR_ATTACHMENT_READ, THSVS_ACCESS_COLOR_ATTACHMENT_WRITE},
5461 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
5462 THSVS_IMAGE_LAYOUT_OPTIMAL,
5463 THSVS_IMAGE_LAYOUT_OPTIMAL,
5469 applyImageLayoutChange(0, colorBufferTexture.frameBufferUnbindImageLayoutChange, &colorBufferTexture,
false);
5471 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(frameBufferId) +
": color buffer: unbinding: " + to_string(colorBufferTextureId) +
" / " + to_string(depthBufferTextureId));
5474 auto depthBufferTextureId = frameBuffer->depthTextureId;
5475 auto colorBufferTextureId1 = frameBuffer->gbufferColorBufferTextureId1;
5476 auto colorBufferTextureId2 = frameBuffer->gbufferColorBufferTextureId2;
5477 auto colorBufferTextureId3 = frameBuffer->gbufferColorBufferTextureId3;
5478 auto colorBufferTextureId4 = frameBuffer->gbufferColorBufferTextureId4;
5479 auto colorBufferTextureId5 = frameBuffer->gbufferColorBufferTextureId5;
5480 if (depthBufferTextureId !=
ID_NONE) {
5481 auto& depthBufferTexture = *
textures[depthBufferTextureId];
5482 if (depthBufferTexture.frameBufferUnbindImageLayoutChange.valid ==
false) {
5484 depthBufferTexture.frameBufferUnbindImageLayoutChange,
5485 &depthBufferTexture,
5486 { THSVS_ACCESS_DEPTH_STENCIL_ATTACHMENT_READ, THSVS_ACCESS_DEPTH_STENCIL_ATTACHMENT_WRITE },
5487 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
5488 THSVS_IMAGE_LAYOUT_OPTIMAL,
5489 THSVS_IMAGE_LAYOUT_OPTIMAL,
5495 applyImageLayoutChange(0, depthBufferTexture.frameBufferUnbindImageLayoutChange, &depthBufferTexture,
false);
5497 array<texture_type*, 8> colorBufferTextures = {
5498 textures[frameBuffer->gbufferGeometryBufferTextureId1],
5499 textures[frameBuffer->gbufferGeometryBufferTextureId2],
5500 textures[frameBuffer->gbufferGeometryBufferTextureId3],
5501 textures[frameBuffer->gbufferColorBufferTextureId1],
5502 textures[frameBuffer->gbufferColorBufferTextureId2],
5503 textures[frameBuffer->gbufferColorBufferTextureId3],
5504 textures[frameBuffer->gbufferColorBufferTextureId4],
5505 textures[frameBuffer->gbufferColorBufferTextureId5]
5507 array<image_layout_change, 8> colorBufferTexturesImageLayoutChanges;
5509 for (
auto colorBufferTexture: colorBufferTextures) {
5510 if (colorBufferTexture->frameBufferUnbindImageLayoutChange.valid ==
false) {
5512 colorBufferTexture->frameBufferUnbindImageLayoutChange,
5514 { THSVS_ACCESS_COLOR_ATTACHMENT_READ, THSVS_ACCESS_COLOR_ATTACHMENT_WRITE},
5515 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
5516 THSVS_IMAGE_LAYOUT_OPTIMAL,
5517 THSVS_IMAGE_LAYOUT_OPTIMAL,
5523 colorBufferTexturesImageLayoutChanges[i++] = colorBufferTexture->frameBufferUnbindImageLayoutChange;
5526 if (
VERBOSE ==
true) Console::println(
5527 "VKRenderer::" +
string(__FUNCTION__) +
"(): " +
5528 to_string(frameBufferId) +
5529 ": geometry buffer: unbinding: " +
5530 to_string(depthBufferTextureId) +
", " +
5531 to_string(colorBufferTextures[0]->
id) +
", " +
5532 to_string(colorBufferTextures[1]->
id) +
", " +
5533 to_string(colorBufferTextures[2]->
id) +
", " +
5534 to_string(colorBufferTextures[3]->
id) +
", " +
5535 to_string(colorBufferTextures[4]->
id) +
", " +
5536 to_string(colorBufferTextures[5]->
id) +
", " +
5537 to_string(colorBufferTextures[6]->
id) +
", " +
5538 to_string(colorBufferTextures[7]->
id)
5545 if (frameBufferId !=
ID_NONE) {
5547 if (frameBuffer ==
nullptr) {
5548 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): framebuffer not found: " + to_string(frameBufferId));
5557 if (frameBuffer ==
nullptr) {
5558 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): framebuffer not found: " + to_string(
boundFrameBufferId));
5561 auto depthBufferTextureId = frameBuffer->depthTextureId;
5562 auto colorBufferTextureId = frameBuffer->colorTextureId;
5563 if (depthBufferTextureId !=
ID_NONE) {
5564 auto& depthBufferTexture = *
textures[depthBufferTextureId];
5565 if (depthBufferTexture.frameBufferBindImageLayoutChange.valid ==
false) {
5567 depthBufferTexture.frameBufferBindImageLayoutChange,
5568 &depthBufferTexture,
5569 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
5570 { THSVS_ACCESS_DEPTH_STENCIL_ATTACHMENT_READ, THSVS_ACCESS_DEPTH_STENCIL_ATTACHMENT_WRITE },
5571 THSVS_IMAGE_LAYOUT_OPTIMAL,
5572 THSVS_IMAGE_LAYOUT_OPTIMAL,
5578 applyImageLayoutChange(0, depthBufferTexture.frameBufferBindImageLayoutChange, &depthBufferTexture,
false);
5580 if (colorBufferTextureId !=
ID_NONE) {
5581 auto& colorBufferTexture = *
textures[colorBufferTextureId];
5582 if (colorBufferTexture.frameBufferBindImageLayoutChange.valid ==
false) {
5584 colorBufferTexture.frameBufferBindImageLayoutChange,
5585 &colorBufferTexture,
5586 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
5587 { THSVS_ACCESS_COLOR_ATTACHMENT_READ, THSVS_ACCESS_COLOR_ATTACHMENT_WRITE},
5588 THSVS_IMAGE_LAYOUT_OPTIMAL,
5589 THSVS_IMAGE_LAYOUT_OPTIMAL,
5595 applyImageLayoutChange(0, colorBufferTexture.frameBufferBindImageLayoutChange, &colorBufferTexture,
false);
5597 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(frameBufferId) +
": binding: " + to_string(colorBufferTextureId) +
" / " + to_string(depthBufferTextureId));
5600 auto depthBufferTextureId = frameBuffer->depthTextureId;
5601 auto colorBufferTextureId1 = frameBuffer->gbufferColorBufferTextureId1;
5602 auto colorBufferTextureId2 = frameBuffer->gbufferColorBufferTextureId2;
5603 auto colorBufferTextureId3 = frameBuffer->gbufferColorBufferTextureId3;
5604 auto colorBufferTextureId4 = frameBuffer->gbufferColorBufferTextureId4;
5605 auto colorBufferTextureId5 = frameBuffer->gbufferColorBufferTextureId5;
5606 if (depthBufferTextureId !=
ID_NONE) {
5607 auto& depthBufferTexture = *
textures[depthBufferTextureId];
5608 if (depthBufferTexture.frameBufferBindImageLayoutChange.valid ==
false) {
5610 depthBufferTexture.frameBufferBindImageLayoutChange,
5611 &depthBufferTexture,
5612 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
5613 { THSVS_ACCESS_DEPTH_STENCIL_ATTACHMENT_READ, THSVS_ACCESS_DEPTH_STENCIL_ATTACHMENT_WRITE },
5614 THSVS_IMAGE_LAYOUT_OPTIMAL,
5615 THSVS_IMAGE_LAYOUT_OPTIMAL,
5621 applyImageLayoutChange(0, depthBufferTexture.frameBufferBindImageLayoutChange, &depthBufferTexture,
false);
5623 array<texture_type*, 8> colorBufferTextures = {
5624 textures[frameBuffer->gbufferGeometryBufferTextureId1],
5625 textures[frameBuffer->gbufferGeometryBufferTextureId2],
5626 textures[frameBuffer->gbufferGeometryBufferTextureId3],
5627 textures[frameBuffer->gbufferColorBufferTextureId1],
5628 textures[frameBuffer->gbufferColorBufferTextureId2],
5629 textures[frameBuffer->gbufferColorBufferTextureId3],
5630 textures[frameBuffer->gbufferColorBufferTextureId4],
5631 textures[frameBuffer->gbufferColorBufferTextureId5]
5633 array<image_layout_change, 8> colorBufferTexturesImageLayoutChanges;
5635 for (
auto colorBufferTexture: colorBufferTextures) {
5636 if (colorBufferTexture->frameBufferBindImageLayoutChange.valid ==
false) {
5638 colorBufferTexture->frameBufferBindImageLayoutChange,
5640 { THSVS_ACCESS_FRAGMENT_SHADER_READ_SAMPLED_IMAGE_OR_UNIFORM_TEXEL_BUFFER, THSVS_ACCESS_NONE },
5641 { THSVS_ACCESS_COLOR_ATTACHMENT_READ, THSVS_ACCESS_COLOR_ATTACHMENT_WRITE },
5642 THSVS_IMAGE_LAYOUT_OPTIMAL,
5643 THSVS_IMAGE_LAYOUT_OPTIMAL,
5649 colorBufferTexturesImageLayoutChanges[i++] = colorBufferTexture->frameBufferBindImageLayoutChange;
5652 if (
VERBOSE ==
true) Console::println(
5653 "VKRenderer::" +
string(__FUNCTION__) +
"(): " +
5654 to_string(frameBufferId) +
5655 ": geometry buffer: binding: " +
5656 to_string(depthBufferTextureId) +
", " +
5657 to_string(colorBufferTextures[0]->
id) +
", " +
5658 to_string(colorBufferTextures[1]->
id) +
", " +
5659 to_string(colorBufferTextures[2]->
id) +
", " +
5660 to_string(colorBufferTextures[3]->
id) +
", " +
5661 to_string(colorBufferTextures[4]->
id) +
", " +
5662 to_string(colorBufferTextures[5]->
id) +
", " +
5663 to_string(colorBufferTextures[6]->
id) +
", " +
5664 to_string(colorBufferTextures[7]->
id)
5676 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(frameBufferId));
5678 if (frameBuffer ==
nullptr) {
5679 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): framebuffer not found: " + to_string(frameBufferId));
5682 vkDestroyRenderPass(
device, frameBuffer->renderPass,
nullptr);
5683 vkDestroyFramebuffer(
device, frameBuffer->frameBuffer,
nullptr);
5692 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
5693 vector<int32_t> bufferIds;
5696 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): coud not allocate buffer object, maximum is " + to_string(
BUFFERS_MAX));
5700 for (
auto i = 0; i < bufferCount; i++) {
5702 auto& buffer = *bufferPtr;
5703 auto reuseBufferId = -1;
5709 buffer.id = reuseBufferId != -1?reuseBufferId:
bufferIdx++;
5710 buffer.useGPUMemory = useGPUMemory;
5711 buffer.shared = shared;
5712 buffer.frameUsedLast = -1LL;
5713 buffer.frameCleanedLast =
frame;
5715 buffers[buffer.id] = bufferPtr;
5716 bufferIds.push_back(buffer.id);
5725 auto buffer = bufferObject->bindBuffer;
5726 buffer->frameUsedLast =
frame;
5727 size = buffer->size;
5732 return bufferObjectId > 0 && bufferObjectId <=
BUFFERS_MAX?
buffers[bufferObjectId]:
nullptr;
5735inline void VKRenderer::createBuffer(VkDeviceSize size, VkBufferUsageFlags usage, VkMemoryPropertyFlags properties, VkBuffer& buffer, VmaAllocation& allocation, VmaAllocationInfo& allocationInfo) {
5739 const VkBufferCreateInfo bufferCreateInfo = {
5740 .sType = VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO,
5743 .size =
static_cast<uint32_t
>(size),
5745 .sharingMode = VK_SHARING_MODE_EXCLUSIVE,
5746 .queueFamilyIndexCount = 0,
5747 .pQueueFamilyIndices =
nullptr
5750 VmaAllocationCreateInfo allocationCreateInfo = {};
5751 allocationCreateInfo.flags = (properties & VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT) == VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT?VMA_ALLOCATION_CREATE_MAPPED_BIT:0;
5752 allocationCreateInfo.usage = VMA_MEMORY_USAGE_UNKNOWN;
5753 allocationCreateInfo.requiredFlags = properties;
5756 err = vmaCreateBuffer(
vmaAllocator, &bufferCreateInfo, &allocationCreateInfo, &buffer, &allocation, &allocationInfo);
5761 if (size == 0)
return;
5764 if (buffer ==
nullptr)
return;
5778 VmaAllocationInfo dstAllocationInfo {};
5779 vmaGetAllocationInfo(
vmaAllocator, allocationDst, &dstAllocationInfo);
5782 auto remainingSize = size;
5783 auto offset = _offset;
5785 auto dst =
static_cast<uint8_t*
>(dstAllocationInfo.pMappedData) + offset;
5786 while (remainingSize >= 8) {
5787 *(uint64_t*)dst = *(uint64_t*)src;
5792 while (remainingSize >= 4) {
5793 *(uint32_t*)dst = *(uint32_t*)src;
5803 if (size == 0)
return;
5814 vector<int> buffersToRemove;
5817 vector<int32_t> buffersToRemove;
5818 for (
auto& reusableBufferCandidate: buffer->
buffers) {
5819 if (
frame >= reusableBufferCandidate.frameUsedLast + 10) {
5820 if (reusableBufferCandidate.memoryMappable ==
true) vmaUnmapMemory(
vmaAllocator, reusableBufferCandidate.allocation);
5821 vmaDestroyBuffer(
vmaAllocator, reusableBufferCandidate.buf, reusableBufferCandidate.allocation);
5822 buffersToRemove.push_back(i - buffersToRemove.size());
5826 for (
auto bufferToRemove: buffersToRemove) {
5827 auto it = buffer->
buffers.begin();
5828 for (
auto i = 0; i < bufferToRemove; i++) ++it;
5840 for (
auto& reusableBufferCandidate: buffer->
buffers) {
5852 if (reusableBufferCandidate->
size >= size &&
5854 reusableBuffer = reusableBufferCandidate;
5862 if (reusableBuffer ==
nullptr) {
5864 reusableBuffer = &buffer->
buffers.back();
5869 if (reusableBuffer->
size == 0) {
5870 VmaAllocationInfo allocationInfo = {};
5871 createBuffer(size, usage, buffer->
useGPUMemory ==
true?VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT:VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT, reusableBuffer->
buf, reusableBuffer->
allocation, allocationInfo);
5872 reusableBuffer->
size = size;
5874 VkMemoryPropertyFlags memoryFlags;
5875 vmaGetMemoryTypeProperties(
vmaAllocator, allocationInfo.memoryType, &memoryFlags);
5876 reusableBuffer->
memoryMappable = (memoryFlags & VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT) == VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT;
5891 VkBuffer stagingBuffer;
5892 VmaAllocation stagingBufferAllocation;
5893 VkDeviceSize stagingBufferAllocationSize;
5894 VmaAllocationInfo stagingBufferAllocationInfo = {};
5895 createBuffer(size, VK_BUFFER_USAGE_TRANSFER_SRC_BIT, VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT, stagingBuffer, stagingBufferAllocation, stagingBufferAllocationInfo);
5900 .buffer = stagingBuffer,
5901 .allocation = stagingBufferAllocation
5907 vmaMapMemory(
vmaAllocator, stagingBufferAllocation, &mmData);
5910 vmaMemCpy(stagingBufferAllocation, data, size);
5913 VkBufferCopy copyRegion = {
5916 .size =
static_cast<VkDeviceSize
>(size)
5918 vkCmdCopyBuffer(
contexts[contextIdx].setupCommandInUse, stagingBuffer, reusableBuffer->
buf, 1, ©Region);
5938 uploadBufferObjectInternal(contextIdx, bufferObjectId, size, data->
getBuffer(), (VkBufferUsageFlagBits)(VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT | VK_BUFFER_USAGE_STORAGE_BUFFER_BIT));
5943 uploadBufferObjectInternal(contextIdx, bufferObjectId, size, data->
getBuffer(), (VkBufferUsageFlagBits)(VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT | VK_BUFFER_USAGE_STORAGE_BUFFER_BIT));
5948 uploadBufferObjectInternal(contextIdx, bufferObjectId, size, data->
getBuffer(), (VkBufferUsageFlagBits)(VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT | VK_BUFFER_USAGE_STORAGE_BUFFER_BIT));
5953 uploadBufferObjectInternal(contextIdx, bufferObjectId, size, data->
getBuffer(), (VkBufferUsageFlagBits)(VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_INDEX_BUFFER_BIT));
5958 uploadBufferObjectInternal(contextIdx, bufferObjectId, size, data->
getBuffer(), (VkBufferUsageFlagBits)(VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_INDEX_BUFFER_BIT));
5967 if (textureObject ==
nullptr)
return nullptr;
5968 return textureObject->bindTexture;
5978 return framebufferPipelinesCandidate;
5990 framebufferPipelines->pipelines.fill(VK_NULL_HANDLE);
5992 return framebufferPipelines;
5996 auto& currentContext =
contexts[contextIdx];
6000 if (framebufferPipelines ==
nullptr)
return VK_NULL_HANDLE;
6001 return framebufferPipelines->pipelines[pipelineIdx];
6006 auto& currentContext =
contexts[contextIdx];
6007 uint32_t bufferSize = 0;
6013 auto& currentContext =
contexts[contextIdx];
6019 auto& currentContext =
contexts[contextIdx];
6025 auto& currentContext =
contexts[contextIdx];
6031 auto& currentContext =
contexts[contextIdx];
6037 auto& currentContext =
contexts[contextIdx];
6043 auto& currentContext =
contexts[contextIdx];
6049 auto& currentContext =
contexts[contextIdx];
6055 auto& currentContext =
contexts[contextIdx];
6061 auto& currentContext =
contexts[contextIdx];
6066 auto& currentContext =
contexts[contextIdx];
6071 auto& currentContext =
contexts[contextIdx];
6076 auto& currentContext =
contexts[contextIdx];
6081 auto& currentContext =
contexts[contextIdx];
6091 auto& currentContext =
contexts[contextIdx];
6092 auto& programContext = currentContext.program->contexts[currentContext.idx];
6093 auto& programCommandBuffer = programContext.commandBuffers[currentContext.currentCommandBuffer];
6096 auto uboDescriptorSet = programCommandBuffer.uboDescriptorSets[programCommandBuffer.uboDescriptorSetsIdx];
6097 auto texturesDescriptorSetUncached = programCommandBuffer.texturesDescriptorSetsUncached[programCommandBuffer.texturesDescriptorSetsIdxUncached];
6111 #if defined(CPU_64BIT)
6120 auto samplerIdx = 0;
6121 for (
auto shader: currentContext.program->shaders) {
6123 for (
auto uniform: shader->samplerUniformList) {
6125 if (uniform->textureUnit == -1) {
6126 textureIds[samplerIdx] =
ID_NONE;
6128 auto& boundTexture = currentContext.boundTextures[uniform->textureUnit];
6129 if (boundTexture.view == VK_NULL_HANDLE) {
6130 textureIds[samplerIdx] =
ID_NONE;
6132 textureIds[samplerIdx] = boundTexture.id;
6141 if (shader->uboBindingIdx == -1) {
6147 auto& uniformBuffer = shader->uniformBuffers[currentContext.idx];
6148 auto& src = uniformBuffer.buffers[uniformBuffer.bufferIdx];
6149 auto uboBuffer = src.buffer;
6150 uniformBuffer.bufferIdx = (uniformBuffer.bufferIdx + 1) % uniformBuffer.buffers.size();
6151 vmaMemCpy(src.allocation, uniformBuffer.uniformBufferData.data(), uniformBuffer.size);
6154 currentContext.descriptorBufferInfos[shader->uboBindingIdx] = {
6155 .buffer = uboBuffer,
6157 .range = shader->uboSize
6161 currentContext.descriptorWriteSets[shader->uboBindingIdx] = {
6162 .sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
6164 .dstSet = uboDescriptorSet,
6165 .dstBinding =
static_cast<uint32_t
>(shader->uboBindingIdx),
6166 .dstArrayElement = 0,
6167 .descriptorCount = 1,
6168 .descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER,
6169 .pImageInfo =
nullptr,
6170 .pBufferInfo = ¤tContext.descriptorBufferInfos[shader->uboBindingIdx],
6171 .pTexelBufferView =
nullptr
6178 samplers = samplerIdx;
6182 auto& textureDescriptorSetCache = programContext.texturesDescriptorSetsCache;
6183 auto textureDescriptorSetCacheIt = samplers >
SAMPLER_HASH_MAX?textureDescriptorSetCache.end():textureDescriptorSetCache.find(textureDescriptorSetCacheId);
6184 auto textureDescriptorSetCacheHit = textureDescriptorSetCacheIt != textureDescriptorSetCache.end();
6185 if (textureDescriptorSetCacheHit ==
false) {
6187 auto textureDescriptorSetsIdx = -1;
6188 if (programContext.freeTextureDescriptorSetsIds.empty() ==
false) {
6189 auto freeTextureDescriptorSetsIdsIdx = programContext.freeTextureDescriptorSetsIds.size() - 1;
6190 textureDescriptorSetsIdx = programContext.freeTextureDescriptorSetsIds[freeTextureDescriptorSetsIdsIdx];
6191 programContext.freeTextureDescriptorSetsIds.erase(programContext.freeTextureDescriptorSetsIds.begin() + freeTextureDescriptorSetsIdsIdx);
6193 textureDescriptorSetsIdx = programContext.descriptorSets2Idx++;
6195 texturesDescriptorSetUncached = programContext.descriptorSets2[textureDescriptorSetsIdx];
6196 textureDescriptorSetCache[textureDescriptorSetCacheId] = textureDescriptorSetsIdx;
6197 for (
auto textureId: textureIds) programContext.texturesDescriptorSetsCacheTextureIds[textureId].insert(textureDescriptorSetCacheId);
6199 programCommandBuffer.texturesDescriptorSetsIdxUncached = (programCommandBuffer.texturesDescriptorSetsIdxUncached + 1) % programCommandBuffer.texturesDescriptorSetsUncached.size();
6201 auto samplerIdx = 0;
6202 for (
auto shader: currentContext.program->shaders) {
6204 for (
auto uniform: shader->samplerUniformList) {
6205 if (uniform->textureUnit == -1) {
6206 switch(uniform->type) {
6208 currentContext.descriptorImageInfos[samplerIdx] = {
6215 currentContext.descriptorImageInfos[samplerIdx] = {
6222 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): object command: unknown sampler: " + to_string(uniform->type));
6226 auto& boundTexture = currentContext.boundTextures[uniform->textureUnit];
6227 if (boundTexture.view == VK_NULL_HANDLE) {
6228 switch(uniform->type) {
6230 currentContext.descriptorImageInfos[samplerIdx] = {
6237 currentContext.descriptorImageInfos[samplerIdx] = {
6244 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): object command: unknown sampler: " + to_string(uniform->type));
6248 currentContext.descriptorImageInfos[samplerIdx] = {
6249 .sampler = boundTexture.sampler,
6250 .imageView = boundTexture.view,
6251 .imageLayout = boundTexture.layout
6255 currentContext.descriptorWriteSets[uniform->position] = {
6256 .sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
6258 .dstSet = texturesDescriptorSetUncached,
6259 .dstBinding =
static_cast<uint32_t
>(uniform->position),
6260 .dstArrayElement = 0,
6261 .descriptorCount = 1,
6262 .descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER,
6263 .pImageInfo = ¤tContext.descriptorImageInfos[samplerIdx],
6264 .pBufferInfo = VK_NULL_HANDLE,
6265 .pTexelBufferView = VK_NULL_HANDLE
6271 texturesDescriptorSetUncached = programContext.descriptorSets2[textureDescriptorSetCacheIt->second];
6275 vkUpdateDescriptorSets(
device, textureDescriptorSetCacheHit ==
true?uboIdx:currentContext.program->layoutBindings, currentContext.descriptorWriteSets.data(), 0,
nullptr);
6278 array<VkDescriptorSet, 2> descSets { uboDescriptorSet, texturesDescriptorSetUncached };
6281 auto& drawCommand = currentContext.commandBuffers[currentContext.currentCommandBuffer].drawCommand;
6282 vkCmdBindDescriptorSets(drawCommand, VK_PIPELINE_BIND_POINT_GRAPHICS, currentContext.program->pipelineLayout, 0, descSets.size(), descSets.data(), 0,
nullptr);
6285 if (indicesBuffer != VK_NULL_HANDLE) {
6286 vkCmdBindIndexBuffer(drawCommand, indicesBuffer, 0, VK_INDEX_TYPE_UINT32);
6290 vkCmdBindVertexBuffers(drawCommand, 0,
OBJECTS_VERTEX_BUFFER_COUNT, currentContext.boundBuffers.data(), currentContext.boundBufferOffsets.data());
6293 if (indicesBuffer != VK_NULL_HANDLE) {
6294 vkCmdDrawIndexed(drawCommand, triangles * 3, instances, trianglesOffset * 3, 0, 0);
6296 vkCmdDraw(drawCommand, triangles * 3, instances, trianglesOffset * 3, 0);
6300 programCommandBuffer.uboDescriptorSetsIdx = (programCommandBuffer.uboDescriptorSetsIdx + 1) % programCommandBuffer.uboDescriptorSets.size();
6301 currentContext.commandCount++;
6314 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
6327 auto currentBufferIdx = context.currentCommandBuffer;
6329 if (commandBuffer != VK_NULL_HANDLE) {
6338 auto& currentContext =
contexts[contextIdx];
6342 if (currentContext.commandCount >= commandsMax) {
6344 auto currentBufferIdx = currentContext.currentCommandBuffer;
6346 if (commandBuffer != VK_NULL_HANDLE)
submitDrawCommandBuffers(1, &commandBuffer, currentContext.commandBuffers[currentBufferIdx].drawFence);
6347 currentContext.commandCount = 0;
6363 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
6366 auto& currentContext =
contexts[contextIdx];
6367 auto& programContext = currentContext.program->contexts[currentContext.idx];
6368 auto& programCommandBuffer = programContext.commandBuffers[currentContext.currentCommandBuffer];
6371 auto uboDescriptorSet = programCommandBuffer.uboDescriptorSets[programCommandBuffer.uboDescriptorSetsIdx];
6372 auto texturesDescriptorSet = programCommandBuffer.texturesDescriptorSetsUncached[programCommandBuffer.texturesDescriptorSetsIdxUncached];
6386 auto samplerIdx = 0;
6387 for (
auto shader: currentContext.program->shaders) {
6389 for (
auto uniform: shader->samplerUniformList) {
6390 if (uniform->textureUnit == -1) {
6391 switch(uniform->type) {
6393 currentContext.descriptorImageInfos[samplerIdx] = {
6400 currentContext.descriptorImageInfos[samplerIdx] = {
6407 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): object command: unknown sampler: " + to_string(uniform->type));
6411 auto& texture = currentContext.boundTextures[uniform->textureUnit];
6412 if (texture.view == VK_NULL_HANDLE) {
6413 switch(uniform->type) {
6415 currentContext.descriptorImageInfos[samplerIdx] = {
6422 currentContext.descriptorImageInfos[samplerIdx] = {
6429 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): object command: unknown sampler: " + to_string(uniform->type));
6433 currentContext.descriptorImageInfos[samplerIdx] = {
6434 .sampler = texture.sampler,
6435 .imageView = texture.view,
6436 .imageLayout = texture.layout
6440 currentContext.descriptorWriteSets[uniform->position] = {
6441 .sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
6443 .dstSet = texturesDescriptorSet,
6444 .dstBinding =
static_cast<uint32_t
>(uniform->position),
6445 .dstArrayElement = 0,
6446 .descriptorCount = 1,
6447 .descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER,
6448 .pImageInfo = ¤tContext.descriptorImageInfos[samplerIdx],
6449 .pBufferInfo = VK_NULL_HANDLE,
6450 .pTexelBufferView = VK_NULL_HANDLE
6456 if (shader->uboBindingIdx == -1) {
6462 auto& uniformBuffer = shader->uniformBuffers[currentContext.idx];
6463 auto& src = uniformBuffer.buffers[uniformBuffer.bufferIdx];
6464 auto uboBuffer = src.buffer;
6465 uniformBuffer.bufferIdx = (uniformBuffer.bufferIdx + 1) % uniformBuffer.buffers.size();
6466 vmaMemCpy(src.allocation, uniformBuffer.uniformBufferData.data(), uniformBuffer.size);
6469 currentContext.descriptorBufferInfos[shader->uboBindingIdx] = {
6470 .buffer = uboBuffer,
6472 .range = shader->uboSize
6476 currentContext.descriptorWriteSets[shader->uboBindingIdx] = {
6477 .sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
6479 .dstSet = uboDescriptorSet,
6480 .dstBinding =
static_cast<uint32_t
>(shader->uboBindingIdx),
6481 .dstArrayElement = 0,
6482 .descriptorCount = 1,
6483 .descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER,
6484 .pImageInfo =
nullptr,
6485 .pBufferInfo = ¤tContext.descriptorBufferInfos[shader->uboBindingIdx],
6486 .pTexelBufferView =
nullptr
6495 vkUpdateDescriptorSets(
device, currentContext.program->layoutBindings, currentContext.descriptorWriteSets.data(), 0,
nullptr);
6498 auto& drawCommand = currentContext.commandBuffers[currentContext.currentCommandBuffer].drawCommand;
6499 array<VkDescriptorSet, 2> descriptorSets { uboDescriptorSet, texturesDescriptorSet };
6500 vkCmdBindDescriptorSets(drawCommand, VK_PIPELINE_BIND_POINT_GRAPHICS, currentContext.program->pipelineLayout, 0, descriptorSets.size(), descriptorSets.data(), 0,
nullptr);
6501 vkCmdBindVertexBuffers(drawCommand, 0,
POINTS_VERTEX_BUFFER_COUNT, currentContext.boundBuffers.data(), currentContext.boundBufferOffsets.data());
6502 vkCmdDraw(drawCommand, points, 1, pointsOffset, 0);
6505 programCommandBuffer.uboDescriptorSetsIdx = (programCommandBuffer.uboDescriptorSetsIdx + 1) % programCommandBuffer.uboDescriptorSets.size();
6506 programCommandBuffer.texturesDescriptorSetsIdxUncached = (programCommandBuffer.texturesDescriptorSetsIdxUncached + 1) % programCommandBuffer.texturesDescriptorSetsUncached.size();
6507 currentContext.commandCount++;
6524 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
6527 auto& currentContext =
contexts[contextIdx];
6528 auto& programCommandBuffer = currentContext.program->contexts[currentContext.idx].commandBuffers[currentContext.currentCommandBuffer];
6531 auto uboDescriptorSet = programCommandBuffer.uboDescriptorSets[programCommandBuffer.uboDescriptorSetsIdx];
6532 auto textureDescriptorSet = programCommandBuffer.texturesDescriptorSetsUncached[programCommandBuffer.texturesDescriptorSetsIdxUncached];
6546 auto samplerIdx = 0;
6547 for (
auto shader: currentContext.program->shaders) {
6549 for (
auto uniform: shader->samplerUniformList) {
6550 if (uniform->textureUnit == -1) {
6551 switch(uniform->type) {
6553 currentContext.descriptorImageInfos[samplerIdx] = {
6560 currentContext.descriptorImageInfos[samplerIdx] = {
6567 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): object command: unknown sampler: " + to_string(uniform->type));
6571 auto& texture = currentContext.boundTextures[uniform->textureUnit];
6572 if (texture.view == VK_NULL_HANDLE) {
6573 switch(uniform->type) {
6575 currentContext.descriptorImageInfos[samplerIdx] = {
6582 currentContext.descriptorImageInfos[samplerIdx] = {
6589 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): object command: unknown sampler: " + to_string(uniform->type));
6593 currentContext.descriptorImageInfos[samplerIdx] = {
6594 .sampler = texture.sampler,
6595 .imageView = texture.view,
6596 .imageLayout = texture.layout
6600 currentContext.descriptorWriteSets[uniform->position] = {
6601 .sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
6603 .dstSet = textureDescriptorSet,
6604 .dstBinding =
static_cast<uint32_t
>(uniform->position),
6605 .dstArrayElement = 0,
6606 .descriptorCount = 1,
6607 .descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER,
6608 .pImageInfo = ¤tContext.descriptorImageInfos[samplerIdx],
6609 .pBufferInfo = VK_NULL_HANDLE,
6610 .pTexelBufferView = VK_NULL_HANDLE
6616 if (shader->uboBindingIdx == -1) {
6622 auto& uniformBuffer = shader->uniformBuffers[currentContext.idx];
6623 auto& src = uniformBuffer.buffers[uniformBuffer.bufferIdx];
6624 auto uboBuffer = src.buffer;
6625 uniformBuffer.bufferIdx = (uniformBuffer.bufferIdx + 1) % uniformBuffer.buffers.size();
6626 vmaMemCpy(src.allocation, uniformBuffer.uniformBufferData.data(), uniformBuffer.size);
6629 currentContext.descriptorBufferInfos[shader->uboBindingIdx] = {
6630 .buffer = uboBuffer,
6632 .range = shader->uboSize
6636 currentContext.descriptorWriteSets[shader->uboBindingIdx] = {
6637 .sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
6639 .dstSet = uboDescriptorSet,
6640 .dstBinding =
static_cast<uint32_t
>(shader->uboBindingIdx),
6641 .dstArrayElement = 0,
6642 .descriptorCount = 1,
6643 .descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER,
6644 .pImageInfo =
nullptr,
6645 .pBufferInfo = ¤tContext.descriptorBufferInfos[shader->uboBindingIdx],
6646 .pTexelBufferView =
nullptr
6655 vkUpdateDescriptorSets(
device, currentContext.program->layoutBindings, currentContext.descriptorWriteSets.data(), 0,
nullptr);
6658 auto& drawCommand = currentContext.commandBuffers[currentContext.currentCommandBuffer].drawCommand;
6659 array<VkDescriptorSet, 2> descriptorSets { uboDescriptorSet, textureDescriptorSet };
6660 vkCmdBindDescriptorSets(drawCommand, VK_PIPELINE_BIND_POINT_GRAPHICS, currentContext.program->pipelineLayout, 0, descriptorSets.size(), descriptorSets.data(), 0,
nullptr);
6661 vkCmdBindVertexBuffers(drawCommand, 0,
LINES_VERTEX_BUFFER_COUNT, currentContext.boundBuffers.data(), currentContext.boundBufferOffsets.data());
6662 vkCmdSetLineWidth(drawCommand,
lineWidth);
6663 vkCmdDraw(drawCommand, points, 1, pointsOffset, 0);
6666 programCommandBuffer.uboDescriptorSetsIdx = (programCommandBuffer.uboDescriptorSetsIdx + 1) % programCommandBuffer.uboDescriptorSets.size();
6667 programCommandBuffer.texturesDescriptorSetsIdxUncached = (programCommandBuffer.texturesDescriptorSetsIdxUncached + 1) % programCommandBuffer.texturesDescriptorSetsUncached.size();
6668 currentContext.commandCount++;
6680 auto& currentContext =
contexts[contextIdx];
6681 uint32_t bufferSize = 0;
6683 currentContext.boundIndicesBuffer = VK_NULL_HANDLE;
6684 currentContext.boundBuffers.fill(defaultBuffer);
6685 currentContext.boundBufferSizes.fill(bufferSize);
6691 for (
auto bufferObjectId: bufferObjectIds) {
6699 return contexts[contextIdx].activeTextureUnit;
6704 contexts[contextIdx].activeTextureUnit = textureUnit;
6709 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
6712 auto pixelDepth = -1.0f;
6715 VkFormat usedFormat = VK_FORMAT_UNDEFINED;
6716 VkImage usedImage = VK_NULL_HANDLE;
6717 uint32_t usedWidth = 0;
6718 uint32_t usedHeight = -1;
6719 array<ThsvsAccessType, 2> usedAccessTypes;
6720 ThsvsImageLayout usedImageLayout = THSVS_IMAGE_LAYOUT_OPTIMAL;
6722 if (frameBuffer ==
nullptr) {
6724 if (depthBufferTexture ==
nullptr) {
6725 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): depth buffer: depth buffer texture not found: " + to_string(
depthBufferDefault));
6728 usedFormat = depthBufferTexture->format;
6729 usedImage = depthBufferTexture->image;
6730 usedWidth = depthBufferTexture->width;
6731 usedHeight = depthBufferTexture->height;
6732 usedAccessTypes = depthBufferTexture->accessTypes[0];
6733 usedImageLayout = depthBufferTexture->svsLayout;
6736 if (depthBufferTexture ==
nullptr) {
6737 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): depth buffer: depth buffer texture not found: " + to_string(frameBuffer->depthBufferTextureId));
6740 usedFormat = depthBufferTexture->format;
6741 usedImage = depthBufferTexture->image;
6742 usedWidth = depthBufferTexture->width;
6743 usedHeight = depthBufferTexture->height;
6744 usedAccessTypes = depthBufferTexture->accessTypes[0];
6745 usedImageLayout = depthBufferTexture->svsLayout;
6753 VmaAllocationInfo allocationInfo = {};
6754 VkBuffer buffer = VK_NULL_HANDLE;
6755 VmaAllocation allocation = VK_NULL_HANDLE;
6758 VK_BUFFER_USAGE_TRANSFER_DST_BIT,
6759 VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT,
6764 VkMemoryPropertyFlags memoryFlags;
6765 vmaGetMemoryTypeProperties(
vmaAllocator, allocationInfo.memoryType, &memoryFlags);
6766 auto memoryMapped = (memoryFlags & VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT) == VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT;
6767 if (memoryMapped ==
false) {
6769 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): Could not create memory mappable buffer");
6777 array<ThsvsAccessType, 2> nextAccessTypes = { THSVS_ACCESS_TRANSFER_READ, THSVS_ACCESS_NONE };
6778 setImageLayout3(currentContext.idx, usedImage, VK_IMAGE_ASPECT_DEPTH_BIT, usedAccessTypes, nextAccessTypes, usedImageLayout, THSVS_IMAGE_LAYOUT_OPTIMAL);
6781 VkBufferImageCopy bufferImageCopy = {
6783 .bufferRowLength = 0,
6784 .bufferImageHeight = 0,
6785 .imageSubresource = {
6786 .aspectMask = VK_IMAGE_ASPECT_DEPTH_BIT,
6788 .baseArrayLayer = 0,
6792 .x =
static_cast<int32_t
>(x),
6793 .y =
static_cast<int32_t
>(usedHeight - 1 - y),
6805 vkCmdCopyImageToBuffer(
6806 currentContext.setupCommandInUse,
6808 VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
6820 pixelDepth =
static_cast<float*
>(data)[0];
6828 array<ThsvsAccessType, 2> lastAccessTypes = { THSVS_ACCESS_TRANSFER_READ, THSVS_ACCESS_NONE };
6829 setImageLayout3(currentContext.idx, usedImage, VK_IMAGE_ASPECT_DEPTH_BIT, lastAccessTypes, usedAccessTypes, THSVS_IMAGE_LAYOUT_OPTIMAL, usedImageLayout);
6837 .allocation = allocation
6843 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(pixelDepth));
6851 if (
VERBOSE ==
true) Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"()");
6854 VkFormat usedFormat = VK_FORMAT_UNDEFINED;
6855 VkImage usedImage = VK_NULL_HANDLE;
6856 uint32_t usedWidth = 0;
6857 uint32_t usedHeight = -1;
6858 array<ThsvsAccessType, 2> usedAccessTypes;
6859 ThsvsImageLayout usedImageLayout = THSVS_IMAGE_LAYOUT_OPTIMAL;
6861 if (frameBuffer ==
nullptr) {
6864 usedImage = swapchainBuffer.image;
6865 usedWidth = swapchainBuffer.width;
6866 usedHeight = swapchainBuffer.height;
6867 usedAccessTypes = swapchainBuffer.accessTypes;
6868 usedImageLayout = swapchainBuffer.svsLayout;
6871 if (colorBufferTexture ==
nullptr) {
6872 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): color buffer: color buffer texture not found: " + to_string(frameBuffer->colorTextureId));
6875 usedFormat = colorBufferTexture->format;
6876 usedImage = colorBufferTexture->image;
6877 usedWidth = colorBufferTexture->width;
6878 usedHeight = colorBufferTexture->height;
6879 usedAccessTypes = colorBufferTexture->accessTypes[0];
6880 usedImageLayout = colorBufferTexture->svsLayout;
6885 VkImage image = VK_NULL_HANDLE;
6886 VmaAllocation allocation = VK_NULL_HANDLE;
6887 auto requiredFlags = VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT;
6890 const VkImageCreateInfo imageCreateInfo = {
6891 .sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO,
6894 .imageType = VK_IMAGE_TYPE_2D,
6895 .format = usedFormat,
6897 .width =
static_cast<uint32_t
>(width),
6898 .height =
static_cast<uint32_t
>(height),
6903 .samples = VK_SAMPLE_COUNT_1_BIT,
6904 .tiling = VK_IMAGE_TILING_LINEAR,
6905 .usage = VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT,
6906 .sharingMode = VK_SHARING_MODE_EXCLUSIVE,
6907 .queueFamilyIndexCount = 0,
6908 .pQueueFamilyIndices = 0,
6909 .initialLayout = VK_IMAGE_LAYOUT_PREINITIALIZED
6912 VmaAllocationCreateInfo imageAllocCreateInfo = {};
6913 imageAllocCreateInfo.usage = VMA_MEMORY_USAGE_UNKNOWN;
6914 imageAllocCreateInfo.requiredFlags = requiredFlags;
6916 VmaAllocationInfo allocationInfo = {};
6919 err = vmaCreateImage(
vmaAllocator, &imageCreateInfo, &imageAllocCreateInfo, &image, &allocation, &allocationInfo);
6922 if ((requiredFlags & VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT) == VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT) {
6923 VkImageCopy imageCopy = {
6925 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
6927 .baseArrayLayer = 0,
6936 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
6938 .baseArrayLayer = 0,
6947 .width =
static_cast<uint32_t
>(width),
6948 .height =
static_cast<uint32_t
>(height),
6955 array<ThsvsAccessType, 2> nextAccessTypes = { THSVS_ACCESS_TRANSFER_READ, THSVS_ACCESS_NONE };
6956 setImageLayout3(currentContext.idx, usedImage, VK_IMAGE_ASPECT_COLOR_BIT, usedAccessTypes, nextAccessTypes, usedImageLayout, THSVS_IMAGE_LAYOUT_OPTIMAL);
6960 array<ThsvsAccessType, 2> accessTypes = { THSVS_ACCESS_HOST_PREINITIALIZED, THSVS_ACCESS_NONE };
6961 array<ThsvsAccessType, 2> nextAccessTypes = { THSVS_ACCESS_TRANSFER_WRITE, THSVS_ACCESS_NONE };
6962 setImageLayout3(currentContext.idx, image, VK_IMAGE_ASPECT_COLOR_BIT, accessTypes, nextAccessTypes, THSVS_IMAGE_LAYOUT_OPTIMAL, THSVS_IMAGE_LAYOUT_OPTIMAL);
6967 currentContext.setupCommandInUse,
6969 VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
6971 VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
6977 const VkImageSubresource imageSubResource = {
6978 .aspectMask = VK_IMAGE_ASPECT_COLOR_BIT,
6982 VkSubresourceLayout subResourceLayout;
6983 vkGetImageSubresourceLayout(
device, image, &imageSubResource, &subResourceLayout);
6992 auto pixelBuffer = ByteBuffer::allocate(width * height * 4);
6993 for (
int y = height - 1; y >= 0; y--) {
6994 auto row =
static_cast<uint8_t*
>(
static_cast<uint8_t*
>(data) + subResourceLayout.offset + subResourceLayout.rowPitch * y);
6995 for (
auto x = 0; x < width; x++) {
6996 pixelBuffer->put(
static_cast<uint8_t
>(row[x * 4 + 2]));
6997 pixelBuffer->put(
static_cast<uint8_t
>(row[x * 4 + 1]));
6998 pixelBuffer->put(
static_cast<uint8_t
>(row[x * 4 + 0]));
6999 pixelBuffer->put(
static_cast<uint8_t
>(row[x * 4 + 3]));
7009 array<ThsvsAccessType, 2> lastAccessTypes = { THSVS_ACCESS_TRANSFER_READ, THSVS_ACCESS_NONE };
7010 setImageLayout3(currentContext.idx, usedImage, VK_IMAGE_ASPECT_COLOR_BIT, lastAccessTypes, usedAccessTypes, THSVS_IMAGE_LAYOUT_OPTIMAL, usedImageLayout);
7018 .allocation = allocation,
7019 .imageView = VK_NULL_HANDLE,
7020 .sampler = VK_NULL_HANDLE,
7050 auto& currentContext =
contexts[contextIdx];
7051 auto& programCommandBuffer = currentContext.program->contexts[currentContext.idx].commandBuffers[currentContext.currentCommandBuffer];
7054 auto uboDescriptorSet = programCommandBuffer.uboDescriptorSets[programCommandBuffer.uboDescriptorSetsIdx];
7065 for (
auto shader: currentContext.program->shaders) {
7066 for (
int i = 0; i <= shader->maxBindings; i++) {
7067 currentContext.descriptorBufferInfos[i] = {
7068 .buffer = currentContext.boundBuffers[i],
7070 .range = currentContext.boundBufferSizes[i]
7072 currentContext.descriptorWriteSets[i] = {
7073 .sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
7075 .dstSet = uboDescriptorSet,
7076 .dstBinding =
static_cast<uint32_t
>(i),
7077 .dstArrayElement = 0,
7078 .descriptorCount = 1,
7079 .descriptorType = VK_DESCRIPTOR_TYPE_STORAGE_BUFFER,
7080 .pImageInfo =
nullptr,
7081 .pBufferInfo = ¤tContext.descriptorBufferInfos[i],
7082 .pTexelBufferView =
nullptr
7087 if (shader->uboBindingIdx == -1) {
7094 auto& uniformBuffer = shader->uniformBuffers[currentContext.idx];
7095 auto& src = uniformBuffer.buffers[uniformBuffer.bufferIdx];
7096 auto uboBuffer = src.buffer;
7097 uniformBuffer.bufferIdx = (uniformBuffer.bufferIdx + 1) % uniformBuffer.buffers.size();
7098 vmaMemCpy(src.allocation, uniformBuffer.uniformBufferData.data(), uniformBuffer.size);
7101 currentContext.descriptorBufferInfos[shader->uboBindingIdx] = {
7102 .buffer = uboBuffer,
7104 .range = shader->uboSize
7108 currentContext.descriptorWriteSets[shader->uboBindingIdx] = {
7109 .sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET,
7111 .dstSet = uboDescriptorSet,
7112 .dstBinding =
static_cast<uint32_t
>(shader->uboBindingIdx),
7113 .dstArrayElement = 0,
7114 .descriptorCount = 1,
7115 .descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER,
7116 .pImageInfo =
nullptr,
7117 .pBufferInfo = ¤tContext.descriptorBufferInfos[shader->uboBindingIdx],
7118 .pTexelBufferView =
nullptr,
7126 vkUpdateDescriptorSets(
device, currentContext.program->layoutBindings, currentContext.descriptorWriteSets.data(), 0,
nullptr);
7129 auto& drawCommand = currentContext.commandBuffers[currentContext.currentCommandBuffer].drawCommand;
7130 vkCmdBindDescriptorSets(drawCommand, VK_PIPELINE_BIND_POINT_COMPUTE, currentContext.program->pipelineLayout, 0, 1, &uboDescriptorSet, 0,
nullptr);
7131 vkCmdDispatch(drawCommand, numGroupsX, numGroupsY, numGroupsZ);
7134 programCommandBuffer.uboDescriptorSetsIdx = (programCommandBuffer.uboDescriptorSetsIdx + 1) % programCommandBuffer.uboDescriptorSets.size();
7135 currentContext.commandCount++;
7149 VkResult fenceResult;
7152 }
while (fenceResult == VK_TIMEOUT);
7155 for (
auto& context:
contexts) context.computeRenderBarrierBuffers.clear();
7166 auto prevAccesses = THSVS_ACCESS_COMPUTE_SHADER_WRITE;
7167 auto nextAccesses = THSVS_ACCESS_VERTEX_BUFFER;
7169 for (
auto buffer: context.computeRenderBarrierBuffers) {
7170 ThsvsBufferBarrier svsBufferBarrier = {
7171 .prevAccessCount = 1,
7172 .pPrevAccesses = &prevAccesses,
7173 .nextAccessCount = 1,
7174 .pNextAccesses = &nextAccesses,
7175 .srcQueueFamilyIndex = 0,
7176 .dstQueueFamilyIndex = 0,
7179 .size = VK_WHOLE_SIZE
7181 VkBufferMemoryBarrier vkBufferMemoryBarrier;
7182 VkPipelineStageFlags srcStages;
7183 VkPipelineStageFlags dstStages;
7184 thsvsGetVulkanBufferMemoryBarrier(
7188 &vkBufferMemoryBarrier
7191 vkCmdPipelineBarrier(context.setupCommandInUse, srcStages, dstStages, 0, 0,
nullptr, 1, &vkBufferMemoryBarrier, 0,
nullptr);
7194 context.computeRenderBarrierBuffers.clear();
7199 uploadBufferObjectInternal(contextIdx, bufferObjectId, size, data->
getBuffer(), (VkBufferUsageFlagBits)(VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT | VK_BUFFER_USAGE_STORAGE_BUFFER_BIT));
7203 auto& currentContext =
contexts[contextIdx];
7204 uploadBufferObjectInternal(contextIdx, bufferObjectId, size, data->
getBuffer(), (VkBufferUsageFlagBits)(VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT | VK_BUFFER_USAGE_STORAGE_BUFFER_BIT));
7208 auto& currentContext =
contexts[contextIdx];
7213 auto& currentContext =
contexts[contextIdx];
7218 auto& currentContext =
contexts[contextIdx];
7223 auto& currentContext =
contexts[contextIdx];
7228 auto& currentContext =
contexts[contextIdx];
7233 auto& currentContext =
contexts[contextIdx];
7235 currentContext.computeRenderBarrierBuffers.push_back(currentContext.boundBuffers[5]);
7237 auto prevAccesses = THSVS_ACCESS_VERTEX_BUFFER;
7238 auto nextAccesses = THSVS_ACCESS_COMPUTE_SHADER_WRITE;
7239 ThsvsBufferBarrier svsbufferBarrier = {
7240 .prevAccessCount = 1,
7241 .pPrevAccesses = &prevAccesses,
7242 .nextAccessCount = 1,
7243 .pNextAccesses = &nextAccesses,
7244 .srcQueueFamilyIndex = 0,
7245 .dstQueueFamilyIndex = 0,
7246 .buffer = currentContext.boundBuffers[5],
7248 .size = VK_WHOLE_SIZE
7250 VkBufferMemoryBarrier vkBufferMemoryBarrier;
7251 VkPipelineStageFlags srcStages;
7252 VkPipelineStageFlags dstStages;
7253 thsvsGetVulkanBufferMemoryBarrier(
7257 &vkBufferMemoryBarrier
7260 vkCmdPipelineBarrier(currentContext.setupCommandInUse, srcStages, dstStages, 0, 0,
nullptr, 1, &vkBufferMemoryBarrier, 0,
nullptr);
7265 auto& currentContext =
contexts[contextIdx];
7267 currentContext.computeRenderBarrierBuffers.push_back(currentContext.boundBuffers[6]);
7268 auto prevAccesses = THSVS_ACCESS_VERTEX_BUFFER;
7269 auto nextAccesses = THSVS_ACCESS_COMPUTE_SHADER_WRITE;
7270 ThsvsBufferBarrier svsbufferBarrier = {
7271 .prevAccessCount = 1,
7272 .pPrevAccesses = &prevAccesses,
7273 .nextAccessCount = 1,
7274 .pNextAccesses = &nextAccesses,
7275 .srcQueueFamilyIndex = 0,
7276 .dstQueueFamilyIndex = 0,
7277 .buffer = currentContext.boundBuffers[6],
7279 .size = VK_WHOLE_SIZE
7281 VkBufferMemoryBarrier vkBufferMemoryBarrier;
7282 VkPipelineStageFlags srcStages;
7283 VkPipelineStageFlags dstStages;
7284 thsvsGetVulkanBufferMemoryBarrier(
7288 &vkBufferMemoryBarrier
7291 vkCmdPipelineBarrier(currentContext.setupCommandInUse, srcStages, dstStages, 0, 0,
nullptr, 1, &vkBufferMemoryBarrier, 0,
nullptr);
7296 auto& currentContext =
contexts[contextIdx];
7301 Console::println(
"VKRenderer::" +
string(__FUNCTION__) +
"(): " + to_string(this->vSync) +
" --> " + to_string(
vSync));
7302 if (this->vSync ==
vSync)
return;
7304 this->vSync =
vSync;
7308 array<VmaBudget, VK_MAX_MEMORY_HEAPS> budget;
7312 stats.memoryUsageShared = budget[1].allocationBytes;
#define ERR_EXIT(err_msg, err_class)
#define GET_INSTANCE_PROC_ADDR(inst, entrypoint)
#define GET_DEVICE_PROC_ADDR(dev, entrypoint)
#define SAMPLER_HASH_TYPE
Application base class, please make sure to allocate application on heap to have correct application ...
void reshape(int32_t width, int32_t height)
Reshape.
static TextureManager * getTextureManager()
static int getThreadCount()
static Engine * getInstance()
Returns engine instance.
TDME2 engine entity shader parameters.
const string & getId() const
ByteBuffer * getTextureData()
int32_t getAtlasSize() const
int32_t getTextureHeight() const
ClampMode getClampMode() const
int32_t getTextureWidth() const
int32_t addCubeMapTexture(const string &id, Texture *textureLeft, Texture *textureRight, Texture *textureTop, Texture *textureBottom, Texture *textureFront, Texture *textureBack, int contextIdx=0)
Adds a cube map texture to manager.
TextureManager_TextureManaged * addTexture(const string &id, bool &created)
Adds a texture to manager.
int32_t CUBEMAPTEXTUREINDEX_NEGATIVE_Z
int32_t SHADER_FRAGMENT_SHADER
Renderer_Statistics statistics
int32_t SHADER_COMPUTE_SHADER
int32_t SHADER_VERTEX_SHADER
int32_t CLEAR_COLOR_BUFFER_BIT
int32_t CUBEMAPTEXTUREINDEX_NEGATIVE_X
int32_t CUBEMAPTEXTUREINDEX_POSITIVE_Y
vector< Renderer_Context > rendererContexts
int32_t CUBEMAPTEXTUREINDEX_NEGATIVE_Y
virtual void onBindTexture(int contextIdx, int32_t textureId)=0
On bind texture event.
RendererType rendererType
int32_t DEPTHFUNCTION_ALWAYS
int32_t DEPTHFUNCTION_LESSEQUAL
int32_t DEPTHFUNCTION_GREATEREQUAL
int32_t CUBEMAPTEXTUREINDEX_POSITIVE_Z
int32_t CONTEXTINDEX_DEFAULT
int32_t DEPTHFUNCTION_EQUAL
int32_t FRAMEBUFFER_DEFAULT
int32_t CUBEMAPTEXTUREINDEX_POSITIVE_X
int32_t CLEAR_DEPTH_BUFFER_BIT
GL3/Core -> Vulkan shader program.
static void loadShader(VKRenderer::shader_type &shader, int32_t type, const string &pathName, const string &fileName, const string &definitions=string(), const string &functions=string())
Loads a shader.
static bool linkProgram(VKRenderer::program_type &program)
Links attached shaders to a program.
void setClearColor(float red, float green, float blue, float alpha) override
Set up clear color.
void prepareTextureImage(int contextIdx, struct texture_type *textureObject, VkImageTiling tiling, VkImageUsageFlags usage, VkFlags requiredFlags, Texture *texture, const array< ThsvsAccessType, 2 > &nextAccesses, ThsvsImageLayout imageLayout, bool disableMipMaps=true, uint32_t baseLevel=0, uint32_t levelCount=1)
void enableDepthBufferWriting() override
Enable depth buffer writing.
const string getShaderVersion() override
void unsetPipeline(int contextIdx)
void bindSpriteSheetDimensionBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind sprite sheet dimension buffer object.
VkColorSpaceKHR windowColorSpace
void bindTexture(int contextIdx, int32_t textureId) override
Binds a texture with given id or unbinds when using ID_NONE.
uint32_t currentWindowFramebufferIdx
void clear(int32_t mask) override
Clear render buffer with given mask.
void createSkinningComputingProgram(program_type *program)
static constexpr int POINTS_VERTEX_BUFFER_COUNT
void bindTangentsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind tangents buffer object.
void disposeBufferObjects(vector< int32_t > &bufferObjectIds) override
Disposes a frame buffer object.
void invalidateTextureDescriptorCaches(int textureId)
int32_t getTextureUnit(int contextIdx) override
Get texture unit.
VkSwapchainKHR windowSwapchain
void setProgramUniformFloatMatrix3x3(int contextIdx, int32_t uniformId, const array< float, 9 > &data) override
Set up a float matrix 3x3 uniform value.
VkPhysicalDeviceProperties gpuProperties
void doneGuiMode() override
Set up renderer for 3d rendering.
static constexpr int DESC_MAX_UNCACHED
void setColorMask(bool red, bool green, bool blue, bool alpha) override
Set up GL rendering colormask.
void bindTextureCoordinatesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind texture coordinates buffer object.
void submitDrawCommandBuffers(int commandBufferCount, VkCommandBuffer *commandBuffers, VkFence &fence)
void setImageLayout2(int contextIdx, texture_type *textureObject, const array< ThsvsAccessType, 2 > &accessTypes, const array< ThsvsAccessType, 2 > &nextAccessTypes, ThsvsImageLayout layout, ThsvsImageLayout nextLayout, bool discardContent, uint32_t baseMipLevel, uint32_t levelCount, uint32_t baseArrayLayer, uint32_t layerCount, bool updateTextureObject)
void bindSkinningVertexJointWeightsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning vertex joint weights buffer object.
bool isGLCLAvailable() override
vector< delete_buffer_type > deleteBuffers
static constexpr int PROGRAMS_MAX
void attachShaderToProgram(int32_t programId, int32_t shaderId) override
Attaches a shader to a program.
void setupObjectsRenderingPipeline(int contextIdx, program_type *program)
void dispatchCompute(int contextIdx, int32_t numNodesX, int32_t numNodesY, int32_t numNodesZ) override
Dispatch compute.
void setFrontFace(int contextIdx, int32_t frontFace) override
Set up clock wise or counter clock wise faces as front face.
void setTextureUnit(int contextIdx, int32_t textureUnit) override
Sets up texture unit.
buffer_object_type * emptyVertexBuffer
void drawLinesFromBufferObjects(int contextIdx, int32_t points, int32_t pointsOffset) override
Draw lines from buffer objects.
VkDescriptorPool descriptorPool2
void createBuffer(VkDeviceSize size, VkBufferUsageFlags usage, VkMemoryPropertyFlags properties, VkBuffer &buffer, VmaAllocation &allocation, VmaAllocationInfo &allocationInfo)
void createLinesRenderingPipeline(int contextIdx, program_type *program)
void disableBlending() override
Disables blending.
PFN_vkDestroySwapchainKHR fpDestroySwapchainKHR
void setProgramAttributeLocation(int32_t programId, int32_t location, const string &name) override
Bind attribute to a input location.
VkBool32 checkLayers(uint32_t checkCount, const char **checkNames, const vector< VkLayerProperties > &instanceLayers)
void setViewPort(int32_t width, int32_t height) override
Set up viewport parameter.
uint64_t framebufferPipelinesId
PFN_vkAcquireNextImageKHR fpAcquireNextImageKHR
bool isSupportingMultithreadedRendering() override
VkCullModeFlagBits cullMode
int32_t getTextureUnits() override
void bindSkinningVerticesResultBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning vertices result buffer object.
void setProgramUniformFloat(int contextIdx, int32_t uniformId, float value) override
Set up a float uniform value.
const string getRenderer() override
const Renderer_Statistics getStatistics() override
void uploadCubeMapSingleTexture(int contextIdx, texture_type *cubemapTextureType, Texture *texture, uint32_t baseArrayLayer)
void initialize() override
Initialize renderer.
void setVSync(bool vSync) override
Enable/Disable v-sync.
vector< int32_t > disposeBuffers
int32_t createGBufferColorTexture(int32_t width, int32_t height) override
Creates a geometry buffer color RGBA texture.
PFN_vkCreateSwapchainKHR fpCreateSwapchainKHR
unordered_map< int32_t, shader_type * > shaders
vector< int32_t > freeTextureIds
uint16_t createPipelineIndex(program_type *program, int contextIdx)
void setProgramUniformFloatVec3(int contextIdx, int32_t uniformId, const array< float, 3 > &data) override
Set up a float vec3 uniform value.
bool isSupportingIntegerProgramAttributes() override
static constexpr int SHADERS_COMPUTE_MAX
bool isBufferObjectsAvailable() override
Checks if buffer objects is available.
void disposeFrameBufferObject(int32_t frameBufferId) override
Disposes a frame buffer object.
const string getVendor() override
void uploadBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, FloatBuffer *data) override
Uploads buffer data to buffer object.
void uploadBufferObjectInternal(int contextIdx, buffer_object_type *buffer, int32_t size, const uint8_t *data, VkBufferUsageFlagBits usage)
void setProgramUniformInternal(int contextIdx, int32_t uniformId, uint8_t *data, int32_t size)
void memoryBarrier() override
Memory barrier.
void createBufferTexture(int32_t textureId, int32_t width, int32_t height, int32_t cubeMapTextureId, int32_t cubeMapTextureIndex, VkFormat format)
void setupSkinningComputingPipeline(int contextIdx, program_type *program)
void setProgramUniformInteger(int contextIdx, int32_t uniformId, int32_t value) override
Set up a integer uniform value.
int32_t getProgramUniformLocation(int32_t programId, const string &name) override
Returns location of given uniform variable.
void disableDepthBufferWriting() override
Disable depth buffer writing.
SpinLock pipelinesSpinLock
void createRenderProgram(program_type *program)
void setImageLayout(int contextIdx, texture_type *textureObject, const array< ThsvsAccessType, 2 > &nextAccessTypes, ThsvsImageLayout nextLayout, bool discardContent, uint32_t baseMipLevel=0, uint32_t levelCount=1, bool submit=true)
void bindSkinningVertexJointIdxsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning vertex joint indices buffer object.
void invalidatePipelines()
void applyImageLayoutChange(int contextIdx, const image_layout_change &imageLayoutChange, texture_type *textureObject, bool submit=true)
void bindNormalsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind normals buffer object.
int whiteTextureSampler2dDefaultId
void drawPointsFromBufferObjects(int contextIdx, int32_t points, int32_t pointsOffset) override
Draw points from buffer objects.
void vmaMemCpy(VmaAllocation allocationDst, const uint8_t *src, uint32_t size, uint32_t offset=0)
void disposeTexture(int32_t textureId) override
Dispose a texture.
int32_t createColorBufferTexture(int32_t width, int32_t height, int32_t cubeMapTextureId, int32_t cubeMapTextureIndex) override
Creates a color buffer texture.
void disableCulling(int contextIdx) override
Disable culling.
VkBuffer getBindBufferObjectInternal(int32_t bufferObjectId, uint32_t &size)
PFN_vkGetPhysicalDeviceSurfaceCapabilitiesKHR fpGetPhysicalDeviceSurfaceCapabilitiesKHR
static constexpr int SHADERS_MAX
void bindEffectColorMulsBufferObject(int contextIdx, int32_t bufferObjectId, int32_t divisor) override
Bind effect color muls buffer object.
int32_t createCubeMapTexture(int contextIdx, int32_t width, int32_t height) override
Create cube map texture from frame buffers.
static constexpr int COMMANDS_MAX_GRAPHICS
void bindCubeMapTexture(int contextIdx, int32_t textureId) override
Binds a cube map texture with given id or unbinds when using ID_NONE.
bool isPBRAvailable() override
VkSemaphore drawCompleteSemaphore
framebuffer_pipelines_type * framebufferPipelinesCache
void setProgramUniformFloatVec4(int contextIdx, int32_t uniformId, const array< float, 4 > &data) override
Set up a float vec4 uniform value.
void setDepthFunction(int32_t depthFunction) override
Set up depth function.
void uploadCubeMapTexture(int contextIdx, Texture *textureLeft, Texture *textureRight, Texture *textureTop, Texture *textureBottom, Texture *textureFront, Texture *textureBack) override
Uploads cube map texture data to current bound texture.
uint32_t graphicsQueueNodeIndex
void bindFrameBuffer(int32_t frameBufferId) override
Enables a framebuffer to be rendered.
static constexpr bool VERBOSE
void initializeRenderPass()
int32_t loadShader(int32_t type, const string &pathName, const string &fileName, const string &definitions=string(), const string &functions=string()) override
Loads a shader.
void initializeFrameBuffers()
bool isNormalMappingAvailable() override
static constexpr int DRAW_COMMANDBUFFER_MAX
vector< framebuffer_pipelines_type * > framebuffersPipelines
void createDepthStencilStateCreateInfo(VkPipelineDepthStencilStateCreateInfo &depthStencilStateCreateInfo)
bool isDepthTextureAvailable() override
Checks if depth texture is available.
void resizeDepthBufferTexture(int32_t textureId, int32_t width, int32_t height) override
Resizes a depth texture.
void resizeGBufferGeometryTexture(int32_t textureId, int32_t width, int32_t height) override
Resizes a geometry buffer geometry texture.
int whiteTextureSamplerCubeDefaultId
uint64_t createPipelineFramebufferId()
void finishFrame() override
Finish frame.
bool isInstancedRenderingAvailable() override
Checks if instanced rendering is available.
void drawTrianglesFromBufferObjects(int contextIdx, int32_t triangles, int32_t trianglesOffset) override
Draw triangles from buffer objects.
PFN_vkQueuePresentKHR fpQueuePresentKHR
vector< context_type > contexts
VmaAllocator vmaAllocator
static constexpr int TEXTUREUNITS_MAX
void endRenderPass(int contextIdx)
void startRenderPass(int contextIdx)
void createDepthBufferTexture(int32_t textureId, int32_t width, int32_t height, int32_t cubeMapTextureId, int32_t cubeMapTextureIndex)
bool isSpecularMappingAvailable() override
void setImageLayout3(int contextIdx, VkImage image, VkImageAspectFlags aspectMask, const array< ThsvsAccessType, 2 > &accessTypes, const array< ThsvsAccessType, 2 > &nextAccessTypes, ThsvsImageLayout layout, ThsvsImageLayout nextLayout)
void initGuiMode() override
Set up renderer for GUI rendering.
framebuffer_pipelines_type * createFramebufferPipelines(uint64_t framebufferPipelinesId)
void applyImageLayoutChanges(int contextIdx, const array< image_layout_change, 8 > imageLayoutChanges, array< texture_type *, 8 > textureObjects, bool submit=true)
void finishSetupCommandBuffer(int contextIdx)
void bindSkinningVertexJointsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning vertex joints buffer object.
void setCullFace(int32_t cullFace) override
Sets up which face will be culled.
vector< int32_t > createBufferObjects(int32_t bufferCount, bool useGPUMemory, bool shared) override
Generate buffer objects for vertex data and such.
void enableBlending() override
Enables blending.
void setupPointsRenderingPipeline(int contextIdx, program_type *program)
static constexpr int COMPUTE_STORAGE_BUFFER_COUNT
VkPresentModeKHR lastSwapchainPresentMode
void setupLinesRenderingPipeline(int contextIdx, program_type *program)
void createObjectsRenderingPipeline(int contextIdx, program_type *program)
void uploadIndicesBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, ShortBuffer *data) override
Uploads buffer data to buffer object.
VkPhysicalDeviceFeatures gpuFeatures
void drawInstancedTrianglesFromBufferObjects(int contextIdx, int32_t triangles, int32_t trianglesOffset, VkBuffer indicesBuffer, int32_t instances)
bool linkProgram(int32_t programId) override
Links attached shaders to a program.
void setProgramUniformFloatMatrix4x4(int contextIdx, int32_t uniformId, const array< float, 16 > &data) override
Set up a float matrix 4x4 uniform value.
void enableCulling(int contextIdx) override
Enable culling.
static constexpr int CUBEMAPTEXTUREINDEX_MIN
int32_t boundFrameBufferId
vector< delete_image_type > deleteImages
PFN_vkGetPhysicalDeviceSurfacePresentModesKHR fpGetPhysicalDeviceSurfacePresentModesKHR
uint32_t windowSwapchainImageCount
vector< program_type * > programVector
bool beginDrawCommandBuffer(int contextIdx, int bufferId=-1)
void endDrawCommandsAllContexts()
array< buffer_object_type *, BUFFERS_MAX+1 > buffers
void bindOriginsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind origins buffer object.
static constexpr int SHADERSSTAGES_MAX
int32_t createProgram(int type) override
Creates a shader program.
static constexpr int LINES_VERTEX_BUFFER_COUNT
void bindIndicesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind indices buffer object.
VkDescriptorPool descriptorPool1
bool isComputeShaderAvailable() override
static constexpr int DESC_MAX_CACHED
void initializeFrame() override
Pre Frame Initialization.
int32_t createGBufferGeometryTexture(int32_t width, int32_t height) override
Creates a geometry buffer geometry texture.
void uploadSkinningBufferObject(int contextIdx, int32_t bufferObjectId, int32_t size, FloatBuffer *data) override
Upload skinning buffer object.
static constexpr int BUFFERS_MAX
void enableAdditionBlending() override
Enable blending with c = a + b.
framebuffer_pipelines_type * getFramebufferPipelines(uint64_t framebufferPipelinesId)
void bindVerticesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind vertices buffer object.
texture_type * getBindTextureInternal(int32_t textureId)
vector< int32_t > freeBufferIds
void createRasterizationStateCreateInfo(int contextIdx, VkPipelineRasterizationStateCreateInfo &rasterizationStateCreateInfo)
void drawInstancedIndexedTrianglesFromBufferObjects(int contextIdx, int32_t triangles, int32_t trianglesOffset, int32_t instances) override
Draw instanced indexed triangles from buffer objects.
VkPhysicalDevice physicalDevice
BlendingMode blendingMode
PFN_vkGetSwapchainImagesKHR fpGetSwapchainImagesKHR
void useProgram(int contextIdx, int32_t programId) override
Use shader program.
void bindColorsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind colors buffer object.
VkPresentModeKHR swapchainPresentMode
void bindPointSizesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind point sizes buffer object.
void bindSkinningVerticesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning vertices buffer object.
void createColorBlendAttachmentState(VkPipelineColorBlendAttachmentState &blendAttachmentState)
void bindBitangentsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind bitangents buffer object.
vector< VkPipeline > disposePipelines
void bindSkinningNormalsResultBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning normals result buffer object.
void setProgramUniformFloatMatrices4x4(int contextIdx, int32_t uniformId, int32_t count, FloatBuffer *data) override
Set up a float matrices 4x4 uniform values.
void resizeGBufferColorTexture(int32_t textureId, int32_t width, int32_t height) override
Resizes a geometry buffer color RGBA texture.
void bindSkinningMatricesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning matrices result buffer object.
void initializeSwapChain()
int32_t createGeometryBufferObject(int32_t depthBufferTextureId, int32_t geometryBufferTextureId1, int32_t geometryBufferTextureId2, int32_t geometryBufferTextureId3, int32_t colorBufferTextureId1, int32_t colorBufferTextureId2, int32_t colorBufferTextureId3, int32_t colorBufferTextureId4, int32_t colorBufferTextureId5) override
Creates a geometry frame buffer object.
vector< window_frambuffer_buffer_type > windowFramebufferBuffers
void updateViewPort() override
Update viewport.
vector< VkFence > contextsDrawFences
void requestSubmitDrawBuffers(int contextIdx)
void createFramebufferObject(int32_t frameBufferId)
VkPipeline getPipelineInternal(int contextIdx, program_type *programm, uint64_t framebuffePipelineId, uint32_t pipelineIdx)
uint32_t getMipLevels(Texture *texture)
void bindModelMatricesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind model matrices buffer object.
void createPointsRenderingPipeline(int contextIdx, program_type *program)
VkSemaphore imageAcquiredSemaphore
void drawIndexedTrianglesFromBufferObjects(int contextIdx, int32_t triangles, int32_t trianglesOffset) override
Draw indexed triangles from buffer objects.
void disableDepthBufferTest() override
Disable depth buffer test.
void bindSkinningNormalsBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind skinning normal buffer object.
void unbindBufferObjects(int contextIdx) override
Unbind buffer objects.
array< texture_type *, TEXTURES_MAX+1 > textures
uint32_t lastWindowFramebufferIdx
static constexpr int TEXTURES_MAX
VkPhysicalDeviceMemoryProperties memoryProperties
texture_type * getTextureInternal(int32_t textureId)
void bindEffectColorAddsBufferObject(int contextIdx, int32_t bufferObjectIdd, int32_t divisor) override
Bind effect color adds buffer object.
int32_t createTexture() override
Creates a texture.
vector< int32_t > disposeTextures
VkQueueFamilyProperties * queueProperties
float readPixelDepth(int32_t x, int32_t y) override
Reads a pixel depth.
void uploadTexture(int contextIdx, Texture *texture) override
Uploads texture data to current bound texture.
void getImageLayoutChange(image_layout_change &imageLayoutChange, texture_type *textureObject, const array< ThsvsAccessType, 2 > &prevAccessTypes, const array< ThsvsAccessType, 2 > &nextAccessTypes, ThsvsImageLayout prevLayout, ThsvsImageLayout nextLayout, bool discardContent, uint32_t baseMipLevel=0, uint32_t levelCount=1)
VkCommandBuffer endDrawCommandBuffer(int contextIdx, int bufferId=-1, bool cycleBuffers=true)
texture_type * whiteTextureSampler2dDefault
bool isUsingShortIndices() override
void setLineWidth(float lineWidth) override
Set line width.
ByteBuffer * readPixels(int32_t x, int32_t y, int32_t width, int32_t height) override
Read pixels.
void enableDepthBufferTest() override
Enable depth buffer test.
void resizeColorBufferTexture(int32_t textureId, int32_t width, int32_t height) override
Resize color buffer texture.
static constexpr int OBJECTS_VERTEX_BUFFER_COUNT
void bindTextureSpriteIndicesBufferObject(int contextIdx, int32_t bufferObjectId) override
Bind texture and sprite indices buffer object.
void setProgramUniformFloatVec2(int contextIdx, int32_t uniformId, const array< float, 2 > &data) override
Set up a float vec2 uniform value.
void finishSetupCommandBuffers()
bool isDeferredShadingAvailable() override
PFN_vkGetPhysicalDeviceSurfaceFormatsKHR fpGetPhysicalDeviceSurfaceFormatsKHR
buffer_object_type * getBufferObjectInternal(int32_t bufferObjectId)
void prepareSetupCommandBuffer(int contextIdx)
texture_type * whiteTextureSamplerCubeDefault
PFN_vkGetPhysicalDeviceSurfaceSupportKHR fpGetPhysicalDeviceSurfaceSupportKHR
vector< framebuffer_object_type * > framebuffers
bool isUsingProgramAttributeLocation() override
File system singleton class.
void unlock()
Unlocks this mutex.
void lock()
Locks the mutex, additionally mutex locks will block until other locks have been unlocked.
Implementation for read/write lock.
void unlock()
Unlocks this spin lock.
void lock()
Locks the spin lock, additionally spin lock locks will block until other locks have been unlocked.
uint8_t get(int32_t position)
Run time type information utility class.
std::exception Exception
Exception base class.
Bean holding light properties.
uint32_t disposedTextures
int64_t memoryUsageShared
reusable_buffer * bindBuffer
vector< reusable_buffer * > frameFreeBuffers
reusable_buffer * currentBuffer
list< reusable_buffer > buffers
VkPipelineStageFlags dstStages
VkPipelineStageFlags srcStages
VkImageMemoryBarrier vkImageMemoryBarrier
array< ThsvsAccessType, 2 > accessTypes
ThsvsImageLayout svsLayout
vector< context > contexts
vector< shader_type * > shaders
VkPipelineLayout pipelineLayout
VkDescriptorSetLayout texturesDescriptorSetLayout
VkDescriptorSetLayout uboDescriptorSetLayout
VkImageAspectFlags aspectMask
array< array< ThsvsAccessType, 2 >, 6 > accessTypes
int32_t frameBufferObjectId
int32_t cubemapTextureIndex
texture_type * cubemapBufferTexture
image_layout_change frameBufferUnbindImageLayoutChange
ThsvsImageLayout svsLayout
image_layout_change frameBufferBindImageLayoutChange