2018-04-08 21:17:35 +02:00
|
|
|
using ChocolArm64.Memory;
|
|
|
|
using Ryujinx.Graphics.Gal;
|
|
|
|
using System;
|
|
|
|
|
2018-06-24 02:39:25 +02:00
|
|
|
namespace Ryujinx.HLE.Gpu.Texture
|
2018-04-08 21:17:35 +02:00
|
|
|
{
|
2018-06-09 02:15:56 +02:00
|
|
|
static class TextureReader
|
2018-04-08 21:17:35 +02:00
|
|
|
{
|
2018-06-24 02:39:25 +02:00
|
|
|
public static byte[] Read(IAMemory Memory, TextureInfo Texture)
|
2018-04-08 21:17:35 +02:00
|
|
|
{
|
|
|
|
switch (Texture.Format)
|
|
|
|
{
|
2018-05-30 01:37:10 +02:00
|
|
|
case GalTextureFormat.R32G32B32A32: return Read16Bpp (Memory, Texture);
|
|
|
|
case GalTextureFormat.R16G16B16A16: return Read8Bpp (Memory, Texture);
|
|
|
|
case GalTextureFormat.A8B8G8R8: return Read4Bpp (Memory, Texture);
|
|
|
|
case GalTextureFormat.R32: return Read4Bpp (Memory, Texture);
|
|
|
|
case GalTextureFormat.A1B5G5R5: return Read5551 (Memory, Texture);
|
|
|
|
case GalTextureFormat.B5G6R5: return Read565 (Memory, Texture);
|
|
|
|
case GalTextureFormat.G8R8: return Read2Bpp (Memory, Texture);
|
2018-06-12 14:29:44 +02:00
|
|
|
case GalTextureFormat.R16: return Read2Bpp (Memory, Texture);
|
2018-05-30 01:37:10 +02:00
|
|
|
case GalTextureFormat.R8: return Read1Bpp (Memory, Texture);
|
2018-06-10 06:00:43 +02:00
|
|
|
case GalTextureFormat.BC7U: return Read16Bpt4x4(Memory, Texture);
|
2018-05-30 01:37:10 +02:00
|
|
|
case GalTextureFormat.BC1: return Read8Bpt4x4 (Memory, Texture);
|
|
|
|
case GalTextureFormat.BC2: return Read16Bpt4x4(Memory, Texture);
|
|
|
|
case GalTextureFormat.BC3: return Read16Bpt4x4(Memory, Texture);
|
|
|
|
case GalTextureFormat.BC4: return Read8Bpt4x4 (Memory, Texture);
|
|
|
|
case GalTextureFormat.BC5: return Read16Bpt4x4(Memory, Texture);
|
2018-06-02 05:50:56 +02:00
|
|
|
case GalTextureFormat.Astc2D4x4: return Read16Bpt4x4(Memory, Texture);
|
2018-04-08 21:17:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
throw new NotImplementedException(Texture.Format.ToString());
|
|
|
|
}
|
|
|
|
|
2018-06-24 02:39:25 +02:00
|
|
|
private unsafe static byte[] Read1Bpp(IAMemory Memory, TextureInfo Texture)
|
2018-05-30 01:37:10 +02:00
|
|
|
{
|
|
|
|
int Width = Texture.Width;
|
|
|
|
int Height = Texture.Height;
|
|
|
|
|
|
|
|
byte[] Output = new byte[Width * Height];
|
|
|
|
|
|
|
|
ISwizzle Swizzle = TextureHelper.GetSwizzle(Texture, Width, 1);
|
|
|
|
|
|
|
|
(AMemory CpuMem, long Position) = TextureHelper.GetMemoryAndPosition(
|
|
|
|
Memory,
|
|
|
|
Texture.Position);
|
|
|
|
|
|
|
|
fixed (byte* BuffPtr = Output)
|
|
|
|
{
|
|
|
|
long OutOffs = 0;
|
|
|
|
|
|
|
|
for (int Y = 0; Y < Height; Y++)
|
|
|
|
for (int X = 0; X < Width; X++)
|
|
|
|
{
|
|
|
|
long Offset = (uint)Swizzle.GetSwizzleOffset(X, Y);
|
|
|
|
|
|
|
|
byte Pixel = CpuMem.ReadByteUnchecked(Position + Offset);
|
|
|
|
|
|
|
|
*(BuffPtr + OutOffs) = Pixel;
|
|
|
|
|
|
|
|
OutOffs++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Output;
|
|
|
|
}
|
|
|
|
|
2018-06-24 02:39:25 +02:00
|
|
|
private unsafe static byte[] Read5551(IAMemory Memory, TextureInfo Texture)
|
2018-05-30 01:37:10 +02:00
|
|
|
{
|
|
|
|
int Width = Texture.Width;
|
|
|
|
int Height = Texture.Height;
|
|
|
|
|
|
|
|
byte[] Output = new byte[Width * Height * 2];
|
|
|
|
|
|
|
|
ISwizzle Swizzle = TextureHelper.GetSwizzle(Texture, Width, 2);
|
|
|
|
|
|
|
|
(AMemory CpuMem, long Position) = TextureHelper.GetMemoryAndPosition(
|
|
|
|
Memory,
|
|
|
|
Texture.Position);
|
|
|
|
|
|
|
|
fixed (byte* BuffPtr = Output)
|
|
|
|
{
|
|
|
|
long OutOffs = 0;
|
|
|
|
|
|
|
|
for (int Y = 0; Y < Height; Y++)
|
|
|
|
for (int X = 0; X < Width; X++)
|
|
|
|
{
|
|
|
|
long Offset = (uint)Swizzle.GetSwizzleOffset(X, Y);
|
|
|
|
|
|
|
|
uint Pixel = (uint)CpuMem.ReadInt16Unchecked(Position + Offset);
|
|
|
|
|
|
|
|
Pixel = (Pixel & 0x001f) << 11 |
|
|
|
|
(Pixel & 0x03e0) << 1 |
|
|
|
|
(Pixel & 0x7c00) >> 9 |
|
|
|
|
(Pixel & 0x8000) >> 15;
|
|
|
|
|
|
|
|
*(short*)(BuffPtr + OutOffs) = (short)Pixel;
|
|
|
|
|
|
|
|
OutOffs += 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Output;
|
|
|
|
}
|
|
|
|
|
2018-06-24 02:39:25 +02:00
|
|
|
private unsafe static byte[] Read565(IAMemory Memory, TextureInfo Texture)
|
2018-05-30 01:37:10 +02:00
|
|
|
{
|
|
|
|
int Width = Texture.Width;
|
|
|
|
int Height = Texture.Height;
|
|
|
|
|
|
|
|
byte[] Output = new byte[Width * Height * 2];
|
|
|
|
|
|
|
|
ISwizzle Swizzle = TextureHelper.GetSwizzle(Texture, Width, 2);
|
|
|
|
|
|
|
|
(AMemory CpuMem, long Position) = TextureHelper.GetMemoryAndPosition(
|
|
|
|
Memory,
|
|
|
|
Texture.Position);
|
|
|
|
|
|
|
|
fixed (byte* BuffPtr = Output)
|
|
|
|
{
|
|
|
|
long OutOffs = 0;
|
|
|
|
|
|
|
|
for (int Y = 0; Y < Height; Y++)
|
|
|
|
for (int X = 0; X < Width; X++)
|
|
|
|
{
|
|
|
|
long Offset = (uint)Swizzle.GetSwizzleOffset(X, Y);
|
|
|
|
|
|
|
|
uint Pixel = (uint)CpuMem.ReadInt16Unchecked(Position + Offset);
|
|
|
|
|
|
|
|
Pixel = (Pixel & 0x001f) << 11 |
|
|
|
|
(Pixel & 0x07e0) |
|
|
|
|
(Pixel & 0xf800) >> 11;
|
|
|
|
|
|
|
|
*(short*)(BuffPtr + OutOffs) = (short)Pixel;
|
|
|
|
|
|
|
|
OutOffs += 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Output;
|
|
|
|
}
|
|
|
|
|
2018-06-24 02:39:25 +02:00
|
|
|
private unsafe static byte[] Read2Bpp(IAMemory Memory, TextureInfo Texture)
|
2018-04-08 21:17:35 +02:00
|
|
|
{
|
|
|
|
int Width = Texture.Width;
|
|
|
|
int Height = Texture.Height;
|
|
|
|
|
2018-04-10 21:50:32 +02:00
|
|
|
byte[] Output = new byte[Width * Height * 2];
|
2018-04-08 21:17:35 +02:00
|
|
|
|
2018-04-26 04:11:26 +02:00
|
|
|
ISwizzle Swizzle = TextureHelper.GetSwizzle(Texture, Width, 2);
|
2018-04-08 21:17:35 +02:00
|
|
|
|
NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 20:53:23 +02:00
|
|
|
(AMemory CpuMem, long Position) = TextureHelper.GetMemoryAndPosition(
|
|
|
|
Memory,
|
|
|
|
Texture.Position);
|
|
|
|
|
2018-04-08 21:17:35 +02:00
|
|
|
fixed (byte* BuffPtr = Output)
|
|
|
|
{
|
|
|
|
long OutOffs = 0;
|
|
|
|
|
|
|
|
for (int Y = 0; Y < Height; Y++)
|
|
|
|
for (int X = 0; X < Width; X++)
|
|
|
|
{
|
|
|
|
long Offset = (uint)Swizzle.GetSwizzleOffset(X, Y);
|
|
|
|
|
NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 20:53:23 +02:00
|
|
|
short Pixel = CpuMem.ReadInt16Unchecked(Position + Offset);
|
2018-04-08 21:17:35 +02:00
|
|
|
|
2018-04-10 21:50:32 +02:00
|
|
|
*(short*)(BuffPtr + OutOffs) = Pixel;
|
2018-04-08 21:17:35 +02:00
|
|
|
|
2018-04-10 21:50:32 +02:00
|
|
|
OutOffs += 2;
|
2018-04-08 21:17:35 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Output;
|
|
|
|
}
|
|
|
|
|
2018-06-24 02:39:25 +02:00
|
|
|
private unsafe static byte[] Read4Bpp(IAMemory Memory, TextureInfo Texture)
|
2018-04-08 23:08:36 +02:00
|
|
|
{
|
|
|
|
int Width = Texture.Width;
|
|
|
|
int Height = Texture.Height;
|
|
|
|
|
2018-04-10 21:50:32 +02:00
|
|
|
byte[] Output = new byte[Width * Height * 4];
|
2018-04-08 23:08:36 +02:00
|
|
|
|
2018-04-26 04:11:26 +02:00
|
|
|
ISwizzle Swizzle = TextureHelper.GetSwizzle(Texture, Width, 4);
|
2018-04-08 23:08:36 +02:00
|
|
|
|
NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 20:53:23 +02:00
|
|
|
(AMemory CpuMem, long Position) = TextureHelper.GetMemoryAndPosition(
|
|
|
|
Memory,
|
|
|
|
Texture.Position);
|
|
|
|
|
2018-04-08 23:08:36 +02:00
|
|
|
fixed (byte* BuffPtr = Output)
|
|
|
|
{
|
|
|
|
long OutOffs = 0;
|
|
|
|
|
|
|
|
for (int Y = 0; Y < Height; Y++)
|
|
|
|
for (int X = 0; X < Width; X++)
|
|
|
|
{
|
|
|
|
long Offset = (uint)Swizzle.GetSwizzleOffset(X, Y);
|
|
|
|
|
NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 20:53:23 +02:00
|
|
|
int Pixel = CpuMem.ReadInt32Unchecked(Position + Offset);
|
2018-04-08 23:08:36 +02:00
|
|
|
|
2018-04-10 21:50:32 +02:00
|
|
|
*(int*)(BuffPtr + OutOffs) = Pixel;
|
2018-04-08 23:08:36 +02:00
|
|
|
|
2018-04-10 21:50:32 +02:00
|
|
|
OutOffs += 4;
|
2018-04-08 23:08:36 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Output;
|
|
|
|
}
|
|
|
|
|
2018-06-24 02:39:25 +02:00
|
|
|
private unsafe static byte[] Read8Bpp(IAMemory Memory, TextureInfo Texture)
|
2018-05-30 01:37:10 +02:00
|
|
|
{
|
|
|
|
int Width = Texture.Width;
|
|
|
|
int Height = Texture.Height;
|
|
|
|
|
|
|
|
byte[] Output = new byte[Width * Height * 8];
|
|
|
|
|
|
|
|
ISwizzle Swizzle = TextureHelper.GetSwizzle(Texture, Width, 8);
|
|
|
|
|
|
|
|
(AMemory CpuMem, long Position) = TextureHelper.GetMemoryAndPosition(
|
|
|
|
Memory,
|
|
|
|
Texture.Position);
|
|
|
|
|
|
|
|
fixed (byte* BuffPtr = Output)
|
|
|
|
{
|
|
|
|
long OutOffs = 0;
|
|
|
|
|
|
|
|
for (int Y = 0; Y < Height; Y++)
|
|
|
|
for (int X = 0; X < Width; X++)
|
|
|
|
{
|
|
|
|
long Offset = (uint)Swizzle.GetSwizzleOffset(X, Y);
|
|
|
|
|
|
|
|
long Pixel = CpuMem.ReadInt64Unchecked(Position + Offset);
|
|
|
|
|
|
|
|
*(long*)(BuffPtr + OutOffs) = Pixel;
|
|
|
|
|
|
|
|
OutOffs += 8;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Output;
|
|
|
|
}
|
|
|
|
|
2018-06-24 02:39:25 +02:00
|
|
|
private unsafe static byte[] Read16Bpp(IAMemory Memory, TextureInfo Texture)
|
2018-05-30 01:37:10 +02:00
|
|
|
{
|
|
|
|
int Width = Texture.Width;
|
|
|
|
int Height = Texture.Height;
|
|
|
|
|
|
|
|
byte[] Output = new byte[Width * Height * 16];
|
|
|
|
|
|
|
|
ISwizzle Swizzle = TextureHelper.GetSwizzle(Texture, Width, 16);
|
|
|
|
|
|
|
|
(AMemory CpuMem, long Position) = TextureHelper.GetMemoryAndPosition(
|
|
|
|
Memory,
|
|
|
|
Texture.Position);
|
|
|
|
|
|
|
|
fixed (byte* BuffPtr = Output)
|
|
|
|
{
|
|
|
|
long OutOffs = 0;
|
|
|
|
|
|
|
|
for (int Y = 0; Y < Height; Y++)
|
|
|
|
for (int X = 0; X < Width; X++)
|
|
|
|
{
|
|
|
|
long Offset = (uint)Swizzle.GetSwizzleOffset(X, Y);
|
|
|
|
|
|
|
|
long PxLow = CpuMem.ReadInt64Unchecked(Position + Offset + 0);
|
|
|
|
long PxHigh = CpuMem.ReadInt64Unchecked(Position + Offset + 8);
|
|
|
|
|
|
|
|
*(long*)(BuffPtr + OutOffs + 0) = PxLow;
|
|
|
|
*(long*)(BuffPtr + OutOffs + 8) = PxHigh;
|
|
|
|
|
|
|
|
OutOffs += 16;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Output;
|
|
|
|
}
|
|
|
|
|
2018-06-24 02:39:25 +02:00
|
|
|
private unsafe static byte[] Read8Bpt4x4(IAMemory Memory, TextureInfo Texture)
|
2018-04-08 21:17:35 +02:00
|
|
|
{
|
|
|
|
int Width = (Texture.Width + 3) / 4;
|
|
|
|
int Height = (Texture.Height + 3) / 4;
|
|
|
|
|
|
|
|
byte[] Output = new byte[Width * Height * 8];
|
|
|
|
|
2018-04-26 04:11:26 +02:00
|
|
|
ISwizzle Swizzle = TextureHelper.GetSwizzle(Texture, Width, 8);
|
2018-04-08 21:17:35 +02:00
|
|
|
|
NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 20:53:23 +02:00
|
|
|
(AMemory CpuMem, long Position) = TextureHelper.GetMemoryAndPosition(
|
|
|
|
Memory,
|
|
|
|
Texture.Position);
|
|
|
|
|
2018-04-08 21:17:35 +02:00
|
|
|
fixed (byte* BuffPtr = Output)
|
|
|
|
{
|
|
|
|
long OutOffs = 0;
|
|
|
|
|
|
|
|
for (int Y = 0; Y < Height; Y++)
|
|
|
|
for (int X = 0; X < Width; X++)
|
|
|
|
{
|
|
|
|
long Offset = (uint)Swizzle.GetSwizzleOffset(X, Y);
|
|
|
|
|
NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 20:53:23 +02:00
|
|
|
long Tile = CpuMem.ReadInt64Unchecked(Position + Offset);
|
2018-04-08 21:17:35 +02:00
|
|
|
|
|
|
|
*(long*)(BuffPtr + OutOffs) = Tile;
|
|
|
|
|
|
|
|
OutOffs += 8;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Output;
|
|
|
|
}
|
|
|
|
|
2018-06-24 02:39:25 +02:00
|
|
|
private unsafe static byte[] Read16Bpt4x4(IAMemory Memory, TextureInfo Texture)
|
2018-04-08 21:17:35 +02:00
|
|
|
{
|
|
|
|
int Width = (Texture.Width + 3) / 4;
|
|
|
|
int Height = (Texture.Height + 3) / 4;
|
|
|
|
|
|
|
|
byte[] Output = new byte[Width * Height * 16];
|
|
|
|
|
2018-04-26 04:11:26 +02:00
|
|
|
ISwizzle Swizzle = TextureHelper.GetSwizzle(Texture, Width, 16);
|
2018-04-08 21:17:35 +02:00
|
|
|
|
NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 20:53:23 +02:00
|
|
|
(AMemory CpuMem, long Position) = TextureHelper.GetMemoryAndPosition(
|
|
|
|
Memory,
|
|
|
|
Texture.Position);
|
|
|
|
|
2018-04-08 21:17:35 +02:00
|
|
|
fixed (byte* BuffPtr = Output)
|
|
|
|
{
|
|
|
|
long OutOffs = 0;
|
|
|
|
|
|
|
|
for (int Y = 0; Y < Height; Y++)
|
|
|
|
for (int X = 0; X < Width; X++)
|
|
|
|
{
|
|
|
|
long Offset = (uint)Swizzle.GetSwizzleOffset(X, Y);
|
|
|
|
|
NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 20:53:23 +02:00
|
|
|
long Tile0 = CpuMem.ReadInt64Unchecked(Position + Offset + 0);
|
|
|
|
long Tile1 = CpuMem.ReadInt64Unchecked(Position + Offset + 8);
|
2018-04-08 21:17:35 +02:00
|
|
|
|
|
|
|
*(long*)(BuffPtr + OutOffs + 0) = Tile0;
|
|
|
|
*(long*)(BuffPtr + OutOffs + 8) = Tile1;
|
|
|
|
|
|
|
|
OutOffs += 16;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Output;
|
|
|
|
}
|
|
|
|
}
|
2018-04-08 23:08:36 +02:00
|
|
|
}
|