79da90cea8
Silence three warnings and make them errors to avoid introducing more in the future.
864 lines
31 KiB
C++
864 lines
31 KiB
C++
// Copyright 2020 yuzu Emulator Project
|
|
// Licensed under GPLv2 or any later version
|
|
// Refer to the license.txt file included.
|
|
|
|
#include <algorithm>
|
|
#include <exception>
|
|
#include <memory>
|
|
#include <optional>
|
|
#include <string_view>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include "common/common_types.h"
|
|
#include "common/logging/log.h"
|
|
|
|
#include "video_core/renderer_vulkan/wrapper.h"
|
|
|
|
namespace Vulkan::vk {
|
|
|
|
namespace {
|
|
|
|
template <typename Func>
|
|
void SortPhysicalDevices(std::vector<VkPhysicalDevice>& devices, const InstanceDispatch& dld,
|
|
Func&& func) {
|
|
// Calling GetProperties calls Vulkan more than needed. But they are supposed to be cheap
|
|
// functions.
|
|
std::stable_sort(devices.begin(), devices.end(),
|
|
[&dld, &func](VkPhysicalDevice lhs, VkPhysicalDevice rhs) {
|
|
return func(vk::PhysicalDevice(lhs, dld).GetProperties(),
|
|
vk::PhysicalDevice(rhs, dld).GetProperties());
|
|
});
|
|
}
|
|
|
|
void SortPhysicalDevicesPerVendor(std::vector<VkPhysicalDevice>& devices,
|
|
const InstanceDispatch& dld,
|
|
std::initializer_list<u32> vendor_ids) {
|
|
for (auto it = vendor_ids.end(); it != vendor_ids.begin();) {
|
|
--it;
|
|
SortPhysicalDevices(devices, dld, [id = *it](const auto& lhs, const auto& rhs) {
|
|
return lhs.vendorID == id && rhs.vendorID != id;
|
|
});
|
|
}
|
|
}
|
|
|
|
void SortPhysicalDevices(std::vector<VkPhysicalDevice>& devices, const InstanceDispatch& dld) {
|
|
// Sort by name, this will set a base and make GPUs with higher numbers appear first
|
|
// (e.g. GTX 1650 will intentionally be listed before a GTX 1080).
|
|
SortPhysicalDevices(devices, dld, [](const auto& lhs, const auto& rhs) {
|
|
return std::string_view{lhs.deviceName} > std::string_view{rhs.deviceName};
|
|
});
|
|
// Prefer discrete over non-discrete
|
|
SortPhysicalDevices(devices, dld, [](const auto& lhs, const auto& rhs) {
|
|
return lhs.deviceType == VK_PHYSICAL_DEVICE_TYPE_DISCRETE_GPU &&
|
|
rhs.deviceType != VK_PHYSICAL_DEVICE_TYPE_DISCRETE_GPU;
|
|
});
|
|
// Prefer Nvidia over AMD, AMD over Intel, Intel over the rest.
|
|
SortPhysicalDevicesPerVendor(devices, dld, {0x10DE, 0x1002, 0x8086});
|
|
}
|
|
|
|
template <typename T>
|
|
bool Proc(T& result, const InstanceDispatch& dld, const char* proc_name,
|
|
VkInstance instance = nullptr) noexcept {
|
|
result = reinterpret_cast<T>(dld.vkGetInstanceProcAddr(instance, proc_name));
|
|
return result != nullptr;
|
|
}
|
|
|
|
template <typename T>
|
|
void Proc(T& result, const DeviceDispatch& dld, const char* proc_name, VkDevice device) noexcept {
|
|
result = reinterpret_cast<T>(dld.vkGetDeviceProcAddr(device, proc_name));
|
|
}
|
|
|
|
void Load(VkDevice device, DeviceDispatch& dld) noexcept {
|
|
#define X(name) Proc(dld.name, dld, #name, device)
|
|
X(vkAcquireNextImageKHR);
|
|
X(vkAllocateCommandBuffers);
|
|
X(vkAllocateDescriptorSets);
|
|
X(vkAllocateMemory);
|
|
X(vkBeginCommandBuffer);
|
|
X(vkBindBufferMemory);
|
|
X(vkBindImageMemory);
|
|
X(vkCmdBeginQuery);
|
|
X(vkCmdBeginRenderPass);
|
|
X(vkCmdBeginTransformFeedbackEXT);
|
|
X(vkCmdBindDescriptorSets);
|
|
X(vkCmdBindIndexBuffer);
|
|
X(vkCmdBindPipeline);
|
|
X(vkCmdBindTransformFeedbackBuffersEXT);
|
|
X(vkCmdBindVertexBuffers);
|
|
X(vkCmdBlitImage);
|
|
X(vkCmdClearAttachments);
|
|
X(vkCmdCopyBuffer);
|
|
X(vkCmdCopyBufferToImage);
|
|
X(vkCmdCopyImage);
|
|
X(vkCmdCopyImageToBuffer);
|
|
X(vkCmdDispatch);
|
|
X(vkCmdDraw);
|
|
X(vkCmdDrawIndexed);
|
|
X(vkCmdEndQuery);
|
|
X(vkCmdEndRenderPass);
|
|
X(vkCmdEndTransformFeedbackEXT);
|
|
X(vkCmdFillBuffer);
|
|
X(vkCmdPipelineBarrier);
|
|
X(vkCmdPushConstants);
|
|
X(vkCmdSetBlendConstants);
|
|
X(vkCmdSetDepthBias);
|
|
X(vkCmdSetDepthBounds);
|
|
X(vkCmdSetEvent);
|
|
X(vkCmdSetScissor);
|
|
X(vkCmdSetStencilCompareMask);
|
|
X(vkCmdSetStencilReference);
|
|
X(vkCmdSetStencilWriteMask);
|
|
X(vkCmdSetViewport);
|
|
X(vkCmdWaitEvents);
|
|
X(vkCmdBindVertexBuffers2EXT);
|
|
X(vkCmdSetCullModeEXT);
|
|
X(vkCmdSetDepthBoundsTestEnableEXT);
|
|
X(vkCmdSetDepthCompareOpEXT);
|
|
X(vkCmdSetDepthTestEnableEXT);
|
|
X(vkCmdSetDepthWriteEnableEXT);
|
|
X(vkCmdSetFrontFaceEXT);
|
|
X(vkCmdSetPrimitiveTopologyEXT);
|
|
X(vkCmdSetStencilOpEXT);
|
|
X(vkCmdSetStencilTestEnableEXT);
|
|
X(vkCreateBuffer);
|
|
X(vkCreateBufferView);
|
|
X(vkCreateCommandPool);
|
|
X(vkCreateComputePipelines);
|
|
X(vkCreateDescriptorPool);
|
|
X(vkCreateDescriptorSetLayout);
|
|
X(vkCreateDescriptorUpdateTemplateKHR);
|
|
X(vkCreateEvent);
|
|
X(vkCreateFence);
|
|
X(vkCreateFramebuffer);
|
|
X(vkCreateGraphicsPipelines);
|
|
X(vkCreateImage);
|
|
X(vkCreateImageView);
|
|
X(vkCreatePipelineLayout);
|
|
X(vkCreateQueryPool);
|
|
X(vkCreateRenderPass);
|
|
X(vkCreateSampler);
|
|
X(vkCreateSemaphore);
|
|
X(vkCreateShaderModule);
|
|
X(vkCreateSwapchainKHR);
|
|
X(vkDestroyBuffer);
|
|
X(vkDestroyBufferView);
|
|
X(vkDestroyCommandPool);
|
|
X(vkDestroyDescriptorPool);
|
|
X(vkDestroyDescriptorSetLayout);
|
|
X(vkDestroyDescriptorUpdateTemplateKHR);
|
|
X(vkDestroyEvent);
|
|
X(vkDestroyFence);
|
|
X(vkDestroyFramebuffer);
|
|
X(vkDestroyImage);
|
|
X(vkDestroyImageView);
|
|
X(vkDestroyPipeline);
|
|
X(vkDestroyPipelineLayout);
|
|
X(vkDestroyQueryPool);
|
|
X(vkDestroyRenderPass);
|
|
X(vkDestroySampler);
|
|
X(vkDestroySemaphore);
|
|
X(vkDestroyShaderModule);
|
|
X(vkDestroySwapchainKHR);
|
|
X(vkDeviceWaitIdle);
|
|
X(vkEndCommandBuffer);
|
|
X(vkFreeCommandBuffers);
|
|
X(vkFreeDescriptorSets);
|
|
X(vkFreeMemory);
|
|
X(vkGetBufferMemoryRequirements);
|
|
X(vkGetDeviceQueue);
|
|
X(vkGetEventStatus);
|
|
X(vkGetFenceStatus);
|
|
X(vkGetImageMemoryRequirements);
|
|
X(vkGetQueryPoolResults);
|
|
X(vkGetSemaphoreCounterValueKHR);
|
|
X(vkMapMemory);
|
|
X(vkQueueSubmit);
|
|
X(vkResetFences);
|
|
X(vkResetQueryPoolEXT);
|
|
X(vkUnmapMemory);
|
|
X(vkUpdateDescriptorSetWithTemplateKHR);
|
|
X(vkUpdateDescriptorSets);
|
|
X(vkWaitForFences);
|
|
X(vkWaitSemaphoresKHR);
|
|
#undef X
|
|
}
|
|
|
|
} // Anonymous namespace
|
|
|
|
bool Load(InstanceDispatch& dld) noexcept {
|
|
#define X(name) Proc(dld.name, dld, #name)
|
|
return X(vkCreateInstance) && X(vkEnumerateInstanceExtensionProperties) &&
|
|
X(vkEnumerateInstanceLayerProperties);
|
|
#undef X
|
|
}
|
|
|
|
bool Load(VkInstance instance, InstanceDispatch& dld) noexcept {
|
|
#define X(name) Proc(dld.name, dld, #name, instance)
|
|
// These functions may fail to load depending on the enabled extensions.
|
|
// Don't return a failure on these.
|
|
X(vkCreateDebugUtilsMessengerEXT);
|
|
X(vkDestroyDebugUtilsMessengerEXT);
|
|
X(vkDestroySurfaceKHR);
|
|
X(vkGetPhysicalDeviceFeatures2KHR);
|
|
X(vkGetPhysicalDeviceProperties2KHR);
|
|
X(vkGetPhysicalDeviceSurfaceCapabilitiesKHR);
|
|
X(vkGetPhysicalDeviceSurfaceFormatsKHR);
|
|
X(vkGetPhysicalDeviceSurfacePresentModesKHR);
|
|
X(vkGetPhysicalDeviceSurfaceSupportKHR);
|
|
X(vkGetSwapchainImagesKHR);
|
|
X(vkQueuePresentKHR);
|
|
|
|
return X(vkCreateDevice) && X(vkDestroyDevice) && X(vkDestroyDevice) &&
|
|
X(vkEnumerateDeviceExtensionProperties) && X(vkEnumeratePhysicalDevices) &&
|
|
X(vkGetDeviceProcAddr) && X(vkGetPhysicalDeviceFormatProperties) &&
|
|
X(vkGetPhysicalDeviceMemoryProperties) && X(vkGetPhysicalDeviceProperties) &&
|
|
X(vkGetPhysicalDeviceQueueFamilyProperties);
|
|
#undef X
|
|
}
|
|
|
|
const char* Exception::what() const noexcept {
|
|
return ToString(result);
|
|
}
|
|
|
|
const char* ToString(VkResult result) noexcept {
|
|
switch (result) {
|
|
case VkResult::VK_SUCCESS:
|
|
return "VK_SUCCESS";
|
|
case VkResult::VK_NOT_READY:
|
|
return "VK_NOT_READY";
|
|
case VkResult::VK_TIMEOUT:
|
|
return "VK_TIMEOUT";
|
|
case VkResult::VK_EVENT_SET:
|
|
return "VK_EVENT_SET";
|
|
case VkResult::VK_EVENT_RESET:
|
|
return "VK_EVENT_RESET";
|
|
case VkResult::VK_INCOMPLETE:
|
|
return "VK_INCOMPLETE";
|
|
case VkResult::VK_ERROR_OUT_OF_HOST_MEMORY:
|
|
return "VK_ERROR_OUT_OF_HOST_MEMORY";
|
|
case VkResult::VK_ERROR_OUT_OF_DEVICE_MEMORY:
|
|
return "VK_ERROR_OUT_OF_DEVICE_MEMORY";
|
|
case VkResult::VK_ERROR_INITIALIZATION_FAILED:
|
|
return "VK_ERROR_INITIALIZATION_FAILED";
|
|
case VkResult::VK_ERROR_DEVICE_LOST:
|
|
return "VK_ERROR_DEVICE_LOST";
|
|
case VkResult::VK_ERROR_MEMORY_MAP_FAILED:
|
|
return "VK_ERROR_MEMORY_MAP_FAILED";
|
|
case VkResult::VK_ERROR_LAYER_NOT_PRESENT:
|
|
return "VK_ERROR_LAYER_NOT_PRESENT";
|
|
case VkResult::VK_ERROR_EXTENSION_NOT_PRESENT:
|
|
return "VK_ERROR_EXTENSION_NOT_PRESENT";
|
|
case VkResult::VK_ERROR_FEATURE_NOT_PRESENT:
|
|
return "VK_ERROR_FEATURE_NOT_PRESENT";
|
|
case VkResult::VK_ERROR_INCOMPATIBLE_DRIVER:
|
|
return "VK_ERROR_INCOMPATIBLE_DRIVER";
|
|
case VkResult::VK_ERROR_TOO_MANY_OBJECTS:
|
|
return "VK_ERROR_TOO_MANY_OBJECTS";
|
|
case VkResult::VK_ERROR_FORMAT_NOT_SUPPORTED:
|
|
return "VK_ERROR_FORMAT_NOT_SUPPORTED";
|
|
case VkResult::VK_ERROR_FRAGMENTED_POOL:
|
|
return "VK_ERROR_FRAGMENTED_POOL";
|
|
case VkResult::VK_ERROR_OUT_OF_POOL_MEMORY:
|
|
return "VK_ERROR_OUT_OF_POOL_MEMORY";
|
|
case VkResult::VK_ERROR_INVALID_EXTERNAL_HANDLE:
|
|
return "VK_ERROR_INVALID_EXTERNAL_HANDLE";
|
|
case VkResult::VK_ERROR_SURFACE_LOST_KHR:
|
|
return "VK_ERROR_SURFACE_LOST_KHR";
|
|
case VkResult::VK_ERROR_NATIVE_WINDOW_IN_USE_KHR:
|
|
return "VK_ERROR_NATIVE_WINDOW_IN_USE_KHR";
|
|
case VkResult::VK_SUBOPTIMAL_KHR:
|
|
return "VK_SUBOPTIMAL_KHR";
|
|
case VkResult::VK_ERROR_OUT_OF_DATE_KHR:
|
|
return "VK_ERROR_OUT_OF_DATE_KHR";
|
|
case VkResult::VK_ERROR_INCOMPATIBLE_DISPLAY_KHR:
|
|
return "VK_ERROR_INCOMPATIBLE_DISPLAY_KHR";
|
|
case VkResult::VK_ERROR_VALIDATION_FAILED_EXT:
|
|
return "VK_ERROR_VALIDATION_FAILED_EXT";
|
|
case VkResult::VK_ERROR_INVALID_SHADER_NV:
|
|
return "VK_ERROR_INVALID_SHADER_NV";
|
|
case VkResult::VK_ERROR_INVALID_DRM_FORMAT_MODIFIER_PLANE_LAYOUT_EXT:
|
|
return "VK_ERROR_INVALID_DRM_FORMAT_MODIFIER_PLANE_LAYOUT_EXT";
|
|
case VkResult::VK_ERROR_FRAGMENTATION_EXT:
|
|
return "VK_ERROR_FRAGMENTATION_EXT";
|
|
case VkResult::VK_ERROR_NOT_PERMITTED_EXT:
|
|
return "VK_ERROR_NOT_PERMITTED_EXT";
|
|
case VkResult::VK_ERROR_INVALID_DEVICE_ADDRESS_EXT:
|
|
return "VK_ERROR_INVALID_DEVICE_ADDRESS_EXT";
|
|
case VkResult::VK_ERROR_FULL_SCREEN_EXCLUSIVE_MODE_LOST_EXT:
|
|
return "VK_ERROR_FULL_SCREEN_EXCLUSIVE_MODE_LOST_EXT";
|
|
case VkResult::VK_ERROR_UNKNOWN:
|
|
return "VK_ERROR_UNKNOWN";
|
|
case VkResult::VK_ERROR_INCOMPATIBLE_VERSION_KHR:
|
|
return "VK_ERROR_INCOMPATIBLE_VERSION_KHR";
|
|
case VkResult::VK_THREAD_IDLE_KHR:
|
|
return "VK_THREAD_IDLE_KHR";
|
|
case VkResult::VK_THREAD_DONE_KHR:
|
|
return "VK_THREAD_DONE_KHR";
|
|
case VkResult::VK_OPERATION_DEFERRED_KHR:
|
|
return "VK_OPERATION_DEFERRED_KHR";
|
|
case VkResult::VK_OPERATION_NOT_DEFERRED_KHR:
|
|
return "VK_OPERATION_NOT_DEFERRED_KHR";
|
|
case VkResult::VK_PIPELINE_COMPILE_REQUIRED_EXT:
|
|
return "VK_PIPELINE_COMPILE_REQUIRED_EXT";
|
|
case VkResult::VK_RESULT_MAX_ENUM:
|
|
return "VK_RESULT_MAX_ENUM";
|
|
}
|
|
return "Unknown";
|
|
}
|
|
|
|
void Destroy(VkInstance instance, const InstanceDispatch& dld) noexcept {
|
|
dld.vkDestroyInstance(instance, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, const InstanceDispatch& dld) noexcept {
|
|
dld.vkDestroyDevice(device, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkBuffer handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyBuffer(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkBufferView handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyBufferView(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkCommandPool handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyCommandPool(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkDescriptorPool handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyDescriptorPool(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkDescriptorSetLayout handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyDescriptorSetLayout(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkDescriptorUpdateTemplateKHR handle,
|
|
const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyDescriptorUpdateTemplateKHR(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkDeviceMemory handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkFreeMemory(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkEvent handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyEvent(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkFence handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyFence(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkFramebuffer handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyFramebuffer(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkImage handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyImage(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkImageView handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyImageView(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkPipeline handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyPipeline(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkPipelineLayout handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyPipelineLayout(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkQueryPool handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyQueryPool(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkRenderPass handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyRenderPass(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkSampler handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroySampler(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkSwapchainKHR handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroySwapchainKHR(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkSemaphore handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroySemaphore(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkDevice device, VkShaderModule handle, const DeviceDispatch& dld) noexcept {
|
|
dld.vkDestroyShaderModule(device, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkInstance instance, VkDebugUtilsMessengerEXT handle,
|
|
const InstanceDispatch& dld) noexcept {
|
|
dld.vkDestroyDebugUtilsMessengerEXT(instance, handle, nullptr);
|
|
}
|
|
|
|
void Destroy(VkInstance instance, VkSurfaceKHR handle, const InstanceDispatch& dld) noexcept {
|
|
dld.vkDestroySurfaceKHR(instance, handle, nullptr);
|
|
}
|
|
|
|
VkResult Free(VkDevice device, VkDescriptorPool handle, Span<VkDescriptorSet> sets,
|
|
const DeviceDispatch& dld) noexcept {
|
|
return dld.vkFreeDescriptorSets(device, handle, sets.size(), sets.data());
|
|
}
|
|
|
|
VkResult Free(VkDevice device, VkCommandPool handle, Span<VkCommandBuffer> buffers,
|
|
const DeviceDispatch& dld) noexcept {
|
|
dld.vkFreeCommandBuffers(device, handle, buffers.size(), buffers.data());
|
|
return VK_SUCCESS;
|
|
}
|
|
|
|
Instance Instance::Create(u32 version, Span<const char*> layers, Span<const char*> extensions,
|
|
InstanceDispatch& dld) noexcept {
|
|
const VkApplicationInfo application_info{
|
|
.sType = VK_STRUCTURE_TYPE_APPLICATION_INFO,
|
|
.pNext = nullptr,
|
|
.pApplicationName = "yuzu Emulator",
|
|
.applicationVersion = VK_MAKE_VERSION(0, 1, 0),
|
|
.pEngineName = "yuzu Emulator",
|
|
.engineVersion = VK_MAKE_VERSION(0, 1, 0),
|
|
.apiVersion = version,
|
|
};
|
|
const VkInstanceCreateInfo ci{
|
|
.sType = VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO,
|
|
.pNext = nullptr,
|
|
.flags = 0,
|
|
.pApplicationInfo = &application_info,
|
|
.enabledLayerCount = layers.size(),
|
|
.ppEnabledLayerNames = layers.data(),
|
|
.enabledExtensionCount = extensions.size(),
|
|
.ppEnabledExtensionNames = extensions.data(),
|
|
};
|
|
|
|
VkInstance instance;
|
|
if (dld.vkCreateInstance(&ci, nullptr, &instance) != VK_SUCCESS) {
|
|
// Failed to create the instance.
|
|
return {};
|
|
}
|
|
if (!Proc(dld.vkDestroyInstance, dld, "vkDestroyInstance", instance)) {
|
|
// We successfully created an instance but the destroy function couldn't be loaded.
|
|
// This is a good moment to panic.
|
|
return {};
|
|
}
|
|
|
|
return Instance(instance, dld);
|
|
}
|
|
|
|
std::optional<std::vector<VkPhysicalDevice>> Instance::EnumeratePhysicalDevices() {
|
|
u32 num;
|
|
if (dld->vkEnumeratePhysicalDevices(handle, &num, nullptr) != VK_SUCCESS) {
|
|
return std::nullopt;
|
|
}
|
|
std::vector<VkPhysicalDevice> physical_devices(num);
|
|
if (dld->vkEnumeratePhysicalDevices(handle, &num, physical_devices.data()) != VK_SUCCESS) {
|
|
return std::nullopt;
|
|
}
|
|
SortPhysicalDevices(physical_devices, *dld);
|
|
return std::make_optional(std::move(physical_devices));
|
|
}
|
|
|
|
DebugCallback Instance::TryCreateDebugCallback(
|
|
PFN_vkDebugUtilsMessengerCallbackEXT callback) noexcept {
|
|
const VkDebugUtilsMessengerCreateInfoEXT ci{
|
|
.sType = VK_STRUCTURE_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT,
|
|
.pNext = nullptr,
|
|
.flags = 0,
|
|
.messageSeverity = VK_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT |
|
|
VK_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT |
|
|
VK_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT |
|
|
VK_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT,
|
|
.messageType = VK_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT |
|
|
VK_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT |
|
|
VK_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT,
|
|
.pfnUserCallback = callback,
|
|
.pUserData = nullptr,
|
|
};
|
|
|
|
VkDebugUtilsMessengerEXT messenger;
|
|
if (dld->vkCreateDebugUtilsMessengerEXT(handle, &ci, nullptr, &messenger) != VK_SUCCESS) {
|
|
return {};
|
|
}
|
|
return DebugCallback(messenger, handle, *dld);
|
|
}
|
|
|
|
void Buffer::BindMemory(VkDeviceMemory memory, VkDeviceSize offset) const {
|
|
Check(dld->vkBindBufferMemory(owner, handle, memory, offset));
|
|
}
|
|
|
|
void Image::BindMemory(VkDeviceMemory memory, VkDeviceSize offset) const {
|
|
Check(dld->vkBindImageMemory(owner, handle, memory, offset));
|
|
}
|
|
|
|
DescriptorSets DescriptorPool::Allocate(const VkDescriptorSetAllocateInfo& ai) const {
|
|
const std::size_t num = ai.descriptorSetCount;
|
|
std::unique_ptr sets = std::make_unique<VkDescriptorSet[]>(num);
|
|
switch (const VkResult result = dld->vkAllocateDescriptorSets(owner, &ai, sets.get())) {
|
|
case VK_SUCCESS:
|
|
return DescriptorSets(std::move(sets), num, owner, handle, *dld);
|
|
case VK_ERROR_OUT_OF_POOL_MEMORY:
|
|
return {};
|
|
default:
|
|
throw Exception(result);
|
|
}
|
|
}
|
|
|
|
CommandBuffers CommandPool::Allocate(std::size_t num_buffers, VkCommandBufferLevel level) const {
|
|
const VkCommandBufferAllocateInfo ai{
|
|
.sType = VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO,
|
|
.pNext = nullptr,
|
|
.commandPool = handle,
|
|
.level = level,
|
|
.commandBufferCount = static_cast<u32>(num_buffers),
|
|
};
|
|
|
|
std::unique_ptr buffers = std::make_unique<VkCommandBuffer[]>(num_buffers);
|
|
switch (const VkResult result = dld->vkAllocateCommandBuffers(owner, &ai, buffers.get())) {
|
|
case VK_SUCCESS:
|
|
return CommandBuffers(std::move(buffers), num_buffers, owner, handle, *dld);
|
|
case VK_ERROR_OUT_OF_POOL_MEMORY:
|
|
return {};
|
|
default:
|
|
throw Exception(result);
|
|
}
|
|
}
|
|
|
|
std::vector<VkImage> SwapchainKHR::GetImages() const {
|
|
u32 num;
|
|
Check(dld->vkGetSwapchainImagesKHR(owner, handle, &num, nullptr));
|
|
std::vector<VkImage> images(num);
|
|
Check(dld->vkGetSwapchainImagesKHR(owner, handle, &num, images.data()));
|
|
return images;
|
|
}
|
|
|
|
Device Device::Create(VkPhysicalDevice physical_device, Span<VkDeviceQueueCreateInfo> queues_ci,
|
|
Span<const char*> enabled_extensions, const void* next,
|
|
DeviceDispatch& dld) noexcept {
|
|
const VkDeviceCreateInfo ci{
|
|
.sType = VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO,
|
|
.pNext = next,
|
|
.flags = 0,
|
|
.queueCreateInfoCount = queues_ci.size(),
|
|
.pQueueCreateInfos = queues_ci.data(),
|
|
.enabledLayerCount = 0,
|
|
.ppEnabledLayerNames = nullptr,
|
|
.enabledExtensionCount = enabled_extensions.size(),
|
|
.ppEnabledExtensionNames = enabled_extensions.data(),
|
|
.pEnabledFeatures = nullptr,
|
|
};
|
|
|
|
VkDevice device;
|
|
if (dld.vkCreateDevice(physical_device, &ci, nullptr, &device) != VK_SUCCESS) {
|
|
return {};
|
|
}
|
|
Load(device, dld);
|
|
return Device(device, dld);
|
|
}
|
|
|
|
Queue Device::GetQueue(u32 family_index) const noexcept {
|
|
VkQueue queue;
|
|
dld->vkGetDeviceQueue(handle, family_index, 0, &queue);
|
|
return Queue(queue, *dld);
|
|
}
|
|
|
|
Buffer Device::CreateBuffer(const VkBufferCreateInfo& ci) const {
|
|
VkBuffer object;
|
|
Check(dld->vkCreateBuffer(handle, &ci, nullptr, &object));
|
|
return Buffer(object, handle, *dld);
|
|
}
|
|
|
|
BufferView Device::CreateBufferView(const VkBufferViewCreateInfo& ci) const {
|
|
VkBufferView object;
|
|
Check(dld->vkCreateBufferView(handle, &ci, nullptr, &object));
|
|
return BufferView(object, handle, *dld);
|
|
}
|
|
|
|
Image Device::CreateImage(const VkImageCreateInfo& ci) const {
|
|
VkImage object;
|
|
Check(dld->vkCreateImage(handle, &ci, nullptr, &object));
|
|
return Image(object, handle, *dld);
|
|
}
|
|
|
|
ImageView Device::CreateImageView(const VkImageViewCreateInfo& ci) const {
|
|
VkImageView object;
|
|
Check(dld->vkCreateImageView(handle, &ci, nullptr, &object));
|
|
return ImageView(object, handle, *dld);
|
|
}
|
|
|
|
Semaphore Device::CreateSemaphore() const {
|
|
static constexpr VkSemaphoreCreateInfo ci{
|
|
.sType = VK_STRUCTURE_TYPE_SEMAPHORE_CREATE_INFO,
|
|
.pNext = nullptr,
|
|
.flags = 0,
|
|
};
|
|
return CreateSemaphore(ci);
|
|
}
|
|
|
|
Semaphore Device::CreateSemaphore(const VkSemaphoreCreateInfo& ci) const {
|
|
VkSemaphore object;
|
|
Check(dld->vkCreateSemaphore(handle, &ci, nullptr, &object));
|
|
return Semaphore(object, handle, *dld);
|
|
}
|
|
|
|
Fence Device::CreateFence(const VkFenceCreateInfo& ci) const {
|
|
VkFence object;
|
|
Check(dld->vkCreateFence(handle, &ci, nullptr, &object));
|
|
return Fence(object, handle, *dld);
|
|
}
|
|
|
|
DescriptorPool Device::CreateDescriptorPool(const VkDescriptorPoolCreateInfo& ci) const {
|
|
VkDescriptorPool object;
|
|
Check(dld->vkCreateDescriptorPool(handle, &ci, nullptr, &object));
|
|
return DescriptorPool(object, handle, *dld);
|
|
}
|
|
|
|
RenderPass Device::CreateRenderPass(const VkRenderPassCreateInfo& ci) const {
|
|
VkRenderPass object;
|
|
Check(dld->vkCreateRenderPass(handle, &ci, nullptr, &object));
|
|
return RenderPass(object, handle, *dld);
|
|
}
|
|
|
|
DescriptorSetLayout Device::CreateDescriptorSetLayout(
|
|
const VkDescriptorSetLayoutCreateInfo& ci) const {
|
|
VkDescriptorSetLayout object;
|
|
Check(dld->vkCreateDescriptorSetLayout(handle, &ci, nullptr, &object));
|
|
return DescriptorSetLayout(object, handle, *dld);
|
|
}
|
|
|
|
PipelineLayout Device::CreatePipelineLayout(const VkPipelineLayoutCreateInfo& ci) const {
|
|
VkPipelineLayout object;
|
|
Check(dld->vkCreatePipelineLayout(handle, &ci, nullptr, &object));
|
|
return PipelineLayout(object, handle, *dld);
|
|
}
|
|
|
|
Pipeline Device::CreateGraphicsPipeline(const VkGraphicsPipelineCreateInfo& ci) const {
|
|
VkPipeline object;
|
|
Check(dld->vkCreateGraphicsPipelines(handle, nullptr, 1, &ci, nullptr, &object));
|
|
return Pipeline(object, handle, *dld);
|
|
}
|
|
|
|
Pipeline Device::CreateComputePipeline(const VkComputePipelineCreateInfo& ci) const {
|
|
VkPipeline object;
|
|
Check(dld->vkCreateComputePipelines(handle, nullptr, 1, &ci, nullptr, &object));
|
|
return Pipeline(object, handle, *dld);
|
|
}
|
|
|
|
Sampler Device::CreateSampler(const VkSamplerCreateInfo& ci) const {
|
|
VkSampler object;
|
|
Check(dld->vkCreateSampler(handle, &ci, nullptr, &object));
|
|
return Sampler(object, handle, *dld);
|
|
}
|
|
|
|
Framebuffer Device::CreateFramebuffer(const VkFramebufferCreateInfo& ci) const {
|
|
VkFramebuffer object;
|
|
Check(dld->vkCreateFramebuffer(handle, &ci, nullptr, &object));
|
|
return Framebuffer(object, handle, *dld);
|
|
}
|
|
|
|
CommandPool Device::CreateCommandPool(const VkCommandPoolCreateInfo& ci) const {
|
|
VkCommandPool object;
|
|
Check(dld->vkCreateCommandPool(handle, &ci, nullptr, &object));
|
|
return CommandPool(object, handle, *dld);
|
|
}
|
|
|
|
DescriptorUpdateTemplateKHR Device::CreateDescriptorUpdateTemplateKHR(
|
|
const VkDescriptorUpdateTemplateCreateInfoKHR& ci) const {
|
|
VkDescriptorUpdateTemplateKHR object;
|
|
Check(dld->vkCreateDescriptorUpdateTemplateKHR(handle, &ci, nullptr, &object));
|
|
return DescriptorUpdateTemplateKHR(object, handle, *dld);
|
|
}
|
|
|
|
QueryPool Device::CreateQueryPool(const VkQueryPoolCreateInfo& ci) const {
|
|
VkQueryPool object;
|
|
Check(dld->vkCreateQueryPool(handle, &ci, nullptr, &object));
|
|
return QueryPool(object, handle, *dld);
|
|
}
|
|
|
|
ShaderModule Device::CreateShaderModule(const VkShaderModuleCreateInfo& ci) const {
|
|
VkShaderModule object;
|
|
Check(dld->vkCreateShaderModule(handle, &ci, nullptr, &object));
|
|
return ShaderModule(object, handle, *dld);
|
|
}
|
|
|
|
Event Device::CreateEvent() const {
|
|
static constexpr VkEventCreateInfo ci{
|
|
.sType = VK_STRUCTURE_TYPE_EVENT_CREATE_INFO,
|
|
.pNext = nullptr,
|
|
.flags = 0,
|
|
};
|
|
|
|
VkEvent object;
|
|
Check(dld->vkCreateEvent(handle, &ci, nullptr, &object));
|
|
return Event(object, handle, *dld);
|
|
}
|
|
|
|
SwapchainKHR Device::CreateSwapchainKHR(const VkSwapchainCreateInfoKHR& ci) const {
|
|
VkSwapchainKHR object;
|
|
Check(dld->vkCreateSwapchainKHR(handle, &ci, nullptr, &object));
|
|
return SwapchainKHR(object, handle, *dld);
|
|
}
|
|
|
|
DeviceMemory Device::TryAllocateMemory(const VkMemoryAllocateInfo& ai) const noexcept {
|
|
VkDeviceMemory memory;
|
|
if (dld->vkAllocateMemory(handle, &ai, nullptr, &memory) != VK_SUCCESS) {
|
|
return {};
|
|
}
|
|
return DeviceMemory(memory, handle, *dld);
|
|
}
|
|
|
|
DeviceMemory Device::AllocateMemory(const VkMemoryAllocateInfo& ai) const {
|
|
VkDeviceMemory memory;
|
|
Check(dld->vkAllocateMemory(handle, &ai, nullptr, &memory));
|
|
return DeviceMemory(memory, handle, *dld);
|
|
}
|
|
|
|
VkMemoryRequirements Device::GetBufferMemoryRequirements(VkBuffer buffer) const noexcept {
|
|
VkMemoryRequirements requirements;
|
|
dld->vkGetBufferMemoryRequirements(handle, buffer, &requirements);
|
|
return requirements;
|
|
}
|
|
|
|
VkMemoryRequirements Device::GetImageMemoryRequirements(VkImage image) const noexcept {
|
|
VkMemoryRequirements requirements;
|
|
dld->vkGetImageMemoryRequirements(handle, image, &requirements);
|
|
return requirements;
|
|
}
|
|
|
|
void Device::UpdateDescriptorSets(Span<VkWriteDescriptorSet> writes,
|
|
Span<VkCopyDescriptorSet> copies) const noexcept {
|
|
dld->vkUpdateDescriptorSets(handle, writes.size(), writes.data(), copies.size(), copies.data());
|
|
}
|
|
|
|
VkPhysicalDeviceProperties PhysicalDevice::GetProperties() const noexcept {
|
|
VkPhysicalDeviceProperties properties;
|
|
dld->vkGetPhysicalDeviceProperties(physical_device, &properties);
|
|
return properties;
|
|
}
|
|
|
|
void PhysicalDevice::GetProperties2KHR(VkPhysicalDeviceProperties2KHR& properties) const noexcept {
|
|
dld->vkGetPhysicalDeviceProperties2KHR(physical_device, &properties);
|
|
}
|
|
|
|
VkPhysicalDeviceFeatures PhysicalDevice::GetFeatures() const noexcept {
|
|
VkPhysicalDeviceFeatures2KHR features2;
|
|
features2.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_FEATURES_2_KHR;
|
|
features2.pNext = nullptr;
|
|
dld->vkGetPhysicalDeviceFeatures2KHR(physical_device, &features2);
|
|
return features2.features;
|
|
}
|
|
|
|
void PhysicalDevice::GetFeatures2KHR(VkPhysicalDeviceFeatures2KHR& features) const noexcept {
|
|
dld->vkGetPhysicalDeviceFeatures2KHR(physical_device, &features);
|
|
}
|
|
|
|
VkFormatProperties PhysicalDevice::GetFormatProperties(VkFormat format) const noexcept {
|
|
VkFormatProperties properties;
|
|
dld->vkGetPhysicalDeviceFormatProperties(physical_device, format, &properties);
|
|
return properties;
|
|
}
|
|
|
|
std::vector<VkExtensionProperties> PhysicalDevice::EnumerateDeviceExtensionProperties() const {
|
|
u32 num;
|
|
dld->vkEnumerateDeviceExtensionProperties(physical_device, nullptr, &num, nullptr);
|
|
std::vector<VkExtensionProperties> properties(num);
|
|
dld->vkEnumerateDeviceExtensionProperties(physical_device, nullptr, &num, properties.data());
|
|
return properties;
|
|
}
|
|
|
|
std::vector<VkQueueFamilyProperties> PhysicalDevice::GetQueueFamilyProperties() const {
|
|
u32 num;
|
|
dld->vkGetPhysicalDeviceQueueFamilyProperties(physical_device, &num, nullptr);
|
|
std::vector<VkQueueFamilyProperties> properties(num);
|
|
dld->vkGetPhysicalDeviceQueueFamilyProperties(physical_device, &num, properties.data());
|
|
return properties;
|
|
}
|
|
|
|
bool PhysicalDevice::GetSurfaceSupportKHR(u32 queue_family_index, VkSurfaceKHR surface) const {
|
|
VkBool32 supported;
|
|
Check(dld->vkGetPhysicalDeviceSurfaceSupportKHR(physical_device, queue_family_index, surface,
|
|
&supported));
|
|
return supported == VK_TRUE;
|
|
}
|
|
|
|
VkSurfaceCapabilitiesKHR PhysicalDevice::GetSurfaceCapabilitiesKHR(VkSurfaceKHR surface) const {
|
|
VkSurfaceCapabilitiesKHR capabilities;
|
|
Check(dld->vkGetPhysicalDeviceSurfaceCapabilitiesKHR(physical_device, surface, &capabilities));
|
|
return capabilities;
|
|
}
|
|
|
|
std::vector<VkSurfaceFormatKHR> PhysicalDevice::GetSurfaceFormatsKHR(VkSurfaceKHR surface) const {
|
|
u32 num;
|
|
Check(dld->vkGetPhysicalDeviceSurfaceFormatsKHR(physical_device, surface, &num, nullptr));
|
|
std::vector<VkSurfaceFormatKHR> formats(num);
|
|
Check(
|
|
dld->vkGetPhysicalDeviceSurfaceFormatsKHR(physical_device, surface, &num, formats.data()));
|
|
return formats;
|
|
}
|
|
|
|
std::vector<VkPresentModeKHR> PhysicalDevice::GetSurfacePresentModesKHR(
|
|
VkSurfaceKHR surface) const {
|
|
u32 num;
|
|
Check(dld->vkGetPhysicalDeviceSurfacePresentModesKHR(physical_device, surface, &num, nullptr));
|
|
std::vector<VkPresentModeKHR> modes(num);
|
|
Check(dld->vkGetPhysicalDeviceSurfacePresentModesKHR(physical_device, surface, &num,
|
|
modes.data()));
|
|
return modes;
|
|
}
|
|
|
|
VkPhysicalDeviceMemoryProperties PhysicalDevice::GetMemoryProperties() const noexcept {
|
|
VkPhysicalDeviceMemoryProperties properties;
|
|
dld->vkGetPhysicalDeviceMemoryProperties(physical_device, &properties);
|
|
return properties;
|
|
}
|
|
|
|
u32 AvailableVersion(const InstanceDispatch& dld) noexcept {
|
|
PFN_vkEnumerateInstanceVersion vkEnumerateInstanceVersion;
|
|
if (!Proc(vkEnumerateInstanceVersion, dld, "vkEnumerateInstanceVersion")) {
|
|
// If the procedure is not found, Vulkan 1.0 is assumed
|
|
return VK_API_VERSION_1_0;
|
|
}
|
|
u32 version;
|
|
if (const VkResult result = vkEnumerateInstanceVersion(&version); result != VK_SUCCESS) {
|
|
LOG_ERROR(Render_Vulkan, "vkEnumerateInstanceVersion returned {}, assuming Vulkan 1.1",
|
|
ToString(result));
|
|
return VK_API_VERSION_1_1;
|
|
}
|
|
return version;
|
|
}
|
|
|
|
std::optional<std::vector<VkExtensionProperties>> EnumerateInstanceExtensionProperties(
|
|
const InstanceDispatch& dld) {
|
|
u32 num;
|
|
if (dld.vkEnumerateInstanceExtensionProperties(nullptr, &num, nullptr) != VK_SUCCESS) {
|
|
return std::nullopt;
|
|
}
|
|
std::vector<VkExtensionProperties> properties(num);
|
|
if (dld.vkEnumerateInstanceExtensionProperties(nullptr, &num, properties.data()) !=
|
|
VK_SUCCESS) {
|
|
return std::nullopt;
|
|
}
|
|
return properties;
|
|
}
|
|
|
|
std::optional<std::vector<VkLayerProperties>> EnumerateInstanceLayerProperties(
|
|
const InstanceDispatch& dld) {
|
|
u32 num;
|
|
if (dld.vkEnumerateInstanceLayerProperties(&num, nullptr) != VK_SUCCESS) {
|
|
return std::nullopt;
|
|
}
|
|
std::vector<VkLayerProperties> properties(num);
|
|
if (dld.vkEnumerateInstanceLayerProperties(&num, properties.data()) != VK_SUCCESS) {
|
|
return std::nullopt;
|
|
}
|
|
return properties;
|
|
}
|
|
|
|
} // namespace Vulkan::vk
|