forked from Mirror/Ryujinx
5001f78b1d
* Implement faster address translation and write tracking on the MMU * Rename MemoryAlloc to MemoryManagement, and other nits * Support multi-level page tables * Fix typo * Reword comment a bit * Support scalar vector loads/stores on the memory fast path, and minor fixes * Add missing cast * Alignment * Fix VirtualFree function signature * Change MemoryProtection enum to uint aswell for consistency
79 lines
No EOL
2.2 KiB
C#
79 lines
No EOL
2.2 KiB
C#
using ChocolArm64.Memory;
|
|
using System.Collections.Concurrent;
|
|
|
|
namespace Ryujinx.Graphics.Memory
|
|
{
|
|
class NvGpuVmmCache
|
|
{
|
|
private const int PageBits = MemoryManager.PageBits;
|
|
|
|
private const long PageSize = MemoryManager.PageSize;
|
|
private const long PageMask = MemoryManager.PageMask;
|
|
|
|
private ConcurrentDictionary<long, int>[] CachedPages;
|
|
|
|
private MemoryManager _memory;
|
|
|
|
public NvGpuVmmCache(MemoryManager memory)
|
|
{
|
|
_memory = memory;
|
|
|
|
CachedPages = new ConcurrentDictionary<long, int>[1 << 20];
|
|
}
|
|
|
|
public bool IsRegionModified(long position, long size, NvGpuBufferType bufferType)
|
|
{
|
|
long va = position;
|
|
|
|
long pa = _memory.GetPhysicalAddress(va);
|
|
|
|
long endAddr = (va + size + PageMask) & ~PageMask;
|
|
|
|
long addrTruncated = va & ~PageMask;
|
|
|
|
bool modified = _memory.IsRegionModified(addrTruncated, endAddr - addrTruncated);
|
|
|
|
int newBuffMask = 1 << (int)bufferType;
|
|
|
|
long cachedPagesCount = 0;
|
|
|
|
while (va < endAddr)
|
|
{
|
|
long page = _memory.GetPhysicalAddress(va) >> PageBits;
|
|
|
|
ConcurrentDictionary<long, int> dictionary = CachedPages[page];
|
|
|
|
if (dictionary == null)
|
|
{
|
|
dictionary = new ConcurrentDictionary<long, int>();
|
|
|
|
CachedPages[page] = dictionary;
|
|
}
|
|
else if (modified)
|
|
{
|
|
CachedPages[page].Clear();
|
|
}
|
|
|
|
if (dictionary.TryGetValue(pa, out int currBuffMask))
|
|
{
|
|
if ((currBuffMask & newBuffMask) != 0)
|
|
{
|
|
cachedPagesCount++;
|
|
}
|
|
else
|
|
{
|
|
dictionary[pa] |= newBuffMask;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
dictionary[pa] = newBuffMask;
|
|
}
|
|
|
|
va += PageSize;
|
|
}
|
|
|
|
return cachedPagesCount != (endAddr - addrTruncated) >> PageBits;
|
|
}
|
|
}
|
|
} |