* Ported the old Vulkan PR

* chop-chop

* Support for external objects in vulkan backend

* Fixed structure type

* Removed debug code

* sln fix

* Don't force vulkan on windows
This commit is contained in:
Nikita Tsukanov 2024-04-29 14:47:49 +05:00 коммит произвёл GitHub
Родитель 76a1c70758
Коммит 3a1a69b145
Не найден ключ, соответствующий данной подписи
Идентификатор ключа GPG: B5690EEEBB952194
57 изменённых файлов: 6239 добавлений и 50 удалений

Просмотреть файл

@ -30,6 +30,7 @@
"src\\Avalonia.MicroCom\\Avalonia.MicroCom.csproj",
"src\\Avalonia.Native\\Avalonia.Native.csproj",
"src\\Avalonia.OpenGL\\Avalonia.OpenGL.csproj",
"src\\Avalonia.Vulkan\\Avalonia.Vulkan.csproj",
"src\\Avalonia.ReactiveUI\\Avalonia.ReactiveUI.csproj",
"src\\Avalonia.Remote.Protocol\\Avalonia.Remote.Protocol.csproj",
"src\\Avalonia.Themes.Fluent\\Avalonia.Themes.Fluent.csproj",

Просмотреть файл

@ -298,6 +298,8 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "UnloadableAssemblyLoadConte
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "UnloadableAssemblyLoadContextPlug", "samples\UnloadableAssemblyLoadContext\UnloadableAssemblyLoadContextPlug\UnloadableAssemblyLoadContextPlug.csproj", "{DA5F1FF9-4259-4C54-B443-85CFA226EE6A}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Avalonia.Vulkan", "src\Avalonia.Vulkan\Avalonia.Vulkan.csproj", "{3E2DE2B6-13BC-4C27-BCB9-A423B86CAF77}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Avalonia.RenderTests.WpfCompare", "tests\Avalonia.RenderTests.WpfCompare\Avalonia.RenderTests.WpfCompare.csproj", "{9AE1B827-21AC-4063-AB22-C8804B7F931E}"
EndProject
Global
@ -676,6 +678,10 @@ Global
{60B4ED1F-ECFA-453B-8A70-1788261C8355}.Debug|Any CPU.Build.0 = Debug|Any CPU
{60B4ED1F-ECFA-453B-8A70-1788261C8355}.Release|Any CPU.ActiveCfg = Release|Any CPU
{60B4ED1F-ECFA-453B-8A70-1788261C8355}.Release|Any CPU.Build.0 = Release|Any CPU
{3E2DE2B6-13BC-4C27-BCB9-A423B86CAF77}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{3E2DE2B6-13BC-4C27-BCB9-A423B86CAF77}.Debug|Any CPU.Build.0 = Debug|Any CPU
{3E2DE2B6-13BC-4C27-BCB9-A423B86CAF77}.Release|Any CPU.ActiveCfg = Release|Any CPU
{3E2DE2B6-13BC-4C27-BCB9-A423B86CAF77}.Release|Any CPU.Build.0 = Release|Any CPU
{B0FD6A48-FBAB-4676-B36A-DE76B0922B12}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{B0FD6A48-FBAB-4676-B36A-DE76B0922B12}.Debug|Any CPU.Build.0 = Debug|Any CPU
{B0FD6A48-FBAB-4676-B36A-DE76B0922B12}.Release|Any CPU.ActiveCfg = Release|Any CPU

Просмотреть файл

@ -39,4 +39,5 @@
<s:Boolean x:Key="/Default/Environment/SettingsMigration/IsMigratorApplied/=JetBrains_002EReSharper_002EFeature_002EServices_002EDaemon_002ESettings_002EMigration_002ESwaWarningsModeSettingsMigrate/@EntryIndexedValue">True</s:Boolean>
<s:Boolean x:Key="/Default/UserDictionary/Words/=Activatable/@EntryIndexedValue">True</s:Boolean>
<s:Boolean x:Key="/Default/UserDictionary/Words/=Avalonia/@EntryIndexedValue">True</s:Boolean>
<s:Boolean x:Key="/Default/UserDictionary/Words/=Fcitx/@EntryIndexedValue">True</s:Boolean></wpf:ResourceDictionary>
<s:Boolean x:Key="/Default/UserDictionary/Words/=Fcitx/@EntryIndexedValue">True</s:Boolean>
<s:Boolean x:Key="/Default/UserDictionary/Words/=swapchain/@EntryIndexedValue">True</s:Boolean></wpf:ResourceDictionary>

Просмотреть файл

@ -5,6 +5,7 @@
<ProjectReference Include="$(MSBuildThisFileDirectory)/../src/Avalonia.DesignerSupport/Avalonia.DesignerSupport.csproj" />
<ProjectReference Include="$(MSBuildThisFileDirectory)/../src/Avalonia.OpenGL/Avalonia.OpenGL.csproj" />
<ProjectReference Include="$(MSBuildThisFileDirectory)/../src/Avalonia.Metal/Avalonia.Metal.csproj" />
<ProjectReference Include="$(MSBuildThisFileDirectory)/../src/Avalonia.Vulkan/Avalonia.Vulkan.csproj" />
<ProjectReference Include="$(MSBuildThisFileDirectory)/../src/Avalonia.Dialogs/Avalonia.Dialogs.csproj" />
<ProjectReference Include="$(MSBuildThisFileDirectory)/../src/Markup/Avalonia.Markup/Avalonia.Markup.csproj" />
<ProjectReference Include="$(MSBuildThisFileDirectory)/../src/Markup/Avalonia.Markup.Xaml/Avalonia.Markup.Xaml.csproj" />

Просмотреть файл

@ -13,7 +13,7 @@ using Avalonia.LinuxFramebuffer.Output;
using Avalonia.LogicalTree;
using Avalonia.Rendering.Composition;
using Avalonia.Threading;
using Avalonia.Vulkan;
using ControlCatalog.Pages;
namespace ControlCatalog.NetCore
@ -133,7 +133,15 @@ namespace ControlCatalog.NetCore
{
EnableMultiTouch = true,
UseDBusMenu = true,
EnableIme = true
EnableIme = true,
})
.With(new VulkanOptions
{
VulkanInstanceCreationOptions = new ()
{
UseDebug = true
}
})
.With(new CompositionOptions()
{

Просмотреть файл

@ -1,5 +1,8 @@
global using System.Reactive.Disposables;
using Avalonia;
using Avalonia.Logging;
using Avalonia.Vulkan;
namespace GpuInterop
{
public class Program
@ -10,6 +13,21 @@ namespace GpuInterop
public static AppBuilder BuildAvaloniaApp() =>
AppBuilder.Configure<App>()
.UsePlatformDetect()
.LogToTrace();
.With(new Win32PlatformOptions
{
RenderingMode = new []
{
Win32RenderingMode.Vulkan
}
})
.With(new X11PlatformOptions(){RenderingMode =new[] { X11RenderingMode.Vulkan } })
.With(new VulkanOptions()
{
VulkanInstanceCreationOptions = new VulkanInstanceCreationOptions()
{
UseDebug = true
}
})
.LogToTrace(LogEventLevel.Debug, "Vulkan");
}
}

Просмотреть файл

@ -397,7 +397,7 @@ unsafe class VulkanContent : IDisposable
Matrix4x4.CreatePerspectiveFieldOfView((float)(Math.PI / 4), (float)size.Width / size.Height,
0.01f, 1000);
_colorAttachment = new VulkanImage(_context, (uint)Format.R8G8B8A8Unorm, size, false);
_colorAttachment = new VulkanImage(_context, (uint)Format.R8G8B8A8Unorm, size, false, Array.Empty<string>());
CreateDepthAttachment(size);
var api = _context.Api;

Просмотреть файл

@ -101,8 +101,10 @@ public unsafe class VulkanContext : IDisposable
if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
{
if (!gpuInterop.SupportedImageHandleTypes.Contains(KnownPlatformGraphicsExternalImageHandleTypes
if (!(gpuInterop.SupportedImageHandleTypes.Contains(KnownPlatformGraphicsExternalImageHandleTypes
.D3D11TextureGlobalSharedHandle)
|| gpuInterop.SupportedImageHandleTypes.Contains(KnownPlatformGraphicsExternalImageHandleTypes
.VulkanOpaqueNtHandle))
)
return (null, "Image sharing is not supported by the current backend");
requireDeviceExtensions.Add(KhrExternalMemoryWin32.ExtensionName);
@ -240,10 +242,13 @@ public unsafe class VulkanContext : IDisposable
}
});
D3DDevice? d3dDevice = null;
if (physicalDeviceIDProperties.DeviceLuidvalid &&
RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
RuntimeInformation.IsOSPlatform(OSPlatform.Windows) &&
!gpuInterop.SupportedImageHandleTypes.Contains(KnownPlatformGraphicsExternalImageHandleTypes.VulkanOpaqueNtHandle)
)
d3dDevice = D3DMemoryHelper.CreateDeviceByLuid(
new Span<byte>(physicalDeviceIDProperties.DeviceLuid, 8));

Просмотреть файл

@ -1,5 +1,7 @@
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Runtime.InteropServices;
using Avalonia;
using Avalonia.Platform;
@ -44,7 +46,7 @@ public unsafe class VulkanImage : IDisposable
public uint CurrentLayout => (uint) _currentLayout;
public VulkanImage(VulkanContext vk, uint format, PixelSize size,
bool exportable, uint mipLevels = 0)
bool exportable, IReadOnlyList<string> supportedHandleTypes)
{
_vk = vk;
_instance = vk.Instance;
@ -62,8 +64,12 @@ public unsafe class VulkanImage : IDisposable
//MipLevels = MipLevels != 0 ? MipLevels : (uint)Math.Floor(Math.Log(Math.Max(Size.Width, Size.Height), 2));
var handleType = RuntimeInformation.IsOSPlatform(OSPlatform.Windows) ?
ExternalMemoryHandleTypeFlags.D3D11TextureBit :
(supportedHandleTypes.Contains(KnownPlatformGraphicsExternalImageHandleTypes.D3D11TextureNtHandle)
&& !supportedHandleTypes.Contains(KnownPlatformGraphicsExternalImageHandleTypes.VulkanOpaqueNtHandle) ?
ExternalMemoryHandleTypeFlags.D3D11TextureBit :
ExternalMemoryHandleTypeFlags.OpaqueWin32Bit) :
ExternalMemoryHandleTypeFlags.OpaqueFDBit;
var externalMemoryCreateInfo = new ExternalMemoryImageCreateInfo
{
SType = StructureType.ExternalMemoryImageCreateInfo,
@ -96,35 +102,37 @@ public unsafe class VulkanImage : IDisposable
Api.GetImageMemoryRequirements(_device, InternalHandle,
out var memoryRequirements);
var fdExport = new ExportMemoryAllocateInfo
{
HandleTypes = handleType, SType = StructureType.ExportMemoryAllocateInfo
};
var dedicatedAllocation = new MemoryDedicatedAllocateInfoKHR
{
SType = StructureType.MemoryDedicatedAllocateInfoKhr,
Image = image
};
ImportMemoryWin32HandleInfoKHR handleImport = default;
if (exportable && RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
var fdExport = new ExportMemoryAllocateInfo
{
_d3dTexture2D = D3DMemoryHelper.CreateMemoryHandle(vk.D3DDevice!, size, Format);
using var dxgi = _d3dTexture2D.QueryInterface<SharpDX.DXGI.Resource1>();
handleImport = new ImportMemoryWin32HandleInfoKHR
{
PNext = &dedicatedAllocation,
SType = StructureType.ImportMemoryWin32HandleInfoKhr,
HandleType = ExternalMemoryHandleTypeFlags.D3D11TextureBit,
Handle = dxgi.CreateSharedHandle(null, SharedResourceFlags.Read | SharedResourceFlags.Write),
};
HandleTypes = handleType, SType = StructureType.ExportMemoryAllocateInfo,
PNext = &dedicatedAllocation
};
ImportMemoryWin32HandleInfoKHR handleImport = default;
if (handleType == ExternalMemoryHandleTypeFlags.D3D11TextureBit && exportable)
{
_d3dTexture2D = D3DMemoryHelper.CreateMemoryHandle(vk.D3DDevice, size, Format);
using var dxgi = _d3dTexture2D.QueryInterface<SharpDX.DXGI.Resource1>();
handleImport = new ImportMemoryWin32HandleInfoKHR
{
PNext = &dedicatedAllocation,
SType = StructureType.ImportMemoryWin32HandleInfoKhr,
HandleType = ExternalMemoryHandleTypeFlags.D3D11TextureBit,
Handle = dxgi.CreateSharedHandle(null, SharedResourceFlags.Read | SharedResourceFlags.Write),
};
}
var memoryAllocateInfo = new MemoryAllocateInfo
{
PNext =
exportable ? RuntimeInformation.IsOSPlatform(OSPlatform.Windows) ? &handleImport : &fdExport : null,
exportable ? handleImport.Handle != IntPtr.Zero ? &handleImport : &fdExport : null,
SType = StructureType.MemoryAllocateInfo,
AllocationSize = memoryRequirements.Size,
MemoryTypeIndex = (uint)VulkanMemoryHelper.FindSuitableMemoryTypeIndex(
@ -187,14 +195,34 @@ public unsafe class VulkanImage : IDisposable
return fd;
}
public IntPtr ExportOpaqueNtHandle()
{
if (!Api.TryGetDeviceExtension<KhrExternalMemoryWin32>(_instance, _device, out var ext))
throw new InvalidOperationException();
var info = new MemoryGetWin32HandleInfoKHR()
{
Memory = _imageMemory,
SType = StructureType.MemoryGetWin32HandleInfoKhr,
HandleType = ExternalMemoryHandleTypeFlags.OpaqueWin32Bit
};
ext.GetMemoryWin32Handle(_device, info, out var fd).ThrowOnError();
return fd;
}
public IPlatformHandle Export()
{
if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
{
using var dxgi = _d3dTexture2D!.QueryInterface<Resource1>();
return new PlatformHandle(
dxgi.CreateSharedHandle(null, SharedResourceFlags.Read | SharedResourceFlags.Write),
KnownPlatformGraphicsExternalImageHandleTypes.D3D11TextureNtHandle);
if (_d3dTexture2D != null)
{
using var dxgi = _d3dTexture2D!.QueryInterface<Resource1>();
return new PlatformHandle(
dxgi.CreateSharedHandle(null, SharedResourceFlags.Read | SharedResourceFlags.Write),
KnownPlatformGraphicsExternalImageHandleTypes.D3D11TextureNtHandle);
}
return new PlatformHandle(ExportOpaqueNtHandle(),
KnownPlatformGraphicsExternalImageHandleTypes.VulkanOpaqueNtHandle);
}
else
return new PlatformHandle(new IntPtr(ExportFd()),
@ -203,7 +231,7 @@ public unsafe class VulkanImage : IDisposable
public ImageTiling Tiling => ImageTiling.Optimal;
public bool IsDirectXBacked => _d3dTexture2D != null;
internal void TransitionLayout(CommandBuffer commandBuffer,
ImageLayout fromLayout, AccessFlags fromAccessFlags,

Просмотреть файл

@ -1,4 +1,6 @@
using System;
using System.Runtime.InteropServices;
using Avalonia.Platform;
using Silk.NET.Vulkan;
using Silk.NET.Vulkan.Extensions.KHR;
using SilkNetDemo;
@ -16,7 +18,9 @@ class VulkanSemaphorePair : IDisposable
var semaphoreExportInfo = new ExportSemaphoreCreateInfo
{
SType = StructureType.ExportSemaphoreCreateInfo,
HandleTypes = ExternalSemaphoreHandleTypeFlags.OpaqueFDBit
HandleTypes = RuntimeInformation.IsOSPlatform(OSPlatform.Windows) ?
ExternalSemaphoreHandleTypeFlags.OpaqueWin32Bit :
ExternalSemaphoreHandleTypeFlags.OpaqueFDBit
};
var semaphoreCreateInfo = new SemaphoreCreateInfo
@ -46,6 +50,30 @@ class VulkanSemaphorePair : IDisposable
ext.GetSemaphoreF(_resources.Device, info, out var fd).ThrowOnError();
return fd;
}
public IntPtr ExportWin32(bool renderFinished)
{
if (!_resources.Api.TryGetDeviceExtension<KhrExternalSemaphoreWin32>(_resources.Instance, _resources.Device,
out var ext))
throw new InvalidOperationException();
var info = new SemaphoreGetWin32HandleInfoKHR()
{
SType = StructureType.SemaphoreGetWin32HandleInfoKhr,
Semaphore = renderFinished ? RenderFinishedSemaphore : ImageAvailableSemaphore,
HandleType = ExternalSemaphoreHandleTypeFlags.OpaqueWin32Bit
};
ext.GetSemaphoreWin32Handle(_resources.Device, info, out var fd).ThrowOnError();
return fd;
}
public IPlatformHandle Export(bool renderFinished)
{
if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
return new PlatformHandle(ExportWin32(renderFinished),
KnownPlatformGraphicsExternalSemaphoreHandleTypes.VulkanOpaqueNtHandle);
return new PlatformHandle(new IntPtr(ExportFd(renderFinished)),
KnownPlatformGraphicsExternalSemaphoreHandleTypes.VulkanOpaquePosixFileDescriptor);
}
internal Semaphore ImageAvailableSemaphore { get; }
internal Semaphore RenderFinishedSemaphore { get; }
@ -55,4 +83,4 @@ class VulkanSemaphorePair : IDisposable
_resources.Api.DestroySemaphore(_resources.Device, ImageAvailableSemaphore, null);
_resources.Api.DestroySemaphore(_resources.Device, RenderFinishedSemaphore, null);
}
}
}

Просмотреть файл

@ -52,7 +52,7 @@ class VulkanSwapchainImage : ISwapchainImage
_interop = interop;
_target = target;
Size = size;
_image = new VulkanImage(vk, (uint)Format.R8G8B8A8Unorm, size, true);
_image = new VulkanImage(vk, (uint)Format.R8G8B8A8Unorm, size, true, interop.SupportedImageHandleTypes);
_semaphorePair = new VulkanSemaphorePair(vk, true);
}
@ -83,7 +83,7 @@ class VulkanSwapchainImage : ISwapchainImage
ImageLayout.Undefined, AccessFlags.None,
ImageLayout.ColorAttachmentOptimal, AccessFlags.ColorAttachmentReadBit);
if(RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
if(_image.IsDirectXBacked)
buffer.Submit(null,null,null, null, new VulkanCommandBufferPool.VulkanCommandBuffer.KeyedMutexSubmitInfo
{
AcquireKey = 0,
@ -111,8 +111,7 @@ class VulkanSwapchainImage : ISwapchainImage
_image.TransitionLayout(buffer.InternalHandle, ImageLayout.TransferSrcOptimal, AccessFlags.TransferWriteBit);
if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
if (_image.IsDirectXBacked)
{
buffer.Submit(null, null, null, null,
new VulkanCommandBufferPool.VulkanCommandBuffer.KeyedMutexSubmitInfo
@ -123,15 +122,11 @@ class VulkanSwapchainImage : ISwapchainImage
else
buffer.Submit(null, null, new[] { _semaphorePair.RenderFinishedSemaphore });
if (!RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
if (!_image.IsDirectXBacked)
{
_availableSemaphore ??= _interop.ImportSemaphore(new PlatformHandle(
new IntPtr(_semaphorePair.ExportFd(false)),
KnownPlatformGraphicsExternalSemaphoreHandleTypes.VulkanOpaquePosixFileDescriptor));
_availableSemaphore ??= _interop.ImportSemaphore(_semaphorePair.Export(false));
_renderCompletedSemaphore ??= _interop.ImportSemaphore(new PlatformHandle(
new IntPtr(_semaphorePair.ExportFd(true)),
KnownPlatformGraphicsExternalSemaphoreHandleTypes.VulkanOpaquePosixFileDescriptor));
_renderCompletedSemaphore ??= _interop.ImportSemaphore(_semaphorePair.Export(true));
}
_importedImage ??= _interop.ImportImage(_image.Export(),
@ -143,7 +138,7 @@ class VulkanSwapchainImage : ISwapchainImage
MemorySize = _image.MemorySize
});
if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
if (_image.IsDirectXBacked)
_lastPresent = _target.UpdateWithKeyedMutexAsync(_importedImage, 1, 0);
else
_lastPresent = _target.UpdateWithSemaphoresAsync(_importedImage, _renderCompletedSemaphore!, _availableSemaphore!);

Просмотреть файл

@ -34,6 +34,7 @@
<InternalsVisibleTo Include="Avalonia.Markup.Xaml, PublicKey=$(AvaloniaPublicKey)" />
<InternalsVisibleTo Include="Avalonia.Markup.Xaml.Loader, PublicKey=$(AvaloniaPublicKey)" />
<InternalsVisibleTo Include="Avalonia.OpenGL, PublicKey=$(AvaloniaPublicKey)" />
<InternalsVisibleTo Include="Avalonia.Vulkan, PublicKey=$(AvaloniaPublicKey)" />
<InternalsVisibleTo Include="Avalonia.Skia, PublicKey=$(AvaloniaPublicKey)" />
<InternalsVisibleTo Include="Avalonia.Controls.ColorPicker, PublicKey=$(AvaloniaPublicKey)" />
<InternalsVisibleTo Include="Avalonia.Controls.DataGrid, PublicKey=$(AvaloniaPublicKey)" />

Просмотреть файл

@ -56,5 +56,8 @@ namespace Avalonia.Platform.Interop
ArrayPool<byte>.Shared.Return(bytes);
}
}
public static implicit operator IntPtr(Utf8Buffer b) => b.handle;
public static unsafe implicit operator byte*(Utf8Buffer b) => (byte*)b.handle;
}
}

Просмотреть файл

@ -35,6 +35,14 @@ public static class KnownPlatformGraphicsExternalImageHandleTypes
/// A POSIX file descriptor that's exported by Vulkan using VK_EXTERNAL_MEMORY_HANDLE_TYPE_OPAQUE_FD_BIT or in a compatible way
/// </summary>
public const string VulkanOpaquePosixFileDescriptor = nameof(VulkanOpaquePosixFileDescriptor);
/// <summary>
/// A NT handle that's been exported by Vulkan using VK_EXTERNAL_MEMORY_HANDLE_TYPE_OPAQUE_WIN32_BIT or in a compatible way
/// </summary>
public const string VulkanOpaqueNtHandle = nameof(VulkanOpaqueNtHandle);
// A global shared handle that's been exported by Vulkan using VK_EXTERNAL_MEMORY_HANDLE_TYPE_OPAQUE_WIN32_KMT_BIT or in a compatible way
public const string VulkanOpaqueKmtHandle = nameof(VulkanOpaqueKmtHandle);
}
/// <summary>

Просмотреть файл

@ -0,0 +1,17 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFrameworks>net6.0;netstandard2.0</TargetFrameworks>
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
</PropertyGroup>
<ItemGroup>
<ProjectReference Include="..\Avalonia.Base\Avalonia.Base.csproj" />
<ProjectReference Include="..\Avalonia.Controls\Avalonia.Controls.csproj" />
</ItemGroup>
<Import Project="..\..\build\DevAnalyzers.props" />
<Import Project="..\..\build\SourceGenerators.props" />
<Import Project="..\..\build\TrimmingEnable.props" />
<Import Project="..\..\build\NullableEnable.props" />
</Project>

Просмотреть файл

@ -0,0 +1,33 @@
using System;
using System.Collections.Generic;
using Avalonia.Metadata;
using Avalonia.Platform;
using Avalonia.Rendering.Composition;
namespace Avalonia.Vulkan;
[Unstable]
public interface IVulkanContextExternalObjectsFeature
{
IReadOnlyList<string> SupportedImageHandleTypes { get; }
IReadOnlyList<string> SupportedSemaphoreTypes { get; }
byte[]? DeviceUuid { get; }
byte[]? DeviceLuid { get; }
CompositionGpuImportedImageSynchronizationCapabilities GetSynchronizationCapabilities(string imageHandleType);
IVulkanExternalImage ImportImage(IPlatformHandle handle, PlatformGraphicsExternalImageProperties properties);
IVulkanExternalSemaphore ImportSemaphore(IPlatformHandle handle);
}
[Unstable]
public interface IVulkanExternalSemaphore : IDisposable
{
ulong Handle { get; }
void SubmitWaitSemaphore();
void SubmitSignalSemaphore();
}
[Unstable]
public interface IVulkanExternalImage : IDisposable
{
VulkanImageInfo Info { get; }
}

Просмотреть файл

@ -0,0 +1,41 @@
using System;
using System.Collections.Generic;
using Avalonia.Metadata;
using Avalonia.Platform;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan;
public interface IVulkanDevice : IDisposable, IOptionalFeatureProvider
{
public IntPtr Handle { get; }
public IntPtr PhysicalDeviceHandle { get; }
public IntPtr MainQueueHandle { get; }
public uint GraphicsQueueFamilyIndex { get; }
public IVulkanInstance Instance { get; }
bool IsLost { get; }
public IDisposable Lock();
public IEnumerable<string> EnabledExtensions { get; }
}
public interface IVulkanInstance
{
public IntPtr Handle { get; }
public IntPtr GetInstanceProcAddress(IntPtr instance, string name);
public IntPtr GetDeviceProcAddress(IntPtr device, string name);
public IEnumerable<string> EnabledExtensions { get; }
}
[NotClientImplementable]
public interface IVulkanPlatformGraphicsContext : IPlatformGraphicsContext
{
IVulkanDevice Device { get; }
IVulkanInstance Instance { get; }
internal VulkanInstanceApi InstanceApi { get; }
internal VulkanDeviceApi DeviceApi { get; }
internal VkDevice DeviceHandle { get; }
internal VkPhysicalDevice PhysicalDeviceHandle { get; }
internal VkInstance InstanceHandle { get; }
internal VkQueue MainQueueHandle { get; }
internal uint GraphicsQueueFamilyIndex { get; }
IVulkanRenderTarget CreateRenderTarget(IEnumerable<object> surfaces);
}

Просмотреть файл

@ -0,0 +1,15 @@
using System;
namespace Avalonia.Vulkan;
public interface IVulkanKhrSurfacePlatformSurface : IDisposable
{
double Scaling { get; }
PixelSize Size { get; }
ulong CreateSurface(IVulkanPlatformGraphicsContext context);
}
public interface IVulkanKhrSurfacePlatformSurfaceFactory
{
bool CanRenderToSurface(IVulkanPlatformGraphicsContext context, object surface);
IVulkanKhrSurfacePlatformSurface CreateSurface(IVulkanPlatformGraphicsContext context, object surface);
}

Просмотреть файл

@ -0,0 +1,20 @@
using System;
using Avalonia.Metadata;
namespace Avalonia.Vulkan;
[NotClientImplementable]
public interface IVulkanRenderTarget : IDisposable
{
IVulkanRenderSession BeginDraw();
}
[NotClientImplementable]
public interface IVulkanRenderSession : IDisposable
{
double Scaling { get; }
PixelSize Size { get; }
public bool IsYFlipped { get; }
VulkanImageInfo ImageInfo { get; }
bool IsRgba { get; }
}

Просмотреть файл

@ -0,0 +1,107 @@
using System;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan.Interop;
internal class VulkanCommandBuffer : IDisposable
{
private readonly VulkanCommandBufferPool _pool;
private VkCommandBuffer _handle;
private readonly IVulkanPlatformGraphicsContext _context;
private VulkanFence _fence;
private bool _hasEnded;
private bool _hasStarted;
public VkCommandBuffer Handle => _handle;
public VulkanCommandBuffer(VulkanCommandBufferPool pool, VkCommandBuffer handle, IVulkanPlatformGraphicsContext context)
{
_pool = pool;
_handle = handle;
_context = context;
_fence = new VulkanFence(context, VkFenceCreateFlags.VK_FENCE_CREATE_SIGNALED_BIT);
}
public unsafe void Dispose()
{
if (_fence.Handle.Handle != 0)
_fence.Wait();
_fence.Dispose();
if (_handle.Handle != IntPtr.Zero)
{
VkCommandBuffer buf = _handle;
_context.DeviceApi.FreeCommandBuffers(_context.DeviceHandle, _pool.Handle, 1, &buf);
_handle = default;
}
}
public bool IsFinished => _fence.IsSignaled;
public void BeginRecording()
{
if (_hasStarted)
return;
var beginInfo = new VkCommandBufferBeginInfo
{
sType = VkStructureType.VK_STRUCTURE_TYPE_COMMAND_BUFFER_BEGIN_INFO,
flags = VkCommandBufferUsageFlags.VK_COMMAND_BUFFER_USAGE_ONE_TIME_SUBMIT_BIT
};
_context.DeviceApi.BeginCommandBuffer(_handle, ref beginInfo).ThrowOnError("vkBeginCommandBuffer");
_hasStarted = true;
}
public void EndRecording()
{
if (_hasStarted && !_hasEnded)
{
_context.DeviceApi.EndCommandBuffer(_handle).ThrowOnError("vkEndCommandBuffer");
_hasEnded = true;
}
}
public void Submit()
{
Submit(null, null, null);
}
public unsafe void Submit(
ReadOnlySpan<VulkanSemaphore> waitSemaphores,
ReadOnlySpan<VkPipelineStageFlags> waitDstStageMask,
ReadOnlySpan<VulkanSemaphore> signalSemaphores,
VulkanFence? fence = null)
{
EndRecording();
VkFence fenceHandle = (fence ?? _fence).Handle;
_context.DeviceApi.ResetFences(_context.DeviceHandle, 1, &fenceHandle)
.ThrowOnError("vkResetFences");
var pWaitSempaphores = stackalloc VkSemaphore[waitSemaphores.Length];
for (var c = 0; c < waitSemaphores.Length; c++)
pWaitSempaphores[c] = waitSemaphores[c].Handle;
var pSignalSemaphores = stackalloc VkSemaphore[signalSemaphores.Length];
for (var c = 0; c < signalSemaphores.Length; c++)
pSignalSemaphores[c] = signalSemaphores[c].Handle;
VkCommandBuffer commandBuffer = _handle;
fixed (VkPipelineStageFlags* flags = waitDstStageMask)
{
var submitInfo = new VkSubmitInfo
{
sType = VkStructureType.VK_STRUCTURE_TYPE_SUBMIT_INFO,
waitSemaphoreCount = (uint)waitSemaphores.Length,
pWaitSemaphores = pWaitSempaphores,
signalSemaphoreCount = (uint)signalSemaphores.Length,
pSignalSemaphores = pSignalSemaphores,
commandBufferCount = 1,
pCommandBuffers = &commandBuffer,
pWaitDstStageMask = flags
};
_context.DeviceApi.QueueSubmit(_context.MainQueueHandle, 1, &submitInfo, fenceHandle)
.ThrowOnError("vkQueueSubmit");
}
_pool.AddSubmittedCommandBuffer(this);
}
}

Просмотреть файл

@ -0,0 +1,71 @@
using System;
using System.Collections.Generic;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan.Interop;
internal class VulkanCommandBufferPool : IDisposable
{
private readonly IVulkanPlatformGraphicsContext _context;
private readonly Queue<VulkanCommandBuffer> _commandBuffers = new();
private VkCommandPool _handle;
public VkCommandPool Handle => _handle;
public VulkanCommandBufferPool(IVulkanPlatformGraphicsContext context)
{
_context = context;
var createInfo = new VkCommandPoolCreateInfo
{
sType = VkStructureType.VK_STRUCTURE_TYPE_COMMAND_POOL_CREATE_INFO,
flags = VkCommandPoolCreateFlags.VK_COMMAND_POOL_CREATE_RESET_COMMAND_BUFFER_BIT,
queueFamilyIndex = context.GraphicsQueueFamilyIndex
};
_context.DeviceApi.CreateCommandPool(_context.DeviceHandle, ref createInfo, IntPtr.Zero, out _handle)
.ThrowOnError("vkCreateCommandPool");
}
public void FreeUsedCommandBuffers()
{
while (_commandBuffers.Count > 0)
_commandBuffers.Dequeue().Dispose();
}
public void FreeFinishedCommandBuffers()
{
while (_commandBuffers.Count > 0)
{
var next = _commandBuffers.Peek();
if(!next.IsFinished)
return;
_commandBuffers.Dequeue();
next.Dispose();
}
}
public void Dispose()
{
FreeUsedCommandBuffers();
if (_handle.Handle != 0)
_context.DeviceApi.DestroyCommandPool(_context.DeviceHandle, _handle, IntPtr.Zero);
_handle = default;
}
public unsafe VulkanCommandBuffer CreateCommandBuffer()
{
var commandBufferAllocateInfo = new VkCommandBufferAllocateInfo
{
sType = VkStructureType.VK_STRUCTURE_TYPE_COMMAND_BUFFER_ALLOCATE_INFO,
commandPool = _handle,
commandBufferCount = 1,
level = VkCommandBufferLevel.VK_COMMAND_BUFFER_LEVEL_PRIMARY
};
VkCommandBuffer bufferHandle = default;
_context.DeviceApi.AllocateCommandBuffers(_context.DeviceHandle, ref commandBufferAllocateInfo,
&bufferHandle).ThrowOnError("vkAllocateCommandBuffers");
return new VulkanCommandBuffer(this, bufferHandle, _context);
}
public void AddSubmittedCommandBuffer(VulkanCommandBuffer buffer) => _commandBuffers.Enqueue(buffer);
}

Просмотреть файл

@ -0,0 +1,37 @@
using System;
using System.Runtime.InteropServices;
using Avalonia.Logging;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan.Interop;
unsafe static class VulkanDebugLogger
{
private static VkDebugUtilsMessengerCallbackEXTDelegate s_Delegate = WriteLogEvent;
public static IntPtr CallbackPtr { get; } = Marshal.GetFunctionPointerForDelegate(s_Delegate);
private static uint WriteLogEvent(VkDebugUtilsMessageSeverityFlagsEXT messageSeverity,
VkDebugUtilsMessageTypeFlagsEXT messagetypes,
VkDebugUtilsMessengerCallbackDataEXT* pCallbackData,
void* puserdata)
{
var level = messageSeverity switch
{
VkDebugUtilsMessageSeverityFlagsEXT.VK_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT => LogEventLevel.Error,
VkDebugUtilsMessageSeverityFlagsEXT.VK_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT =>
LogEventLevel.Warning,
VkDebugUtilsMessageSeverityFlagsEXT.VK_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT =>
LogEventLevel.Verbose,
VkDebugUtilsMessageSeverityFlagsEXT.VK_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT => LogEventLevel
.Information,
_ => LogEventLevel.Information
};
if (Logger.TryGet(level, "Vulkan", out var logger))
{
var msg =Marshal.PtrToStringAnsi((nint)pCallbackData->pMessage);
logger.Log("Vulkan", "Vulkan: {0}", msg);
}
return 0;
}
}

Просмотреть файл

@ -0,0 +1,166 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.InteropServices;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan.Interop;
internal unsafe partial class VulkanDevice
{
public static IVulkanDevice Create(IVulkanInstance instance,
VulkanDeviceCreationOptions options, VulkanPlatformSpecificOptions platformOptions)
{
uint deviceCount = 0;
var api = new VulkanInstanceApi(instance);
var vkInstance = new VkInstance(instance.Handle);
api.EnumeratePhysicalDevices(vkInstance, ref deviceCount, null)
.ThrowOnError("vkEnumeratePhysicalDevices");
if (deviceCount == 0)
throw new VulkanException("No devices found");
var devices = stackalloc VkPhysicalDevice[(int)deviceCount];
api.EnumeratePhysicalDevices(vkInstance, ref deviceCount, devices)
.ThrowOnError("vkEnumeratePhysicalDevices");
var surfaceForProbePtr = platformOptions.DeviceCheckSurfaceFactory?.Invoke(api.Instance);
var surfaceForProbe = surfaceForProbePtr.HasValue && surfaceForProbePtr.Value != 0
? new VkSurfaceKHR(surfaceForProbePtr.Value)
: (VkSurfaceKHR?)null;
DeviceInfo? compatibleDevice = null, discreteDevice = null;
for (var c = 0; c < deviceCount; c++)
{
var info = CheckDevice(api, devices[c], options, surfaceForProbe);
if (info != null)
{
compatibleDevice ??= info;
if (info.Value.Type == VkPhysicalDeviceType.VK_PHYSICAL_DEVICE_TYPE_DISCRETE_GPU)
discreteDevice ??= info;
}
if (compatibleDevice != null && (discreteDevice != null || !options.PreferDiscreteGpu))
break;
}
if (options.PreferDiscreteGpu && discreteDevice != null)
compatibleDevice = discreteDevice;
if (compatibleDevice == null)
throw new VulkanException("No compatible devices found");
var dev = compatibleDevice.Value;
var queuePriorities = stackalloc float[(int)dev.QueueCount];
for (var c = 0; c < dev.QueueCount; c++)
queuePriorities[c] = 1f;
var queueCreateInfo = new VkDeviceQueueCreateInfo
{
sType = VkStructureType.VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO,
queueFamilyIndex = dev.QueueFamilyIndex,
queueCount = dev.QueueCount,
pQueuePriorities = queuePriorities,
};
var enableExtensions =
new HashSet<string>(options.DeviceExtensions.Concat(VulkanExternalObjectsFeature.RequiredDeviceExtensions));
var enabledExtensions = enableExtensions
.Intersect(dev.Extensions).Append(VK_KHR_swapchain).Distinct().ToArray();
using var pEnabledExtensions = new Utf8BufferArray(enabledExtensions);
var createInfo = new VkDeviceCreateInfo
{
sType = VkStructureType.VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO,
queueCreateInfoCount = 1,
pQueueCreateInfos = &queueCreateInfo,
ppEnabledExtensionNames = pEnabledExtensions,
enabledExtensionCount = pEnabledExtensions.UCount,
};
api.CreateDevice(dev.PhysicalDevice, ref createInfo, IntPtr.Zero, out var createdDevice)
.ThrowOnError("vkCreateDevice");
api.GetDeviceQueue(createdDevice, dev.QueueFamilyIndex, 0, out var createdQueue);
return new VulkanDevice(api.Instance, createdDevice, dev.PhysicalDevice, createdQueue,
dev.QueueFamilyIndex, enabledExtensions);
}
struct DeviceInfo
{
public VkPhysicalDevice PhysicalDevice;
public uint QueueFamilyIndex;
public VkPhysicalDeviceType Type;
public List<string> Extensions;
public uint QueueCount;
}
static List<string> GetDeviceExtensions(VulkanInstanceApi instance, VkPhysicalDevice physicalDevice)
{
uint propertyCount = 0;
instance.EnumerateDeviceExtensionProperties(physicalDevice, null, ref propertyCount, null);
var extensionProps = new VkExtensionProperties[propertyCount];
var extensions = new List<string>((int)propertyCount);
if (propertyCount != 0)
fixed (VkExtensionProperties* ptr = extensionProps)
{
instance.EnumerateDeviceExtensionProperties(physicalDevice, null, ref propertyCount, ptr);
for (var c = 0; c < propertyCount; c++)
extensions.Add(Marshal.PtrToStringAnsi(new IntPtr(ptr[c].extensionName))!);
}
return extensions;
}
private const string VK_KHR_swapchain = "VK_KHR_swapchain";
static unsafe DeviceInfo? CheckDevice(VulkanInstanceApi instance, VkPhysicalDevice physicalDevice,
VulkanDeviceCreationOptions options, VkSurfaceKHR? surface)
{
instance.GetPhysicalDeviceProperties(physicalDevice, out var properties);
var supportedExtensions = GetDeviceExtensions(instance, physicalDevice);
if (!supportedExtensions.Contains(VK_KHR_swapchain))
return null;
uint familyCount = 0;
instance.GetPhysicalDeviceQueueFamilyProperties(physicalDevice, ref familyCount, null);
var familyProperties = stackalloc VkQueueFamilyProperties[(int)familyCount];
instance.GetPhysicalDeviceQueueFamilyProperties(physicalDevice, ref familyCount, familyProperties);
var requredFlags = VkQueueFlags.VK_QUEUE_GRAPHICS_BIT;
if (options.RequireComputeBit)
requredFlags |= VkQueueFlags.VK_QUEUE_COMPUTE_BIT;
for (var c = 0; c < familyCount; c++)
{
if ((familyProperties[c].queueFlags & requredFlags) != requredFlags)
continue;
if (surface.HasValue)
{
instance.GetPhysicalDeviceSurfaceSupportKHR(physicalDevice, (uint)c, surface.Value, out var supported)
.ThrowOnError("vkGetPhysicalDeviceSurfaceSupportKHR");
if (supported == 0)
continue;
}
return new DeviceInfo
{
PhysicalDevice = physicalDevice,
Extensions = supportedExtensions,
Type = properties.deviceType,
QueueFamilyIndex = (uint)c,
QueueCount = familyProperties[c].queueCount
};
}
return null;
}
}

Просмотреть файл

@ -0,0 +1,66 @@
using System;
using System.Collections.Generic;
using Avalonia.Reactive;
using System.Threading;
using Avalonia.Platform;
using Avalonia.Vulkan.Interop;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan.Interop;
internal partial class VulkanDevice : IVulkanDevice
{
private readonly VkDevice _handle;
private readonly VkPhysicalDevice _physicalDeviceHandle;
private readonly VkQueue _mainQueue;
private readonly uint _graphicsQueueIndex;
private readonly object _lock = new();
private Thread? _lockedByThread;
private int _lockCount;
private VulkanDevice(IVulkanInstance instance, VkDevice handle, VkPhysicalDevice physicalDeviceHandle,
VkQueue mainQueue, uint graphicsQueueIndex, string[] enabledExtensions)
{
_handle = handle;
_physicalDeviceHandle = physicalDeviceHandle;
_mainQueue = mainQueue;
_graphicsQueueIndex = graphicsQueueIndex;
Instance = instance;
EnabledExtensions = enabledExtensions;
}
T CheckAccess<T>(T f)
{
if (_lockedByThread != Thread.CurrentThread)
throw new InvalidOperationException("This class is only usable when locked");
return f;
}
public IDisposable Lock()
{
Monitor.Enter(_lock);
_lockCount++;
_lockedByThread = Thread.CurrentThread;
return Disposable.Create(() =>
{
_lockCount--;
if (_lockCount == 0)
_lockedByThread = null;
Monitor.Exit(_lock);
});
}
public IEnumerable<string> EnabledExtensions { get; }
public bool IsLost => false;
public IntPtr Handle => _handle.Handle;
public IntPtr PhysicalDeviceHandle => _physicalDeviceHandle.Handle;
public IntPtr MainQueueHandle => CheckAccess(_mainQueue).Handle;
public uint GraphicsQueueFamilyIndex => _graphicsQueueIndex;
public IVulkanInstance Instance { get; }
public void Dispose()
{
// TODO
}
public object? TryGetFeature(Type featureType) => null;
}

Просмотреть файл

@ -0,0 +1,328 @@
using System;
using System.Linq;
using System.Threading;
using Avalonia.Vulkan.UnmanagedInterop;
// ReSharper disable FieldCanBeMadeReadOnly.Local
// ReSharper disable IdentifierTypo
// ReSharper disable StringLiteralTypo
namespace Avalonia.Vulkan.Interop;
internal class VulkanDisplay : IDisposable
{
private IVulkanPlatformGraphicsContext _context;
private VulkanSemaphorePair _semaphorePair;
private uint _nextImage;
private readonly VulkanKhrSurface _surface;
private VkSurfaceFormatKHR _surfaceFormat;
private VkSwapchainKHR _swapchain;
private VkExtent2D _swapchainExtent;
private VkImage[] _swapchainImages = Array.Empty<VkImage>();
private VkImageView[] _swapchainImageViews = Array.Empty<VkImageView>();
public VulkanCommandBufferPool CommandBufferPool { get; private set; }
public PixelSize Size { get; private set; }
private VulkanDisplay(IVulkanPlatformGraphicsContext context, VulkanKhrSurface surface, VkSwapchainKHR swapchain,
VkExtent2D swapchainExtent)
{
_context = context;
_surface = surface;
_swapchain = swapchain;
_swapchainExtent = swapchainExtent;
_semaphorePair = new VulkanSemaphorePair(_context);
CommandBufferPool = new VulkanCommandBufferPool(_context);
CreateSwapchainImages();
}
internal VkSurfaceFormatKHR SurfaceFormat
{
get
{
if (_surfaceFormat.format == VkFormat.VK_FORMAT_UNDEFINED)
_surfaceFormat = _surface.GetSurfaceFormat();
return _surfaceFormat;
}
}
private static unsafe VkSwapchainKHR CreateSwapchain(IVulkanPlatformGraphicsContext context,
VulkanKhrSurface surface, out VkExtent2D swapchainExtent, VulkanDisplay? oldDisplay = null)
{
while (!surface.CanSurfacePresent())
Thread.Sleep(16);
context.InstanceApi.GetPhysicalDeviceSurfaceCapabilitiesKHR(context.PhysicalDeviceHandle,
surface.Handle, out var capabilities)
.ThrowOnError("vkGetPhysicalDeviceSurfaceCapabilitiesKHR");
uint presentModesCount = 0;
context.InstanceApi.GetPhysicalDeviceSurfacePresentModesKHR(context.PhysicalDeviceHandle,
surface.Handle, ref presentModesCount, null)
.ThrowOnError("vkGetPhysicalDeviceSurfacePresentModesKHR");
var modes = new VkPresentModeKHR[(int)presentModesCount];
fixed (VkPresentModeKHR* pModes = modes)
context.InstanceApi.GetPhysicalDeviceSurfacePresentModesKHR(context.PhysicalDeviceHandle,
surface.Handle, ref presentModesCount, pModes)
.ThrowOnError("vkGetPhysicalDeviceSurfacePresentModesKHR");
var imageCount = capabilities.minImageCount + 1;
if (capabilities.maxImageCount > 0 && imageCount > capabilities.maxImageCount)
imageCount = capabilities.maxImageCount;
var surfaceFormat = surface.GetSurfaceFormat();
bool supportsIdentityTransform = capabilities.supportedTransforms.HasAllFlags(
VkSurfaceTransformFlagsKHR.VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR);
bool isRotated =
capabilities.currentTransform.HasAllFlags(VkSurfaceTransformFlagsKHR.VK_SURFACE_TRANSFORM_ROTATE_90_BIT_KHR)
|| capabilities.currentTransform.HasAllFlags(VkSurfaceTransformFlagsKHR
.VK_SURFACE_TRANSFORM_ROTATE_270_BIT_KHR);
if (capabilities.currentExtent.width != uint.MaxValue)
swapchainExtent = capabilities.currentExtent;
else
{
var surfaceSize = surface.Size;
var width = Math.Max(capabilities.minImageExtent.width,
Math.Min(capabilities.maxImageExtent.width, (uint)surfaceSize.Width));
var height = Math.Max(capabilities.minImageExtent.height,
Math.Min(capabilities.maxImageExtent.height, (uint)surfaceSize.Height));
swapchainExtent = new VkExtent2D
{
width = width,
height = height
};
}
VkPresentModeKHR presentMode;
if (modes.Contains(VkPresentModeKHR.VK_PRESENT_MODE_MAILBOX_KHR))
presentMode = VkPresentModeKHR.VK_PRESENT_MODE_MAILBOX_KHR;
else if (modes.Contains(VkPresentModeKHR.VK_PRESENT_MODE_FIFO_KHR))
presentMode = VkPresentModeKHR.VK_PRESENT_MODE_FIFO_KHR;
else
presentMode = VkPresentModeKHR.VK_PRESENT_MODE_IMMEDIATE_KHR;
var swapchainCreateInfo = new VkSwapchainCreateInfoKHR
{
sType = VkStructureType.VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR,
surface = surface.Handle,
minImageCount = imageCount,
imageFormat = surfaceFormat.format,
imageColorSpace = surfaceFormat.colorSpace,
imageExtent = swapchainExtent,
imageUsage = VkImageUsageFlags.VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT |
VkImageUsageFlags.VK_IMAGE_USAGE_TRANSFER_DST_BIT,
imageSharingMode = VkSharingMode.VK_SHARING_MODE_EXCLUSIVE,
imageArrayLayers = 1,
preTransform = supportsIdentityTransform && isRotated
? VkSurfaceTransformFlagsKHR.VK_SURFACE_TRANSFORM_IDENTITY_BIT_KHR
: capabilities.currentTransform,
compositeAlpha = VkCompositeAlphaFlagsKHR.VK_COMPOSITE_ALPHA_OPAQUE_BIT_KHR,
presentMode = presentMode,
clipped = 1,
oldSwapchain = oldDisplay?._swapchain ?? default
};
context.DeviceApi.CreateSwapchainKHR(context.DeviceHandle, ref swapchainCreateInfo, IntPtr.Zero,
out var swapchain).ThrowOnError("vkCreateSwapchainKHR");
oldDisplay?.DestroySwapchain();
return swapchain;
}
private void DestroySwapchain()
{
if(_swapchain.Handle != 0)
_context.DeviceApi.DestroySwapchainKHR(_context.DeviceHandle, _swapchain, IntPtr.Zero);
_swapchain = default;
}
internal static VulkanDisplay CreateDisplay(IVulkanPlatformGraphicsContext context, VulkanKhrSurface surface)
{
var swapchain = CreateSwapchain(context, surface, out var extent);
return new VulkanDisplay(context, surface, swapchain, extent);
}
private void DestroyCurrentImageViews()
{
if (_swapchainImageViews.Length <= 0)
return;
foreach (var imageView in _swapchainImageViews)
_context.DeviceApi.DestroyImageView(_context.DeviceHandle, imageView, IntPtr.Zero);
_swapchainImageViews = Array.Empty<VkImageView>();
}
private unsafe void CreateSwapchainImages()
{
DestroyCurrentImageViews();
Size = new PixelSize((int)_swapchainExtent.width, (int)_swapchainExtent.height);
uint imageCount = 0;
_context.DeviceApi.GetSwapchainImagesKHR(_context.DeviceHandle, _swapchain, ref imageCount, null)
.ThrowOnError("vkGetSwapchainImagesKHR");
_swapchainImages = new VkImage[imageCount];
fixed (VkImage* pImages = _swapchainImages)
_context.DeviceApi.GetSwapchainImagesKHR(_context.DeviceHandle, _swapchain, ref imageCount,
pImages).ThrowOnError("vkGetSwapchainImagesKHR");
_swapchainImageViews = new VkImageView[imageCount];
for (var c = 0; c < imageCount; c++)
_swapchainImageViews[c] = CreateSwapchainImageView(_swapchainImages[c], SurfaceFormat.format);
}
private VkImageView CreateSwapchainImageView(VkImage swapchainImage, VkFormat format)
{
var imageViewCreateInfo = new VkImageViewCreateInfo
{
sType = VkStructureType.VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO,
subresourceRange =
{
aspectMask = VkImageAspectFlags.VK_IMAGE_ASPECT_COLOR_BIT,
levelCount = 1,
layerCount = 1
},
format = format,
image = swapchainImage,
viewType = VkImageViewType.VK_IMAGE_VIEW_TYPE_2D,
};
_context.DeviceApi.CreateImageView(_context.DeviceHandle, ref imageViewCreateInfo,
IntPtr.Zero, out var imageView).ThrowOnError("vkCreateImageView");
return imageView;
}
private void Recreate()
{
_context.DeviceApi.DeviceWaitIdle(_context.DeviceHandle);
_swapchain = CreateSwapchain(_context, _surface, out var extent, this);
_swapchainExtent = extent;
CreateSwapchainImages();
}
public bool EnsureSwapchainAvailable()
{
if (Size != _surface.Size)
{
Recreate();
return true;
}
return false;
}
public VulkanCommandBuffer StartPresentation()
{
_nextImage = 0;
while (true)
{
var acquireResult = _context.DeviceApi.AcquireNextImageKHR(
_context.DeviceHandle,
_swapchain,
ulong.MaxValue,
_semaphorePair.ImageAvailableSemaphore.Handle,
default, out _nextImage);
if (acquireResult is VkResult.VK_ERROR_OUT_OF_DATE_KHR or VkResult.VK_SUBOPTIMAL_KHR)
Recreate();
else
{
acquireResult.ThrowOnError("vkAcquireNextImageKHR");
break;
}
}
var commandBuffer = CommandBufferPool.CreateCommandBuffer();
commandBuffer.BeginRecording();
VulkanMemoryHelper.TransitionLayout(_context, commandBuffer,
_swapchainImages[_nextImage], VkImageLayout.VK_IMAGE_LAYOUT_UNDEFINED,
VkAccessFlags.VK_ACCESS_NONE, VkImageLayout.VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
VkAccessFlags.VK_ACCESS_TRANSFER_WRITE_BIT, 1);
return commandBuffer;
}
internal unsafe void BlitImageToCurrentImage(VulkanCommandBuffer commandBuffer, VulkanImage image)
{
VulkanMemoryHelper.TransitionLayout(_context, commandBuffer,
image.Handle, image.CurrentLayout, VkAccessFlags.VK_ACCESS_NONE,
VkImageLayout.VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
VkAccessFlags.VK_ACCESS_TRANSFER_READ_BIT,
image.MipLevels);
var srcBlitRegion = new VkImageBlit
{
srcOffsets2 =
{
x = image.Size.Width,
y = image.Size.Height,
z = 1
},
dstOffsets2 =
{
x = Size.Width,
y = Size.Height,
z = 1
},
srcSubresource =
{
aspectMask = VkImageAspectFlags.VK_IMAGE_ASPECT_COLOR_BIT,
layerCount = 1
},
dstSubresource =
{
aspectMask = VkImageAspectFlags.VK_IMAGE_ASPECT_COLOR_BIT,
layerCount = 1
}
};
_context.DeviceApi.CmdBlitImage(commandBuffer.Handle, image.Handle,
VkImageLayout.VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
_swapchainImages[_nextImage],
VkImageLayout.VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
1, &srcBlitRegion, VkFilter.VK_FILTER_LINEAR);
VulkanMemoryHelper.TransitionLayout(_context, commandBuffer,
image.Handle, VkImageLayout.VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
VkAccessFlags.VK_ACCESS_TRANSFER_READ_BIT,
image.CurrentLayout, VkAccessFlags.VK_ACCESS_NONE, image.MipLevels);
}
internal unsafe void EndPresentation(VulkanCommandBuffer commandBuffer)
{
VulkanMemoryHelper.TransitionLayout(_context, commandBuffer,
_swapchainImages[_nextImage],
VkImageLayout.VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL,
VkAccessFlags.VK_ACCESS_NONE,
VkImageLayout.VK_IMAGE_LAYOUT_PRESENT_SRC_KHR,
VkAccessFlags.VK_ACCESS_NONE,
1);
commandBuffer.Submit(new[] { _semaphorePair.ImageAvailableSemaphore },
new[] { VkPipelineStageFlags.VK_PIPELINE_STAGE_COLOR_ATTACHMENT_OUTPUT_BIT },
new[] { _semaphorePair.RenderFinishedSemaphore });
var semaphore = _semaphorePair.RenderFinishedSemaphore.Handle;
var swapchain = _swapchain;
var nextImage = _nextImage;
VkResult result;
var presentInfo = new VkPresentInfoKHR
{
sType = VkStructureType.VK_STRUCTURE_TYPE_PRESENT_INFO_KHR,
waitSemaphoreCount = 1,
pWaitSemaphores = &semaphore,
swapchainCount = 1,
pSwapchains = &swapchain,
pImageIndices = &nextImage,
pResults = &result
};
_context.DeviceApi.vkQueuePresentKHR(_context.MainQueueHandle, ref presentInfo)
.ThrowOnError("vkQueuePresentKHR");
result.ThrowOnError("vkQueuePresentKHR");
}
public void Dispose()
{
_context.DeviceApi.DeviceWaitIdle(_context.DeviceHandle);
_semaphorePair?.Dispose();
DestroyCurrentImageViews();
DestroySwapchain();
CommandBufferPool?.Dispose();
CommandBufferPool = null!;
}
}

Просмотреть файл

@ -0,0 +1,40 @@
using System;
using Avalonia.Vulkan.Interop;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan;
internal struct VulkanFence : IDisposable
{
private readonly IVulkanPlatformGraphicsContext _context;
private VkFence _handle;
public VulkanFence(IVulkanPlatformGraphicsContext context, VkFenceCreateFlags flags)
{
_context = context;
var fenceCreateInfo = new VkFenceCreateInfo
{
sType = VkStructureType.VK_STRUCTURE_TYPE_FENCE_CREATE_INFO,
flags = flags
};
_context.DeviceApi.CreateFence(_context.DeviceHandle, ref fenceCreateInfo, IntPtr.Zero, out _handle)
.ThrowOnError("vkCreateFence");
}
public VkFence Handle => _handle;
public void Dispose()
{
_context.DeviceApi.DestroyFence(_context.DeviceHandle, _handle, IntPtr.Zero);
_handle = default;
}
public bool IsSignaled => _context.DeviceApi.GetFenceStatus(_context.DeviceHandle, _handle) == VkResult.VK_SUCCESS;
public unsafe void Wait(ulong timeout = ulong.MaxValue)
{
VkFence fence = _handle;
_context.DeviceApi.WaitForFences(_context.DeviceHandle, 1, &fence, 1, timeout).ThrowOnError("vkWaitForFences");
}
}

Просмотреть файл

@ -0,0 +1,197 @@
using System;
using System.Diagnostics;
using Avalonia.Vulkan.Interop;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan;
internal class VulkanImageBase : IDisposable
{
private IVulkanPlatformGraphicsContext _context;
private VkAccessFlags _currentAccessFlags;
public VkImageUsageFlags UsageFlags { get; }
private VkImageView _imageView;
private VkDeviceMemory _imageMemory;
private VkImage _handle;
private VulkanCommandBufferPool _commandBufferPool;
internal VkImage Handle => _handle;
internal VkFormat Format { get; }
internal VkImageAspectFlags AspectFlags { get; private set; }
public uint MipLevels { get; private set; }
public PixelSize Size { get; }
public ulong MemorySize { get; private set; }
public VkImageLayout CurrentLayout { get; protected set; }
public VkDeviceMemory MemoryHandle => _imageMemory;
public VkImageTiling Tiling => VkImageTiling.VK_IMAGE_TILING_OPTIMAL;
public VulkanImageInfo ImageInfo => new()
{
Handle = Handle.Handle,
PixelSize = Size,
Format = (uint)Format,
MemoryHandle = MemoryHandle.Handle,
MemorySize = MemorySize,
ViewHandle = _imageView.Handle,
UsageFlags = (uint)UsageFlags,
Layout = (uint)CurrentLayout,
Tiling = (uint)Tiling,
LevelCount = MipLevels,
SampleCount = 1,
IsProtected = false
};
public struct MemoryImportInfo
{
public IntPtr Next;
public ulong MemorySize;
public ulong MemoryOffset;
}
public VulkanImageBase(IVulkanPlatformGraphicsContext context,
VulkanCommandBufferPool commandBufferPool,
VkFormat format, PixelSize size, uint mipLevels = 0)
{
Format = format;
Size = size;
MipLevels = mipLevels;
_context = context;
_commandBufferPool = commandBufferPool;
UsageFlags = VkImageUsageFlags.VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT
| VkImageUsageFlags.VK_IMAGE_USAGE_TRANSFER_DST_BIT
| VkImageUsageFlags.VK_IMAGE_USAGE_TRANSFER_SRC_BIT
| VkImageUsageFlags.VK_IMAGE_USAGE_SAMPLED_BIT;
}
protected virtual VkDeviceMemory CreateMemory(VkImage image, ulong size, uint memoryTypeBits)
{
var memoryAllocateInfo = new VkMemoryAllocateInfo
{
sType = VkStructureType.VK_STRUCTURE_TYPE_MEMORY_ALLOCATE_INFO,
allocationSize = size,
memoryTypeIndex = (uint)VulkanMemoryHelper.FindSuitableMemoryTypeIndex(_context,
memoryTypeBits,
VkMemoryPropertyFlags.VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT),
};
_context.DeviceApi.AllocateMemory(_context.DeviceHandle, ref memoryAllocateInfo, IntPtr.Zero,
out var imageMemory).ThrowOnError("vkAllocateMemory");
return imageMemory;
}
public unsafe void Initialize(void* pNext)
{
if (Handle.Handle != 0)
return;
MipLevels = MipLevels != 0 ? MipLevels : (uint)Math.Floor(Math.Log(Math.Max(Size.Width, Size.Height), 2));
var createInfo = new VkImageCreateInfo
{
pNext = pNext,
sType = VkStructureType.VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO,
imageType = VkImageType.VK_IMAGE_TYPE_2D,
format = Format,
extent = new VkExtent3D
{
depth = 1,
width = (uint)Size.Width,
height = (uint)Size.Height
},
mipLevels = MipLevels,
arrayLayers = 1,
samples = VkSampleCountFlags.VK_SAMPLE_COUNT_1_BIT,
tiling = Tiling,
usage = UsageFlags,
sharingMode = VkSharingMode.VK_SHARING_MODE_EXCLUSIVE,
initialLayout = VkImageLayout.VK_IMAGE_LAYOUT_UNDEFINED,
flags = VkImageCreateFlags.VK_IMAGE_CREATE_MUTABLE_FORMAT_BIT
};
_context.DeviceApi.CreateImage(_context.DeviceHandle, ref createInfo, IntPtr.Zero, out _handle)
.ThrowOnError("vkCreateImage");
_context.DeviceApi.GetImageMemoryRequirements(_context.DeviceHandle, _handle, out var memoryRequirements);
try
{
_imageMemory = CreateMemory(_handle, memoryRequirements.size, memoryRequirements.memoryTypeBits);
}
catch
{
_context.DeviceApi.DestroyImage(_context.DeviceHandle, _handle, IntPtr.Zero);
throw;
}
_context.DeviceApi.BindImageMemory(_context.DeviceHandle, _handle, _imageMemory, 0)
.ThrowOnError("vkBindImageMemory");
MemorySize = memoryRequirements.size;
AspectFlags = VkImageAspectFlags.VK_IMAGE_ASPECT_COLOR_BIT;
var imageViewCreateInfo = new VkImageViewCreateInfo
{
sType = VkStructureType.VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO,
components = new(),
subresourceRange = new()
{
aspectMask = AspectFlags,
levelCount = MipLevels,
layerCount = 1
},
format = Format,
image = _handle,
viewType = VkImageViewType.VK_IMAGE_VIEW_TYPE_2D
};
_context.DeviceApi.CreateImageView(_context.DeviceHandle, ref imageViewCreateInfo,
IntPtr.Zero, out _imageView).ThrowOnError("vkCreateImageView");
CurrentLayout = VkImageLayout.VK_IMAGE_LAYOUT_UNDEFINED;
}
internal void TransitionLayout(VkImageLayout destinationLayout, VkAccessFlags destinationAccessFlags)
{
var commandBuffer = _commandBufferPool!.CreateCommandBuffer();
commandBuffer.BeginRecording();
VulkanMemoryHelper.TransitionLayout(_context, commandBuffer, Handle,
CurrentLayout, _currentAccessFlags, destinationLayout, destinationAccessFlags,
MipLevels);
commandBuffer.EndRecording();
commandBuffer.Submit();
CurrentLayout = destinationLayout;
_currentAccessFlags = destinationAccessFlags;
}
public void TransitionLayout(uint destinationLayout, uint destinationAccessFlags)
{
TransitionLayout((VkImageLayout)destinationLayout, (VkAccessFlags)destinationAccessFlags);
}
public void Dispose()
{
var api = _context.DeviceApi;
var d = _context.DeviceHandle;
if (_imageView.Handle != 0)
{
api.DestroyImageView(d, _imageView, IntPtr.Zero);
_imageView = default;
}
if (_handle.Handle != 0)
{
api.DestroyImage(d, _handle, IntPtr.Zero);
_handle = default;
}
if (_imageMemory.Handle != 0)
{
api.FreeMemory(d, _imageMemory, IntPtr.Zero);
_imageMemory = default;
}
}
}
unsafe class VulkanImage : VulkanImageBase
{
public VulkanImage(IVulkanPlatformGraphicsContext context, VulkanCommandBufferPool commandBufferPool,
VkFormat format, PixelSize size, uint mipLevels = 0) : base(context, commandBufferPool, format, size, mipLevels)
{
Initialize(null);
TransitionLayout(VkImageLayout.VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL, VkAccessFlags.VK_ACCESS_NONE_KHR);
}
}

Просмотреть файл

@ -0,0 +1,179 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.CompilerServices;
using Avalonia.Reactive;
using System.Runtime.InteropServices;
using Avalonia.Platform.Interop;
using Avalonia.Vulkan.Interop;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan;
internal class VulkanInstance : IVulkanInstance
{
private readonly VkGetInstanceProcAddressDelegate _getProcAddress;
private readonly VulkanInstanceApi _api;
public VulkanInstance(VkInstance handle, VkGetInstanceProcAddressDelegate getProcAddress, string[] enabledExtensions)
{
Handle = handle;
_getProcAddress = getProcAddress;
_api = new VulkanInstanceApi(this);
EnabledExtensions = enabledExtensions;
}
internal static unsafe IVulkanInstance Create(
VulkanInstanceCreationOptions options,
VulkanPlatformSpecificOptions platformOptions)
{
var getProcAddress = options.CustomGetProcAddressDelegate ??
platformOptions.GetProcAddressDelegate ??
throw new ArgumentException("No VkGetInstanceProcAddr provided");
using var name = new Utf8Buffer(options.ApplicationName ?? "AvaloniaUI");
using var engineName = new Utf8Buffer("AvaloniaUI");
var gapi = new UnmanagedInterop.VulkanGlobalApi(getProcAddress);
var supportedExtensions = GetSupportedExtensions(gapi);
var appInfo = new VkApplicationInfo()
{
sType = VkStructureType.VK_STRUCTURE_TYPE_APPLICATION_INFO,
apiVersion = VulkanHelpers.MakeVersion(options.VulkanVersion),
applicationVersion = VulkanHelpers.MakeVersion(1, 0, 0),
engineVersion = VulkanHelpers.MakeVersion(1, 0, 0),
pApplicationName = name,
pEngineName = engineName,
};
var enabledExtensions = new HashSet<string>(options.InstanceExtensions
.Concat(platformOptions.RequiredInstanceExtensions)
.Append("VK_KHR_surface"));
var enabledLayers = options.EnabledLayers.ToList();
void AddExtensionsIfSupported(params string[] names)
{
if(names.All(n=>supportedExtensions.Contains(n)))
foreach (var n in names)
enabledExtensions.Add(n);
}
if (options.UseDebug)
{
AddExtensionsIfSupported("VK_EXT_debug_utils");
if (IsLayerAvailable(gapi, "VK_LAYER_KHRONOS_validation"))
enabledLayers.Add("VK_LAYER_KHRONOS_validation");
}
AddExtensionsIfSupported(VulkanExternalObjectsFeature.RequiredInstanceExtensions);
using var enabledExtensionBuffers = new Utf8BufferArray(
enabledExtensions
.Where(x => !string.IsNullOrWhiteSpace(x))
.Distinct());
using var enabledLayersBuffers = new Utf8BufferArray(enabledLayers
.Where(x => !string.IsNullOrWhiteSpace(x))
.Distinct());
var createInfo = new VkInstanceCreateInfo()
{
sType = VkStructureType.VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO,
pApplicationInfo = &appInfo,
ppEnabledLayerNames = enabledLayersBuffers,
enabledLayerCount = enabledLayersBuffers.UCount,
ppEnabledExtensionNames = enabledExtensionBuffers,
enabledExtensionCount = enabledExtensionBuffers.UCount
};
gapi.vkCreateInstance(ref createInfo, IntPtr.Zero, out var pInstance)
.ThrowOnError(nameof(gapi.vkCreateInstance));
var instance = new VulkanInstance(pInstance, getProcAddress, enabledExtensions.ToArray());
var instanceApi = new VulkanInstanceApi(instance);
if (options.UseDebug)
{
var debugCreateInfo = new VkDebugUtilsMessengerCreateInfoEXT
{
sType = VkStructureType.VK_STRUCTURE_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT,
messageSeverity = VkDebugUtilsMessageSeverityFlagsEXT.VK_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT
|VkDebugUtilsMessageSeverityFlagsEXT.VK_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT
|VkDebugUtilsMessageSeverityFlagsEXT.VK_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT
|VkDebugUtilsMessageSeverityFlagsEXT.VK_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT,
messageType = VkDebugUtilsMessageTypeFlagsEXT.VK_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT
|VkDebugUtilsMessageTypeFlagsEXT.VK_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT
|VkDebugUtilsMessageTypeFlagsEXT.VK_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT,
pfnUserCallback = VulkanDebugLogger.CallbackPtr
};
instanceApi.CreateDebugUtilsMessengerEXT(pInstance, ref debugCreateInfo, IntPtr.Zero, out var messenger);
}
return instance;
}
private static unsafe bool IsLayerAvailable(VulkanGlobalApi api, string layerName)
{
uint layerPropertiesCount = 0;
api.EnumerateInstanceLayerProperties(ref layerPropertiesCount, null)
.ThrowOnError("vkEnumerateInstanceLayerProperties");
var layerProperties = new VkLayerProperties[layerPropertiesCount];
fixed (VkLayerProperties* pLayerProperties = layerProperties)
{
api.EnumerateInstanceLayerProperties(ref layerPropertiesCount, pLayerProperties)
.ThrowOnError("vkEnumerateInstanceLayerProperties");
for (var i = 0; i < layerPropertiesCount; i++)
{
var currentLayerName = Marshal.PtrToStringAnsi((IntPtr)pLayerProperties[i].layerName);
if (currentLayerName == layerName) return true;
}
}
return false;
}
private static unsafe HashSet<string> GetSupportedExtensions(VulkanGlobalApi api)
{
var supportedExtensions = new HashSet<string>();
uint supportedExtensionCount = 0;
api.vkEnumerateInstanceExtensionProperties(IntPtr.Zero, &supportedExtensionCount, null);
if (supportedExtensionCount > 0)
{
var ptr = (VkExtensionProperties*)Marshal.AllocHGlobal(Unsafe.SizeOf<VkExtensionProperties>() *
(int)supportedExtensionCount);
try
{
api.vkEnumerateInstanceExtensionProperties(IntPtr.Zero, &supportedExtensionCount, ptr)
.ThrowOnError("vkEnumerateInstanceExtensionProperties");
for (var c = 0; c < supportedExtensionCount; c++)
supportedExtensions.Add(Marshal.PtrToStringAnsi((IntPtr)ptr[c].extensionName) ?? "");
}
finally
{
Marshal.FreeHGlobal((IntPtr)ptr);
}
}
return supportedExtensions;
}
public VkInstance Handle { get; }
IntPtr IVulkanInstance.Handle => Handle.Handle;
public IntPtr GetInstanceProcAddress(IntPtr instance, string name) => _getProcAddress(instance, name);
public IntPtr GetDeviceProcAddress(IntPtr device, string name)
{
using var buf = new Utf8Buffer(name);
return _api.GetDeviceProcAddr(new(device), buf);
}
public IEnumerable<string> EnabledExtensions { get; }
}

Просмотреть файл

@ -0,0 +1,69 @@
using System;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan.Interop;
internal class VulkanKhrSurface : IDisposable
{
private readonly IVulkanPlatformGraphicsContext _context;
private readonly IVulkanKhrSurfacePlatformSurface _surfaceInfo;
private VkSurfaceKHR _handle;
public VkSurfaceKHR Handle => _handle;
public VulkanKhrSurface(IVulkanPlatformGraphicsContext context, IVulkanKhrSurfacePlatformSurface surfaceInfo)
{
_context = context;
_surfaceInfo = surfaceInfo;
_handle = new VkSurfaceKHR(surfaceInfo.CreateSurface(context));
}
internal bool CanSurfacePresent()
{
_context.InstanceApi.GetPhysicalDeviceSurfaceSupportKHR(_context.PhysicalDeviceHandle,
_context.GraphicsQueueFamilyIndex, _handle, out var isSupported)
.ThrowOnError("vkGetPhysicalDeviceSurfaceSupportKHR");
return isSupported != 0;
}
internal unsafe VkSurfaceFormatKHR GetSurfaceFormat()
{
uint surfaceFormatsCount = 0;
_context.InstanceApi.GetPhysicalDeviceSurfaceFormatsKHR(_context.PhysicalDeviceHandle,
_handle, ref surfaceFormatsCount, null)
.ThrowOnError("vkGetPhysicalDeviceSurfaceFormatsKHR");
if (surfaceFormatsCount == 0)
throw new VulkanException("vkGetPhysicalDeviceSurfaceFormatsKHR returned 0 formats");
var surfaceFormats = stackalloc VkSurfaceFormatKHR[(int)surfaceFormatsCount];
_context.InstanceApi.GetPhysicalDeviceSurfaceFormatsKHR(_context.PhysicalDeviceHandle,
_handle, ref surfaceFormatsCount, surfaceFormats)
.ThrowOnError("vkGetPhysicalDeviceSurfaceFormatsKHR");
if (surfaceFormatsCount == 1 && surfaceFormats[0].format == VkFormat.VK_FORMAT_UNDEFINED)
return new VkSurfaceFormatKHR
{
format = VkFormat.VK_FORMAT_B8G8R8A8_UNORM,
colorSpace = VkColorSpaceKHR.VK_COLOR_SPACE_SRGB_NONLINEAR_KHR
};
for (var c = 0; c < surfaceFormatsCount; c++)
{
if (surfaceFormats[c].format == VkFormat.VK_FORMAT_B8G8R8A8_UNORM
&& surfaceFormats[c].colorSpace == VkColorSpaceKHR.VK_COLOR_SPACE_SRGB_NONLINEAR_KHR)
return surfaceFormats[c];
}
return surfaceFormats[0];
}
public PixelSize Size => _surfaceInfo.Size;
public void Dispose()
{
if (_handle.Handle != 0)
_context.InstanceApi.DestroySurfaceKHR(_context.InstanceHandle, _handle, IntPtr.Zero);
_handle = default;
}
}

Просмотреть файл

@ -0,0 +1,67 @@
using System;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan.Interop;
internal static class VulkanMemoryHelper
{
internal static int FindSuitableMemoryTypeIndex(IVulkanPlatformGraphicsContext context, uint memoryTypeBits,
VkMemoryPropertyFlags flags)
{
context.InstanceApi.GetPhysicalDeviceMemoryProperties(context.PhysicalDeviceHandle,
out var properties);
for (var i = 0; i < properties.memoryTypeCount; i++)
{
var type = properties.memoryTypes[i];
if ((memoryTypeBits & (1 << i)) != 0 && type.propertyFlags.HasAllFlags(flags))
return i;
}
return -1;
}
internal static unsafe void TransitionLayout(IVulkanPlatformGraphicsContext context,
VulkanCommandBuffer commandBuffer,
VkImage image,
VkImageLayout sourceLayout,
VkAccessFlags sourceAccessMask,
VkImageLayout destinationLayout,
VkAccessFlags destinationAccessMask,
uint mipLevels)
{
var subresourceRange = new VkImageSubresourceRange
{
aspectMask = VkImageAspectFlags.VK_IMAGE_ASPECT_COLOR_BIT,
levelCount = mipLevels,
layerCount = 1
};
var barrier = new VkImageMemoryBarrier
{
sType = VkStructureType.VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER,
srcAccessMask = sourceAccessMask,
dstAccessMask = destinationAccessMask,
oldLayout = sourceLayout,
newLayout = destinationLayout,
srcQueueFamilyIndex = VulkanHelpers.QueueFamilyIgnored,
dstQueueFamilyIndex = VulkanHelpers.QueueFamilyIgnored,
image = image,
subresourceRange = subresourceRange
};
context.DeviceApi.CmdPipelineBarrier(
commandBuffer.Handle,
VkPipelineStageFlags.VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
VkPipelineStageFlags.VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
0,
0,
IntPtr.Zero,
0,
IntPtr.Zero,
1,
&barrier);
}
}

Просмотреть файл

@ -0,0 +1,58 @@
using System;
using Avalonia.Vulkan.Interop;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan;
class VulkanSemaphore : IDisposable
{
private readonly IVulkanPlatformGraphicsContext _context;
private VkSemaphore _handle;
public VkSemaphore Handle => _handle;
public VulkanSemaphore(IVulkanPlatformGraphicsContext context)
{
_context = context;
var info = new VkSemaphoreCreateInfo
{
sType = VkStructureType.VK_STRUCTURE_TYPE_SEMAPHORE_CREATE_INFO
};
_context.DeviceApi.CreateSemaphore(context.DeviceHandle, ref info, IntPtr.Zero, out _handle)
.ThrowOnError("vkCreateSemaphore");
}
public VulkanSemaphore(IVulkanPlatformGraphicsContext context, VkSemaphore handle)
{
_context = context;
_handle = handle;
}
public void Dispose()
{
if (_handle.Handle != 0)
{
_context.DeviceApi.DestroySemaphore(_context.DeviceHandle, _handle, IntPtr.Zero);
_handle = default;
}
}
}
internal class VulkanSemaphorePair : IDisposable
{
public unsafe VulkanSemaphorePair(IVulkanPlatformGraphicsContext context)
{
ImageAvailableSemaphore = new VulkanSemaphore(context);
RenderFinishedSemaphore = new VulkanSemaphore(context);
}
internal VulkanSemaphore ImageAvailableSemaphore { get; }
internal VulkanSemaphore RenderFinishedSemaphore { get; }
public void Dispose()
{
ImageAvailableSemaphore.Dispose();
RenderFinishedSemaphore.Dispose();
}
}

Просмотреть файл

@ -0,0 +1,46 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.InteropServices;
using Avalonia.Platform.Interop;
namespace Avalonia.Vulkan.Interop;
internal unsafe class Utf8BufferArray : IDisposable
{
private readonly List<Utf8Buffer> _buffers;
private byte** _bufferArray;
public Utf8BufferArray(IEnumerable<string> strings)
{
_buffers = strings.Select(x => new Utf8Buffer(x)).ToList();
_bufferArray = (byte**)Marshal.AllocHGlobal(_buffers.Count * IntPtr.Size);
for (var c = 0; c < _buffers.Count; c++)
_bufferArray[c] = _buffers[c];
}
public static unsafe implicit operator byte**(Utf8BufferArray a) => a._bufferArray;
public int Count => _buffers.Count;
public uint UCount => (uint)Count;
public void Dispose() => Dispose(true);
void Dispose(bool disposing)
{
if (_bufferArray != null)
Marshal.FreeHGlobal(new IntPtr(_bufferArray));
_bufferArray = null;
if (disposing)
{
foreach (var b in _buffers)
b.Dispose();
_buffers.Clear();
GC.SuppressFinalize(this);
}
}
~Utf8BufferArray()
{
Dispose(false);
}
}

Просмотреть файл

@ -0,0 +1,154 @@
using System;
using Avalonia.SourceGenerator;
using Avalonia.Vulkan.Interop;
using uint32_t = System.UInt32;
using uint64_t = System.UInt64;
using VkBool32 = System.UInt32;
using VkDeviceSize = System.UInt64;
namespace Avalonia.Vulkan.UnmanagedInterop;
internal unsafe partial class VulkanDeviceApi
{
public VulkanDeviceApi(IVulkanDevice device)
{
Initialize(name =>
{
var addr = device.Instance.GetDeviceProcAddress(device.Handle, name);
if (addr != IntPtr.Zero)
return addr;
return device.Instance.GetInstanceProcAddress(device.Instance.Handle, name);
});
}
[GetProcAddress("vkCreateFence")]
public partial VkResult CreateFence(VkDevice device,
ref VkFenceCreateInfo pCreateInfo,
IntPtr pAllocator,
out VkFence pFence);
[GetProcAddress("vkDestroyFence")]
public partial void DestroyFence(VkDevice device, VkFence fence, IntPtr pAllocator);
[GetProcAddress("vkCreateCommandPool")]
public partial VkResult CreateCommandPool(VkDevice device, ref VkCommandPoolCreateInfo pCreateInfo,
IntPtr pAllocator, out VkCommandPool pCommandPool);
[GetProcAddress("vkDestroyCommandPool")]
public partial void DestroyCommandPool(VkDevice device, VkCommandPool pool, IntPtr pAllocator);
[GetProcAddress("vkAllocateCommandBuffers")]
public partial VkResult AllocateCommandBuffers(VkDevice device,
ref VkCommandBufferAllocateInfo pAllocateInfo, VkCommandBuffer* pCommandBuffers);
[GetProcAddress("vkFreeCommandBuffers")]
public partial void FreeCommandBuffers(VkDevice device, VkCommandPool commandPool, uint32_t commandBufferCount,
VkCommandBuffer* pCommandBuffers);
[GetProcAddress("vkWaitForFences")]
public partial VkResult WaitForFences(VkDevice device, uint32_t fenceCount, VkFence* pFences, VkBool32 waitAll,
uint64_t timeout);
[GetProcAddress("vkGetFenceStatus")]
public partial VkResult GetFenceStatus(VkDevice device, VkFence fence);
[GetProcAddress("vkBeginCommandBuffer")]
public partial VkResult BeginCommandBuffer(VkCommandBuffer commandBuffer, ref VkCommandBufferBeginInfo pBeginInfo);
[GetProcAddress("vkEndCommandBuffer")]
public partial VkResult EndCommandBuffer(VkCommandBuffer commandBuffer);
[GetProcAddress("vkCreateSemaphore")]
public partial VkResult CreateSemaphore(VkDevice device, ref VkSemaphoreCreateInfo pCreateInfo,
IntPtr pAllocator, out VkSemaphore pSemaphore);
[GetProcAddress("vkDestroySemaphore")]
public partial void DestroySemaphore(VkDevice device, VkSemaphore semaphore, IntPtr pAllocator);
[GetProcAddress("vkResetFences")]
public partial VkResult ResetFences(VkDevice device, uint32_t fenceCount, VkFence* pFences);
[GetProcAddress("vkQueueSubmit")]
public partial VkResult QueueSubmit(VkQueue queue, uint32_t submitCount, VkSubmitInfo* pSubmits,
VkFence fence);
[GetProcAddress("vkCreateImage")]
public partial VkResult CreateImage(VkDevice device, ref VkImageCreateInfo pCreateInfo, IntPtr pAllocator,
out VkImage pImage);
[GetProcAddress("vkDestroyImage")]
public partial void DestroyImage(VkDevice device, VkImage image, IntPtr pAllocator);
[GetProcAddress("vkGetImageMemoryRequirements")]
public partial void GetImageMemoryRequirements(VkDevice device, VkImage image,
out VkMemoryRequirements pMemoryRequirements);
[GetProcAddress("vkAllocateMemory")]
public partial VkResult AllocateMemory(VkDevice device, ref VkMemoryAllocateInfo pAllocateInfo, IntPtr pAllocator,
out VkDeviceMemory pMemory);
[GetProcAddress("vkFreeMemory")]
public partial void FreeMemory(VkDevice device, VkDeviceMemory memory, IntPtr pAllocator);
[GetProcAddress("vkBindImageMemory")]
public partial VkResult BindImageMemory(VkDevice device, VkImage image, VkDeviceMemory memory,
VkDeviceSize memoryOffset);
[GetProcAddress("vkCreateImageView")]
public partial VkResult CreateImageView(VkDevice device, ref VkImageViewCreateInfo pCreateInfo, IntPtr pAllocator,
out VkImageView pView);
[GetProcAddress("vkDestroyImageView")]
public partial void DestroyImageView(VkDevice device, VkImageView imageView, IntPtr pAllocator);
[GetProcAddress("vkCmdPipelineBarrier")]
public partial void CmdPipelineBarrier(VkCommandBuffer commandBuffer,
VkPipelineStageFlags srcStageMask,
VkPipelineStageFlags dstStageMask,
VkDependencyFlags dependencyFlags,
uint32_t memoryBarrierCount,
IntPtr pMemoryBarriers,
uint32_t bufferMemoryBarrierCount,
IntPtr pBufferMemoryBarriers,
uint32_t imageMemoryBarrierCount,
VkImageMemoryBarrier* pImageMemoryBarriers);
[GetProcAddress("vkCreateSwapchainKHR")]
public partial VkResult CreateSwapchainKHR(VkDevice device, ref VkSwapchainCreateInfoKHR pCreateInfo,
IntPtr pAllocator, out VkSwapchainKHR pSwapchain);
[GetProcAddress("vkDestroySwapchainKHR")]
public partial void DestroySwapchainKHR(VkDevice device, VkSwapchainKHR swapchain, IntPtr pAllocator);
[GetProcAddress("vkGetSwapchainImagesKHR")]
public partial VkResult GetSwapchainImagesKHR(VkDevice device, VkSwapchainKHR swapchain, ref uint32_t pSwapchainImageCount,
VkImage* pSwapchainImages);
[GetProcAddress("vkDeviceWaitIdle")]
public partial VkResult DeviceWaitIdle(VkDevice device);
[GetProcAddress("vkQueueWaitIdle")]
public partial VkResult QueueWaitIdle(VkQueue queue);
[GetProcAddress("vkAcquireNextImageKHR")]
public partial VkResult AcquireNextImageKHR(VkDevice device, VkSwapchainKHR swapchain, uint64_t timeout,
VkSemaphore semaphore, VkFence fence, out uint32_t pImageIndex);
[GetProcAddress("vkCmdBlitImage")]
public partial void CmdBlitImage(VkCommandBuffer commandBuffer, VkImage srcImage, VkImageLayout srcImageLayout,
VkImage dstImage, VkImageLayout dstImageLayout, uint32_t regionCount, VkImageBlit* pRegions, VkFilter filter);
[GetProcAddress("vkQueuePresentKHR")]
public partial VkResult vkQueuePresentKHR(VkQueue queue, ref VkPresentInfoKHR pPresentInfo);
[GetProcAddress("vkImportSemaphoreFdKHR", true)]
public partial VkResult ImportSemaphoreFdKHR(VkDevice device, VkImportSemaphoreFdInfoKHR* pImportSemaphoreFdInfo);
[GetProcAddress("vkImportSemaphoreWin32HandleKHR", true)]
public partial VkResult ImportSemaphoreWin32HandleKHR(VkDevice device,
VkImportSemaphoreWin32HandleInfoKHR* pImportSemaphoreWin32HandleInfo);
}

Разница между файлами не показана из-за своего большого размера Загрузить разницу

Просмотреть файл

@ -0,0 +1,29 @@
using System;
using Avalonia.SourceGenerator;
using Avalonia.Vulkan.Interop;
namespace Avalonia.Vulkan.UnmanagedInterop;
internal unsafe partial class VulkanGlobalApi
{
private readonly VkGetInstanceProcAddressDelegate _vkGetProcAddress;
public VulkanGlobalApi(VkGetInstanceProcAddressDelegate vkGetProcAddress)
{
_vkGetProcAddress = vkGetProcAddress;
Initialize(name => vkGetProcAddress(IntPtr.Zero, name));
}
public IntPtr GetProcAddress(VkInstance instance, string name) => _vkGetProcAddress(instance.Handle, name);
[GetProcAddress("vkEnumerateInstanceLayerProperties")]
public partial VkResult EnumerateInstanceLayerProperties(ref uint pPropertyCount, VkLayerProperties* pProperties);
[GetProcAddress("vkCreateInstance")]
public partial VkResult vkCreateInstance(ref VkInstanceCreateInfo pCreateInfo, IntPtr pAllocator, out VkInstance pInstance);
[GetProcAddress("vkEnumerateInstanceExtensionProperties")]
public partial VkResult vkEnumerateInstanceExtensionProperties(IntPtr pLayerName, uint* pPropertyCount,
VkExtensionProperties* pProperties);
}

Просмотреть файл

@ -0,0 +1,82 @@
using System;
using Avalonia.SourceGenerator;
using Avalonia.Vulkan.Interop;
using Avalonia.Vulkan.UnmanagedInterop;
using uint32_t = System.UInt32;
using VkBool32 = System.UInt32;
namespace Avalonia.Vulkan;
internal unsafe partial class VulkanInstanceApi
{
public IVulkanInstance Instance { get; }
public VulkanInstanceApi(IVulkanInstance instance)
{
Instance = instance;
Initialize(name => instance.GetInstanceProcAddress(instance.Handle, name));
}
[GetProcAddress("vkCreateDebugUtilsMessengerEXT", true)]
public partial VkResult CreateDebugUtilsMessengerEXT(VkInstance instance,
ref VkDebugUtilsMessengerCreateInfoEXT pCreateInfo, IntPtr pAllocator, out VkDebugUtilsMessengerEXT pMessenger);
[GetProcAddress("vkEnumeratePhysicalDevices")]
public partial VkResult EnumeratePhysicalDevices(VkInstance instance, ref uint32_t pPhysicalDeviceCount,
VkPhysicalDevice* pPhysicalDevices);
[GetProcAddress("vkGetPhysicalDeviceProperties")]
public partial void GetPhysicalDeviceProperties(VkPhysicalDevice physicalDevice,
out VkPhysicalDeviceProperties pProperties);
[GetProcAddress("vkEnumerateDeviceExtensionProperties")]
public partial VkResult EnumerateDeviceExtensionProperties(VkPhysicalDevice physicalDevice, byte* pLayerName,
ref uint32_t pPropertyCount, VkExtensionProperties* pProperties);
[GetProcAddress("vkGetPhysicalDeviceSurfaceSupportKHR")]
public partial VkResult GetPhysicalDeviceSurfaceSupportKHR(VkPhysicalDevice physicalDevice,
uint32_t queueFamilyIndex,
VkSurfaceKHR surface, out VkBool32 pSupported);
[GetProcAddress("vkGetPhysicalDeviceQueueFamilyProperties")]
public partial void GetPhysicalDeviceQueueFamilyProperties(VkPhysicalDevice physicalDevice,
ref uint32_t pQueueFamilyPropertyCount, VkQueueFamilyProperties* pQueueFamilyProperties);
[GetProcAddress("vkCreateDevice")]
public partial VkResult CreateDevice(VkPhysicalDevice physicalDevice, ref VkDeviceCreateInfo pCreateInfo,
IntPtr pAllocator, out VkDevice pDevice);
[GetProcAddress("vkGetDeviceQueue")]
public partial void GetDeviceQueue(VkDevice device, uint32_t queueFamilyIndex, uint32_t queueIndex,
out VkQueue pQueue);
[GetProcAddress("vkGetDeviceProcAddr")]
public partial IntPtr GetDeviceProcAddr(VkDevice device, IntPtr pName);
[GetProcAddress("vkDestroySurfaceKHR")]
public partial void DestroySurfaceKHR(VkInstance instance, VkSurfaceKHR surface, IntPtr pAllocator);
[GetProcAddress("vkGetPhysicalDeviceSurfaceFormatsKHR")]
public partial VkResult GetPhysicalDeviceSurfaceFormatsKHR(
VkPhysicalDevice physicalDevice,
VkSurfaceKHR surface,
ref uint32_t pSurfaceFormatCount,
VkSurfaceFormatKHR* pSurfaceFormats);
[GetProcAddress("vkGetPhysicalDeviceMemoryProperties")]
public partial void GetPhysicalDeviceMemoryProperties(VkPhysicalDevice physicalDevice,
out VkPhysicalDeviceMemoryProperties pMemoryProperties);
[GetProcAddress("vkGetPhysicalDeviceSurfaceCapabilitiesKHR")]
public partial VkResult GetPhysicalDeviceSurfaceCapabilitiesKHR(VkPhysicalDevice physicalDevice, VkSurfaceKHR surface,
out VkSurfaceCapabilitiesKHR pSurfaceCapabilities);
[GetProcAddress("vkGetPhysicalDeviceSurfacePresentModesKHR")]
public partial VkResult GetPhysicalDeviceSurfacePresentModesKHR(VkPhysicalDevice physicalDevice, VkSurfaceKHR surface,
ref uint32_t pPresentModeCount, VkPresentModeKHR* pPresentModes);
[GetProcAddress("vkGetPhysicalDeviceProperties2", true)]
public partial void GetPhysicalDeviceProperties2(
VkPhysicalDevice physicalDevice,
VkPhysicalDeviceProperties2* pProperties);
}

Просмотреть файл

@ -0,0 +1,767 @@
// ReSharper disable IdentifierTypo
// ReSharper disable InconsistentNaming
// ReSharper disable NotAccessedField.Global
#pragma warning disable CS0649
#pragma warning disable CS0169
#pragma warning disable CA1823
using System;
using uint32_t = System.UInt32;
using VkSampleCountFlags = System.UInt32;
using int32_t = System.Int32;
using VkBool32 = System.UInt32;
using uint8_t = System.Byte;
using size_t = System.IntPtr;
using VkDeviceSize = System.UInt64;
// ReSharper disable RedundantUnsafeContext
namespace Avalonia.Vulkan.UnmanagedInterop
{
struct VkInstance
{
public IntPtr Handle;
public VkInstance(IntPtr handle)
{
Handle = handle;
}
}
struct VkPhysicalDevice
{
public IntPtr Handle;
public VkPhysicalDevice(IntPtr handle)
{
Handle = handle;
}
}
struct VkDevice
{
public IntPtr Handle;
public VkDevice(IntPtr handle)
{
Handle = handle;
}
}
struct VkSwapchainKHR
{
public ulong Handle;
}
struct VkSemaphore
{
public ulong Handle;
}
struct VkFence
{
public ulong Handle;
}
struct VkImage
{
public ulong Handle;
}
struct VkImageView
{
public ulong Handle;
}
struct VkDeviceMemory
{
public ulong Handle;
}
struct VkQueue
{
public IntPtr Handle;
public VkQueue(IntPtr handle)
{
Handle = handle;
}
}
struct VkCommandPool
{
public ulong Handle;
}
struct VkCommandBuffer
{
public IntPtr Handle;
}
struct VkSurfaceKHR
{
public ulong Handle;
public VkSurfaceKHR(ulong handle)
{
Handle = handle;
}
}
struct VkDebugUtilsMessengerEXT
{
public IntPtr Handle;
}
unsafe struct VkLayerProperties
{
public fixed byte layerName[256];
public uint32_t specVersion;
public uint32_t implementationVersion;
public fixed byte description[256];
}
unsafe struct VkDebugUtilsLabelEXT
{
public VkStructureType sType;
public IntPtr pNext;
public IntPtr pLabelName;
public fixed float color[4];
}
struct VkDebugUtilsObjectNameInfoEXT
{
public VkStructureType sType;
public IntPtr pNext;
public VkObjectType objectType;
public ulong objectHandle;
public IntPtr pObjectName;
}
unsafe struct VkDebugUtilsMessengerCallbackDataEXT
{
public VkStructureType sType;
public IntPtr pNext;
public uint flags;
public IntPtr pMessageIdName;
public int32_t messageIdNumber;
public IntPtr pMessage;
public uint32_t queueLabelCount;
public VkDebugUtilsLabelEXT* pQueueLabels;
public uint32_t cmdBufLabelCount;
public VkDebugUtilsLabelEXT* pCmdBufLabels;
public uint32_t objectCount;
public VkDebugUtilsObjectNameInfoEXT* pObjects;
}
unsafe delegate VkBool32 VkDebugUtilsMessengerCallbackEXTDelegate(
VkDebugUtilsMessageSeverityFlagsEXT messageSeverity,
VkDebugUtilsMessageTypeFlagsEXT messageTypes,
VkDebugUtilsMessengerCallbackDataEXT* pCallbackData,
void* pUserData);
struct VkDebugUtilsMessengerCreateInfoEXT
{
public VkStructureType sType;
public IntPtr pNext;
public uint flags;
public VkDebugUtilsMessageSeverityFlagsEXT messageSeverity;
public VkDebugUtilsMessageTypeFlagsEXT messageType;
public IntPtr pfnUserCallback;
public IntPtr pUserData;
}
unsafe struct VkInstanceCreateInfo
{
public VkStructureType sType;
public IntPtr pNext;
public int flags;
public VkApplicationInfo* pApplicationInfo;
public uint32_t enabledLayerCount;
public byte** ppEnabledLayerNames;
public uint32_t enabledExtensionCount;
public byte** ppEnabledExtensionNames;
}
unsafe struct VkApplicationInfo
{
public VkStructureType sType;
public void* pNext;
public byte* pApplicationName;
public uint32_t applicationVersion;
public byte* pEngineName;
public uint32_t engineVersion;
public uint32_t apiVersion;
}
unsafe struct VkPhysicalDeviceSparseProperties
{
public VkBool32 residencyStandard2DBlockShape;
public VkBool32 residencyStandard2DMultisampleBlockShape;
public VkBool32 residencyStandard3DBlockShape;
public VkBool32 residencyAlignedMipSize;
public VkBool32 residencyNonResidentStrict;
}
unsafe struct VkPhysicalDeviceProperties
{
public uint32_t apiVersion;
public uint32_t driverVersion;
public uint32_t vendorID;
public uint32_t deviceID;
public VkPhysicalDeviceType deviceType;
public fixed byte deviceName[256];
public fixed uint8_t pipelineCacheUUID[16];
public VkPhysicalDeviceLimits limits;
public VkPhysicalDeviceSparseProperties sparseProperties;
}
unsafe struct VkPhysicalDeviceProperties2 {
public VkStructureType sType;
public void* pNext;
public VkPhysicalDeviceProperties properties;
}
unsafe struct VkPhysicalDeviceIDProperties {
public VkStructureType sType;
public void* pNext;
public fixed uint8_t deviceUUID[16];
public fixed uint8_t driverUUID[16];
public fixed uint8_t deviceLUID[8];
public uint32_t deviceNodeMask;
public VkBool32 deviceLUIDValid;
}
unsafe struct VkPhysicalDeviceLimits
{
public uint32_t maxImageDimension1D;
public uint32_t maxImageDimension2D;
public uint32_t maxImageDimension3D;
public uint32_t maxImageDimensionCube;
public uint32_t maxImageArrayLayers;
public uint32_t maxTexelBufferElements;
public uint32_t maxUniformBufferRange;
public uint32_t maxStorageBufferRange;
public uint32_t maxPushConstantsSize;
public uint32_t maxMemoryAllocationCount;
public uint32_t maxSamplerAllocationCount;
public VkDeviceSize bufferImageGranularity;
public VkDeviceSize sparseAddressSpaceSize;
public uint32_t maxBoundDescriptorSets;
public uint32_t maxPerStageDescriptorSamplers;
public uint32_t maxPerStageDescriptorUniformBuffers;
public uint32_t maxPerStageDescriptorStorageBuffers;
public uint32_t maxPerStageDescriptorSampledImages;
public uint32_t maxPerStageDescriptorStorageImages;
public uint32_t maxPerStageDescriptorInputAttachments;
public uint32_t maxPerStageResources;
public uint32_t maxDescriptorSetSamplers;
public uint32_t maxDescriptorSetUniformBuffers;
public uint32_t maxDescriptorSetUniformBuffersDynamic;
public uint32_t maxDescriptorSetStorageBuffers;
public uint32_t maxDescriptorSetStorageBuffersDynamic;
public uint32_t maxDescriptorSetSampledImages;
public uint32_t maxDescriptorSetStorageImages;
public uint32_t maxDescriptorSetInputAttachments;
public uint32_t maxVertexInputAttributes;
public uint32_t maxVertexInputBindings;
public uint32_t maxVertexInputAttributeOffset;
public uint32_t maxVertexInputBindingStride;
public uint32_t maxVertexOutputComponents;
public uint32_t maxTessellationGenerationLevel;
public uint32_t maxTessellationPatchSize;
public uint32_t maxTessellationControlPerVertexInputComponents;
public uint32_t maxTessellationControlPerVertexOutputComponents;
public uint32_t maxTessellationControlPerPatchOutputComponents;
public uint32_t maxTessellationControlTotalOutputComponents;
public uint32_t maxTessellationEvaluationInputComponents;
public uint32_t maxTessellationEvaluationOutputComponents;
public uint32_t maxGeometryShaderInvocations;
public uint32_t maxGeometryInputComponents;
public uint32_t maxGeometryOutputComponents;
public uint32_t maxGeometryOutputVertices;
public uint32_t maxGeometryTotalOutputComponents;
public uint32_t maxFragmentInputComponents;
public uint32_t maxFragmentOutputAttachments;
public uint32_t maxFragmentDualSrcAttachments;
public uint32_t maxFragmentCombinedOutputResources;
public uint32_t maxComputeSharedMemorySize;
public fixed uint32_t maxComputeWorkGroupCount[3];
public uint32_t maxComputeWorkGroupInvocations;
public fixed uint32_t maxComputeWorkGroupSize[3];
public uint32_t subPixelPrecisionBits;
public uint32_t subTexelPrecisionBits;
public uint32_t mipmapPrecisionBits;
public uint32_t maxDrawIndexedIndexValue;
public uint32_t maxDrawIndirectCount;
public float maxSamplerLodBias;
public float maxSamplerAnisotropy;
public uint32_t maxViewports;
public fixed uint32_t maxViewportDimensions[2];
public fixed float viewportBoundsRange[2];
public uint32_t viewportSubPixelBits;
public size_t minMemoryMapAlignment;
public VkDeviceSize minTexelBufferOffsetAlignment;
public VkDeviceSize minUniformBufferOffsetAlignment;
public VkDeviceSize minStorageBufferOffsetAlignment;
public int32_t minTexelOffset;
public uint32_t maxTexelOffset;
public int32_t minTexelGatherOffset;
public uint32_t maxTexelGatherOffset;
public float minInterpolationOffset;
public float maxInterpolationOffset;
public uint32_t subPixelInterpolationOffsetBits;
public uint32_t maxFramebufferWidth;
public uint32_t maxFramebufferHeight;
public uint32_t maxFramebufferLayers;
public VkSampleCountFlags framebufferColorSampleCounts;
public VkSampleCountFlags framebufferDepthSampleCounts;
public VkSampleCountFlags framebufferStencilSampleCounts;
public VkSampleCountFlags framebufferNoAttachmentsSampleCounts;
public uint32_t maxColorAttachments;
public VkSampleCountFlags sampledImageColorSampleCounts;
public VkSampleCountFlags sampledImageIntegerSampleCounts;
public VkSampleCountFlags sampledImageDepthSampleCounts;
public VkSampleCountFlags sampledImageStencilSampleCounts;
public VkSampleCountFlags storageImageSampleCounts;
public uint32_t maxSampleMaskWords;
public VkBool32 timestampComputeAndGraphics;
public float timestampPeriod;
public uint32_t maxClipDistances;
public uint32_t maxCullDistances;
public uint32_t maxCombinedClipAndCullDistances;
public uint32_t discreteQueuePriorities;
public fixed float pointSizeRange[2];
public fixed float lineWidthRange[2];
public float pointSizeGranularity;
public float lineWidthGranularity;
public VkBool32 strictLines;
public VkBool32 standardSampleLocations;
public VkDeviceSize optimalBufferCopyOffsetAlignment;
public VkDeviceSize optimalBufferCopyRowPitchAlignment;
public VkDeviceSize nonCoherentAtomSize;
}
internal unsafe struct VkExtensionProperties
{
public fixed byte extensionName[256];
public uint32_t specVersion;
}
struct VkQueueFamilyProperties
{
public VkQueueFlags queueFlags;
public uint32_t queueCount;
public uint32_t timestampValidBits;
public VkExtent3D minImageTransferGranularity;
}
struct VkExtent2D
{
public uint32_t width;
public uint32_t height;
}
struct VkExtent3D
{
public uint32_t width;
public uint32_t height;
public uint32_t depth;
}
unsafe struct VkDeviceQueueCreateInfo
{
public VkStructureType sType;
public IntPtr pNext;
public VkDeviceQueueCreateFlags flags;
public uint32_t queueFamilyIndex;
public uint32_t queueCount;
public float* pQueuePriorities;
}
unsafe struct VkDeviceCreateInfo
{
public VkStructureType sType;
public IntPtr pNext;
public uint flags;
public uint32_t queueCreateInfoCount;
public VkDeviceQueueCreateInfo* pQueueCreateInfos;
public uint32_t enabledLayerCount;
public byte** ppEnabledLayerNames;
public uint32_t enabledExtensionCount;
public byte** ppEnabledExtensionNames;
public IntPtr pEnabledFeatures;
}
struct VkFenceCreateInfo
{
public VkStructureType sType;
IntPtr pNext;
public VkFenceCreateFlags flags;
}
struct VkCommandPoolCreateInfo
{
public VkStructureType sType;
IntPtr pNext;
public VkCommandPoolCreateFlags flags;
public uint32_t queueFamilyIndex;
}
struct VkCommandBufferAllocateInfo
{
public VkStructureType sType;
IntPtr pNext;
public VkCommandPool commandPool;
public VkCommandBufferLevel level;
public uint32_t commandBufferCount;
}
struct VkCommandBufferBeginInfo
{
public VkStructureType sType;
IntPtr pNext;
public VkCommandBufferUsageFlags flags;
IntPtr pInheritanceInfo;
}
struct VkSemaphoreCreateInfo
{
public VkStructureType sType;
IntPtr pNext;
uint flags;
}
unsafe struct VkSubmitInfo
{
public VkStructureType sType;
public IntPtr pNext;
public uint32_t waitSemaphoreCount;
public VkSemaphore* pWaitSemaphores;
public VkPipelineStageFlags* pWaitDstStageMask;
public uint32_t commandBufferCount;
public VkCommandBuffer* pCommandBuffers;
public uint32_t signalSemaphoreCount;
public VkSemaphore* pSignalSemaphores;
}
struct VkSurfaceFormatKHR
{
public VkFormat format;
public VkColorSpaceKHR colorSpace;
}
struct VkMemoryType
{
public VkMemoryPropertyFlags propertyFlags;
public uint32_t heapIndex;
}
struct VkMemoryHeap
{
public VkDeviceSize size;
public VkMemoryHeapFlags flags;
}
unsafe struct VkPhysicalDeviceMemoryProperties
{
public uint32_t memoryTypeCount;
public VkMemoryTypesBuffer memoryTypes;
public uint32_t memoryHeapCount;
public VkMemoryHeapsBuffer memoryHeaps;
public struct VkMemoryTypesBuffer
{
public VkMemoryType Element0;
public VkMemoryType Element1;
public VkMemoryType Element2;
public VkMemoryType Element3;
public VkMemoryType Element4;
public VkMemoryType Element5;
public VkMemoryType Element6;
public VkMemoryType Element7;
public VkMemoryType Element8;
public VkMemoryType Element9;
public VkMemoryType Element10;
public VkMemoryType Element11;
public VkMemoryType Element12;
public VkMemoryType Element13;
public VkMemoryType Element14;
public VkMemoryType Element15;
public VkMemoryType Element16;
public VkMemoryType Element17;
public VkMemoryType Element18;
public VkMemoryType Element19;
public VkMemoryType Element20;
public VkMemoryType Element21;
public VkMemoryType Element22;
public VkMemoryType Element23;
public VkMemoryType Element24;
public VkMemoryType Element25;
public VkMemoryType Element26;
public VkMemoryType Element27;
public VkMemoryType Element28;
public VkMemoryType Element29;
public VkMemoryType Element30;
public VkMemoryType Element31;
public ref VkMemoryType this[int index]
{
get
{
if (index > 31 || index < 0)
{
throw new ArgumentOutOfRangeException(nameof(index));
}
fixed (VkMemoryType* ptr = &Element0)
{
return ref ptr[index];
}
}
}
}
public struct VkMemoryHeapsBuffer
{
public VkMemoryHeap Element0;
public VkMemoryHeap Element1;
public VkMemoryHeap Element2;
public VkMemoryHeap Element3;
public VkMemoryHeap Element4;
public VkMemoryHeap Element5;
public VkMemoryHeap Element6;
public VkMemoryHeap Element7;
public VkMemoryHeap Element8;
public VkMemoryHeap Element9;
public VkMemoryHeap Element10;
public VkMemoryHeap Element11;
public VkMemoryHeap Element12;
public VkMemoryHeap Element13;
public VkMemoryHeap Element14;
public VkMemoryHeap Element15;
public ref VkMemoryHeap this[int index]
{
get
{
if (index > 15 || index < 0)
{
throw new ArgumentOutOfRangeException(nameof(index));
}
fixed (VkMemoryHeap* ptr = &Element0)
{
return ref ptr[index];
}
}
}
}
}
unsafe struct VkImageCreateInfo
{
public VkStructureType sType;
public void* pNext;
public VkImageCreateFlags flags;
public VkImageType imageType;
public VkFormat format;
public VkExtent3D extent;
public uint32_t mipLevels;
public uint32_t arrayLayers;
public VkSampleCountFlags samples;
public VkImageTiling tiling;
public VkImageUsageFlags usage;
public VkSharingMode sharingMode;
public uint32_t queueFamilyIndexCount;
public uint32_t* pQueueFamilyIndices;
public VkImageLayout initialLayout;
}
struct VkMemoryRequirements
{
public VkDeviceSize size;
public VkDeviceSize alignment;
public uint32_t memoryTypeBits;
}
struct VkMemoryAllocateInfo
{
public VkStructureType sType;
public IntPtr pNext;
public VkDeviceSize allocationSize;
public uint32_t memoryTypeIndex;
}
struct VkComponentMapping
{
public VkComponentSwizzle r;
public VkComponentSwizzle g;
public VkComponentSwizzle b;
public VkComponentSwizzle a;
}
struct VkImageSubresourceRange
{
public VkImageAspectFlags aspectMask;
public uint32_t baseMipLevel;
public uint32_t levelCount;
public uint32_t baseArrayLayer;
public uint32_t layerCount;
}
struct VkImageViewCreateInfo
{
public VkStructureType sType;
public IntPtr pNext;
public VkImageViewCreateFlags flags;
public VkImage image;
public VkImageViewType viewType;
public VkFormat format;
public VkComponentMapping components;
public VkImageSubresourceRange subresourceRange;
}
struct VkImageMemoryBarrier
{
public VkStructureType sType;
IntPtr pNext;
public VkAccessFlags srcAccessMask;
public VkAccessFlags dstAccessMask;
public VkImageLayout oldLayout;
public VkImageLayout newLayout;
public uint32_t srcQueueFamilyIndex;
public uint32_t dstQueueFamilyIndex;
public VkImage image;
public VkImageSubresourceRange subresourceRange;
}
struct VkSurfaceCapabilitiesKHR
{
public uint32_t minImageCount;
public uint32_t maxImageCount;
public VkExtent2D currentExtent;
public VkExtent2D minImageExtent;
public VkExtent2D maxImageExtent;
public uint32_t maxImageArrayLayers;
public VkSurfaceTransformFlagsKHR supportedTransforms;
public VkSurfaceTransformFlagsKHR currentTransform;
public VkCompositeAlphaFlagsKHR supportedCompositeAlpha;
public VkImageUsageFlags supportedUsageFlags;
}
unsafe struct VkSwapchainCreateInfoKHR
{
public VkStructureType sType;
public IntPtr pNext;
public VkSwapchainCreateFlagsKHR flags;
public VkSurfaceKHR surface;
public uint32_t minImageCount;
public VkFormat imageFormat;
public VkColorSpaceKHR imageColorSpace;
public VkExtent2D imageExtent;
public uint32_t imageArrayLayers;
public VkImageUsageFlags imageUsage;
public VkSharingMode imageSharingMode;
public uint32_t queueFamilyIndexCount;
public uint32_t* pQueueFamilyIndices;
public VkSurfaceTransformFlagsKHR preTransform;
public VkCompositeAlphaFlagsKHR compositeAlpha;
public VkPresentModeKHR presentMode;
public VkBool32 clipped;
public VkSwapchainKHR oldSwapchain;
}
struct VkOffset3D
{
public int32_t x;
public int32_t y;
public int32_t z;
}
struct VkImageSubresourceLayers
{
public VkImageAspectFlags aspectMask;
public uint32_t mipLevel;
public uint32_t baseArrayLayer;
public uint32_t layerCount;
}
struct VkImageBlit
{
public VkImageSubresourceLayers srcSubresource;
public VkOffset3D srcOffsets1;
public VkOffset3D srcOffsets2;
public VkImageSubresourceLayers dstSubresource;
public VkOffset3D dstOffsets1;
public VkOffset3D dstOffsets2;
}
unsafe struct VkPresentInfoKHR
{
public VkStructureType sType;
public IntPtr pNext;
public uint32_t waitSemaphoreCount;
public VkSemaphore* pWaitSemaphores;
public uint32_t swapchainCount;
public VkSwapchainKHR* pSwapchains;
public uint32_t* pImageIndices;
public VkResult* pResults;
}
unsafe struct VkImportSemaphoreFdInfoKHR
{
public VkStructureType sType;
public void* pNext;
public VkSemaphore semaphore;
public VkSemaphoreImportFlags flags;
public VkExternalSemaphoreHandleTypeFlags handleType;
public int fd;
}
unsafe struct VkImportSemaphoreWin32HandleInfoKHR
{
public VkStructureType sType;
public void* pNext;
public VkSemaphore semaphore;
public VkSemaphoreImportFlags flags;
public VkExternalSemaphoreHandleTypeFlags handleType;
public IntPtr handle;
public IntPtr name;
}
unsafe struct VkImportMemoryFdInfoKHR
{
public VkStructureType sType;
public void* pNext;
public VkExternalMemoryHandleTypeFlagBits handleType;
public int fd;
}
unsafe struct VkImportMemoryWin32HandleInfoKHR
{
public VkStructureType sType;
public void* pNext;
public VkExternalMemoryHandleTypeFlagBits handleType;
public IntPtr handle;
public IntPtr name;
}
unsafe struct VkMemoryDedicatedAllocateInfo
{
public VkStructureType sType;
public void* pNext;
public VkImage image;
public IntPtr buffer;
}
unsafe struct VkExternalMemoryImageCreateInfo
{
public VkStructureType sType;
public void* pNext;
public VkExternalMemoryHandleTypeFlagBits handleTypes;
}
}

Просмотреть файл

@ -0,0 +1,19 @@
using System;
using System.Collections.Generic;
using System.Runtime.InteropServices;
using Avalonia.Vulkan.Interop;
namespace Avalonia.Vulkan;
static class VulkanHelpers
{
public static uint MakeVersion(Version v) => MakeVersion(v.Major, v.Minor, v.Build);
public static uint MakeVersion(int major, int minor, int patch)
{
return (uint)((major << 22) | (minor << 12) | patch);
}
public const uint QueueFamilyIgnored = 4294967295;
}

Просмотреть файл

@ -0,0 +1,73 @@
using System;
using System.Collections.Generic;
using System.Linq;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan;
internal class VulkanContext : IVulkanPlatformGraphicsContext
{
private readonly IVulkanKhrSurfacePlatformSurfaceFactory? _surfaceFactory;
private readonly VulkanExternalObjectsFeature? _externalObjectsFeature;
public IVulkanDevice Device { get; }
public IVulkanInstance Instance => Device.Instance;
public VulkanContext(IVulkanDevice device, Dictionary<Type, object> platformFeatures)
{
Device = device;
using (device.Lock())
{
InstanceApi = new VulkanInstanceApi(device.Instance);
DeviceApi = new VulkanDeviceApi(device);
if (platformFeatures.TryGetValue(typeof(IVulkanKhrSurfacePlatformSurfaceFactory), out var factory))
_surfaceFactory = (IVulkanKhrSurfacePlatformSurfaceFactory)factory;
}
if (
VulkanExternalObjectsFeature.RequiredInstanceExtensions.All(ext => Instance.EnabledExtensions.Contains(ext))
&& VulkanExternalObjectsFeature.RequiredDeviceExtensions.All(ext => Device.EnabledExtensions.Contains(ext)))
{
_externalObjectsFeature = new VulkanExternalObjectsFeature(this);
}
}
public void Dispose()
{
}
public object? TryGetFeature(Type featureType)
{
if (featureType == typeof(IVulkanContextExternalObjectsFeature))
return _externalObjectsFeature;
return null;
}
public bool IsLost => Device.IsLost;
public IDisposable EnsureCurrent() => Device.Lock();
public VkDevice DeviceHandle => new (Device.Handle);
public VkPhysicalDevice PhysicalDeviceHandle => new (Device.PhysicalDeviceHandle);
public VkInstance InstanceHandle => new(Instance.Handle);
public VkQueue MainQueueHandle => new(Device.MainQueueHandle);
public uint GraphicsQueueFamilyIndex => Device.GraphicsQueueFamilyIndex;
public VulkanInstanceApi InstanceApi { get; }
public VulkanDeviceApi DeviceApi { get; }
public IVulkanRenderTarget CreateRenderTarget(IEnumerable<object> surfaces)
{
foreach (var surf in surfaces)
{
IVulkanKhrSurfacePlatformSurface khrSurface;
if (surf is IVulkanKhrSurfacePlatformSurface khr)
khrSurface = khr;
else if (_surfaceFactory?.CanRenderToSurface(this, surf) == true)
khrSurface = _surfaceFactory.CreateSurface(this, surf);
else
continue;
return new VulkanKhrRenderTarget(khrSurface, this);
}
throw new VulkanException("Unable to find a suitable platform surface");
}
}

Просмотреть файл

@ -0,0 +1,34 @@
using System;
using Avalonia.Vulkan.Interop;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan;
public class VulkanException : Exception
{
public VulkanException(string message) : base(message)
{
}
public VulkanException(string funcName, int res) : this(funcName, (VkResult)res)
{
}
internal VulkanException(string funcName, VkResult res) : base($"{funcName} returned {res}")
{
}
public static void ThrowOnError(string funcName, int res) => ((VkResult)res).ThrowOnError(funcName);
}
internal static class VulkanExceptionExtensions
{
public static void ThrowOnError(this VkResult res, string funcName)
{
if (res != VkResult.VK_SUCCESS)
throw new VulkanException(funcName, res);
}
}

Просмотреть файл

@ -0,0 +1,311 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.InteropServices;
using Avalonia.Platform;
using Avalonia.Rendering.Composition;
using Avalonia.Vulkan.Interop;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan;
internal unsafe class VulkanExternalObjectsFeature : IVulkanContextExternalObjectsFeature
{
public static string[] RequiredInstanceExtensions = {
"VK_KHR_get_physical_device_properties2",
"VK_KHR_external_memory_capabilities",
"VK_KHR_external_semaphore_capabilities"
};
private static string[] s_requiredCommonDeviceExtensions =
{
"VK_KHR_external_memory",
"VK_KHR_external_semaphore",
"VK_KHR_dedicated_allocation",
};
private static string[] s_requiredLinuxDeviceExtensions =
s_requiredCommonDeviceExtensions.Concat(new[]
{
"VK_KHR_external_semaphore_fd",
"VK_KHR_external_memory_fd"
}).ToArray();
private static string[] s_requiredWin32DeviceExtensions = s_requiredCommonDeviceExtensions.Concat(new[]
{
"VK_KHR_external_semaphore_win32",
"VK_KHR_external_memory_win32"
}).ToArray();
public static string[] RequiredDeviceExtensions = RuntimeInformation.IsOSPlatform(OSPlatform.Windows)
? s_requiredWin32DeviceExtensions
: s_requiredLinuxDeviceExtensions;
private readonly VulkanContext _context;
private readonly VulkanCommandBufferPool _pool;
public VulkanExternalObjectsFeature(VulkanContext context)
{
_context = context;
if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
{
//TODO: keyed muted
SupportedImageHandleTypes = new[]
{
KnownPlatformGraphicsExternalImageHandleTypes.VulkanOpaqueNtHandle,
KnownPlatformGraphicsExternalImageHandleTypes.VulkanOpaqueKmtHandle,
};
SupportedSemaphoreTypes = new[]
{
KnownPlatformGraphicsExternalSemaphoreHandleTypes.VulkanOpaqueNtHandle,
KnownPlatformGraphicsExternalSemaphoreHandleTypes.VulkanOpaqueKmtHandle
};
}
else
{
SupportedImageHandleTypes = new[]
{
KnownPlatformGraphicsExternalImageHandleTypes.VulkanOpaquePosixFileDescriptor
};
SupportedSemaphoreTypes = new[]
{
KnownPlatformGraphicsExternalSemaphoreHandleTypes.VulkanOpaquePosixFileDescriptor
};
}
var physicalDeviceIDProperties = new VkPhysicalDeviceIDProperties()
{
sType = VkStructureType.VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_ID_PROPERTIES
};
var physicalDeviceProperties2 = new VkPhysicalDeviceProperties2()
{
sType = VkStructureType.VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROPERTIES_2,
pNext = &physicalDeviceIDProperties
};
_context.InstanceApi.GetPhysicalDeviceProperties2(_context.PhysicalDeviceHandle, &physicalDeviceProperties2);
var luid = new Span<byte>(physicalDeviceIDProperties.deviceLUID, 8).ToArray();
if (luid.Any(b => b != 0))
DeviceLuid = luid;
var uuid = new Span<byte>(physicalDeviceIDProperties.deviceUUID, 16).ToArray();
if (uuid.Any(b => b != 0))
DeviceUuid = uuid;
_pool = new VulkanCommandBufferPool(_context);
}
public IReadOnlyList<string> SupportedImageHandleTypes { get; }
public IReadOnlyList<string> SupportedSemaphoreTypes { get; }
public byte[]? DeviceUuid { get; }
public byte[]? DeviceLuid { get; }
public CompositionGpuImportedImageSynchronizationCapabilities GetSynchronizationCapabilities(string imageHandleType)
{
if (!SupportedImageHandleTypes.Contains(imageHandleType))
throw new ArgumentException();
//TODO: keyed muted
return CompositionGpuImportedImageSynchronizationCapabilities.Semaphores;
}
public IVulkanExternalImage ImportImage(IPlatformHandle handle, PlatformGraphicsExternalImageProperties properties)
{
_pool.FreeFinishedCommandBuffers();
if (!SupportedImageHandleTypes.Contains(handle.HandleDescriptor))
throw new NotSupportedException();
return new ImportedImage(_context, _pool, handle, properties);
}
public IVulkanExternalSemaphore ImportSemaphore(IPlatformHandle handle)
{
if (!SupportedSemaphoreTypes.Contains(handle.HandleDescriptor))
throw new NotSupportedException();
var typeBit = handle.HandleDescriptor switch
{
KnownPlatformGraphicsExternalSemaphoreHandleTypes.VulkanOpaquePosixFileDescriptor =>
VkExternalSemaphoreHandleTypeFlags.VK_EXTERNAL_SEMAPHORE_HANDLE_TYPE_OPAQUE_FD_BIT,
KnownPlatformGraphicsExternalSemaphoreHandleTypes.VulkanOpaqueKmtHandle =>
VkExternalSemaphoreHandleTypeFlags.VK_EXTERNAL_SEMAPHORE_HANDLE_TYPE_OPAQUE_WIN32_KMT_BIT,
KnownPlatformGraphicsExternalSemaphoreHandleTypes.VulkanOpaqueNtHandle =>
VkExternalSemaphoreHandleTypeFlags.VK_EXTERNAL_SEMAPHORE_HANDLE_TYPE_OPAQUE_WIN32_BIT,
_ => throw new NotSupportedException()
};
var semaphore = new VulkanSemaphore(_context);
if (handle.HandleDescriptor ==
KnownPlatformGraphicsExternalSemaphoreHandleTypes.VulkanOpaquePosixFileDescriptor)
{
var info = new VkImportSemaphoreFdInfoKHR
{
sType = VkStructureType.VK_STRUCTURE_TYPE_IMPORT_SEMAPHORE_FD_INFO_KHR,
fd = handle.Handle.ToInt32(),
handleType = typeBit,
semaphore = semaphore.Handle
};
var addr = _context.Instance.GetDeviceProcAddress(_context.Device.Handle, "vkImportSemaphoreFdKHR");
if (addr == IntPtr.Zero)
addr = _context.Instance.GetInstanceProcAddress(_context.Instance.Handle, "vkImportSemaphoreFdKHR");
_context.DeviceApi.ImportSemaphoreFdKHR(_context.DeviceHandle, &info)
.ThrowOnError("vkImportSemaphoreFdKHR");
return new ImportedSemaphore(_context, _pool, semaphore);
}
else
{
var info = new VkImportSemaphoreWin32HandleInfoKHR()
{
sType = VkStructureType.VK_STRUCTURE_TYPE_IMPORT_SEMAPHORE_WIN32_HANDLE_INFO_KHR,
handle = handle.Handle,
handleType = typeBit,
semaphore = semaphore.Handle
};
_context.DeviceApi.ImportSemaphoreWin32HandleKHR(_context.DeviceHandle, &info)
.ThrowOnError("vkImportSemaphoreWin32HandleKHR");
return new ImportedSemaphore(_context, _pool, semaphore);
}
}
class ImportedSemaphore : IVulkanExternalSemaphore
{
private readonly VulkanContext _context;
private readonly VulkanCommandBufferPool _pool;
private VulkanSemaphore? _sem;
private VulkanSemaphore Sem => _sem ?? throw new ObjectDisposedException(nameof(ImportedSemaphore));
public ImportedSemaphore(VulkanContext context, VulkanCommandBufferPool pool, VulkanSemaphore sem)
{
_context = context;
_pool = pool;
_sem = sem;
}
public void Dispose()
{
_sem?.Dispose();
_sem = null;
}
public ulong Handle => Sem.Handle.Handle;
void SubmitSemaphore(VulkanSemaphore? wait, VulkanSemaphore? signal)
{
var buf = _pool.CreateCommandBuffer();
buf.BeginRecording();
_context.DeviceApi.CmdPipelineBarrier(
buf.Handle,
VkPipelineStageFlags.VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
VkPipelineStageFlags.VK_PIPELINE_STAGE_ALL_COMMANDS_BIT,
0,
0,
IntPtr.Zero,
0,
IntPtr.Zero,
0,
null);
buf.EndRecording();
buf.Submit(wait != null ? new[] { wait }.AsSpan() : default,
new[] { VkPipelineStageFlags.VK_PIPELINE_STAGE_ALL_GRAPHICS_BIT },
signal != null ? new[] { signal }.AsSpan() : default);
}
public void SubmitWaitSemaphore()
{
SubmitSemaphore(_sem, null);
}
public void SubmitSignalSemaphore()
{
SubmitSemaphore(null, _sem);
}
}
class ImportedImage : VulkanImageBase, IVulkanExternalImage
{
private readonly IVulkanPlatformGraphicsContext _context;
private readonly IPlatformHandle _importHandle;
private readonly PlatformGraphicsExternalImageProperties _properties;
private readonly VkExternalMemoryHandleTypeFlagBits _typeBit;
public VulkanImageInfo Info => ImageInfo;
public ImportedImage(IVulkanPlatformGraphicsContext context, VulkanCommandBufferPool commandBufferPool,
IPlatformHandle importHandle,
PlatformGraphicsExternalImageProperties properties) : base(context, commandBufferPool,
properties.Format switch
{
PlatformGraphicsExternalImageFormat.B8G8R8A8UNorm => VkFormat.VK_FORMAT_B8G8R8A8_UNORM,
PlatformGraphicsExternalImageFormat.R8G8B8A8UNorm => VkFormat.VK_FORMAT_R8G8B8A8_UNORM,
_ => throw new ArgumentException($"Format {properties.Format} is not supported")
}, new PixelSize(properties.Width, properties.Height), 1)
{
_context = context;
_importHandle = importHandle;
_properties = properties;
_typeBit = importHandle.HandleDescriptor switch
{
KnownPlatformGraphicsExternalImageHandleTypes.VulkanOpaquePosixFileDescriptor =>
VkExternalMemoryHandleTypeFlagBits.VK_EXTERNAL_MEMORY_HANDLE_TYPE_OPAQUE_FD_BIT,
KnownPlatformGraphicsExternalImageHandleTypes.VulkanOpaqueKmtHandle =>
VkExternalMemoryHandleTypeFlagBits.VK_EXTERNAL_MEMORY_HANDLE_TYPE_OPAQUE_WIN32_KMT_BIT,
KnownPlatformGraphicsExternalImageHandleTypes.VulkanOpaqueNtHandle =>
VkExternalMemoryHandleTypeFlagBits.VK_EXTERNAL_MEMORY_HANDLE_TYPE_OPAQUE_WIN32_BIT,
_ => throw new NotSupportedException()
};
var externalAlloc = new VkExternalMemoryImageCreateInfo
{
handleTypes = _typeBit,
sType = VkStructureType.VK_STRUCTURE_TYPE_EXTERNAL_MEMORY_IMAGE_CREATE_INFO
};
Initialize(&externalAlloc);
TransitionLayout(VkImageLayout.VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL, VkAccessFlags.VK_ACCESS_TRANSFER_READ_BIT);
this.CurrentLayout = VkImageLayout.VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL;
}
protected override VkDeviceMemory CreateMemory(VkImage image, ulong size, uint memoryTypeBits)
{
var handle = _importHandle;
if (_properties.MemoryOffset != 0 || _properties.MemorySize != size)
throw new Exception("Invalid memory size");
var dedicated = new VkMemoryDedicatedAllocateInfo()
{
image = image,
sType = VkStructureType.VK_STRUCTURE_TYPE_MEMORY_DEDICATED_ALLOCATE_INFO,
};
var isPosixHandle = handle.HandleDescriptor ==
KnownPlatformGraphicsExternalImageHandleTypes.VulkanOpaquePosixFileDescriptor;
var win32Info = new VkImportMemoryWin32HandleInfoKHR
{
sType = VkStructureType.VK_STRUCTURE_TYPE_IMPORT_MEMORY_WIN32_HANDLE_INFO_KHR,
handle = handle.Handle,
handleType = _typeBit,
pNext = &dedicated
};
var posixInfo = new VkImportMemoryFdInfoKHR()
{
sType = VkStructureType.VK_STRUCTURE_TYPE_IMPORT_MEMORY_FD_INFO_KHR,
handleType = _typeBit,
fd = isPosixHandle ? handle.Handle.ToInt32() : 0,
pNext = &dedicated
};
var memoryAllocateInfo = new VkMemoryAllocateInfo
{
pNext = new IntPtr(isPosixHandle ? &posixInfo : &win32Info),
sType = VkStructureType.VK_STRUCTURE_TYPE_MEMORY_ALLOCATE_INFO,
allocationSize = size,
memoryTypeIndex = (uint)VulkanMemoryHelper.FindSuitableMemoryTypeIndex(_context,
memoryTypeBits,
VkMemoryPropertyFlags.VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT),
};
_context.DeviceApi.AllocateMemory(_context.DeviceHandle, ref memoryAllocateInfo, IntPtr.Zero,
out var imageMemory).ThrowOnError("vkAllocateMemory");
return imageMemory;
}
}
}

Просмотреть файл

@ -0,0 +1,19 @@
using System;
namespace Avalonia.Vulkan;
public struct VulkanImageInfo
{
public uint Format { get; set; }
public PixelSize PixelSize { get; set; }
public ulong Handle { get; set; }
public uint Layout { get; set; }
public uint Tiling { get; set; }
public uint UsageFlags { get; set; }
public uint LevelCount { get; set; }
public uint SampleCount { get; set; }
public ulong MemoryHandle { get; set; }
public ulong ViewHandle { get; set; }
public ulong MemorySize { get; set; }
public bool IsProtected { get; set; }
}

Просмотреть файл

@ -0,0 +1,107 @@
using System;
using Avalonia.Vulkan.Interop;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Vulkan;
internal class VulkanKhrRenderTarget : IVulkanRenderTarget
{
private VulkanKhrSurface _khrSurface;
private readonly IVulkanPlatformGraphicsContext _context;
private VulkanDisplay _display;
private VulkanImage? _image;
private readonly IVulkanKhrSurfacePlatformSurface _platformSurface;
public VkFormat Format { get; }
public bool IsRgba { get; }
public VulkanKhrRenderTarget(IVulkanKhrSurfacePlatformSurface surface, IVulkanPlatformGraphicsContext context)
{
_platformSurface = surface;
_khrSurface = new(context, surface);
_display = VulkanDisplay.CreateDisplay(context, _khrSurface);
_context = context;
IsRgba = _display.SurfaceFormat.format >= VkFormat.VK_FORMAT_R8G8B8A8_UNORM &&
_display.SurfaceFormat.format <= VkFormat.VK_FORMAT_R8G8B8A8_SRGB;
// Skia seems to only create surfaces from images with unorm format
Format = IsRgba ? VkFormat.VK_FORMAT_R8G8B8A8_UNORM : VkFormat.VK_FORMAT_B8G8R8A8_UNORM;
}
private void CreateImage()
{
_image = new VulkanImage(_context, _display.CommandBufferPool, Format, _display.Size);
}
private void DestroyImage()
{
_context.DeviceApi.DeviceWaitIdle(_context.DeviceHandle);
_image?.Dispose();
_image = null;
}
public void Dispose()
{
_context.DeviceApi.DeviceWaitIdle(_context.DeviceHandle);
DestroyImage();
_display?.Dispose();
_display = null!;
_khrSurface?.Dispose();
_khrSurface = null!;
}
public IVulkanRenderSession BeginDraw()
{
var l = _context.EnsureCurrent();
_display.CommandBufferPool.FreeUsedCommandBuffers();
if (_display.EnsureSwapchainAvailable() || _image == null)
{
DestroyImage();
CreateImage();
}
else
_image.TransitionLayout(VkImageLayout.VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL,
VkAccessFlags.VK_ACCESS_NONE);
return new RenderingSession(_display, _image!, IsRgba, _platformSurface.Scaling, l);
}
public class RenderingSession : IVulkanRenderSession
{
private readonly VulkanImage _image;
private readonly IDisposable _dispose;
public RenderingSession(VulkanDisplay display, VulkanImage image, bool isRgba, double scaling,
IDisposable dispose)
{
_image = image;
_dispose = dispose;
Display = display;
IsRgba = isRgba;
Scaling = scaling;
}
public VulkanDisplay Display { get; }
public PixelSize Size => _image.Size;
public double Scaling { get; }
public bool IsYFlipped => true;
public VulkanImageInfo ImageInfo => _image.ImageInfo;
public bool IsRgba { get; }
public void Dispose()
{
try
{
var commandBuffer = Display.StartPresentation();
Display.BlitImageToCurrentImage(commandBuffer, _image);
Display.EndPresentation(commandBuffer);
}
finally
{
_dispose.Dispose();
}
}
}
}

Просмотреть файл

@ -0,0 +1,73 @@
using System;
using System.Collections.Generic;
namespace Avalonia.Vulkan;
public class VulkanOptions
{
public VulkanInstanceCreationOptions VulkanInstanceCreationOptions { get; set; } = new();
public VulkanDeviceCreationOptions VulkanDeviceCreationOptions { get; set; } = new();
public IVulkanDevice? CustomSharedDevice { get; set; }
}
public class VulkanInstanceCreationOptions
{
public VkGetInstanceProcAddressDelegate? CustomGetProcAddressDelegate { get; set; }
/// <summary>
/// Sets the application name of the vulkan instance
/// </summary>
public string? ApplicationName { get; set; }
/// <summary>
/// Specifies the vulkan api version to use
/// </summary>
public Version VulkanVersion{ get; set; } = new Version(1, 1, 0);
/// <summary>
/// Specifies additional extensions to enable if available on the instance
/// </summary>
public IList<string> InstanceExtensions { get; set; } = new List<string>();
/// <summary>
/// Specifies layers to enable if available on the instance
/// </summary>
public IList<string> EnabledLayers { get; set; } = new List<string>();
/// <summary>
/// Enables the debug layer
/// </summary>
public bool UseDebug { get; set; }
/*
/// <summary>
/// Sets the presentation mode the swapchain uses if available.
/// </summary>
//public PresentMode PresentMode { get; set; } = PresentMode.Mailbox;*/
}
public class VulkanDeviceCreationOptions
{
/// <summary>
/// Specifies extensions to enable if available on the logical device
/// </summary>
public IList<string> DeviceExtensions { get; set; } = new List<string>();
/// <summary>
/// Selects the first suitable discrete gpu available
/// </summary>
public bool PreferDiscreteGpu { get; set; }
public bool RequireComputeBit { get; set; }
}
public class VulkanPlatformSpecificOptions
{
public IList<string> RequiredInstanceExtensions { get; set; } = new List<string>();
public VkGetInstanceProcAddressDelegate? GetProcAddressDelegate { get; set; }
public Func<IVulkanInstance, ulong>? DeviceCheckSurfaceFactory { get; set; }
public Dictionary<Type, object> PlatformFeatures { get; set; } = new();
}
public delegate IntPtr VkGetInstanceProcAddressDelegate(IntPtr instance, string name);

Просмотреть файл

@ -0,0 +1,103 @@
using System;
using Avalonia.Logging;
using Avalonia.Platform;
namespace Avalonia.Vulkan;
public class VulkanPlatformGraphics : IPlatformGraphics
{
private readonly IVulkanDeviceFactory _factory;
private IVulkanPlatformGraphicsContext? _currentSharedContext;
private readonly VulkanPlatformSpecificOptions _platformOptions;
public VulkanPlatformGraphics(IVulkanDeviceFactory factory, VulkanPlatformSpecificOptions platformOptions)
{
_factory = factory;
_platformOptions = platformOptions;
}
public IPlatformGraphicsContext CreateContext() =>
new VulkanContext(_factory.CreateDevice(_platformOptions), _platformOptions.PlatformFeatures);
public IPlatformGraphicsContext GetSharedContext()
{
if (_currentSharedContext?.IsLost == true)
_currentSharedContext = null;
return _currentSharedContext =
new VulkanContext(_factory.GetSharedDevice(_platformOptions), _platformOptions.PlatformFeatures);
}
public bool UsesSharedContext => _factory.UsesShadedDevice;
public interface IVulkanDeviceFactory
{
bool UsesShadedDevice { get; }
IVulkanDevice GetSharedDevice(VulkanPlatformSpecificOptions platformOptions);
IVulkanDevice CreateDevice(VulkanPlatformSpecificOptions platformOptions);
}
class DefaultDeviceFactory : IVulkanDeviceFactory
{
private readonly IVulkanInstance _instance;
private readonly VulkanDeviceCreationOptions _deviceOptions;
public DefaultDeviceFactory(IVulkanInstance instance, VulkanDeviceCreationOptions deviceOptions)
{
_instance = instance;
_deviceOptions = deviceOptions;
}
public bool UsesShadedDevice => false;
public IVulkanDevice GetSharedDevice(VulkanPlatformSpecificOptions platformOptions) => throw new NotSupportedException();
public IVulkanDevice CreateDevice(VulkanPlatformSpecificOptions platformOptions)
{
return Interop.VulkanDevice.Create(_instance, _deviceOptions, platformOptions);
}
}
class CustomSharedDeviceFactory : IVulkanDeviceFactory
{
private readonly IVulkanDevice _device;
public CustomSharedDeviceFactory(IVulkanDevice device)
{
_device = device;
}
public bool UsesShadedDevice => true;
public IVulkanDevice GetSharedDevice(VulkanPlatformSpecificOptions platformOptions) => _device;
public IVulkanDevice CreateDevice(VulkanPlatformSpecificOptions platformOptions) =>
throw new NotSupportedException();
}
public static VulkanPlatformGraphics? TryCreate(VulkanOptions options, VulkanPlatformSpecificOptions platformOptions)
{
if (options.CustomSharedDevice != null)
return new(new CustomSharedDeviceFactory(options.CustomSharedDevice), platformOptions);
IVulkanInstance? instance = null;
try
{
instance = VulkanInstance.Create(options.VulkanInstanceCreationOptions ?? new(),
platformOptions);
var devOpts = options.VulkanDeviceCreationOptions ?? new();
Interop.VulkanDevice.Create(instance, devOpts, platformOptions)
.Dispose();
return new VulkanPlatformGraphics(new DefaultDeviceFactory(instance, devOpts), platformOptions);
}
catch (Exception e)
{
//instance?.Dispose();
Logger.TryGet(LogEventLevel.Error, "Vulkan")?.Log(null, "Unable to initialize Vulkan rendering: {0}", e);
return null;
}
}
}

Просмотреть файл

@ -0,0 +1,29 @@
using System;
using Avalonia.SourceGenerator;
using Avalonia.Vulkan;
using Avalonia.Vulkan.Interop;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.X11.Vulkan;
partial class X11VulkanInterface
{
public X11VulkanInterface(IVulkanInstance instance)
{
Initialize(name => instance.GetInstanceProcAddress(instance.Handle, name));
}
[GetProcAddress("vkCreateXlibSurfaceKHR")]
public partial int vkCreateXlibSurfaceKHR(IntPtr instance, ref VkXlibSurfaceCreateInfoKHR pCreateInfo,
IntPtr pAllocator, out ulong pSurface);
}
struct VkXlibSurfaceCreateInfoKHR
{
public const uint VK_STRUCTURE_TYPE_XLIB_SURFACE_CREATE_INFO_KHR = 1000004000;
public uint sType;
public IntPtr pNext;
public uint flags;
public IntPtr dpy;
public IntPtr window;
}

Просмотреть файл

@ -0,0 +1,94 @@
using System;
using System.Collections.Generic;
using System.Runtime.InteropServices;
using Avalonia.Platform;
using Avalonia.Vulkan;
namespace Avalonia.X11.Vulkan;
internal class VulkanSupport
{
private static IntPtr s_offscreenWindow;
[DllImport("libvulkan.so.1")]
private static extern IntPtr vkGetInstanceProcAddr(IntPtr instance, string name);
public static VulkanPlatformGraphics? TryInitialize(X11Info info, VulkanOptions options)
{
s_offscreenWindow = XLib.XCreateSimpleWindow(info.DeferredDisplay,
XLib.XDefaultRootWindow(info.DeferredDisplay), 0, 0, 1,
1, 1, IntPtr.Zero, IntPtr.Zero);
XLib.XSync(info.DeferredDisplay, true);
return VulkanPlatformGraphics.TryCreate(options ?? new(), new VulkanPlatformSpecificOptions
{
RequiredInstanceExtensions = { "VK_KHR_xlib_surface" },
GetProcAddressDelegate = vkGetInstanceProcAddr,
DeviceCheckSurfaceFactory = instance => CreateDummySurface(info.DeferredDisplay, instance),
PlatformFeatures = new Dictionary<Type, object>
{
[typeof(IVulkanKhrSurfacePlatformSurfaceFactory)] = new VulkanSurfaceFactory(info.DeferredDisplay)
}
});
}
internal class VulkanSurfaceFactory : IVulkanKhrSurfacePlatformSurfaceFactory
{
private readonly IntPtr _display;
public VulkanSurfaceFactory(IntPtr display)
{
_display = display;
}
public bool CanRenderToSurface(IVulkanPlatformGraphicsContext context, object surface) =>
surface is INativePlatformHandleSurface handle && handle.HandleDescriptor == "XID";
public IVulkanKhrSurfacePlatformSurface CreateSurface(IVulkanPlatformGraphicsContext context, object handle) =>
new XidSurface(_display, (INativePlatformHandleSurface)handle);
}
class XidSurface : IVulkanKhrSurfacePlatformSurface
{
private readonly IntPtr _display;
private readonly INativePlatformHandleSurface _handle;
public XidSurface(IntPtr display, INativePlatformHandleSurface handle)
{
_display = display;
_handle = handle;
}
public void Dispose()
{
// No-op
}
public double Scaling => _handle.Scaling;
public PixelSize Size => _handle.Size;
public ulong CreateSurface(IVulkanPlatformGraphicsContext context) =>
CreateXlibSurface(_display, _handle.Handle, context.Instance);
}
private static ulong CreateDummySurface(IntPtr display, IVulkanInstance instance)
{
var surf = CreateXlibSurface(display, s_offscreenWindow, instance);
XLib.XSync(display, true);
return surf;
}
private static ulong CreateXlibSurface(IntPtr display, IntPtr window, IVulkanInstance instance)
{
var vulkanXlib = new X11VulkanInterface(instance);
var createInfo = new VkXlibSurfaceCreateInfoKHR
{
sType = VkXlibSurfaceCreateInfoKHR.VK_STRUCTURE_TYPE_XLIB_SURFACE_CREATE_INFO_KHR,
dpy = display,
window = window
};
VulkanException.ThrowOnError("vkCreateXlibSurfaceKHR",
vulkanXlib.vkCreateXlibSurfaceKHR(instance.Handle, ref createInfo, IntPtr.Zero, out var surface));
return surface;
}
}

Просмотреть файл

@ -1,4 +1,4 @@
using System;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
@ -15,8 +15,10 @@ using Avalonia.Platform;
using Avalonia.Rendering;
using Avalonia.Rendering.Composition;
using Avalonia.Threading;
using Avalonia.Vulkan;
using Avalonia.X11;
using Avalonia.X11.Glx;
using Avalonia.X11.Vulkan;
using Avalonia.X11.Screens;
using static Avalonia.X11.XLib;
@ -214,6 +216,14 @@ namespace Avalonia.X11
return egl;
}
}
if (renderingMode == X11RenderingMode.Vulkan)
{
var vulkan = VulkanSupport.TryInitialize(info,
AvaloniaLocator.Current.GetService<VulkanOptions>() ?? new());
if (vulkan != null)
return vulkan;
}
}
throw new InvalidOperationException($"{nameof(X11PlatformOptions)}.{nameof(X11PlatformOptions.RenderingMode)} has a value of \"{string.Join(", ", opts.RenderingMode)}\", but no options were applied.");
@ -241,7 +251,12 @@ namespace Avalonia
/// <summary>
/// Enables native Linux EGL rendering.
/// </summary>
Egl = 3
Egl = 3,
/// <summary>
/// Enables Vulkan rendering
/// </summary>
Vulkan = 4
}
/// <summary>

Просмотреть файл

@ -0,0 +1,124 @@
using System;
using System.Collections.Generic;
using Avalonia.Platform;
using Avalonia.Rendering.Composition;
using Avalonia.Vulkan;
using SkiaSharp;
namespace Avalonia.Skia.Vulkan;
internal class VulkanSkiaExternalObjectsFeature : IExternalObjectsRenderInterfaceContextFeature
{
private readonly VulkanSkiaGpu _gpu;
private readonly IVulkanPlatformGraphicsContext _context;
private readonly IVulkanContextExternalObjectsFeature _feature;
public VulkanSkiaExternalObjectsFeature(VulkanSkiaGpu gpu,
IVulkanPlatformGraphicsContext context, IVulkanContextExternalObjectsFeature feature)
{
_gpu = gpu;
_context = context;
_feature = feature;
}
public IPlatformRenderInterfaceImportedImage ImportImage(IPlatformHandle handle,
PlatformGraphicsExternalImageProperties properties) =>
new Image(_gpu, _feature.ImportImage(handle, properties), properties);
public IPlatformRenderInterfaceImportedSemaphore ImportSemaphore(IPlatformHandle handle) => new Semaphore(_feature.ImportSemaphore(handle));
class Semaphore : IPlatformRenderInterfaceImportedSemaphore
{
private IVulkanExternalSemaphore? _inner;
public Semaphore(IVulkanExternalSemaphore inner)
{
_inner = inner;
}
public void Dispose()
{
_inner?.Dispose();
_inner = null;
}
public IVulkanExternalSemaphore Inner =>
_inner ?? throw new ObjectDisposedException(nameof(IVulkanExternalSemaphore));
}
class Image : IPlatformRenderInterfaceImportedImage
{
private readonly VulkanSkiaGpu _gpu;
private IVulkanExternalImage? _inner;
private readonly PlatformGraphicsExternalImageProperties _properties;
public Image(VulkanSkiaGpu gpu, IVulkanExternalImage inner, PlatformGraphicsExternalImageProperties properties)
{
_gpu = gpu;
_inner = inner;
_properties = properties;
}
public void Dispose()
{
_inner?.Dispose();
_inner = null;
}
public IBitmapImpl SnapshotWithKeyedMutex(uint acquireIndex, uint releaseIndex) => throw new NotSupportedException();
public IBitmapImpl SnapshotWithSemaphores(IPlatformRenderInterfaceImportedSemaphore waitForSemaphore,
IPlatformRenderInterfaceImportedSemaphore signalSemaphore)
{
var info = _inner!.Info;
_gpu.GrContext.ResetContext();
((Semaphore)waitForSemaphore).Inner.SubmitWaitSemaphore();
var imageInfo = new GRVkImageInfo
{
CurrentQueueFamily = _gpu.Vulkan.Device.GraphicsQueueFamilyIndex,
Format = info.Format,
Image = (ulong)info.Handle,
ImageLayout = info.Layout,
ImageTiling = info.Tiling,
ImageUsageFlags = info.UsageFlags,
LevelCount = info.LevelCount,
SampleCount = info.SampleCount,
Protected = info.IsProtected,
Alloc = new GRVkAlloc
{
Memory = (ulong)info.MemoryHandle,
Size = info.MemorySize
}
};
using var renderTarget = new GRBackendRenderTarget(_properties.Width, _properties.Height, 1, imageInfo);
using var surface = SKSurface.Create(_gpu.GrContext, renderTarget,
_properties.TopLeftOrigin ? GRSurfaceOrigin.TopLeft : GRSurfaceOrigin.BottomLeft,
_properties.Format == PlatformGraphicsExternalImageFormat.R8G8B8A8UNorm
? SKColorType.Rgba8888
: SKColorType.Bgra8888, SKColorSpace.CreateSrgb());
var image = surface.Snapshot();
_gpu.GrContext.Flush();
//surface.Canvas.Flush();
((Semaphore)signalSemaphore).Inner.SubmitSignalSemaphore();
return new ImmutableBitmap(image);
}
public IBitmapImpl SnapshotWithAutomaticSync() => throw new NotSupportedException();
}
public IPlatformRenderInterfaceImportedImage ImportImage(ICompositionImportableSharedGpuContextImage image) => throw new System.NotSupportedException();
public CompositionGpuImportedImageSynchronizationCapabilities
GetSynchronizationCapabilities(string imageHandleType) => _feature.GetSynchronizationCapabilities(imageHandleType);
public byte[]? DeviceUuid => _feature.DeviceUuid;
public byte[]? DeviceLuid => _feature.DeviceLuid;
public IReadOnlyList<string> SupportedImageHandleTypes => _feature.SupportedImageHandleTypes;
public IReadOnlyList<string> SupportedSemaphoreTypes => _feature.SupportedSemaphoreTypes;
}

Просмотреть файл

@ -0,0 +1,86 @@
using System;
using System.Collections.Generic;
using Avalonia.Vulkan;
using Avalonia.Platform;
using Avalonia.Rendering;
using SkiaSharp;
namespace Avalonia.Skia.Vulkan;
internal class VulkanSkiaGpu : ISkiaGpu
{
private readonly VulkanSkiaExternalObjectsFeature? _externalObjects;
public IVulkanPlatformGraphicsContext Vulkan { get; private set; }
public GRContext GrContext { get; private set; }
public VulkanSkiaGpu(IVulkanPlatformGraphicsContext vulkan, long? maxResourceBytes)
{
Vulkan = vulkan;
var device = vulkan.Device;
using (Vulkan.EnsureCurrent())
{
IntPtr GetProcAddressWrapper(string name, IntPtr instance, IntPtr device)
{
if (device != IntPtr.Zero)
{
var addr = Vulkan.Instance.GetDeviceProcAddress(device, name);
if (addr != IntPtr.Zero)
return addr;
}
if (instance != IntPtr.Zero)
{
var addr = Vulkan.Instance.GetInstanceProcAddress(instance, name);
if (addr != IntPtr.Zero)
return addr;
}
return Vulkan.Instance.GetInstanceProcAddress(IntPtr.Zero, name);
}
var ctx = new GRVkBackendContext
{
VkInstance = device.Instance.Handle,
VkPhysicalDevice = device.PhysicalDeviceHandle,
VkDevice = device.Handle,
VkQueue = device.MainQueueHandle,
GraphicsQueueIndex = device.GraphicsQueueFamilyIndex,
GetProcedureAddress = GetProcAddressWrapper
};
GrContext = GRContext.CreateVulkan(ctx) ??
throw new VulkanException("Unable to create GrContext from IVulkanDevice");
if (maxResourceBytes.HasValue)
GrContext.SetResourceCacheLimit(maxResourceBytes.Value);
}
if (vulkan.TryGetFeature<IVulkanContextExternalObjectsFeature>(out var externalObjects))
_externalObjects = new VulkanSkiaExternalObjectsFeature(this, vulkan, externalObjects);
}
public void Dispose()
{
Vulkan.Dispose();
}
public object? TryGetFeature(Type featureType)
{
if (featureType == typeof(IExternalObjectsRenderInterfaceContextFeature))
return _externalObjects;
return null;
}
public bool IsLost => Vulkan.IsLost;
public IDisposable EnsureCurrent() => Vulkan.EnsureCurrent();
public ISkiaGpuRenderTarget TryCreateRenderTarget(IEnumerable<object> surfaces)
{
var target = Vulkan.CreateRenderTarget(surfaces);
return new VulkanSkiaRenderTarget(this, target);
}
public ISkiaSurface? TryCreateSurface(PixelSize size, ISkiaGpuRenderSession? session) => null;
}

Просмотреть файл

@ -0,0 +1,105 @@
using System;
using Avalonia.Skia.Helpers;
using Avalonia.Vulkan;
using SkiaSharp;
namespace Avalonia.Skia.Vulkan;
class VulkanSkiaRenderTarget : ISkiaGpuRenderTarget
{
private readonly VulkanSkiaGpu _gpu;
private readonly IVulkanRenderTarget _target;
public VulkanSkiaRenderTarget(VulkanSkiaGpu gpu, IVulkanRenderTarget target)
{
_gpu = gpu;
_target = target;
}
public void Dispose()
{
_target.Dispose();
}
public ISkiaGpuRenderSession BeginRenderingSession()
{
var session = _target.BeginDraw();
bool success = false;
try
{
var size = session.Size;
var scaling = session.Scaling;
if (size.Width <= 0 || size.Height <= 0 || scaling < 0)
{
session.Dispose();
throw new InvalidOperationException(
$"Can't create drawing context for surface with {size} size and {scaling} scaling");
}
_gpu.GrContext.ResetContext();
var sessionImageInfo = session.ImageInfo;
var imageInfo = new GRVkImageInfo
{
CurrentQueueFamily = _gpu.Vulkan.Device.GraphicsQueueFamilyIndex,
Format = sessionImageInfo.Format,
Image = (ulong)sessionImageInfo.Handle,
ImageLayout = sessionImageInfo.Layout,
ImageTiling = sessionImageInfo.Tiling,
ImageUsageFlags = sessionImageInfo.UsageFlags,
LevelCount = sessionImageInfo.LevelCount,
SampleCount = sessionImageInfo.SampleCount,
Protected = sessionImageInfo.IsProtected,
Alloc = new GRVkAlloc
{
Memory = (ulong)sessionImageInfo.MemoryHandle,
Size = sessionImageInfo.MemorySize
}
};
using var renderTarget = new GRBackendRenderTarget(size.Width, size.Height, 1, imageInfo);
var surface = SKSurface.Create(_gpu.GrContext, renderTarget,
session.IsYFlipped ? GRSurfaceOrigin.TopLeft : GRSurfaceOrigin.BottomLeft,
session.IsRgba ? SKColorType.Rgba8888 : SKColorType.Bgra8888, SKColorSpace.CreateSrgb());
if (surface == null)
throw new InvalidOperationException(
$"Surface can't be created with the provided render target");
success = true;
return new VulkanSkiaRenderSession(_gpu.GrContext, surface, session);
}
finally
{
if(!success)
session.Dispose();
}
}
public bool IsCorrupted => false;
internal class VulkanSkiaRenderSession : ISkiaGpuRenderSession
{
private readonly IVulkanRenderSession _vulkanSession;
public VulkanSkiaRenderSession(GRContext grContext,
SKSurface surface,
IVulkanRenderSession vulkanSession)
{
GrContext = grContext;
SkSurface = surface;
_vulkanSession = vulkanSession;
SurfaceOrigin = vulkanSession.IsYFlipped ? GRSurfaceOrigin.TopLeft : GRSurfaceOrigin.BottomLeft;
}
public void Dispose()
{
SkSurface.Canvas.Flush();
SkSurface.Dispose();
GrContext.Flush();
_vulkanSession.Dispose();
}
public GRContext GrContext { get; }
public SKSurface SkSurface { get; }
public double ScaleFactor => _vulkanSession.Scaling;
public GRSurfaceOrigin SurfaceOrigin { get; }
}
}

Просмотреть файл

@ -5,6 +5,8 @@ using Avalonia.Media;
using Avalonia.OpenGL;
using Avalonia.Platform;
using Avalonia.Media.Imaging;
using Avalonia.Skia.Vulkan;
using Avalonia.Vulkan;
using SkiaSharp;
using Avalonia.Media.TextFormatting;
using Avalonia.Metal;
@ -36,6 +38,8 @@ namespace Avalonia.Skia
return new SkiaContext(new GlSkiaGpu(gl, _maxResourceBytes));
if (graphicsContext is IMetalDevice metal)
return new SkiaContext(new SkiaMetalGpu(metal, _maxResourceBytes));
if (graphicsContext is IVulkanPlatformGraphicsContext vulkanContext)
return new SkiaContext(new VulkanSkiaGpu(vulkanContext, _maxResourceBytes));
throw new ArgumentException("Graphics context of type is not supported");
}

Просмотреть файл

@ -0,0 +1,27 @@
using System;
using Avalonia.SourceGenerator;
using Avalonia.Vulkan;
using Avalonia.Vulkan.UnmanagedInterop;
namespace Avalonia.Win32.Vulkan;
partial class Win32VulkanInterface
{
public Win32VulkanInterface(IVulkanInstance instance)
{
Initialize(name => instance.GetInstanceProcAddress(instance.Handle, name));
}
[GetProcAddress("vkCreateWin32SurfaceKHR")]
public partial int vkCreateWin32SurfaceKHR(IntPtr instance, ref VkWin32SurfaceCreateInfoKHR pCreateInfo,
IntPtr pAllocator, out ulong pSurface);
}
struct VkWin32SurfaceCreateInfoKHR
{
public const uint VK_STRUCTURE_TYPE_WIN32_SURFACE_CREATE_INFO_KHR = 1000009000;
public uint sType;
public IntPtr pNext;
public uint flags;
public IntPtr hinstance;
public IntPtr hwnd;
}

Просмотреть файл

@ -0,0 +1,71 @@
using System;
using System.Collections.Generic;
using System.Runtime.InteropServices;
using Avalonia.Platform;
using Avalonia.Vulkan;
using Avalonia.Win32.Interop;
namespace Avalonia.Win32.Vulkan;
internal class VulkanSupport
{
[DllImport("vulkan-1.dll")]
private static extern IntPtr vkGetInstanceProcAddr(IntPtr instance, string name);
public static VulkanPlatformGraphics? TryInitialize(VulkanOptions options) =>
VulkanPlatformGraphics.TryCreate(options ?? new(), new VulkanPlatformSpecificOptions
{
RequiredInstanceExtensions = { "VK_KHR_win32_surface" },
GetProcAddressDelegate = vkGetInstanceProcAddr,
DeviceCheckSurfaceFactory = instance => CreateHwndSurface(OffscreenParentWindow.Handle, instance),
PlatformFeatures = new Dictionary<Type, object>
{
[typeof(IVulkanKhrSurfacePlatformSurfaceFactory)] = new VulkanSurfaceFactory()
}
});
internal class VulkanSurfaceFactory : IVulkanKhrSurfacePlatformSurfaceFactory
{
public bool CanRenderToSurface(IVulkanPlatformGraphicsContext context, object surface) =>
surface is INativePlatformHandleSurface handle && handle.HandleDescriptor == "HWND";
public IVulkanKhrSurfacePlatformSurface CreateSurface(IVulkanPlatformGraphicsContext context, object handle) =>
new HwndVulkanSurface((INativePlatformHandleSurface)handle);
}
class HwndVulkanSurface : IVulkanKhrSurfacePlatformSurface
{
private readonly INativePlatformHandleSurface _handle;
public HwndVulkanSurface(INativePlatformHandleSurface handle)
{
_handle = handle;
}
public void Dispose()
{
// No-op
}
public double Scaling => _handle.Scaling;
public PixelSize Size => _handle.Size;
public ulong CreateSurface(IVulkanPlatformGraphicsContext context) =>
CreateHwndSurface(_handle.Handle, context.Instance);
}
private static ulong CreateHwndSurface(IntPtr window, IVulkanInstance instance)
{
var vulkanWin32 = new Win32VulkanInterface(instance);
var createInfo = new VkWin32SurfaceCreateInfoKHR()
{
sType = VkWin32SurfaceCreateInfoKHR.VK_STRUCTURE_TYPE_WIN32_SURFACE_CREATE_INFO_KHR,
hinstance = UnmanagedMethods.GetModuleHandle(null),
hwnd = window
};
VulkanException.ThrowOnError("vkCreateWin32SurfaceKHR",
vulkanWin32.vkCreateWin32SurfaceKHR(instance.Handle, ref createInfo, IntPtr.Zero, out var surface));
return surface;
}
}

Просмотреть файл

@ -3,10 +3,12 @@ using System.Diagnostics.Tracing;
using System.Linq;
using Avalonia.OpenGL;
using Avalonia.Platform;
using Avalonia.Vulkan;
using Avalonia.Win32.DComposition;
using Avalonia.Win32.DirectX;
using Avalonia.Win32.OpenGl;
using Avalonia.Win32.OpenGl.Angle;
using Avalonia.Win32.Vulkan;
using Avalonia.Win32.WinRT.Composition;
namespace Avalonia.Win32;
@ -62,6 +64,13 @@ static class Win32GlManager
return wgl;
}
}
if (renderingMode == Win32RenderingMode.Vulkan)
{
var vulkan = VulkanSupport.TryInitialize(AvaloniaLocator.Current.GetService<VulkanOptions>() ?? new());
if (vulkan != null)
return vulkan;
}
}
throw new InvalidOperationException($"{nameof(Win32PlatformOptions)}.{nameof(Win32PlatformOptions.RenderingMode)} has a value of \"{string.Join(", ", opts.RenderingMode)}\", but no options were applied.");

Просмотреть файл

@ -22,7 +22,11 @@ public enum Win32RenderingMode
/// <summary>
/// Avalonia would try to use native Widows OpenGL with GPU rendering.
/// </summary>
Wgl = 3
Wgl = 3,
/// <summary>
/// Avalonia would try to use native Widows Vulkan with GPU rendering.
/// </summary>
Vulkan = 4
}
/// <summary>

Просмотреть файл

@ -1,4 +1,4 @@
using System;
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Diagnostics.CodeAnalysis;