forked from Mirror/Ryujinx
306 lines
8.8 KiB
C#
306 lines
8.8 KiB
C#
|
using System;
|
|||
|
using System.Collections.Generic;
|
|||
|
|
|||
|
namespace Ryujinx.Graphics.Vulkan
|
|||
|
{
|
|||
|
/// <summary>
|
|||
|
/// A structure tracking pending upload ranges for buffers.
|
|||
|
/// Where a range is present, pending data exists that can either be used to build mirrors
|
|||
|
/// or upload directly to the buffer.
|
|||
|
/// </summary>
|
|||
|
struct BufferMirrorRangeList
|
|||
|
{
|
|||
|
internal readonly struct Range
|
|||
|
{
|
|||
|
public int Offset { get; }
|
|||
|
public int Size { get; }
|
|||
|
|
|||
|
public int End => Offset + Size;
|
|||
|
|
|||
|
public Range(int offset, int size)
|
|||
|
{
|
|||
|
Offset = offset;
|
|||
|
Size = size;
|
|||
|
}
|
|||
|
|
|||
|
public bool OverlapsWith(int offset, int size)
|
|||
|
{
|
|||
|
return Offset < offset + size && offset < Offset + Size;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
private List<Range> _ranges;
|
|||
|
|
|||
|
public readonly IEnumerable<Range> All()
|
|||
|
{
|
|||
|
return _ranges;
|
|||
|
}
|
|||
|
|
|||
|
public readonly bool Remove(int offset, int size)
|
|||
|
{
|
|||
|
var list = _ranges;
|
|||
|
bool removedAny = false;
|
|||
|
if (list != null)
|
|||
|
{
|
|||
|
int overlapIndex = BinarySearch(list, offset, size);
|
|||
|
|
|||
|
if (overlapIndex >= 0)
|
|||
|
{
|
|||
|
// Overlaps with a range. Search back to find the first one it doesn't overlap with.
|
|||
|
|
|||
|
while (overlapIndex > 0 && list[overlapIndex - 1].OverlapsWith(offset, size))
|
|||
|
{
|
|||
|
overlapIndex--;
|
|||
|
}
|
|||
|
|
|||
|
int endOffset = offset + size;
|
|||
|
int startIndex = overlapIndex;
|
|||
|
|
|||
|
var currentOverlap = list[overlapIndex];
|
|||
|
|
|||
|
// Orphan the start of the overlap.
|
|||
|
if (currentOverlap.Offset < offset)
|
|||
|
{
|
|||
|
list[overlapIndex] = new Range(currentOverlap.Offset, offset - currentOverlap.Offset);
|
|||
|
currentOverlap = new Range(offset, currentOverlap.End - offset);
|
|||
|
list.Insert(++overlapIndex, currentOverlap);
|
|||
|
startIndex++;
|
|||
|
|
|||
|
removedAny = true;
|
|||
|
}
|
|||
|
|
|||
|
// Remove any middle overlaps.
|
|||
|
while (currentOverlap.Offset < endOffset)
|
|||
|
{
|
|||
|
if (currentOverlap.End > endOffset)
|
|||
|
{
|
|||
|
// Update the end overlap instead of removing it, if it spans beyond the removed range.
|
|||
|
list[overlapIndex] = new Range(endOffset, currentOverlap.End - endOffset);
|
|||
|
|
|||
|
removedAny = true;
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
if (++overlapIndex >= list.Count)
|
|||
|
{
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
currentOverlap = list[overlapIndex];
|
|||
|
}
|
|||
|
|
|||
|
int count = overlapIndex - startIndex;
|
|||
|
|
|||
|
list.RemoveRange(startIndex, count);
|
|||
|
|
|||
|
removedAny |= count > 0;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return removedAny;
|
|||
|
}
|
|||
|
|
|||
|
public void Add(int offset, int size)
|
|||
|
{
|
|||
|
var list = _ranges;
|
|||
|
if (list != null)
|
|||
|
{
|
|||
|
int overlapIndex = BinarySearch(list, offset, size);
|
|||
|
if (overlapIndex >= 0)
|
|||
|
{
|
|||
|
while (overlapIndex > 0 && list[overlapIndex - 1].OverlapsWith(offset, size))
|
|||
|
{
|
|||
|
overlapIndex--;
|
|||
|
}
|
|||
|
|
|||
|
int endOffset = offset + size;
|
|||
|
int startIndex = overlapIndex;
|
|||
|
|
|||
|
while (overlapIndex < list.Count && list[overlapIndex].OverlapsWith(offset, size))
|
|||
|
{
|
|||
|
var currentOverlap = list[overlapIndex];
|
|||
|
var currentOverlapEndOffset = currentOverlap.Offset + currentOverlap.Size;
|
|||
|
|
|||
|
if (offset > currentOverlap.Offset)
|
|||
|
{
|
|||
|
offset = currentOverlap.Offset;
|
|||
|
}
|
|||
|
|
|||
|
if (endOffset < currentOverlapEndOffset)
|
|||
|
{
|
|||
|
endOffset = currentOverlapEndOffset;
|
|||
|
}
|
|||
|
|
|||
|
overlapIndex++;
|
|||
|
size = endOffset - offset;
|
|||
|
}
|
|||
|
|
|||
|
int count = overlapIndex - startIndex;
|
|||
|
|
|||
|
list.RemoveRange(startIndex, count);
|
|||
|
|
|||
|
overlapIndex = startIndex;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
overlapIndex = ~overlapIndex;
|
|||
|
}
|
|||
|
|
|||
|
list.Insert(overlapIndex, new Range(offset, size));
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
_ranges = new List<Range>
|
|||
|
{
|
|||
|
new Range(offset, size)
|
|||
|
};
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
public readonly bool OverlapsWith(int offset, int size)
|
|||
|
{
|
|||
|
var list = _ranges;
|
|||
|
if (list == null)
|
|||
|
{
|
|||
|
return false;
|
|||
|
}
|
|||
|
|
|||
|
return BinarySearch(list, offset, size) >= 0;
|
|||
|
}
|
|||
|
|
|||
|
public readonly List<Range> FindOverlaps(int offset, int size)
|
|||
|
{
|
|||
|
var list = _ranges;
|
|||
|
if (list == null)
|
|||
|
{
|
|||
|
return null;
|
|||
|
}
|
|||
|
|
|||
|
List<Range> result = null;
|
|||
|
|
|||
|
int index = BinarySearch(list, offset, size);
|
|||
|
|
|||
|
if (index >= 0)
|
|||
|
{
|
|||
|
while (index > 0 && list[index - 1].OverlapsWith(offset, size))
|
|||
|
{
|
|||
|
index--;
|
|||
|
}
|
|||
|
|
|||
|
do
|
|||
|
{
|
|||
|
(result ??= new List<Range>()).Add(list[index++]);
|
|||
|
}
|
|||
|
while (index < list.Count && list[index].OverlapsWith(offset, size));
|
|||
|
}
|
|||
|
|
|||
|
return result;
|
|||
|
}
|
|||
|
|
|||
|
private static int BinarySearch(List<Range> list, int offset, int size)
|
|||
|
{
|
|||
|
int left = 0;
|
|||
|
int right = list.Count - 1;
|
|||
|
|
|||
|
while (left <= right)
|
|||
|
{
|
|||
|
int range = right - left;
|
|||
|
|
|||
|
int middle = left + (range >> 1);
|
|||
|
|
|||
|
var item = list[middle];
|
|||
|
|
|||
|
if (item.OverlapsWith(offset, size))
|
|||
|
{
|
|||
|
return middle;
|
|||
|
}
|
|||
|
|
|||
|
if (offset < item.Offset)
|
|||
|
{
|
|||
|
right = middle - 1;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
left = middle + 1;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return ~left;
|
|||
|
}
|
|||
|
|
|||
|
public readonly void FillData(Span<byte> baseData, Span<byte> modData, int offset, Span<byte> result)
|
|||
|
{
|
|||
|
int size = baseData.Length;
|
|||
|
int endOffset = offset + size;
|
|||
|
|
|||
|
var list = _ranges;
|
|||
|
if (list == null)
|
|||
|
{
|
|||
|
baseData.CopyTo(result);
|
|||
|
}
|
|||
|
|
|||
|
int srcOffset = offset;
|
|||
|
int dstOffset = 0;
|
|||
|
bool activeRange = false;
|
|||
|
|
|||
|
for (int i = 0; i < list.Count; i++)
|
|||
|
{
|
|||
|
var range = list[i];
|
|||
|
|
|||
|
int rangeEnd = range.Offset + range.Size;
|
|||
|
|
|||
|
if (activeRange)
|
|||
|
{
|
|||
|
if (range.Offset >= endOffset)
|
|||
|
{
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
if (rangeEnd <= offset)
|
|||
|
{
|
|||
|
continue;
|
|||
|
}
|
|||
|
|
|||
|
activeRange = true;
|
|||
|
}
|
|||
|
|
|||
|
int baseSize = range.Offset - srcOffset;
|
|||
|
|
|||
|
if (baseSize > 0)
|
|||
|
{
|
|||
|
baseData.Slice(dstOffset, baseSize).CopyTo(result.Slice(dstOffset, baseSize));
|
|||
|
srcOffset += baseSize;
|
|||
|
dstOffset += baseSize;
|
|||
|
}
|
|||
|
|
|||
|
int modSize = Math.Min(rangeEnd - srcOffset, endOffset - srcOffset);
|
|||
|
if (modSize != 0)
|
|||
|
{
|
|||
|
modData.Slice(dstOffset, modSize).CopyTo(result.Slice(dstOffset, modSize));
|
|||
|
srcOffset += modSize;
|
|||
|
dstOffset += modSize;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
int baseSizeEnd = endOffset - srcOffset;
|
|||
|
|
|||
|
if (baseSizeEnd > 0)
|
|||
|
{
|
|||
|
baseData.Slice(dstOffset, baseSizeEnd).CopyTo(result.Slice(dstOffset, baseSizeEnd));
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
public readonly int Count()
|
|||
|
{
|
|||
|
return _ranges?.Count ?? 0;
|
|||
|
}
|
|||
|
|
|||
|
public void Clear()
|
|||
|
{
|
|||
|
_ranges = null;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|