forked from Mirror/Ryujinx
c393cdf8e3
* Keep track mapped buffers with fixed offsets * Started rewriting the memory manager * Initial support for MapPhysicalMemory and UnmapPhysicalMemory, other tweaks * MapPhysicalMemory/UnmapPhysicalMemory support, other tweaks * Rebased * Optimize the map/unmap physical memory svcs * Integrate shared font support * Fix address space reserve alignment * Some fixes related to gpu memory mapping * Some cleanup * Only try uploading const buffers that are really used * Check if memory region is contiguous * Rebased * Add missing count increment on IsRegionModified * Check for reads/writes outside of the address space, optimize translation with a tail call
308 lines
No EOL
7.8 KiB
C#
308 lines
No EOL
7.8 KiB
C#
using ChocolArm64.Memory;
|
|
using Ryujinx.HLE.Memory;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
|
|
namespace Ryujinx.HLE.Gpu.Memory
|
|
{
|
|
class NvGpuVmmCache
|
|
{
|
|
private const int MaxCpCount = 10000;
|
|
private const int MaxCpTimeDelta = 60000;
|
|
|
|
private class CachedPage
|
|
{
|
|
private struct Range
|
|
{
|
|
public long Start;
|
|
public long End;
|
|
|
|
public Range(long Start, long End)
|
|
{
|
|
this.Start = Start;
|
|
this.End = End;
|
|
}
|
|
}
|
|
|
|
private List<Range>[] Regions;
|
|
|
|
private HashSet<long> ResidencyKeys;
|
|
|
|
public LinkedListNode<long> Node { get; set; }
|
|
|
|
public int Timestamp { get; private set; }
|
|
|
|
public CachedPage()
|
|
{
|
|
Regions = new List<Range>[(int)NvGpuBufferType.Count];
|
|
|
|
for (int Index = 0; Index < Regions.Length; Index++)
|
|
{
|
|
Regions[Index] = new List<Range>();
|
|
}
|
|
|
|
ResidencyKeys = new HashSet<long>();
|
|
}
|
|
|
|
public void AddResidency(long Key)
|
|
{
|
|
ResidencyKeys.Add(Key);
|
|
}
|
|
|
|
public void RemoveResidency(HashSet<long>[] Residency, long PageSize)
|
|
{
|
|
for (int i = 0; i < (int)NvGpuBufferType.Count; i++)
|
|
{
|
|
foreach (Range Region in Regions[i])
|
|
{
|
|
foreach (long Key in ResidencyKeys)
|
|
{
|
|
Residency[Region.Start / PageSize].Remove(Key);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public bool AddRange(long Start, long End, NvGpuBufferType BufferType)
|
|
{
|
|
List<Range> BtRegions = Regions[(int)BufferType];
|
|
|
|
for (int Index = 0; Index < BtRegions.Count; Index++)
|
|
{
|
|
Range Rg = BtRegions[Index];
|
|
|
|
if (Start >= Rg.Start && End <= Rg.End)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (Start <= Rg.End && Rg.Start <= End)
|
|
{
|
|
long MinStart = Math.Min(Rg.Start, Start);
|
|
long MaxEnd = Math.Max(Rg.End, End);
|
|
|
|
BtRegions[Index] = new Range(MinStart, MaxEnd);
|
|
|
|
Timestamp = Environment.TickCount;
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
BtRegions.Add(new Range(Start, End));
|
|
|
|
Timestamp = Environment.TickCount;
|
|
|
|
return true;
|
|
}
|
|
|
|
public int GetTotalCount()
|
|
{
|
|
int Count = 0;
|
|
|
|
for (int Index = 0; Index < Regions.Length; Index++)
|
|
{
|
|
Count += Regions[Index].Count;
|
|
}
|
|
|
|
return Count;
|
|
}
|
|
}
|
|
|
|
private Dictionary<long, CachedPage> Cache;
|
|
|
|
private LinkedList<long> SortedCache;
|
|
|
|
private HashSet<long>[] Residency;
|
|
|
|
private long ResidencyPageSize;
|
|
|
|
private int CpCount;
|
|
|
|
public NvGpuVmmCache()
|
|
{
|
|
Cache = new Dictionary<long, CachedPage>();
|
|
|
|
SortedCache = new LinkedList<long>();
|
|
}
|
|
|
|
public bool IsRegionModified(AMemory Memory, NvGpuBufferType BufferType, long PA, long Size)
|
|
{
|
|
(bool[] Modified, long ModifiedCount) = Memory.IsRegionModified(PA, Size);
|
|
|
|
PA = Memory.GetPhysicalAddress(PA);
|
|
|
|
ClearCachedPagesIfNeeded();
|
|
|
|
long PageSize = AMemory.PageSize;
|
|
|
|
EnsureResidencyInitialized(PageSize);
|
|
|
|
bool HasResidents = AddResidency(PA, Size);
|
|
|
|
if (!HasResidents && ModifiedCount == 0)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
long Mask = PageSize - 1;
|
|
|
|
long ResidencyKey = PA;
|
|
|
|
long PAEnd = PA + Size;
|
|
|
|
bool RegMod = false;
|
|
|
|
int Index = 0;
|
|
|
|
while (PA < PAEnd)
|
|
{
|
|
long Key = PA & ~AMemory.PageMask;
|
|
|
|
long PAPgEnd = Math.Min((PA + AMemory.PageSize) & ~AMemory.PageMask, PAEnd);
|
|
|
|
bool IsCached = Cache.TryGetValue(Key, out CachedPage Cp);
|
|
|
|
if (IsCached)
|
|
{
|
|
CpCount -= Cp.GetTotalCount();
|
|
|
|
SortedCache.Remove(Cp.Node);
|
|
}
|
|
else
|
|
{
|
|
Cp = new CachedPage();
|
|
|
|
Cache.Add(Key, Cp);
|
|
}
|
|
|
|
if (Modified[Index++] && IsCached)
|
|
{
|
|
Cp = new CachedPage();
|
|
|
|
Cache[Key] = Cp;
|
|
}
|
|
|
|
Cp.AddResidency(ResidencyKey);
|
|
|
|
Cp.Node = SortedCache.AddLast(Key);
|
|
|
|
RegMod |= Cp.AddRange(PA, PAPgEnd, BufferType);
|
|
|
|
CpCount += Cp.GetTotalCount();
|
|
|
|
PA = PAPgEnd;
|
|
}
|
|
|
|
return RegMod;
|
|
}
|
|
|
|
private bool AddResidency(long PA, long Size)
|
|
{
|
|
long PageSize = ResidencyPageSize;
|
|
|
|
long Mask = PageSize - 1;
|
|
|
|
long Key = PA;
|
|
|
|
bool ResidentFound = false;
|
|
|
|
for (long Cursor = PA & ~Mask; Cursor < ((PA + Size + PageSize - 1) & ~Mask); Cursor += PageSize)
|
|
{
|
|
long PageIndex = Cursor / PageSize;
|
|
|
|
Residency[PageIndex].Add(Key);
|
|
|
|
if (Residency[PageIndex].Count > 1)
|
|
{
|
|
ResidentFound = true;
|
|
}
|
|
}
|
|
|
|
return ResidentFound;
|
|
}
|
|
|
|
private void EnsureResidencyInitialized(long PageSize)
|
|
{
|
|
if (Residency == null)
|
|
{
|
|
Residency = new HashSet<long>[DeviceMemory.RamSize / PageSize];
|
|
|
|
for (int i = 0; i < Residency.Length; i++)
|
|
{
|
|
Residency[i] = new HashSet<long>();
|
|
}
|
|
|
|
ResidencyPageSize = PageSize;
|
|
}
|
|
else
|
|
{
|
|
if (ResidencyPageSize != PageSize)
|
|
{
|
|
throw new InvalidOperationException("Tried to change residency page size");
|
|
}
|
|
}
|
|
}
|
|
|
|
private void ClearCachedPagesIfNeeded()
|
|
{
|
|
if (CpCount <= MaxCpCount)
|
|
{
|
|
return;
|
|
}
|
|
|
|
int Timestamp = Environment.TickCount;
|
|
|
|
int TimeDelta;
|
|
|
|
do
|
|
{
|
|
if (!TryPopOldestCachedPageKey(Timestamp, out long Key))
|
|
{
|
|
break;
|
|
}
|
|
|
|
CachedPage Cp = Cache[Key];
|
|
|
|
Cp.RemoveResidency(Residency, ResidencyPageSize);
|
|
|
|
Cache.Remove(Key);
|
|
|
|
CpCount -= Cp.GetTotalCount();
|
|
|
|
TimeDelta = RingDelta(Cp.Timestamp, Timestamp);
|
|
}
|
|
while (CpCount > (MaxCpCount >> 1) || (uint)TimeDelta > (uint)MaxCpTimeDelta);
|
|
}
|
|
|
|
private bool TryPopOldestCachedPageKey(int Timestamp, out long Key)
|
|
{
|
|
LinkedListNode<long> Node = SortedCache.First;
|
|
|
|
if (Node == null)
|
|
{
|
|
Key = 0;
|
|
|
|
return false;
|
|
}
|
|
|
|
SortedCache.Remove(Node);
|
|
|
|
Key = Node.Value;
|
|
|
|
return true;
|
|
}
|
|
|
|
private int RingDelta(int Old, int New)
|
|
{
|
|
if ((uint)New < (uint)Old)
|
|
{
|
|
return New + (~Old + 1);
|
|
}
|
|
else
|
|
{
|
|
return New - Old;
|
|
}
|
|
}
|
|
}
|
|
} |