NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 20:53:23 +02:00
|
|
|
using System;
|
|
|
|
using System.Collections.Concurrent;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Threading;
|
|
|
|
|
2018-06-11 02:46:42 +02:00
|
|
|
namespace Ryujinx.HLE.OsHle.Services.Nv.NvHostCtrl
|
NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 20:53:23 +02:00
|
|
|
{
|
|
|
|
class NvHostSyncpt
|
|
|
|
{
|
|
|
|
public const int SyncptsCount = 192;
|
|
|
|
|
|
|
|
private int[] CounterMin;
|
|
|
|
private int[] CounterMax;
|
|
|
|
|
|
|
|
private long EventMask;
|
|
|
|
|
|
|
|
private ConcurrentDictionary<EventWaitHandle, int> Waiters;
|
|
|
|
|
|
|
|
public NvHostSyncpt()
|
|
|
|
{
|
|
|
|
CounterMin = new int[SyncptsCount];
|
|
|
|
CounterMax = new int[SyncptsCount];
|
|
|
|
|
|
|
|
Waiters = new ConcurrentDictionary<EventWaitHandle, int>();
|
|
|
|
}
|
|
|
|
|
|
|
|
public int GetMin(int Id)
|
|
|
|
{
|
|
|
|
return CounterMin[Id];
|
|
|
|
}
|
|
|
|
|
|
|
|
public int GetMax(int Id)
|
|
|
|
{
|
|
|
|
return CounterMax[Id];
|
|
|
|
}
|
|
|
|
|
|
|
|
public int Increment(int Id)
|
|
|
|
{
|
|
|
|
if (((EventMask >> Id) & 1) != 0)
|
|
|
|
{
|
|
|
|
Interlocked.Increment(ref CounterMax[Id]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return IncrementMin(Id);
|
|
|
|
}
|
|
|
|
|
|
|
|
public int IncrementMin(int Id)
|
|
|
|
{
|
|
|
|
int Value = Interlocked.Increment(ref CounterMin[Id]);
|
|
|
|
|
|
|
|
WakeUpWaiters(Id, Value);
|
|
|
|
|
|
|
|
return Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
public int IncrementMax(int Id)
|
|
|
|
{
|
|
|
|
return Interlocked.Increment(ref CounterMax[Id]);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void AddWaiter(int Threshold, EventWaitHandle WaitEvent)
|
|
|
|
{
|
|
|
|
if (!Waiters.TryAdd(WaitEvent, Threshold))
|
|
|
|
{
|
|
|
|
throw new InvalidOperationException();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool RemoveWaiter(EventWaitHandle WaitEvent)
|
|
|
|
{
|
|
|
|
return Waiters.TryRemove(WaitEvent, out _);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void WakeUpWaiters(int Id, int NewValue)
|
|
|
|
{
|
|
|
|
foreach (KeyValuePair<EventWaitHandle, int> KV in Waiters)
|
|
|
|
{
|
|
|
|
if (MinCompare(Id, NewValue, CounterMax[Id], KV.Value))
|
|
|
|
{
|
|
|
|
KV.Key.Set();
|
|
|
|
|
|
|
|
Waiters.TryRemove(KV.Key, out _);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool MinCompare(int Id, int Threshold)
|
|
|
|
{
|
|
|
|
return MinCompare(Id, CounterMin[Id], CounterMax[Id], Threshold);
|
|
|
|
}
|
|
|
|
|
|
|
|
private bool MinCompare(int Id, int Min, int Max, int Threshold)
|
|
|
|
{
|
|
|
|
int MinDiff = Min - Threshold;
|
|
|
|
int MaxDiff = Max - Threshold;
|
|
|
|
|
|
|
|
if (((EventMask >> Id) & 1) != 0)
|
|
|
|
{
|
|
|
|
return MinDiff >= 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return (uint)MaxDiff >= (uint)MinDiff;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|