forked from Mirror/Ryujinx
c8f9292bab
* drop split devices, rebase * add fallback to opengl if vulkan is not available * addressed review * ensure present image references are incremented and decremented when necessary * allow changing vsync for vulkan * fix screenshot on avalonia vulkan * save favorite when toggled * improve sync between popups * use separate devices for each new window * fix crash when closing window * addressed review * don't create the main window with immediate mode * change skia vk delegate to method * update vulkan throwonerror * addressed review
135 lines
4.7 KiB
C#
135 lines
4.7 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Runtime.InteropServices;
|
|
using Silk.NET.Core;
|
|
using Silk.NET.Vulkan;
|
|
using Silk.NET.Vulkan.Extensions.EXT;
|
|
|
|
namespace Ryujinx.Ava.Ui.Vulkan
|
|
{
|
|
public class VulkanInstance : IDisposable
|
|
{
|
|
private const string EngineName = "Avalonia Vulkan";
|
|
|
|
private VulkanInstance(Instance apiHandle, Vk api)
|
|
{
|
|
InternalHandle = apiHandle;
|
|
Api = api;
|
|
}
|
|
|
|
public IntPtr Handle => InternalHandle.Handle;
|
|
|
|
internal Instance InternalHandle { get; }
|
|
public Vk Api { get; }
|
|
|
|
internal static IEnumerable<string> RequiredInstanceExtensions
|
|
{
|
|
get
|
|
{
|
|
yield return "VK_KHR_surface";
|
|
if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux))
|
|
{
|
|
yield return "VK_KHR_xlib_surface";
|
|
}
|
|
else if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
|
|
{
|
|
yield return "VK_KHR_win32_surface";
|
|
}
|
|
}
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
Api?.DestroyInstance(InternalHandle, Span<AllocationCallbacks>.Empty);
|
|
Api?.Dispose();
|
|
}
|
|
|
|
internal static unsafe VulkanInstance Create(VulkanOptions options)
|
|
{
|
|
var api = Vk.GetApi();
|
|
var applicationName = Marshal.StringToHGlobalAnsi(options.ApplicationName);
|
|
var engineName = Marshal.StringToHGlobalAnsi(EngineName);
|
|
var enabledExtensions = new List<string>(options.InstanceExtensions);
|
|
|
|
enabledExtensions.AddRange(RequiredInstanceExtensions);
|
|
|
|
var applicationInfo = new ApplicationInfo
|
|
{
|
|
PApplicationName = (byte*)applicationName,
|
|
ApiVersion = Vk.Version12.Value,
|
|
PEngineName = (byte*)engineName,
|
|
EngineVersion = new Version32(1, 0, 0),
|
|
ApplicationVersion = new Version32(1, 0, 0)
|
|
};
|
|
|
|
var enabledLayers = new HashSet<string>();
|
|
|
|
if (options.UseDebug)
|
|
{
|
|
enabledExtensions.Add(ExtDebugUtils.ExtensionName);
|
|
enabledExtensions.Add(ExtDebugReport.ExtensionName);
|
|
if (IsLayerAvailable(api, "VK_LAYER_KHRONOS_validation"))
|
|
enabledLayers.Add("VK_LAYER_KHRONOS_validation");
|
|
}
|
|
|
|
foreach (var layer in options.EnabledLayers)
|
|
enabledLayers.Add(layer);
|
|
|
|
var ppEnabledExtensions = stackalloc IntPtr[enabledExtensions.Count];
|
|
var ppEnabledLayers = stackalloc IntPtr[enabledLayers.Count];
|
|
|
|
for (var i = 0; i < enabledExtensions.Count; i++)
|
|
ppEnabledExtensions[i] = Marshal.StringToHGlobalAnsi(enabledExtensions[i]);
|
|
|
|
var layers = enabledLayers.ToList();
|
|
|
|
for (var i = 0; i < enabledLayers.Count; i++)
|
|
ppEnabledLayers[i] = Marshal.StringToHGlobalAnsi(layers[i]);
|
|
|
|
var instanceCreateInfo = new InstanceCreateInfo
|
|
{
|
|
SType = StructureType.InstanceCreateInfo,
|
|
PApplicationInfo = &applicationInfo,
|
|
PpEnabledExtensionNames = (byte**)ppEnabledExtensions,
|
|
PpEnabledLayerNames = (byte**)ppEnabledLayers,
|
|
EnabledExtensionCount = (uint)enabledExtensions.Count,
|
|
EnabledLayerCount = (uint)enabledLayers.Count
|
|
};
|
|
|
|
api.CreateInstance(in instanceCreateInfo, null, out var instance).ThrowOnError();
|
|
|
|
Marshal.FreeHGlobal(applicationName);
|
|
Marshal.FreeHGlobal(engineName);
|
|
|
|
for (var i = 0; i < enabledExtensions.Count; i++) Marshal.FreeHGlobal(ppEnabledExtensions[i]);
|
|
|
|
for (var i = 0; i < enabledLayers.Count; i++) Marshal.FreeHGlobal(ppEnabledLayers[i]);
|
|
|
|
return new VulkanInstance(instance, api);
|
|
}
|
|
|
|
private static unsafe bool IsLayerAvailable(Vk api, string layerName)
|
|
{
|
|
uint layerPropertiesCount;
|
|
|
|
api.EnumerateInstanceLayerProperties(&layerPropertiesCount, null).ThrowOnError();
|
|
|
|
var layerProperties = new LayerProperties[layerPropertiesCount];
|
|
|
|
fixed (LayerProperties* pLayerProperties = layerProperties)
|
|
{
|
|
api.EnumerateInstanceLayerProperties(&layerPropertiesCount, layerProperties).ThrowOnError();
|
|
|
|
for (var i = 0; i < layerPropertiesCount; i++)
|
|
{
|
|
var currentLayerName = Marshal.PtrToStringAnsi((IntPtr)pLayerProperties[i].LayerName);
|
|
|
|
if (currentLayerName == layerName) return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|
|
}
|