// SPDX-FileCopyrightText: 2019-2023 Connor McLaughlin // SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0) // No better place for this.. #define VMA_IMPLEMENTATION #include "vulkan_loader.h" #include "common/assert.h" #include "common/dynamic_library.h" #include "common/log.h" #include #include #include #include #include Log_SetChannel(VulkanDevice); extern "C" { #define VULKAN_MODULE_ENTRY_POINT(name, required) PFN_##name name; #define VULKAN_INSTANCE_ENTRY_POINT(name, required) PFN_##name name; #define VULKAN_DEVICE_ENTRY_POINT(name, required) PFN_##name name; #include "vulkan_entry_points.inl" #undef VULKAN_DEVICE_ENTRY_POINT #undef VULKAN_INSTANCE_ENTRY_POINT #undef VULKAN_MODULE_ENTRY_POINT } void Vulkan::ResetVulkanLibraryFunctionPointers() { #define VULKAN_MODULE_ENTRY_POINT(name, required) name = nullptr; #define VULKAN_INSTANCE_ENTRY_POINT(name, required) name = nullptr; #define VULKAN_DEVICE_ENTRY_POINT(name, required) name = nullptr; #include "vulkan_entry_points.inl" #undef VULKAN_DEVICE_ENTRY_POINT #undef VULKAN_INSTANCE_ENTRY_POINT #undef VULKAN_MODULE_ENTRY_POINT } static DynamicLibrary s_vulkan_library; bool Vulkan::IsVulkanLibraryLoaded() { return s_vulkan_library.IsOpen(); } bool Vulkan::LoadVulkanLibrary(Error* error) { AssertMsg(!s_vulkan_library.IsOpen(), "Vulkan module is not loaded."); #ifdef __APPLE__ // Check if a path to a specific Vulkan library has been specified. char* libvulkan_env = getenv("LIBVULKAN_PATH"); if (libvulkan_env) s_vulkan_library.Open(libvulkan_env, error); if (!s_vulkan_library.IsOpen() && !s_vulkan_library.Open(DynamicLibrary::GetVersionedFilename("MoltenVK").c_str(), error)) { return false; } #else // try versioned first, then unversioned. if (!s_vulkan_library.Open(DynamicLibrary::GetVersionedFilename("vulkan", 1).c_str(), error) && !s_vulkan_library.Open(DynamicLibrary::GetVersionedFilename("vulkan").c_str(), error)) { return false; } #endif bool required_functions_missing = false; #define VULKAN_MODULE_ENTRY_POINT(name, required) \ if (!s_vulkan_library.GetSymbol(#name, &name)) \ { \ Log_ErrorFmt("Vulkan: Failed to load required module function {}", #name); \ required_functions_missing = true; \ } #include "vulkan_entry_points.inl" #undef VULKAN_MODULE_ENTRY_POINT if (required_functions_missing) { ResetVulkanLibraryFunctionPointers(); s_vulkan_library.Close(); return false; } return true; } void Vulkan::UnloadVulkanLibrary() { ResetVulkanLibraryFunctionPointers(); s_vulkan_library.Close(); } bool Vulkan::LoadVulkanInstanceFunctions(VkInstance instance) { bool required_functions_missing = false; auto LoadFunction = [&](PFN_vkVoidFunction* func_ptr, const char* name, bool is_required) { *func_ptr = vkGetInstanceProcAddr(instance, name); if (!(*func_ptr) && is_required) { std::fprintf(stderr, "Vulkan: Failed to load required instance function %s\n", name); required_functions_missing = true; } }; #define VULKAN_INSTANCE_ENTRY_POINT(name, required) \ LoadFunction(reinterpret_cast(&name), #name, required); #include "vulkan_entry_points.inl" #undef VULKAN_INSTANCE_ENTRY_POINT return !required_functions_missing; } bool Vulkan::LoadVulkanDeviceFunctions(VkDevice device) { bool required_functions_missing = false; auto LoadFunction = [&](PFN_vkVoidFunction* func_ptr, const char* name, bool is_required) { *func_ptr = vkGetDeviceProcAddr(device, name); if (!(*func_ptr) && is_required) { std::fprintf(stderr, "Vulkan: Failed to load required device function %s\n", name); required_functions_missing = true; } }; #define VULKAN_DEVICE_ENTRY_POINT(name, required) \ LoadFunction(reinterpret_cast(&name), #name, required); #include "vulkan_entry_points.inl" #undef VULKAN_DEVICE_ENTRY_POINT return !required_functions_missing; }