mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-25 01:19:19 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			272 lines
		
	
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			272 lines
		
	
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2016 Dolphin Emulator Project
 | |
| // Licensed under GPLv2+
 | |
| // Refer to the license.txt file included.
 | |
| 
 | |
| #include <vector>
 | |
| 
 | |
| #include "Core/Host.h"
 | |
| 
 | |
| #include "VideoBackends/Vulkan/CommandBufferManager.h"
 | |
| #include "VideoBackends/Vulkan/Constants.h"
 | |
| #include "VideoBackends/Vulkan/FramebufferManager.h"
 | |
| #include "VideoBackends/Vulkan/ObjectCache.h"
 | |
| #include "VideoBackends/Vulkan/PerfQuery.h"
 | |
| #include "VideoBackends/Vulkan/Renderer.h"
 | |
| #include "VideoBackends/Vulkan/StateTracker.h"
 | |
| #include "VideoBackends/Vulkan/SwapChain.h"
 | |
| #include "VideoBackends/Vulkan/TextureCache.h"
 | |
| #include "VideoBackends/Vulkan/VertexManager.h"
 | |
| #include "VideoBackends/Vulkan/VideoBackend.h"
 | |
| #include "VideoBackends/Vulkan/VulkanContext.h"
 | |
| 
 | |
| #include "VideoCommon/DriverDetails.h"
 | |
| #include "VideoCommon/OnScreenDisplay.h"
 | |
| #include "VideoCommon/VideoBackendBase.h"
 | |
| #include "VideoCommon/VideoConfig.h"
 | |
| 
 | |
| namespace Vulkan
 | |
| {
 | |
| void VideoBackend::InitBackendInfo()
 | |
| {
 | |
|   VulkanContext::PopulateBackendInfo(&g_Config);
 | |
| 
 | |
|   if (LoadVulkanLibrary())
 | |
|   {
 | |
|     VkInstance temp_instance = VulkanContext::CreateVulkanInstance(false, false);
 | |
|     if (temp_instance)
 | |
|     {
 | |
|       if (LoadVulkanInstanceFunctions(temp_instance))
 | |
|       {
 | |
|         VulkanContext::GPUList gpu_list = VulkanContext::EnumerateGPUs(temp_instance);
 | |
|         VulkanContext::PopulateBackendInfoAdapters(&g_Config, gpu_list);
 | |
| 
 | |
|         if (!gpu_list.empty())
 | |
|         {
 | |
|           // Use the selected adapter, or the first to fill features.
 | |
|           size_t device_index = static_cast<size_t>(g_Config.iAdapter);
 | |
|           if (device_index >= gpu_list.size())
 | |
|             device_index = 0;
 | |
| 
 | |
|           VkPhysicalDevice gpu = gpu_list[device_index];
 | |
|           VkPhysicalDeviceProperties properties;
 | |
|           vkGetPhysicalDeviceProperties(gpu, &properties);
 | |
|           VkPhysicalDeviceFeatures features;
 | |
|           vkGetPhysicalDeviceFeatures(gpu, &features);
 | |
|           VulkanContext::PopulateBackendInfoFeatures(&g_Config, gpu, features);
 | |
|           VulkanContext::PopulateBackendInfoMultisampleModes(&g_Config, gpu, properties);
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       vkDestroyInstance(temp_instance, nullptr);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|       PanicAlert("Failed to create Vulkan instance.");
 | |
|     }
 | |
| 
 | |
|     UnloadVulkanLibrary();
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     PanicAlert("Failed to load Vulkan library.");
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool VideoBackend::Initialize(void* window_handle)
 | |
| {
 | |
|   if (!LoadVulkanLibrary())
 | |
|   {
 | |
|     PanicAlert("Failed to load Vulkan library.");
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // HACK: Use InitBackendInfo to initially populate backend features.
 | |
|   // This is because things like stereo get disabled when the config is validated,
 | |
|   // which happens before our device is created (settings control instance behavior),
 | |
|   // and we don't want that to happen if the device actually supports it.
 | |
|   InitBackendInfo();
 | |
|   InitializeShared();
 | |
| 
 | |
|   // Check for presence of the debug layer before trying to enable it
 | |
|   bool enable_validation_layer = g_Config.bEnableValidationLayer;
 | |
|   if (enable_validation_layer && !VulkanContext::CheckValidationLayerAvailablility())
 | |
|   {
 | |
|     WARN_LOG(VIDEO, "Validation layer requested but not available, disabling.");
 | |
|     enable_validation_layer = false;
 | |
|   }
 | |
| 
 | |
|   // Create Vulkan instance, needed before we can create a surface.
 | |
|   bool enable_surface = (window_handle != nullptr);
 | |
|   VkInstance instance =
 | |
|       VulkanContext::CreateVulkanInstance(enable_surface, enable_validation_layer);
 | |
|   if (instance == VK_NULL_HANDLE)
 | |
|   {
 | |
|     PanicAlert("Failed to create Vulkan instance.");
 | |
|     UnloadVulkanLibrary();
 | |
|     ShutdownShared();
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // Load instance function pointers
 | |
|   if (!LoadVulkanInstanceFunctions(instance))
 | |
|   {
 | |
|     PanicAlert("Failed to load Vulkan instance functions.");
 | |
|     vkDestroyInstance(instance, nullptr);
 | |
|     UnloadVulkanLibrary();
 | |
|     ShutdownShared();
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // Create Vulkan surface
 | |
|   VkSurfaceKHR surface = VK_NULL_HANDLE;
 | |
|   if (enable_surface)
 | |
|   {
 | |
|     surface = SwapChain::CreateVulkanSurface(instance, window_handle);
 | |
|     if (surface == VK_NULL_HANDLE)
 | |
|     {
 | |
|       PanicAlert("Failed to create Vulkan surface.");
 | |
|       vkDestroyInstance(instance, nullptr);
 | |
|       UnloadVulkanLibrary();
 | |
|       ShutdownShared();
 | |
|       return false;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Fill the adapter list, and check if the user has selected an invalid device
 | |
|   // For some reason nvidia's driver crashes randomly if you call vkEnumeratePhysicalDevices
 | |
|   // after creating a device..
 | |
|   VulkanContext::GPUList gpu_list = VulkanContext::EnumerateGPUs(instance);
 | |
|   size_t selected_adapter_index = static_cast<size_t>(g_Config.iAdapter);
 | |
|   if (gpu_list.empty())
 | |
|   {
 | |
|     PanicAlert("No Vulkan physical devices available.");
 | |
|     if (surface != VK_NULL_HANDLE)
 | |
|       vkDestroySurfaceKHR(instance, surface, nullptr);
 | |
| 
 | |
|     vkDestroyInstance(instance, nullptr);
 | |
|     UnloadVulkanLibrary();
 | |
|     ShutdownShared();
 | |
|     return false;
 | |
|   }
 | |
|   else if (selected_adapter_index >= gpu_list.size())
 | |
|   {
 | |
|     WARN_LOG(VIDEO, "Vulkan adapter index out of range, selecting first adapter.");
 | |
|     selected_adapter_index = 0;
 | |
|   }
 | |
| 
 | |
|   // Pass ownership over to VulkanContext, and let it take care of everything.
 | |
|   g_vulkan_context = VulkanContext::Create(instance, gpu_list[selected_adapter_index], surface,
 | |
|                                            &g_Config, enable_validation_layer);
 | |
|   if (!g_vulkan_context)
 | |
|   {
 | |
|     PanicAlert("Failed to create Vulkan device");
 | |
|     UnloadVulkanLibrary();
 | |
|     ShutdownShared();
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // Create command buffers. We do this separately because the other classes depend on it.
 | |
|   g_command_buffer_mgr = std::make_unique<CommandBufferManager>(g_Config.bBackendMultithreading);
 | |
|   if (!g_command_buffer_mgr->Initialize())
 | |
|   {
 | |
|     PanicAlert("Failed to create Vulkan command buffers");
 | |
|     g_command_buffer_mgr.reset();
 | |
|     g_vulkan_context.reset();
 | |
|     UnloadVulkanLibrary();
 | |
|     ShutdownShared();
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // Create main wrapper instances.
 | |
|   g_object_cache = std::make_unique<ObjectCache>();
 | |
|   g_framebuffer_manager = std::make_unique<FramebufferManager>();
 | |
|   g_renderer = std::make_unique<Renderer>();
 | |
| 
 | |
|   // Cast to our wrapper classes, so we can call the init methods.
 | |
|   Renderer* renderer = static_cast<Renderer*>(g_renderer.get());
 | |
|   FramebufferManager* framebuffer_mgr =
 | |
|       static_cast<FramebufferManager*>(g_framebuffer_manager.get());
 | |
| 
 | |
|   // Invoke init methods on main wrapper classes.
 | |
|   // These have to be done before the others because the destructors
 | |
|   // for the remaining classes may call methods on these.
 | |
|   if (!g_object_cache->Initialize() || !framebuffer_mgr->Initialize() ||
 | |
|       !renderer->Initialize(framebuffer_mgr, window_handle, surface))
 | |
|   {
 | |
|     PanicAlert("Failed to initialize Vulkan classes.");
 | |
|     g_renderer.reset();
 | |
|     g_object_cache.reset();
 | |
|     g_command_buffer_mgr.reset();
 | |
|     g_vulkan_context.reset();
 | |
|     UnloadVulkanLibrary();
 | |
|     ShutdownShared();
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // Create remaining wrapper instances.
 | |
|   g_vertex_manager = std::make_unique<VertexManager>();
 | |
|   g_texture_cache = std::make_unique<TextureCache>();
 | |
|   g_perf_query = std::make_unique<PerfQuery>();
 | |
|   VertexManager* vertex_manager = static_cast<VertexManager*>(g_vertex_manager.get());
 | |
|   TextureCache* texture_cache = static_cast<TextureCache*>(g_texture_cache.get());
 | |
|   PerfQuery* perf_query = static_cast<PerfQuery*>(g_perf_query.get());
 | |
|   if (!vertex_manager->Initialize(renderer->GetStateTracker()) ||
 | |
|       !texture_cache->Initialize(renderer->GetStateTracker()) ||
 | |
|       !perf_query->Initialize(renderer->GetStateTracker()))
 | |
|   {
 | |
|     PanicAlert("Failed to initialize Vulkan classes.");
 | |
|     g_perf_query.reset();
 | |
|     g_texture_cache.reset();
 | |
|     g_vertex_manager.reset();
 | |
|     g_renderer.reset();
 | |
|     g_object_cache.reset();
 | |
|     g_command_buffer_mgr.reset();
 | |
|     g_vulkan_context.reset();
 | |
|     UnloadVulkanLibrary();
 | |
|     ShutdownShared();
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| // This is called after Initialize() from the Core
 | |
| // Run from the graphics thread
 | |
| void VideoBackend::Video_Prepare()
 | |
| {
 | |
|   // Display the name so the user knows which device was actually created
 | |
|   OSD::AddMessage(StringFromFormat("Using physical adapter %s",
 | |
|                                    g_vulkan_context->GetDeviceProperties().deviceName)
 | |
|                       .c_str(),
 | |
|                   5000);
 | |
| }
 | |
| 
 | |
| void VideoBackend::Shutdown()
 | |
| {
 | |
|   g_command_buffer_mgr->WaitForGPUIdle();
 | |
| 
 | |
|   g_object_cache.reset();
 | |
|   g_command_buffer_mgr.reset();
 | |
|   g_vulkan_context.reset();
 | |
| 
 | |
|   UnloadVulkanLibrary();
 | |
| 
 | |
|   ShutdownShared();
 | |
| }
 | |
| 
 | |
| void VideoBackend::Video_Cleanup()
 | |
| {
 | |
|   g_command_buffer_mgr->WaitForGPUIdle();
 | |
| 
 | |
|   // Save all cached pipelines out to disk for next time.
 | |
|   g_object_cache->SavePipelineCache();
 | |
| 
 | |
|   g_texture_cache.reset();
 | |
|   g_perf_query.reset();
 | |
|   g_vertex_manager.reset();
 | |
|   g_renderer.reset();
 | |
|   g_framebuffer_manager.reset();
 | |
| 
 | |
|   CleanupShared();
 | |
| }
 | |
| }
 |