using Avalonia; using Avalonia.Media; using Avalonia.Platform; using Avalonia.Rendering.SceneGraph; using Avalonia.Skia; using Avalonia.Threading; using Ryujinx.Ava.Ui.Backend.Vulkan; using Ryujinx.Ava.Ui.Vulkan; using Ryujinx.Common.Configuration; using Ryujinx.Graphics.Vulkan; using Silk.NET.Vulkan; using SkiaSharp; using SPB.Windowing; using System; using System.Collections.Concurrent; namespace Ryujinx.Ava.Ui.Controls { internal class VulkanRendererControl : RendererControl { private const int MaxImagesInFlight = 3; private VulkanPlatformInterface _platformInterface; private ConcurrentQueue _imagesInFlight; private PresentImageInfo _currentImage; public VulkanRendererControl(GraphicsDebugLevel graphicsDebugLevel) : base(graphicsDebugLevel) { _platformInterface = AvaloniaLocator.Current.GetService(); _imagesInFlight = new ConcurrentQueue(); } public override void DestroyBackgroundContext() { } protected override ICustomDrawOperation CreateDrawOperation() { return new VulkanDrawOperation(this); } protected override void OnDetachedFromVisualTree(VisualTreeAttachmentEventArgs e) { base.OnDetachedFromVisualTree(e); _imagesInFlight.Clear(); if (_platformInterface.MainSurface.Display != null) { _platformInterface.MainSurface.Display.Presented -= Window_Presented; } _currentImage?.Put(); _currentImage = null; } protected override void OnAttachedToVisualTree(VisualTreeAttachmentEventArgs e) { base.OnAttachedToVisualTree(e); _platformInterface.MainSurface.Display.Presented += Window_Presented; } private void Window_Presented(object sender, EventArgs e) { _platformInterface.MainSurface.Device.QueueWaitIdle(); _currentImage?.Put(); _currentImage = null; } public override void Render(DrawingContext context) { base.Render(context); } protected override void CreateWindow() { } internal override void MakeCurrent() { } internal override void MakeCurrent(SwappableNativeWindowBase window) { } internal override void Present(object image) { Image = image; _imagesInFlight.Enqueue((PresentImageInfo)image); if (_imagesInFlight.Count > MaxImagesInFlight) { _imagesInFlight.TryDequeue(out _); } Dispatcher.UIThread.Post(InvalidateVisual); } private PresentImageInfo GetImage() { lock (_imagesInFlight) { if (!_imagesInFlight.TryDequeue(out _currentImage)) { _currentImage = (PresentImageInfo)Image; } return _currentImage; } } private class VulkanDrawOperation : ICustomDrawOperation { public Rect Bounds { get; } private readonly VulkanRendererControl _control; private bool _isDestroyed; public VulkanDrawOperation(VulkanRendererControl control) { _control = control; Bounds = _control.Bounds; } public void Dispose() { if (_isDestroyed) { return; } _isDestroyed = true; } public bool Equals(ICustomDrawOperation other) { return other is VulkanDrawOperation operation && Equals(this, operation) && operation.Bounds == Bounds; } public bool HitTest(Point p) { return Bounds.Contains(p); } public unsafe void Render(IDrawingContextImpl context) { if (_isDestroyed || _control.Image == null || _control.RenderSize.Width == 0 || _control.RenderSize.Height == 0 || context is not ISkiaDrawingContextImpl skiaDrawingContextImpl) { return; } var image = _control.GetImage(); if (!image.State.IsValid) { _control._currentImage = null; return; } var gpu = AvaloniaLocator.Current.GetService(); image.Get(); var imageInfo = new GRVkImageInfo() { CurrentQueueFamily = _control._platformInterface.PhysicalDevice.QueueFamilyIndex, Format = (uint)Format.R8G8B8A8Unorm, Image = image.Image.Handle, ImageLayout = (uint)ImageLayout.TransferSrcOptimal, ImageTiling = (uint)ImageTiling.Optimal, ImageUsageFlags = (uint)(ImageUsageFlags.ImageUsageColorAttachmentBit | ImageUsageFlags.ImageUsageTransferSrcBit | ImageUsageFlags.ImageUsageTransferDstBit), LevelCount = 1, SampleCount = 1, Protected = false, Alloc = new GRVkAlloc() { Memory = image.Memory.Handle, Flags = 0, Offset = image.MemoryOffset, Size = image.MemorySize } }; using var backendTexture = new GRBackendRenderTarget( (int)image.Extent.Width, (int)image.Extent.Height, 1, imageInfo); var vulkan = AvaloniaLocator.Current.GetService(); using var surface = SKSurface.Create( skiaDrawingContextImpl.GrContext, backendTexture, GRSurfaceOrigin.TopLeft, SKColorType.Rgba8888); if (surface == null) { return; } var rect = new Rect(new Point(), new Size(image.Extent.Width, image.Extent.Height)); using var snapshot = surface.Snapshot(); skiaDrawingContextImpl.SkCanvas.DrawImage(snapshot, rect.ToSKRect(), _control.Bounds.ToSKRect(), new SKPaint()); } } } }