mirror of
https://github.com/Ryujinx/Ryujinx.git
synced 2025-01-19 15:11:31 +00:00
10aa11ce13
* Interrupt GPU command processing when a frame's fence is reached. * Accumulate times rather than %s * Accurate timer for vsync Spin wait for the last .667ms of a frame. Avoids issues caused by signalling 16ms vsync. (periodic stutters in smo) * Use event wait for better timing. * Fix lazy wait Windows doesn't seem to want to do 1ms consistently, so force a spin if we're less than 2ms. * A bit more efficiency on frame waits. Should now wait the remainder 0.6667 instead of 1.6667 sometimes (odd waits above 1ms are reliable, unlike 1ms waits) * Better swap interval 0 solution 737 fps without breaking a sweat. Downside: Vsync can no longer be disabled on games that use the event heavily (link's awakening - which is ok since it breaks anyways) * Fix comment. * Address Comments.
200 lines
6.2 KiB
C#
200 lines
6.2 KiB
C#
using System;
|
|
using System.Collections.Concurrent;
|
|
using System.Runtime.CompilerServices;
|
|
using System.Runtime.InteropServices;
|
|
using System.Threading;
|
|
|
|
namespace Ryujinx.Graphics.Gpu.Engine.GPFifo
|
|
{
|
|
/// <summary>
|
|
/// Represents a GPU General Purpose FIFO device.
|
|
/// </summary>
|
|
public sealed class GPFifoDevice : IDisposable
|
|
{
|
|
/// <summary>
|
|
/// Indicates if the command buffer has pre-fetch enabled.
|
|
/// </summary>
|
|
private enum CommandBufferType
|
|
{
|
|
Prefetch,
|
|
NoPrefetch
|
|
}
|
|
|
|
/// <summary>
|
|
/// Command buffer data.
|
|
/// </summary>
|
|
private struct CommandBuffer
|
|
{
|
|
/// <summary>
|
|
/// The type of the command buffer.
|
|
/// </summary>
|
|
public CommandBufferType Type;
|
|
|
|
/// <summary>
|
|
/// Fetched data.
|
|
/// </summary>
|
|
public int[] Words;
|
|
|
|
/// <summary>
|
|
/// The GPFIFO entry address (used in <see cref="CommandBufferType.NoPrefetch"/> mode).
|
|
/// </summary>
|
|
public ulong EntryAddress;
|
|
|
|
/// <summary>
|
|
/// The count of entries inside this GPFIFO entry.
|
|
/// </summary>
|
|
public uint EntryCount;
|
|
|
|
/// <summary>
|
|
/// Fetch the command buffer.
|
|
/// </summary>
|
|
public void Fetch(GpuContext context)
|
|
{
|
|
if (Words == null)
|
|
{
|
|
Words = MemoryMarshal.Cast<byte, int>(context.MemoryManager.GetSpan(EntryAddress, (int)EntryCount * 4)).ToArray();
|
|
}
|
|
}
|
|
}
|
|
|
|
private readonly ConcurrentQueue<CommandBuffer> _commandBufferQueue;
|
|
|
|
private CommandBuffer _currentCommandBuffer;
|
|
|
|
private readonly bool _ibEnable;
|
|
private readonly GpuContext _context;
|
|
private readonly AutoResetEvent _event;
|
|
private readonly GPFifoProcessor _processor;
|
|
|
|
private bool _interrupt;
|
|
|
|
/// <summary>
|
|
/// Creates a new instance of the GPU General Purpose FIFO device.
|
|
/// </summary>
|
|
/// <param name="context">GPU context that the GPFIFO belongs to</param>
|
|
internal GPFifoDevice(GpuContext context)
|
|
{
|
|
_commandBufferQueue = new ConcurrentQueue<CommandBuffer>();
|
|
_ibEnable = true;
|
|
_context = context;
|
|
_event = new AutoResetEvent(false);
|
|
|
|
_processor = new GPFifoProcessor(context);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Signal the FIFO that there are new entries to process.
|
|
/// </summary>
|
|
public void SignalNewEntries()
|
|
{
|
|
_event.Set();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Push a GPFIFO entry in the form of a prefetched command buffer.
|
|
/// It is intended to be used by nvservices to handle special cases.
|
|
/// </summary>
|
|
/// <param name="commandBuffer">The command buffer containing the prefetched commands</param>
|
|
public void PushHostCommandBuffer(int[] commandBuffer)
|
|
{
|
|
_commandBufferQueue.Enqueue(new CommandBuffer
|
|
{
|
|
Type = CommandBufferType.Prefetch,
|
|
Words = commandBuffer,
|
|
EntryAddress = ulong.MaxValue,
|
|
EntryCount = (uint)commandBuffer.Length
|
|
});
|
|
}
|
|
|
|
/// <summary>
|
|
/// Create a CommandBuffer from a GPFIFO entry.
|
|
/// </summary>
|
|
/// <param name="entry">The GPFIFO entry</param>
|
|
/// <returns>A new CommandBuffer based on the GPFIFO entry</returns>
|
|
private CommandBuffer CreateCommandBuffer(GPEntry entry)
|
|
{
|
|
CommandBufferType type = CommandBufferType.Prefetch;
|
|
|
|
if (entry.Entry1Sync == Entry1Sync.Wait)
|
|
{
|
|
type = CommandBufferType.NoPrefetch;
|
|
}
|
|
|
|
ulong startAddress = ((ulong)entry.Entry0Get << 2) | ((ulong)entry.Entry1GetHi << 32);
|
|
|
|
return new CommandBuffer
|
|
{
|
|
Type = type,
|
|
Words = null,
|
|
EntryAddress = startAddress,
|
|
EntryCount = (uint)entry.Entry1Length
|
|
};
|
|
}
|
|
|
|
/// <summary>
|
|
/// Pushes GPFIFO entries.
|
|
/// </summary>
|
|
/// <param name="entries">GPFIFO entries</param>
|
|
public void PushEntries(ReadOnlySpan<ulong> entries)
|
|
{
|
|
bool beforeBarrier = true;
|
|
|
|
for (int index = 0; index < entries.Length; index++)
|
|
{
|
|
ulong entry = entries[index];
|
|
|
|
CommandBuffer commandBuffer = CreateCommandBuffer(Unsafe.As<ulong, GPEntry>(ref entry));
|
|
|
|
if (beforeBarrier && commandBuffer.Type == CommandBufferType.Prefetch)
|
|
{
|
|
commandBuffer.Fetch(_context);
|
|
}
|
|
|
|
if (commandBuffer.Type == CommandBufferType.NoPrefetch)
|
|
{
|
|
beforeBarrier = false;
|
|
}
|
|
|
|
_commandBufferQueue.Enqueue(commandBuffer);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Waits until commands are pushed to the FIFO.
|
|
/// </summary>
|
|
/// <returns>True if commands were received, false if wait timed out</returns>
|
|
public bool WaitForCommands()
|
|
{
|
|
return !_commandBufferQueue.IsEmpty || (_event.WaitOne(8) && !_commandBufferQueue.IsEmpty);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Processes commands pushed to the FIFO.
|
|
/// </summary>
|
|
public void DispatchCalls()
|
|
{
|
|
while (_ibEnable && !_interrupt && _commandBufferQueue.TryDequeue(out CommandBuffer entry))
|
|
{
|
|
_currentCommandBuffer = entry;
|
|
_currentCommandBuffer.Fetch(_context);
|
|
|
|
_processor.Process(_currentCommandBuffer.Words);
|
|
}
|
|
|
|
_interrupt = false;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Interrupts command processing. This will break out of the DispatchCalls loop.
|
|
/// </summary>
|
|
public void Interrupt()
|
|
{
|
|
_interrupt = true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Disposes of resources used for GPFifo command processing.
|
|
/// </summary>
|
|
public void Dispose() => _event.Dispose();
|
|
}
|
|
}
|