2023-09-17 17:01:29 +00:00
|
|
|
#include "vulkan_util.h"
|
2023-10-26 20:29:05 +00:00
|
|
|
#include <fmt/core.h>
|
2023-09-27 19:47:53 +00:00
|
|
|
#include <Core/PS4/GPU/gpu_memory.h>
|
2023-09-18 11:52:16 +00:00
|
|
|
#include <SDL_vulkan.h>
|
2023-10-15 07:03:26 +00:00
|
|
|
#include <Emulator/Util/singleton.h>
|
2023-09-17 17:01:29 +00:00
|
|
|
#include <Util/log.h>
|
|
|
|
#include <debug.h>
|
2023-09-19 06:15:59 +00:00
|
|
|
#include <vulkan/vk_enum_string_helper.h>
|
2023-09-20 11:19:20 +00:00
|
|
|
#include <vulkan/vulkan_core.h>
|
2023-09-21 21:05:00 +00:00
|
|
|
|
|
|
|
#include <algorithm>
|
2023-09-17 17:01:29 +00:00
|
|
|
|
|
|
|
constexpr bool log_file_vulkanutil = true; // disable it to disable logging
|
|
|
|
|
2023-10-13 06:40:59 +00:00
|
|
|
void Graphics::Vulkan::vulkanCreate(Emu::WindowCtx* ctx) {
|
|
|
|
Emu::VulkanExt ext;
|
2023-09-18 11:52:16 +00:00
|
|
|
vulkanGetInstanceExtensions(&ext);
|
2023-09-17 17:01:29 +00:00
|
|
|
|
|
|
|
VkApplicationInfo app_info{};
|
|
|
|
app_info.sType = VK_STRUCTURE_TYPE_APPLICATION_INFO;
|
|
|
|
app_info.pNext = nullptr;
|
|
|
|
app_info.pApplicationName = "shadps4";
|
|
|
|
app_info.applicationVersion = 1;
|
|
|
|
app_info.pEngineName = "shadps4";
|
|
|
|
app_info.engineVersion = 1;
|
|
|
|
app_info.apiVersion = VK_API_VERSION_1_2;
|
2023-09-17 20:38:16 +00:00
|
|
|
|
|
|
|
VkInstanceCreateInfo inst_info{};
|
|
|
|
inst_info.sType = VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO;
|
|
|
|
inst_info.pNext = nullptr;
|
|
|
|
inst_info.flags = 0;
|
|
|
|
inst_info.pApplicationInfo = &app_info;
|
|
|
|
inst_info.enabledExtensionCount = ext.required_extensions.size();
|
|
|
|
inst_info.ppEnabledExtensionNames = ext.required_extensions.data();
|
|
|
|
inst_info.enabledLayerCount = 0;
|
|
|
|
inst_info.ppEnabledLayerNames = nullptr;
|
|
|
|
|
|
|
|
VkResult result = vkCreateInstance(&inst_info, nullptr, &ctx->m_graphic_ctx.m_instance);
|
|
|
|
if (result == VK_ERROR_INCOMPATIBLE_DRIVER) {
|
|
|
|
LOG_CRITICAL_IF(log_file_vulkanutil, "Can't find an compatiblie vulkan driver\n");
|
|
|
|
std::exit(0);
|
|
|
|
} else if (result != VK_SUCCESS) {
|
|
|
|
LOG_CRITICAL_IF(log_file_vulkanutil, "Can't create an vulkan instance\n");
|
|
|
|
std::exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (SDL_Vulkan_CreateSurface(ctx->m_window, ctx->m_graphic_ctx.m_instance, &ctx->m_surface) == SDL_FALSE) {
|
|
|
|
LOG_CRITICAL_IF(log_file_vulkanutil, "Can't create an vulkan surface\n");
|
|
|
|
std::exit(0);
|
|
|
|
}
|
2023-09-18 11:52:16 +00:00
|
|
|
|
|
|
|
// TODO i am not sure if it's that it is neccesary or if it needs more
|
|
|
|
std::vector<const char*> device_extensions = {VK_KHR_SWAPCHAIN_EXTENSION_NAME, VK_EXT_DEPTH_CLIP_ENABLE_EXTENSION_NAME,
|
|
|
|
VK_EXT_COLOR_WRITE_ENABLE_EXTENSION_NAME, "VK_KHR_maintenance1"};
|
|
|
|
|
2023-10-13 06:40:59 +00:00
|
|
|
ctx->m_surface_capabilities = new Emu::VulkanSurfaceCapabilities{};
|
|
|
|
Emu::VulkanQueues queues;
|
2023-09-18 11:52:16 +00:00
|
|
|
|
|
|
|
vulkanFindCompatiblePhysicalDevice(ctx->m_graphic_ctx.m_instance, ctx->m_surface, device_extensions, ctx->m_surface_capabilities,
|
2023-09-21 09:59:48 +00:00
|
|
|
&ctx->m_graphic_ctx.m_physical_device, &queues);
|
|
|
|
|
|
|
|
if (ctx->m_graphic_ctx.m_physical_device == nullptr) {
|
|
|
|
LOG_CRITICAL_IF(log_file_vulkanutil, "Can't find compatible vulkan device\n");
|
|
|
|
std::exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
VkPhysicalDeviceProperties device_properties{};
|
|
|
|
vkGetPhysicalDeviceProperties(ctx->m_graphic_ctx.m_physical_device, &device_properties);
|
|
|
|
|
2023-09-21 20:48:16 +00:00
|
|
|
LOG_INFO_IF(log_file_vulkanutil, "GFX device to be used : {}\n", device_properties.deviceName);
|
2023-09-21 09:59:48 +00:00
|
|
|
|
2023-09-21 20:48:16 +00:00
|
|
|
ctx->m_graphic_ctx.m_device = vulkanCreateDevice(ctx->m_graphic_ctx.m_physical_device, ctx->m_surface, &ext, queues, device_extensions);
|
|
|
|
if (ctx->m_graphic_ctx.m_device == nullptr) {
|
2023-09-21 09:59:48 +00:00
|
|
|
LOG_CRITICAL_IF(log_file_vulkanutil, "Can't create vulkan device\n");
|
|
|
|
std::exit(0);
|
2023-09-21 20:48:16 +00:00
|
|
|
}
|
2023-09-21 15:20:13 +00:00
|
|
|
|
2023-09-21 20:48:16 +00:00
|
|
|
vulkanCreateQueues(&ctx->m_graphic_ctx, queues);
|
|
|
|
ctx->swapchain = vulkanCreateSwapchain(&ctx->m_graphic_ctx, 2);
|
2023-09-21 15:20:13 +00:00
|
|
|
}
|
|
|
|
|
2023-10-13 06:40:59 +00:00
|
|
|
Emu::VulkanSwapchain* Graphics::Vulkan::vulkanCreateSwapchain(HLE::Libs::Graphics::GraphicCtx* ctx, u32 image_count) {
|
2023-10-15 07:03:26 +00:00
|
|
|
auto* window_ctx = singleton<Emu::WindowCtx>::instance();
|
2023-10-13 06:40:59 +00:00
|
|
|
auto* s = new Emu::VulkanSwapchain;
|
2023-09-21 20:48:16 +00:00
|
|
|
|
|
|
|
VkExtent2D extent{};
|
2023-09-21 21:05:00 +00:00
|
|
|
extent.width = clamp(ctx->screen_width, window_ctx->m_surface_capabilities->capabilities.minImageExtent.width,
|
|
|
|
window_ctx->m_surface_capabilities->capabilities.maxImageExtent.width);
|
|
|
|
extent.height = clamp(ctx->screen_height, window_ctx->m_surface_capabilities->capabilities.minImageExtent.height,
|
|
|
|
window_ctx->m_surface_capabilities->capabilities.maxImageExtent.height);
|
2023-09-21 20:48:16 +00:00
|
|
|
|
2023-09-21 21:05:00 +00:00
|
|
|
image_count = clamp(image_count, window_ctx->m_surface_capabilities->capabilities.minImageCount,
|
|
|
|
window_ctx->m_surface_capabilities->capabilities.maxImageCount);
|
2023-09-21 20:48:16 +00:00
|
|
|
|
|
|
|
VkSwapchainCreateInfoKHR create_info{};
|
|
|
|
create_info.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
|
|
|
|
create_info.pNext = nullptr;
|
|
|
|
create_info.flags = 0;
|
|
|
|
create_info.surface = window_ctx->m_surface;
|
|
|
|
create_info.minImageCount = image_count;
|
|
|
|
|
|
|
|
if (window_ctx->m_surface_capabilities->is_format_unorm_bgra32) {
|
|
|
|
create_info.imageFormat = VK_FORMAT_B8G8R8A8_UNORM;
|
|
|
|
create_info.imageColorSpace = VK_COLOR_SPACE_SRGB_NONLINEAR_KHR;
|
|
|
|
} else if (window_ctx->m_surface_capabilities->is_format_srgb_bgra32) {
|
|
|
|
create_info.imageFormat = VK_FORMAT_B8G8R8A8_SRGB;
|
|
|
|
create_info.imageColorSpace = VK_COLOR_SPACE_SRGB_NONLINEAR_KHR;
|
|
|
|
} else {
|
|
|
|
create_info.imageFormat = window_ctx->m_surface_capabilities->formats.at(0).format;
|
|
|
|
create_info.imageColorSpace = window_ctx->m_surface_capabilities->formats.at(0).colorSpace;
|
|
|
|
}
|
|
|
|
|
|
|
|
create_info.imageExtent = extent;
|
|
|
|
create_info.imageArrayLayers = 1;
|
|
|
|
create_info.imageUsage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT;
|
|
|
|
create_info.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
|
|
|
|
create_info.queueFamilyIndexCount = 0;
|
|
|
|
create_info.pQueueFamilyIndices = nullptr;
|
|
|
|
create_info.preTransform = window_ctx->m_surface_capabilities->capabilities.currentTransform;
|
|
|
|
create_info.compositeAlpha = VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR;
|
|
|
|
create_info.presentMode = VK_PRESENT_MODE_FIFO_KHR;
|
|
|
|
create_info.clipped = VK_TRUE;
|
|
|
|
create_info.oldSwapchain = nullptr;
|
|
|
|
|
|
|
|
s->swapchain_format = create_info.imageFormat;
|
|
|
|
s->swapchain_extent = extent;
|
|
|
|
|
2023-09-21 21:05:00 +00:00
|
|
|
vkCreateSwapchainKHR(ctx->m_device, &create_info, nullptr, &s->swapchain);
|
2023-09-21 20:48:16 +00:00
|
|
|
|
2023-09-21 21:05:00 +00:00
|
|
|
vkGetSwapchainImagesKHR(ctx->m_device, s->swapchain, &s->swapchain_images_count, nullptr);
|
2023-09-21 20:48:16 +00:00
|
|
|
|
|
|
|
s->swapchain_images = new VkImage[s->swapchain_images_count];
|
2023-09-21 21:05:00 +00:00
|
|
|
vkGetSwapchainImagesKHR(ctx->m_device, s->swapchain, &s->swapchain_images_count, s->swapchain_images);
|
2023-09-21 20:48:16 +00:00
|
|
|
|
|
|
|
s->swapchain_image_views = new VkImageView[s->swapchain_images_count];
|
|
|
|
for (uint32_t i = 0; i < s->swapchain_images_count; i++) {
|
|
|
|
VkImageViewCreateInfo create_info{};
|
|
|
|
create_info.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
|
|
|
|
create_info.pNext = nullptr;
|
|
|
|
create_info.flags = 0;
|
|
|
|
create_info.image = (s->swapchain_images)[i];
|
|
|
|
create_info.viewType = VK_IMAGE_VIEW_TYPE_2D;
|
|
|
|
create_info.format = s->swapchain_format;
|
|
|
|
create_info.components.r = VK_COMPONENT_SWIZZLE_IDENTITY;
|
|
|
|
create_info.components.g = VK_COMPONENT_SWIZZLE_IDENTITY;
|
|
|
|
create_info.components.b = VK_COMPONENT_SWIZZLE_IDENTITY;
|
|
|
|
create_info.components.a = VK_COMPONENT_SWIZZLE_IDENTITY;
|
|
|
|
create_info.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
|
|
|
|
create_info.subresourceRange.baseArrayLayer = 0;
|
|
|
|
create_info.subresourceRange.baseMipLevel = 0;
|
|
|
|
create_info.subresourceRange.layerCount = 1;
|
|
|
|
create_info.subresourceRange.levelCount = 1;
|
|
|
|
|
|
|
|
vkCreateImageView(ctx->m_device, &create_info, nullptr, &((s->swapchain_image_views)[i]));
|
|
|
|
}
|
|
|
|
if (s->swapchain == nullptr) {
|
|
|
|
LOG_CRITICAL_IF(log_file_vulkanutil, "Could not create swapchain\n");
|
|
|
|
std::exit(0);
|
|
|
|
}
|
2023-09-21 15:20:13 +00:00
|
|
|
|
2023-09-21 20:48:16 +00:00
|
|
|
s->current_index = static_cast<uint32_t>(-1);
|
2023-09-21 15:20:13 +00:00
|
|
|
|
2023-09-21 20:48:16 +00:00
|
|
|
VkSemaphoreCreateInfo present_complete_info{};
|
|
|
|
present_complete_info.sType = VK_STRUCTURE_TYPE_SEMAPHORE_CREATE_INFO;
|
|
|
|
present_complete_info.pNext = nullptr;
|
|
|
|
present_complete_info.flags = 0;
|
2023-09-21 15:20:13 +00:00
|
|
|
|
2023-09-21 20:48:16 +00:00
|
|
|
auto result = vkCreateSemaphore(ctx->m_device, &present_complete_info, nullptr, &s->present_complete_semaphore);
|
|
|
|
|
|
|
|
VkFenceCreateInfo fence_info{};
|
|
|
|
fence_info.sType = VK_STRUCTURE_TYPE_FENCE_CREATE_INFO;
|
|
|
|
fence_info.pNext = nullptr;
|
|
|
|
fence_info.flags = 0;
|
|
|
|
|
|
|
|
result = vkCreateFence(ctx->m_device, &fence_info, nullptr, &s->present_complete_fence);
|
|
|
|
if (result != VK_SUCCESS) {
|
|
|
|
LOG_CRITICAL_IF(log_file_vulkanutil, "Can't create vulkan fence\n");
|
|
|
|
std::exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
2023-09-27 19:47:53 +00:00
|
|
|
|
2023-10-13 06:40:59 +00:00
|
|
|
void Graphics::Vulkan::vulkanCreateQueues(HLE::Libs::Graphics::GraphicCtx* ctx, const Emu::VulkanQueues& queues) {
|
|
|
|
auto get_queue = [ctx](int id, const Emu::VulkanQueueInfo& info, bool with_mutex = false) {
|
2023-09-21 20:48:16 +00:00
|
|
|
ctx->queues[id].family = info.family;
|
|
|
|
ctx->queues[id].index = info.index;
|
|
|
|
vkGetDeviceQueue(ctx->m_device, ctx->queues[id].family, ctx->queues[id].index, &ctx->queues[id].vk_queue);
|
|
|
|
if (with_mutex) {
|
2023-10-22 14:10:25 +00:00
|
|
|
ctx->queues[id].mutex = new std::mutex;
|
2023-09-21 20:48:16 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
get_queue(VULKAN_QUEUE_GFX, queues.graphics.at(0));
|
|
|
|
get_queue(VULKAN_QUEUE_UTIL, queues.transfer.at(0));
|
|
|
|
get_queue(VULKAN_QUEUE_PRESENT, queues.present.at(0));
|
|
|
|
|
|
|
|
for (int id = 0; id < VULKAN_QUEUE_COMPUTE_NUM; id++) {
|
2023-09-21 15:20:13 +00:00
|
|
|
bool with_mutex = (VULKAN_QUEUE_COMPUTE_NUM == queues.compute.size());
|
|
|
|
get_queue(id, queues.compute.at(id % queues.compute.size()), with_mutex);
|
2023-09-21 20:48:16 +00:00
|
|
|
}
|
2023-09-17 17:01:29 +00:00
|
|
|
}
|
|
|
|
|
2023-10-13 06:40:59 +00:00
|
|
|
VkDevice Graphics::Vulkan::vulkanCreateDevice(VkPhysicalDevice physical_device, VkSurfaceKHR surface, const Emu::VulkanExt* r,
|
|
|
|
const Emu::VulkanQueues& queues, const std::vector<const char*>& device_extensions) {
|
2023-09-21 20:48:16 +00:00
|
|
|
std::vector<VkDeviceQueueCreateInfo> queue_create_info(queues.family_count);
|
|
|
|
std::vector<std::vector<float>> queue_priority(queues.family_count);
|
|
|
|
uint32_t queue_create_info_num = 0;
|
2023-09-21 09:59:48 +00:00
|
|
|
|
2023-09-21 20:48:16 +00:00
|
|
|
for (uint32_t i = 0; i < queues.family_count; i++) {
|
2023-09-21 09:59:48 +00:00
|
|
|
if (queues.family_used[i] != 0) {
|
|
|
|
for (uint32_t pi = 0; pi < queues.family_used[i]; pi++) {
|
|
|
|
queue_priority[queue_create_info_num].push_back(1.0f);
|
|
|
|
}
|
|
|
|
|
|
|
|
queue_create_info[queue_create_info_num].sType = VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO;
|
|
|
|
queue_create_info[queue_create_info_num].pNext = nullptr;
|
|
|
|
queue_create_info[queue_create_info_num].flags = 0;
|
|
|
|
queue_create_info[queue_create_info_num].queueFamilyIndex = i;
|
|
|
|
queue_create_info[queue_create_info_num].queueCount = queues.family_used[i];
|
|
|
|
queue_create_info[queue_create_info_num].pQueuePriorities = queue_priority[queue_create_info_num].data();
|
|
|
|
|
|
|
|
queue_create_info_num++;
|
|
|
|
}
|
2023-09-21 20:48:16 +00:00
|
|
|
}
|
2023-09-21 09:59:48 +00:00
|
|
|
|
2023-09-21 20:48:16 +00:00
|
|
|
VkPhysicalDeviceFeatures device_features{};
|
|
|
|
// TODO add neccesary device features
|
2023-09-21 09:59:48 +00:00
|
|
|
|
2023-09-21 20:48:16 +00:00
|
|
|
VkDeviceCreateInfo create_info{};
|
|
|
|
create_info.sType = VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO;
|
|
|
|
create_info.pNext = nullptr;
|
|
|
|
create_info.flags = 0;
|
|
|
|
create_info.pQueueCreateInfos = queue_create_info.data();
|
|
|
|
create_info.queueCreateInfoCount = queue_create_info_num;
|
|
|
|
create_info.enabledLayerCount = 0;
|
|
|
|
create_info.ppEnabledLayerNames = nullptr;
|
|
|
|
create_info.enabledExtensionCount = device_extensions.size();
|
|
|
|
create_info.ppEnabledExtensionNames = device_extensions.data();
|
|
|
|
create_info.pEnabledFeatures = &device_features;
|
2023-09-21 09:59:48 +00:00
|
|
|
|
2023-09-21 20:48:16 +00:00
|
|
|
VkDevice device = nullptr;
|
2023-09-21 09:59:48 +00:00
|
|
|
|
2023-09-21 20:48:16 +00:00
|
|
|
vkCreateDevice(physical_device, &create_info, nullptr, &device);
|
2023-09-21 09:59:48 +00:00
|
|
|
|
2023-09-21 20:48:16 +00:00
|
|
|
return device;
|
2023-09-21 09:59:48 +00:00
|
|
|
}
|
2023-10-13 06:40:59 +00:00
|
|
|
void Graphics::Vulkan::vulkanGetInstanceExtensions(Emu::VulkanExt* ext) {
|
2023-09-17 17:01:29 +00:00
|
|
|
u32 required_extensions_count = 0;
|
|
|
|
u32 available_extensions_count = 0;
|
|
|
|
u32 available_layers_count = 0;
|
|
|
|
auto result = SDL_Vulkan_GetInstanceExtensions(&required_extensions_count, nullptr);
|
|
|
|
|
|
|
|
ext->required_extensions = std::vector<const char*>(required_extensions_count);
|
|
|
|
|
|
|
|
result = SDL_Vulkan_GetInstanceExtensions(&required_extensions_count, ext->required_extensions.data());
|
|
|
|
|
|
|
|
vkEnumerateInstanceExtensionProperties(nullptr, &available_extensions_count, nullptr);
|
|
|
|
|
|
|
|
ext->available_extensions = std::vector<VkExtensionProperties>(available_extensions_count);
|
|
|
|
|
|
|
|
vkEnumerateInstanceExtensionProperties(nullptr, &available_extensions_count, ext->available_extensions.data());
|
|
|
|
|
|
|
|
vkEnumerateInstanceLayerProperties(&available_layers_count, nullptr);
|
|
|
|
ext->available_layers = std::vector<VkLayerProperties>(available_layers_count);
|
|
|
|
vkEnumerateInstanceLayerProperties(&available_layers_count, ext->available_layers.data());
|
|
|
|
|
|
|
|
for (const char* ext : ext->required_extensions) {
|
|
|
|
LOG_INFO_IF(log_file_vulkanutil, "Vulkan required extension = {}\n", ext);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto& ext : ext->available_extensions) {
|
|
|
|
LOG_INFO_IF(log_file_vulkanutil, "Vulkan available extension: {}, version = {}\n", ext.extensionName, ext.specVersion);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto& l : ext->available_layers) {
|
2023-09-18 11:52:16 +00:00
|
|
|
LOG_INFO_IF(log_file_vulkanutil, "Vulkan available layer: {}, specVersion = {}, implVersion = {}, {}\n", l.layerName, l.specVersion,
|
|
|
|
l.implementationVersion, l.description);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Graphics::Vulkan::vulkanFindCompatiblePhysicalDevice(VkInstance instance, VkSurfaceKHR surface,
|
|
|
|
const std::vector<const char*>& device_extensions,
|
2023-10-13 06:40:59 +00:00
|
|
|
Emu::VulkanSurfaceCapabilities* out_capabilities, VkPhysicalDevice* out_device,
|
|
|
|
Emu::VulkanQueues* out_queues) {
|
2023-09-18 11:52:16 +00:00
|
|
|
u32 count_devices = 0;
|
|
|
|
vkEnumeratePhysicalDevices(instance, &count_devices, nullptr);
|
|
|
|
|
|
|
|
std::vector<VkPhysicalDevice> devices(count_devices);
|
|
|
|
vkEnumeratePhysicalDevices(instance, &count_devices, devices.data());
|
|
|
|
|
|
|
|
VkPhysicalDevice found_best_device = nullptr;
|
2023-10-13 06:40:59 +00:00
|
|
|
Emu::VulkanQueues found_best_queues;
|
2023-09-18 11:52:16 +00:00
|
|
|
|
|
|
|
for (const auto& device : devices) {
|
2023-09-19 06:15:59 +00:00
|
|
|
VkPhysicalDeviceProperties device_properties{};
|
|
|
|
VkPhysicalDeviceFeatures2 device_features2{};
|
|
|
|
|
|
|
|
vkGetPhysicalDeviceProperties(device, &device_properties);
|
|
|
|
vkGetPhysicalDeviceFeatures2(device, &device_features2);
|
|
|
|
if (device_properties.deviceType == VK_PHYSICAL_DEVICE_TYPE_INTEGRATED_GPU) {
|
|
|
|
continue; // we don't want integrated gpu for now .Later we will check the requirements and see what we can support (TODO fix me)
|
|
|
|
}
|
2023-09-20 11:19:20 +00:00
|
|
|
LOG_INFO_IF(log_file_vulkanutil, "Vulkan device: {}\n", device_properties.deviceName);
|
2023-09-19 06:15:59 +00:00
|
|
|
|
|
|
|
auto qs = vulkanFindQueues(device, surface);
|
2023-09-19 11:02:40 +00:00
|
|
|
|
|
|
|
vulkanGetSurfaceCapabilities(device, surface, out_capabilities);
|
2023-09-19 14:13:37 +00:00
|
|
|
|
|
|
|
found_best_device = device;
|
|
|
|
found_best_queues = qs;
|
2023-09-19 06:15:59 +00:00
|
|
|
}
|
2023-09-19 14:13:37 +00:00
|
|
|
*out_device = found_best_device;
|
|
|
|
*out_queues = found_best_queues;
|
2023-09-19 06:15:59 +00:00
|
|
|
}
|
|
|
|
|
2023-10-13 06:40:59 +00:00
|
|
|
Emu::VulkanQueues Graphics::Vulkan::vulkanFindQueues(VkPhysicalDevice device, VkSurfaceKHR surface) {
|
|
|
|
Emu::VulkanQueues qs;
|
2023-09-19 06:15:59 +00:00
|
|
|
|
|
|
|
u32 queue_family_count = 0;
|
|
|
|
vkGetPhysicalDeviceQueueFamilyProperties(device, &queue_family_count, nullptr);
|
|
|
|
std::vector<VkQueueFamilyProperties> queue_families(queue_family_count);
|
|
|
|
vkGetPhysicalDeviceQueueFamilyProperties(device, &queue_family_count, queue_families.data());
|
|
|
|
|
|
|
|
qs.family_count = queue_family_count;
|
|
|
|
|
|
|
|
u32 family = 0;
|
|
|
|
for (auto& f : queue_families) {
|
|
|
|
VkBool32 presentation_supported = VK_FALSE;
|
|
|
|
vkGetPhysicalDeviceSurfaceSupportKHR(device, family, surface, &presentation_supported);
|
|
|
|
|
|
|
|
LOG_INFO_IF(log_file_vulkanutil, "queue family: {}, count = {}, present = {}\n", string_VkQueueFlags(f.queueFlags).c_str(), f.queueCount,
|
2023-09-20 11:19:20 +00:00
|
|
|
(presentation_supported == VK_TRUE ? "true" : "false"));
|
|
|
|
for (uint32_t i = 0; i < f.queueCount; i++) {
|
2023-10-13 06:40:59 +00:00
|
|
|
Emu::VulkanQueueInfo info;
|
2023-09-20 11:19:20 +00:00
|
|
|
info.family = family;
|
|
|
|
info.index = i;
|
|
|
|
info.is_graphics = (f.queueFlags & VK_QUEUE_GRAPHICS_BIT) != 0;
|
|
|
|
info.is_compute = (f.queueFlags & VK_QUEUE_COMPUTE_BIT) != 0;
|
|
|
|
info.is_transfer = (f.queueFlags & VK_QUEUE_TRANSFER_BIT) != 0;
|
|
|
|
info.is_present = (presentation_supported == VK_TRUE);
|
|
|
|
|
|
|
|
qs.available.push_back(info);
|
|
|
|
}
|
|
|
|
|
|
|
|
qs.family_used.push_back(0);
|
2023-09-19 06:15:59 +00:00
|
|
|
|
|
|
|
family++;
|
2023-09-17 17:01:29 +00:00
|
|
|
}
|
2023-09-21 09:41:51 +00:00
|
|
|
u32 index = 0;
|
|
|
|
for (u32 i = 0; i < VULKAN_QUEUE_GRAPHICS_NUM; i++) {
|
|
|
|
for (const auto& idx : qs.available) {
|
2023-09-21 09:59:48 +00:00
|
|
|
if (idx.is_graphics) {
|
2023-09-21 09:41:51 +00:00
|
|
|
qs.family_used[qs.available.at(index).family]++;
|
|
|
|
qs.graphics.push_back(qs.available.at(index));
|
2023-09-21 09:59:48 +00:00
|
|
|
qs.available.erase(qs.available.begin() + index);
|
2023-09-21 09:41:51 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
index = 0;
|
|
|
|
for (u32 i = 0; i < VULKAN_QUEUE_COMPUTE_NUM; i++) {
|
|
|
|
for (const auto& idx : qs.available) {
|
|
|
|
if (idx.is_graphics) {
|
|
|
|
qs.family_used[qs.available.at(index).family]++;
|
|
|
|
qs.compute.push_back(qs.available.at(index));
|
|
|
|
qs.available.erase(qs.available.begin() + index);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
index = 0;
|
|
|
|
for (uint32_t i = 0; i < VULKAN_QUEUE_TRANSFER_NUM; i++) {
|
|
|
|
for (const auto& idx : qs.available) {
|
|
|
|
if (idx.is_graphics) {
|
|
|
|
qs.family_used[qs.available.at(index).family]++;
|
|
|
|
qs.transfer.push_back(qs.available.at(index));
|
|
|
|
qs.available.erase(qs.available.begin() + index);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
index = 0;
|
|
|
|
for (uint32_t i = 0; i < VULKAN_QUEUE_PRESENT_NUM; i++) {
|
|
|
|
for (const auto& idx : qs.available) {
|
|
|
|
if (idx.is_graphics) {
|
|
|
|
qs.family_used[qs.available.at(index).family]++;
|
|
|
|
qs.present.push_back(qs.available.at(index));
|
|
|
|
qs.available.erase(qs.available.begin() + index);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
}
|
2023-09-19 06:15:59 +00:00
|
|
|
return qs;
|
2023-09-17 17:01:29 +00:00
|
|
|
}
|
2023-09-19 11:02:40 +00:00
|
|
|
|
2023-09-20 11:19:20 +00:00
|
|
|
void Graphics::Vulkan::vulkanGetSurfaceCapabilities(VkPhysicalDevice physical_device, VkSurfaceKHR surface,
|
2023-10-13 06:40:59 +00:00
|
|
|
Emu::VulkanSurfaceCapabilities* surfaceCap) {
|
2023-09-19 11:02:40 +00:00
|
|
|
vkGetPhysicalDeviceSurfaceCapabilitiesKHR(physical_device, surface, &surfaceCap->capabilities);
|
|
|
|
|
|
|
|
uint32_t formats_count = 0;
|
|
|
|
vkGetPhysicalDeviceSurfaceFormatsKHR(physical_device, surface, &formats_count, nullptr);
|
|
|
|
|
|
|
|
surfaceCap->formats = std::vector<VkSurfaceFormatKHR>(formats_count);
|
|
|
|
vkGetPhysicalDeviceSurfaceFormatsKHR(physical_device, surface, &formats_count, surfaceCap->formats.data());
|
|
|
|
|
|
|
|
uint32_t present_modes_count = 0;
|
|
|
|
vkGetPhysicalDeviceSurfacePresentModesKHR(physical_device, surface, &present_modes_count, nullptr);
|
|
|
|
|
|
|
|
surfaceCap->present_modes = std::vector<VkPresentModeKHR>(present_modes_count);
|
|
|
|
vkGetPhysicalDeviceSurfacePresentModesKHR(physical_device, surface, &present_modes_count, surfaceCap->present_modes.data());
|
|
|
|
|
|
|
|
for (const auto& f : surfaceCap->formats) {
|
|
|
|
if (f.format == VK_FORMAT_B8G8R8A8_SRGB && f.colorSpace == VK_COLOR_SPACE_SRGB_NONLINEAR_KHR) {
|
|
|
|
surfaceCap->is_format_srgb_bgra32 = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (f.format == VK_FORMAT_B8G8R8A8_UNORM && f.colorSpace == VK_COLOR_SPACE_SRGB_NONLINEAR_KHR) {
|
|
|
|
surfaceCap->is_format_unorm_bgra32 = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2023-09-27 19:47:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void set_image_layout(VkCommandBuffer buffer, HLE::Libs::Graphics::VulkanImage* dst_image, uint32_t base_level, uint32_t levels,
|
|
|
|
VkImageAspectFlags aspect_mask, VkImageLayout old_image_layout, VkImageLayout new_image_layout) {
|
|
|
|
VkImageMemoryBarrier imageMemoryBarrier{};
|
|
|
|
imageMemoryBarrier.sType = VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER;
|
|
|
|
imageMemoryBarrier.pNext = nullptr;
|
|
|
|
imageMemoryBarrier.srcAccessMask = 0;
|
|
|
|
imageMemoryBarrier.dstAccessMask = 0;
|
|
|
|
imageMemoryBarrier.oldLayout = old_image_layout;
|
|
|
|
imageMemoryBarrier.newLayout = new_image_layout;
|
|
|
|
imageMemoryBarrier.srcQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED;
|
|
|
|
imageMemoryBarrier.dstQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED;
|
|
|
|
imageMemoryBarrier.image = dst_image->image;
|
|
|
|
imageMemoryBarrier.subresourceRange.aspectMask = aspect_mask;
|
|
|
|
imageMemoryBarrier.subresourceRange.baseMipLevel = base_level;
|
|
|
|
imageMemoryBarrier.subresourceRange.levelCount = levels;
|
|
|
|
imageMemoryBarrier.subresourceRange.baseArrayLayer = 0;
|
|
|
|
imageMemoryBarrier.subresourceRange.layerCount = 1;
|
|
|
|
|
|
|
|
if (old_image_layout == VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL) {
|
|
|
|
imageMemoryBarrier.srcAccessMask = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (new_image_layout == VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL) {
|
|
|
|
imageMemoryBarrier.dstAccessMask = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (new_image_layout == VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL) {
|
|
|
|
imageMemoryBarrier.dstAccessMask = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_image_layout == VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL) {
|
|
|
|
imageMemoryBarrier.srcAccessMask = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (old_image_layout == VK_IMAGE_LAYOUT_PREINITIALIZED) {
|
|
|
|
imageMemoryBarrier.srcAccessMask = VK_ACCESS_HOST_WRITE_BIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (new_image_layout == VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL) {
|
|
|
|
imageMemoryBarrier.srcAccessMask = 0;
|
|
|
|
imageMemoryBarrier.dstAccessMask = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (new_image_layout == VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL) {
|
|
|
|
imageMemoryBarrier.dstAccessMask = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (new_image_layout == VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL) {
|
|
|
|
imageMemoryBarrier.dstAccessMask = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
VkPipelineStageFlags src_stages = VK_PIPELINE_STAGE_TOP_OF_PIPE_BIT;
|
|
|
|
VkPipelineStageFlags dest_stages = VK_PIPELINE_STAGE_TOP_OF_PIPE_BIT;
|
|
|
|
|
|
|
|
vkCmdPipelineBarrier(buffer, src_stages, dest_stages, 0, 0, nullptr, 0, nullptr, 1, &imageMemoryBarrier);
|
|
|
|
|
|
|
|
dst_image->layout = new_image_layout;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Graphics::Vulkan::vulkanBlitImage(GPU::CommandBuffer* buffer, HLE::Libs::Graphics::VulkanImage* src_image,
|
2023-10-13 06:40:59 +00:00
|
|
|
Emu::VulkanSwapchain* dst_swapchain) {
|
2023-09-27 19:47:53 +00:00
|
|
|
auto* vk_buffer = buffer->getPool()->buffers[buffer->getIndex()];
|
|
|
|
|
|
|
|
HLE::Libs::Graphics::VulkanImage swapchain_image(HLE::Libs::Graphics::VulkanImageType::Unknown);
|
|
|
|
|
|
|
|
swapchain_image.image = dst_swapchain->swapchain_images[dst_swapchain->current_index];
|
|
|
|
swapchain_image.layout = VK_IMAGE_LAYOUT_UNDEFINED;
|
|
|
|
|
|
|
|
set_image_layout(vk_buffer, src_image, 0, 1, VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL,
|
|
|
|
VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL);
|
|
|
|
set_image_layout(vk_buffer, &swapchain_image, 0, 1, VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_UNDEFINED, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL);
|
|
|
|
|
|
|
|
VkImageBlit region{};
|
|
|
|
region.srcSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
|
|
|
|
region.srcSubresource.mipLevel = 0;
|
|
|
|
region.srcSubresource.baseArrayLayer = 0;
|
|
|
|
region.srcSubresource.layerCount = 1;
|
|
|
|
region.srcOffsets[0].x = 0;
|
|
|
|
region.srcOffsets[0].y = 0;
|
|
|
|
region.srcOffsets[0].z = 0;
|
|
|
|
region.srcOffsets[1].x = static_cast<int>(src_image->extent.width);
|
|
|
|
region.srcOffsets[1].y = static_cast<int>(src_image->extent.height);
|
|
|
|
region.srcOffsets[1].z = 1;
|
|
|
|
region.dstSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
|
|
|
|
region.dstSubresource.mipLevel = 0;
|
|
|
|
region.dstSubresource.baseArrayLayer = 0;
|
|
|
|
region.dstSubresource.layerCount = 1;
|
|
|
|
region.dstOffsets[0].x = 0;
|
|
|
|
region.dstOffsets[0].y = 0;
|
|
|
|
region.dstOffsets[0].z = 0;
|
|
|
|
region.dstOffsets[1].x = static_cast<int>(dst_swapchain->swapchain_extent.width);
|
|
|
|
region.dstOffsets[1].y = static_cast<int>(dst_swapchain->swapchain_extent.height);
|
|
|
|
region.dstOffsets[1].z = 1;
|
|
|
|
|
|
|
|
vkCmdBlitImage(vk_buffer, src_image->image, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL, swapchain_image.image, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, 1,
|
|
|
|
®ion, VK_FILTER_LINEAR);
|
|
|
|
|
|
|
|
set_image_layout(vk_buffer, src_image, 0, 1, VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
|
|
|
|
VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Graphics::Vulkan::vulkanFillImage(HLE::Libs::Graphics::GraphicCtx* ctx, HLE::Libs::Graphics::VulkanImage* dst_image, const void* src_data,
|
|
|
|
u64 size, u32 src_pitch, u64 dst_layout) {
|
|
|
|
HLE::Libs::Graphics::VulkanBuffer staging_buffer{};
|
|
|
|
staging_buffer.usage = VK_BUFFER_USAGE_TRANSFER_SRC_BIT;
|
|
|
|
staging_buffer.memory.property = VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT;
|
|
|
|
vulkanCreateBuffer(ctx, size, &staging_buffer);
|
|
|
|
|
|
|
|
void* data = nullptr;
|
|
|
|
vkMapMemory(ctx->m_device, staging_buffer.memory.memory, staging_buffer.memory.offset, staging_buffer.memory.requirements.size, 0, &data);
|
|
|
|
std::memcpy(data, src_data, size);
|
|
|
|
vkUnmapMemory(ctx->m_device, staging_buffer.memory.memory);
|
|
|
|
|
|
|
|
GPU::CommandBuffer buffer(9);
|
|
|
|
|
|
|
|
buffer.begin();
|
|
|
|
vulkanBufferToImage(&buffer, &staging_buffer, src_pitch, dst_image, dst_layout);
|
|
|
|
buffer.end();
|
|
|
|
buffer.execute();
|
|
|
|
buffer.waitForFence();
|
|
|
|
|
|
|
|
vulkanDeleteBuffer(ctx, &staging_buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Graphics::Vulkan::vulkanBufferToImage(GPU::CommandBuffer* buffer, HLE::Libs::Graphics::VulkanBuffer* src_buffer, u32 src_pitch,
|
|
|
|
HLE::Libs::Graphics::VulkanImage* dst_image, u64 dst_layout) {
|
|
|
|
auto* vk_buffer = buffer->getPool()->buffers[buffer->getIndex()];
|
|
|
|
|
|
|
|
set_image_layout(vk_buffer, dst_image, 0, 1, VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_UNDEFINED, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL);
|
|
|
|
|
|
|
|
VkBufferImageCopy region{};
|
|
|
|
region.bufferOffset = 0;
|
|
|
|
region.bufferRowLength = (src_pitch != dst_image->extent.width ? src_pitch : 0);
|
|
|
|
region.bufferImageHeight = 0;
|
|
|
|
|
|
|
|
region.imageSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
|
|
|
|
region.imageSubresource.mipLevel = 0;
|
|
|
|
region.imageSubresource.baseArrayLayer = 0;
|
|
|
|
region.imageSubresource.layerCount = 1;
|
|
|
|
|
|
|
|
region.imageOffset = {0, 0, 0};
|
|
|
|
region.imageExtent = {dst_image->extent.width, dst_image->extent.height, 1};
|
|
|
|
|
|
|
|
vkCmdCopyBufferToImage(vk_buffer, src_buffer->buffer, dst_image->image, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, 1, ®ion);
|
|
|
|
|
|
|
|
set_image_layout(vk_buffer, dst_image, 0, 1, VK_IMAGE_ASPECT_COLOR_BIT, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
|
|
|
|
static_cast<VkImageLayout>(dst_layout));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Graphics::Vulkan::vulkanCreateBuffer(HLE::Libs::Graphics::GraphicCtx* ctx, u64 size, HLE::Libs::Graphics::VulkanBuffer* buffer) {
|
|
|
|
VkBufferCreateInfo buffer_info{};
|
|
|
|
buffer_info.sType = VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO;
|
|
|
|
buffer_info.size = size;
|
|
|
|
buffer_info.usage = buffer->usage;
|
|
|
|
buffer_info.sharingMode = VK_SHARING_MODE_EXCLUSIVE;
|
|
|
|
|
|
|
|
vkCreateBuffer(ctx->m_device, &buffer_info, nullptr, &buffer->buffer);
|
|
|
|
|
|
|
|
vkGetBufferMemoryRequirements(ctx->m_device, buffer->buffer, &buffer->memory.requirements);
|
|
|
|
|
|
|
|
bool allocated = GPU::vulkanAllocateMemory(ctx, &buffer->memory);
|
|
|
|
if (!allocated) {
|
2023-10-26 20:29:05 +00:00
|
|
|
fmt::print("Can't allocate vulkan\n");
|
2023-09-27 19:47:53 +00:00
|
|
|
std::exit(0);
|
|
|
|
}
|
|
|
|
vkBindBufferMemory(ctx->m_device, buffer->buffer, buffer->memory.memory, buffer->memory.offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Graphics::Vulkan::vulkanDeleteBuffer(HLE::Libs::Graphics::GraphicCtx* ctx, HLE::Libs::Graphics::VulkanBuffer* buffer) {
|
|
|
|
vkDestroyBuffer(ctx->m_device, buffer->buffer, nullptr);
|
|
|
|
vkFreeMemory(ctx->m_device, buffer->memory.memory, nullptr);
|
|
|
|
buffer->memory.memory = nullptr;
|
|
|
|
buffer->buffer = nullptr;
|
|
|
|
}
|