ryujinx/Ryujinx.Graphics.Texture/SizeInfo.cs

119 lines
3.2 KiB
C#
Raw Normal View History

2019-10-13 06:02:07 +00:00
using System;
using System.Collections.Generic;
2019-10-13 06:02:07 +00:00
namespace Ryujinx.Graphics.Texture
{
public readonly struct SizeInfo
2019-10-13 06:02:07 +00:00
{
private readonly int[] _mipOffsets;
2019-10-13 06:02:07 +00:00
private readonly int _levels;
private readonly int _depth;
private readonly bool _is3D;
2019-10-13 06:02:07 +00:00
public readonly int[] AllOffsets;
public readonly int[] SliceSizes;
public readonly int[] LevelSizes;
2019-10-13 06:02:07 +00:00
public int LayerSize { get; }
public int TotalSize { get; }
public SizeInfo(int size)
{
_mipOffsets = new int[] { 0 };
AllOffsets = new int[] { 0 };
SliceSizes = new int[] { size };
LevelSizes = new int[] { size };
_depth = 1;
_levels = 1;
LayerSize = size;
TotalSize = size;
_is3D = false;
}
internal SizeInfo(
2019-10-13 06:02:07 +00:00
int[] mipOffsets,
int[] allOffsets,
int[] sliceSizes,
int[] levelSizes,
int depth,
2019-10-13 06:02:07 +00:00
int levels,
int layerSize,
int totalSize,
bool is3D)
2019-10-13 06:02:07 +00:00
{
_mipOffsets = mipOffsets;
AllOffsets = allOffsets;
SliceSizes = sliceSizes;
LevelSizes = levelSizes;
_depth = depth;
2019-10-13 06:02:07 +00:00
_levels = levels;
LayerSize = layerSize;
TotalSize = totalSize;
_is3D = is3D;
2019-10-13 06:02:07 +00:00
}
public int GetMipOffset(int level)
{
if ((uint)level >= _mipOffsets.Length)
2019-10-13 06:02:07 +00:00
{
throw new ArgumentOutOfRangeException(nameof(level));
}
return _mipOffsets[level];
}
public bool FindView(int offset, out int firstLayer, out int firstLevel)
2019-10-13 06:02:07 +00:00
{
int index = Array.BinarySearch(AllOffsets, offset);
2019-10-13 06:02:07 +00:00
if (index < 0)
{
firstLayer = 0;
firstLevel = 0;
return false;
}
if (_is3D)
{
firstLayer = index;
firstLevel = 0;
int levelDepth = _depth;
while (firstLayer >= levelDepth)
{
firstLayer -= levelDepth;
firstLevel++;
levelDepth = Math.Max(levelDepth >> 1, 1);
}
}
else
{
firstLayer = index / _levels;
firstLevel = index - (firstLayer * _levels);
}
2019-10-13 06:02:07 +00:00
return true;
}
public IEnumerable<Region> AllRegions()
{
if (_is3D)
{
for (int i = 0; i < _mipOffsets.Length; i++)
{
int maxSize = TotalSize - _mipOffsets[i];
yield return new Region(_mipOffsets[i], Math.Min(maxSize, LevelSizes[i]));
}
}
else
{
for (int i = 0; i < AllOffsets.Length; i++)
{
yield return new Region(AllOffsets[i], SliceSizes[i % _levels]);
}
}
}
2019-10-13 06:02:07 +00:00
}
}