2019-11-23 21:24:03 -05:00
|
|
|
using Ryujinx.Graphics.GAL;
|
|
|
|
using Ryujinx.Graphics.Gpu.Image;
|
|
|
|
using System;
|
|
|
|
using System.Collections.Concurrent;
|
2020-12-17 13:39:52 -05:00
|
|
|
using System.Threading;
|
2019-11-23 21:24:03 -05:00
|
|
|
|
|
|
|
namespace Ryujinx.Graphics.Gpu
|
|
|
|
{
|
2019-12-29 12:41:50 -05:00
|
|
|
using Texture = Image.Texture;
|
|
|
|
|
2019-12-31 15:08:20 -05:00
|
|
|
/// <summary>
|
|
|
|
/// GPU image presentation window.
|
|
|
|
/// </summary>
|
2019-11-23 21:24:03 -05:00
|
|
|
public class Window
|
|
|
|
{
|
2019-12-29 12:41:50 -05:00
|
|
|
private readonly GpuContext _context;
|
2019-11-23 21:24:03 -05:00
|
|
|
|
2019-12-31 15:08:20 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Texture presented on the window.
|
|
|
|
/// </summary>
|
2019-11-23 21:24:03 -05:00
|
|
|
private struct PresentationTexture
|
|
|
|
{
|
2019-12-31 15:08:20 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Texture information.
|
|
|
|
/// </summary>
|
|
|
|
public TextureInfo Info { get; }
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Texture crop region.
|
|
|
|
/// </summary>
|
|
|
|
public ImageCrop Crop { get; }
|
|
|
|
|
2020-04-18 21:25:57 -04:00
|
|
|
/// <summary>
|
|
|
|
/// Texture acquire callback.
|
|
|
|
/// </summary>
|
|
|
|
public Action<GpuContext, object> AcquireCallback { get; }
|
|
|
|
|
2019-12-31 15:08:20 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Texture release callback.
|
|
|
|
/// </summary>
|
2020-04-18 21:25:57 -04:00
|
|
|
public Action<object> ReleaseCallback { get; }
|
2019-11-23 21:24:03 -05:00
|
|
|
|
2019-12-31 15:08:20 -05:00
|
|
|
/// <summary>
|
2020-04-18 21:25:57 -04:00
|
|
|
/// User defined object, passed to the various callbacks.
|
2019-12-31 15:08:20 -05:00
|
|
|
/// </summary>
|
|
|
|
public object UserObj { get; }
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Creates a new instance of the presentation texture.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="info">Information of the texture to be presented</param>
|
|
|
|
/// <param name="crop">Texture crop region</param>
|
2020-04-18 21:25:57 -04:00
|
|
|
/// <param name="acquireCallback">Texture acquire callback</param>
|
|
|
|
/// <param name="releaseCallback">Texture release callback</param>
|
2019-12-31 15:08:20 -05:00
|
|
|
/// <param name="userObj">User defined object passed to the release callback, can be used to identify the texture</param>
|
2019-11-23 21:24:03 -05:00
|
|
|
public PresentationTexture(
|
2020-04-18 21:25:57 -04:00
|
|
|
TextureInfo info,
|
|
|
|
ImageCrop crop,
|
|
|
|
Action<GpuContext, object> acquireCallback,
|
|
|
|
Action<object> releaseCallback,
|
|
|
|
object userObj)
|
2019-11-23 21:24:03 -05:00
|
|
|
{
|
2020-04-18 21:25:57 -04:00
|
|
|
Info = info;
|
|
|
|
Crop = crop;
|
|
|
|
AcquireCallback = acquireCallback;
|
|
|
|
ReleaseCallback = releaseCallback;
|
|
|
|
UserObj = userObj;
|
2019-11-23 21:24:03 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-29 12:41:50 -05:00
|
|
|
private readonly ConcurrentQueue<PresentationTexture> _frameQueue;
|
2019-11-23 21:24:03 -05:00
|
|
|
|
2020-12-17 13:39:52 -05:00
|
|
|
private int _framesAvailable;
|
|
|
|
|
2019-12-31 15:08:20 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Creates a new instance of the GPU presentation window.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="context">GPU emulation context</param>
|
2019-11-23 21:24:03 -05:00
|
|
|
public Window(GpuContext context)
|
|
|
|
{
|
|
|
|
_context = context;
|
|
|
|
|
|
|
|
_frameQueue = new ConcurrentQueue<PresentationTexture>();
|
|
|
|
}
|
|
|
|
|
2019-12-31 15:08:20 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Enqueues a frame for presentation.
|
|
|
|
/// This method is thread safe and can be called from any thread.
|
|
|
|
/// When the texture is presented and not needed anymore, the release callback is called.
|
|
|
|
/// It's an error to modify the texture after calling this method, before the release callback is called.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="address">CPU virtual address of the texture data</param>
|
|
|
|
/// <param name="width">Texture width</param>
|
|
|
|
/// <param name="height">Texture height</param>
|
|
|
|
/// <param name="stride">Texture stride for linear texture, should be zero otherwise</param>
|
|
|
|
/// <param name="isLinear">Indicates if the texture is linear, normally false</param>
|
|
|
|
/// <param name="gobBlocksInY">GOB blocks in the Y direction, for block linear textures</param>
|
|
|
|
/// <param name="format">Texture format</param>
|
|
|
|
/// <param name="bytesPerPixel">Texture format bytes per pixel (must match the format)</param>
|
|
|
|
/// <param name="crop">Texture crop region</param>
|
2020-04-18 21:25:57 -04:00
|
|
|
/// <param name="acquireCallback">Texture acquire callback</param>
|
|
|
|
/// <param name="releaseCallback">Texture release callback</param>
|
2019-12-31 15:08:20 -05:00
|
|
|
/// <param name="userObj">User defined object passed to the release callback</param>
|
2019-11-23 21:24:03 -05:00
|
|
|
public void EnqueueFrameThreadSafe(
|
2020-04-18 21:25:57 -04:00
|
|
|
ulong address,
|
|
|
|
int width,
|
|
|
|
int height,
|
|
|
|
int stride,
|
|
|
|
bool isLinear,
|
|
|
|
int gobBlocksInY,
|
|
|
|
Format format,
|
|
|
|
int bytesPerPixel,
|
|
|
|
ImageCrop crop,
|
|
|
|
Action<GpuContext, object> acquireCallback,
|
|
|
|
Action<object> releaseCallback,
|
|
|
|
object userObj)
|
2019-11-23 21:24:03 -05:00
|
|
|
{
|
2020-09-10 19:48:48 -04:00
|
|
|
FormatInfo formatInfo = new FormatInfo(format, 1, 1, bytesPerPixel, 4);
|
2019-11-23 21:24:03 -05:00
|
|
|
|
|
|
|
TextureInfo info = new TextureInfo(
|
|
|
|
address,
|
|
|
|
width,
|
|
|
|
height,
|
|
|
|
1,
|
|
|
|
1,
|
|
|
|
1,
|
|
|
|
1,
|
|
|
|
stride,
|
|
|
|
isLinear,
|
|
|
|
gobBlocksInY,
|
|
|
|
1,
|
|
|
|
1,
|
|
|
|
Target.Texture2D,
|
|
|
|
formatInfo);
|
|
|
|
|
2020-04-18 21:25:57 -04:00
|
|
|
_frameQueue.Enqueue(new PresentationTexture(info, crop, acquireCallback, releaseCallback, userObj));
|
2019-11-23 21:24:03 -05:00
|
|
|
}
|
|
|
|
|
2019-12-31 15:08:20 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Presents a texture on the queue.
|
|
|
|
/// If the queue is empty, then no texture is presented.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="swapBuffersCallback">Callback method to call when a new texture should be presented on the screen</param>
|
2019-11-23 21:24:03 -05:00
|
|
|
public void Present(Action swapBuffersCallback)
|
|
|
|
{
|
|
|
|
_context.AdvanceSequence();
|
|
|
|
|
|
|
|
if (_frameQueue.TryDequeue(out PresentationTexture pt))
|
|
|
|
{
|
2020-04-18 21:25:57 -04:00
|
|
|
pt.AcquireCallback(_context, pt.UserObj);
|
|
|
|
|
2020-07-06 22:41:07 -04:00
|
|
|
Texture texture = _context.Methods.TextureManager.FindOrCreateTexture(pt.Info, TextureSearchFlags.WithUpscale);
|
2019-11-23 21:24:03 -05:00
|
|
|
|
|
|
|
texture.SynchronizeMemory();
|
|
|
|
|
|
|
|
_context.Renderer.Window.Present(texture.HostTexture, pt.Crop);
|
|
|
|
|
|
|
|
swapBuffersCallback();
|
|
|
|
|
2020-04-18 21:25:57 -04:00
|
|
|
pt.ReleaseCallback(pt.UserObj);
|
2019-11-23 21:24:03 -05:00
|
|
|
}
|
|
|
|
}
|
2020-12-17 13:39:52 -05:00
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Indicate that a frame on the queue is ready to be acquired.
|
|
|
|
/// </summary>
|
|
|
|
public void SignalFrameReady()
|
|
|
|
{
|
|
|
|
Interlocked.Increment(ref _framesAvailable);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Determine if any frames are available, and decrement the available count if there are.
|
|
|
|
/// </summary>
|
|
|
|
/// <returns>True if a frame is available, false otherwise</returns>
|
|
|
|
public bool ConsumeFrameAvailable()
|
|
|
|
{
|
|
|
|
if (Interlocked.CompareExchange(ref _framesAvailable, 0, 0) != 0)
|
|
|
|
{
|
|
|
|
Interlocked.Decrement(ref _framesAvailable);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2019-11-23 21:24:03 -05:00
|
|
|
}
|
|
|
|
}
|