forked from Mirror/Ryujinx
644de99e86
* Implement GPU syncpoints This adds support for GPU syncpoints on the GPU backend & nvservices. Everything that was implemented here is based on my researches, hardware testing of the GM20B and reversing of nvservices (8.1.0). Thanks to @fincs for the informations about some behaviours of the pusher and for the initial informations about syncpoints. * syncpoint: address gdkchan's comments * Add some missing logic to handle SubmitGpfifo correctly * Handle the NV event API correctly * evnt => hostEvent * Finish addressing gdkchan's comments * nvservices: write the output buffer even when an error is returned * dma pusher: Implemnet prefetch barrier lso fix when the commands should be prefetch. * Partially fix prefetch barrier * Add a missing syncpoint check in QueryEvent of NvHostSyncPt * Address Ac_K's comments and fix GetSyncpoint for ChannelResourcePolicy == Channel * fix SyncptWait & SyncptWaitEx cmds logic * Address ripinperi's comments * Address gdkchan's comments * Move user event management to the control channel * Fix mm implementation, nvdec works again * Address ripinperi's comments * Address gdkchan's comments * Implement nvhost-ctrl close accurately + make nvservices dispose channels when stopping the emulator * Fix typo in MultiMediaOperationType
235 lines
No EOL
7.1 KiB
C#
235 lines
No EOL
7.1 KiB
C#
using Ryujinx.Graphics.Gpu.State;
|
|
using System.IO;
|
|
|
|
namespace Ryujinx.Graphics.Gpu
|
|
{
|
|
/// <summary>
|
|
/// GPU commands FIFO.
|
|
/// </summary>
|
|
class NvGpuFifo
|
|
{
|
|
private const int MacrosCount = 0x80;
|
|
private const int MacroIndexMask = MacrosCount - 1;
|
|
|
|
// Note: The size of the macro memory is unknown, we just make
|
|
// a guess here and use 256kb as the size. Increase if needed.
|
|
private const int MmeWords = 256 * 256;
|
|
|
|
private GpuContext _context;
|
|
|
|
/// <summary>
|
|
/// Cached GPU macro program.
|
|
/// </summary>
|
|
private struct CachedMacro
|
|
{
|
|
/// <summary>
|
|
/// Word offset of the code on the code memory.
|
|
/// </summary>
|
|
public int Position { get; }
|
|
|
|
private bool _executionPending;
|
|
private int _argument;
|
|
|
|
private MacroInterpreter _interpreter;
|
|
|
|
/// <summary>
|
|
/// Creates a new instance of the GPU cached macro program.
|
|
/// </summary>
|
|
/// <param name="position">Macro code start position</param>
|
|
public CachedMacro(int position)
|
|
{
|
|
Position = position;
|
|
|
|
_executionPending = false;
|
|
_argument = 0;
|
|
|
|
_interpreter = new MacroInterpreter();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the first argument for the macro call.
|
|
/// </summary>
|
|
/// <param name="argument">First argument</param>
|
|
public void StartExecution(int argument)
|
|
{
|
|
_argument = argument;
|
|
|
|
_executionPending = true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Starts executing the macro program code.
|
|
/// </summary>
|
|
/// <param name="mme">Program code</param>
|
|
/// <param name="state">Current GPU state</param>
|
|
public void Execute(int[] mme, ShadowRamControl shadowCtrl, GpuState state, GpuState shadowState)
|
|
{
|
|
if (_executionPending)
|
|
{
|
|
_executionPending = false;
|
|
|
|
_interpreter?.Execute(mme, Position, _argument, shadowCtrl, state, shadowState);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Pushes an argument to the macro call argument FIFO.
|
|
/// </summary>
|
|
/// <param name="argument">Argument to be pushed</param>
|
|
public void PushArgument(int argument)
|
|
{
|
|
_interpreter?.Fifo.Enqueue(argument);
|
|
}
|
|
}
|
|
|
|
private int _currMacroPosition;
|
|
private int _currMacroBindIndex;
|
|
|
|
private ShadowRamControl _shadowCtrl;
|
|
|
|
private CachedMacro[] _macros;
|
|
|
|
private int[] _mme;
|
|
|
|
/// <summary>
|
|
/// GPU sub-channel information.
|
|
/// </summary>
|
|
private class SubChannel
|
|
{
|
|
/// <summary>
|
|
/// Sub-channel GPU state.
|
|
/// </summary>
|
|
public GpuState State { get; }
|
|
|
|
/// <summary>
|
|
/// Sub-channel shadow GPU state (used as backup storage to restore MME changes).
|
|
/// </summary>
|
|
public GpuState ShadowState { get; }
|
|
|
|
/// <summary>
|
|
/// Engine bound to the sub-channel.
|
|
/// </summary>
|
|
public ClassId Class { get; set; }
|
|
|
|
/// <summary>
|
|
/// Creates a new instance of the GPU sub-channel.
|
|
/// </summary>
|
|
public SubChannel()
|
|
{
|
|
State = new GpuState();
|
|
ShadowState = new GpuState();
|
|
}
|
|
}
|
|
|
|
private SubChannel[] _subChannels;
|
|
|
|
private SubChannel _fifoChannel;
|
|
|
|
/// <summary>
|
|
/// Creates a new instance of the GPU commands FIFO.
|
|
/// </summary>
|
|
/// <param name="context">GPU emulation context</param>
|
|
public NvGpuFifo(GpuContext context)
|
|
{
|
|
_context = context;
|
|
|
|
_macros = new CachedMacro[MacrosCount];
|
|
|
|
_mme = new int[MmeWords];
|
|
|
|
_fifoChannel = new SubChannel();
|
|
|
|
_context.Methods.RegisterCallbacksForFifo(_fifoChannel.State);
|
|
|
|
_subChannels = new SubChannel[8];
|
|
|
|
for (int index = 0; index < _subChannels.Length; index++)
|
|
{
|
|
_subChannels[index] = new SubChannel();
|
|
|
|
_context.Methods.RegisterCallbacks(_subChannels[index].State);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Send macro code/data to the MME
|
|
/// </summary>
|
|
/// <param name="index">The index in the MME</param>
|
|
/// <param name="data">The data to use</param>
|
|
public void SendMacroCodeData(int index, int data)
|
|
{
|
|
_mme[index] = data;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Bind a macro index to a position for the MME
|
|
/// </summary>
|
|
/// <param name="index">The macro index</param>
|
|
/// <param name="position">The position of the macro</param>
|
|
public void BindMacro(int index, int position)
|
|
{
|
|
_macros[index] = new CachedMacro(position);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Change the shadow RAM setting
|
|
/// </summary>
|
|
/// <param name="shadowCtrl">The new Shadow RAM setting</param>
|
|
public void SetMmeShadowRamControl(ShadowRamControl shadowCtrl)
|
|
{
|
|
_shadowCtrl = shadowCtrl;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Calls a GPU method.
|
|
/// </summary>
|
|
/// <param name="meth">GPU method call parameters</param>
|
|
public void CallMethod(MethodParams meth)
|
|
{
|
|
if ((MethodOffset)meth.Method == MethodOffset.BindChannel)
|
|
{
|
|
_subChannels[meth.SubChannel] = new SubChannel
|
|
{
|
|
Class = (ClassId)meth.Argument
|
|
};
|
|
|
|
_context.Methods.RegisterCallbacks(_subChannels[meth.SubChannel].State);
|
|
}
|
|
else if (meth.Method < 0x60)
|
|
{
|
|
// TODO: check if macros are shared between subchannels or not. For now let's assume they are.
|
|
_fifoChannel.State.CallMethod(meth);
|
|
}
|
|
else if (meth.Method < 0xe00)
|
|
{
|
|
SubChannel sc = _subChannels[meth.SubChannel];
|
|
|
|
sc.ShadowState.Write(meth.Method, meth.Argument);
|
|
|
|
sc.State.CallMethod(meth);
|
|
}
|
|
else
|
|
{
|
|
int macroIndex = (meth.Method >> 1) & MacroIndexMask;
|
|
|
|
if ((meth.Method & 1) != 0)
|
|
{
|
|
_macros[macroIndex].PushArgument(meth.Argument);
|
|
}
|
|
else
|
|
{
|
|
_macros[macroIndex].StartExecution(meth.Argument);
|
|
}
|
|
|
|
if (meth.IsLastCall)
|
|
{
|
|
SubChannel sc = _subChannels[meth.SubChannel];
|
|
|
|
_macros[macroIndex].Execute(_mme, _shadowCtrl, sc.State, sc.ShadowState);
|
|
|
|
_context.Methods.PerformDeferredDraws();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} |