ce1d5be212
* Move GPU LLE emulation from HLE to Graphics * Graphics: Move Gal/Texture to Texture * Remove Engines/ directory and namespace * Use tables for image formats * Abstract OpCode decoding * Simplify image table * Do not leak Read* symbols in TextureReader * Fixups * Rename IGalFrameBuffer -> IGalRenderTarget * Remove MaxBpp hardcoded value * Change yet again texture data and add G8R8 flipping * Rename GalFrameBufferFormat to GalSurfaceFormat * Unident EnsureSetup in ImageHandler * Add IsCompressed * Address some feedback
200 lines
5 KiB
C#
200 lines
5 KiB
C#
using Ryujinx.Graphics.Memory;
|
|
using System.Collections.Generic;
|
|
|
|
namespace Ryujinx.HLE.HOS.Services.Nv.NvGpuAS
|
|
{
|
|
class NvGpuASCtx
|
|
{
|
|
public NvGpuVmm Vmm { get; private set; }
|
|
|
|
private class Range
|
|
{
|
|
public ulong Start { get; private set; }
|
|
public ulong End { get; private set; }
|
|
|
|
public Range(long Position, long Size)
|
|
{
|
|
Start = (ulong)Position;
|
|
End = (ulong)Size + Start;
|
|
}
|
|
}
|
|
|
|
private class MappedMemory : Range
|
|
{
|
|
public long PhysicalAddress { get; private set; }
|
|
public bool VaAllocated { get; private set; }
|
|
|
|
public MappedMemory(
|
|
long Position,
|
|
long Size,
|
|
long PhysicalAddress,
|
|
bool VaAllocated) : base(Position, Size)
|
|
{
|
|
this.PhysicalAddress = PhysicalAddress;
|
|
this.VaAllocated = VaAllocated;
|
|
}
|
|
}
|
|
|
|
private SortedList<long, Range> Maps;
|
|
private SortedList<long, Range> Reservations;
|
|
|
|
public NvGpuASCtx(ServiceCtx Context)
|
|
{
|
|
Vmm = new NvGpuVmm(Context.Memory);
|
|
|
|
Maps = new SortedList<long, Range>();
|
|
Reservations = new SortedList<long, Range>();
|
|
}
|
|
|
|
public bool ValidateFixedBuffer(long Position, long Size)
|
|
{
|
|
long MapEnd = Position + Size;
|
|
|
|
//Check if size is valid (0 is also not allowed).
|
|
if ((ulong)MapEnd <= (ulong)Position)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
//Check if address is page aligned.
|
|
if ((Position & NvGpuVmm.PageMask) != 0)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
//Check if region is reserved.
|
|
if (BinarySearch(Reservations, Position) == null)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
//Check for overlap with already mapped buffers.
|
|
Range Map = BinarySearchLt(Maps, MapEnd);
|
|
|
|
if (Map != null && Map.End > (ulong)Position)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
public void AddMap(
|
|
long Position,
|
|
long Size,
|
|
long PhysicalAddress,
|
|
bool VaAllocated)
|
|
{
|
|
Maps.Add(Position, new MappedMemory(Position, Size, PhysicalAddress, VaAllocated));
|
|
}
|
|
|
|
public bool RemoveMap(long Position, out long Size)
|
|
{
|
|
Size = 0;
|
|
|
|
if (Maps.Remove(Position, out Range Value))
|
|
{
|
|
MappedMemory Map = (MappedMemory)Value;
|
|
|
|
if (Map.VaAllocated)
|
|
{
|
|
Size = (long)(Map.End - Map.Start);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
public bool TryGetMapPhysicalAddress(long Position, out long PhysicalAddress)
|
|
{
|
|
Range Map = BinarySearch(Maps, Position);
|
|
|
|
if (Map != null)
|
|
{
|
|
PhysicalAddress = ((MappedMemory)Map).PhysicalAddress;
|
|
|
|
return true;
|
|
}
|
|
|
|
PhysicalAddress = 0;
|
|
|
|
return false;
|
|
}
|
|
|
|
public void AddReservation(long Position, long Size)
|
|
{
|
|
Reservations.Add(Position, new Range(Position, Size));
|
|
}
|
|
|
|
public bool RemoveReservation(long Position)
|
|
{
|
|
return Reservations.Remove(Position);
|
|
}
|
|
|
|
private Range BinarySearch(SortedList<long, Range> Lst, long Position)
|
|
{
|
|
int Left = 0;
|
|
int Right = Lst.Count - 1;
|
|
|
|
while (Left <= Right)
|
|
{
|
|
int Size = Right - Left;
|
|
|
|
int Middle = Left + (Size >> 1);
|
|
|
|
Range Rg = Lst.Values[Middle];
|
|
|
|
if ((ulong)Position >= Rg.Start && (ulong)Position < Rg.End)
|
|
{
|
|
return Rg;
|
|
}
|
|
|
|
if ((ulong)Position < Rg.Start)
|
|
{
|
|
Right = Middle - 1;
|
|
}
|
|
else
|
|
{
|
|
Left = Middle + 1;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
private Range BinarySearchLt(SortedList<long, Range> Lst, long Position)
|
|
{
|
|
Range LtRg = null;
|
|
|
|
int Left = 0;
|
|
int Right = Lst.Count - 1;
|
|
|
|
while (Left <= Right)
|
|
{
|
|
int Size = Right - Left;
|
|
|
|
int Middle = Left + (Size >> 1);
|
|
|
|
Range Rg = Lst.Values[Middle];
|
|
|
|
if ((ulong)Position < Rg.Start)
|
|
{
|
|
Right = Middle - 1;
|
|
}
|
|
else
|
|
{
|
|
Left = Middle + 1;
|
|
|
|
if ((ulong)Position > Rg.Start)
|
|
{
|
|
LtRg = Rg;
|
|
}
|
|
}
|
|
}
|
|
|
|
return LtRg;
|
|
}
|
|
}
|
|
} |