Ryujinx/src/Ryujinx.Graphics.Metal/Texture.cs

263 lines
9 KiB
C#
Raw Normal View History

2023-08-02 02:36:07 +00:00
using Ryujinx.Common.Logging;
using Ryujinx.Common.Memory;
using Ryujinx.Graphics.GAL;
using SharpMetal.Metal;
using System;
2023-07-29 03:50:00 +00:00
using System.Runtime.CompilerServices;
using System.Runtime.Versioning;
namespace Ryujinx.Graphics.Metal
{
[SupportedOSPlatform("macos")]
2023-08-03 18:50:49 +00:00
class Texture : ITexture, IDisposable
{
private readonly TextureCreateInfo _info;
2023-07-28 02:54:24 +00:00
private readonly Pipeline _pipeline;
2023-07-28 12:20:15 +00:00
private readonly MTLDevice _device;
public MTLTexture MTLTexture;
2023-07-28 19:04:35 +00:00
public TextureCreateInfo Info => _info;
public int Width => Info.Width;
public int Height => Info.Height;
2023-07-29 04:14:21 +00:00
public int Depth => Info.Depth;
2023-07-28 15:55:52 +00:00
public Texture(MTLDevice device, Pipeline pipeline, TextureCreateInfo info)
{
2023-07-28 12:20:15 +00:00
_device = device;
_pipeline = pipeline;
_info = info;
2023-08-03 18:50:49 +00:00
var descriptor = new MTLTextureDescriptor
{
PixelFormat = FormatTable.GetFormat(Info.Format),
2023-08-14 11:12:44 +00:00
Usage = MTLTextureUsage.ShaderRead,
2023-08-03 18:50:49 +00:00
Width = (ulong)Width,
Height = (ulong)Height,
Depth = (ulong)Depth
};
descriptor.Depth = (ulong)Info.Depth;
descriptor.SampleCount = (ulong)Info.Samples;
2023-07-28 02:54:24 +00:00
descriptor.MipmapLevelCount = (ulong)Info.Levels;
descriptor.TextureType = Info.Target.Convert();
descriptor.Swizzle = new MTLTextureSwizzleChannels
{
red = Info.SwizzleR.Convert(),
green = Info.SwizzleG.Convert(),
blue = Info.SwizzleB.Convert(),
alpha = Info.SwizzleA.Convert()
};
2023-07-28 12:20:15 +00:00
MTLTexture = _device.NewTexture(descriptor);
}
public void CopyTo(ITexture destination, int firstLayer, int firstLevel)
{
var blitCommandEncoder = _pipeline.GetOrCreateBlitEncoder();
2023-07-28 20:23:13 +00:00
2023-07-28 02:54:24 +00:00
if (destination is Texture destinationTexture)
{
2023-07-28 20:23:13 +00:00
blitCommandEncoder.CopyFromTexture(
2023-07-28 02:54:24 +00:00
MTLTexture,
(ulong)firstLayer,
(ulong)firstLevel,
destinationTexture.MTLTexture,
(ulong)firstLayer,
(ulong)firstLevel,
MTLTexture.ArrayLength,
MTLTexture.MipmapLevelCount);
}
}
public void CopyTo(ITexture destination, int srcLayer, int dstLayer, int srcLevel, int dstLevel)
{
var blitCommandEncoder = _pipeline.GetOrCreateBlitEncoder();
2023-07-28 20:23:13 +00:00
2023-07-28 02:54:24 +00:00
if (destination is Texture destinationTexture)
{
2023-07-28 20:23:13 +00:00
blitCommandEncoder.CopyFromTexture(
2023-07-28 02:54:24 +00:00
MTLTexture,
(ulong)srcLayer,
(ulong)srcLevel,
destinationTexture.MTLTexture,
(ulong)dstLayer,
(ulong)dstLevel,
MTLTexture.ArrayLength,
MTLTexture.MipmapLevelCount);
}
}
public void CopyTo(ITexture destination, Extents2D srcRegion, Extents2D dstRegion, bool linearFilter)
{
2023-08-03 01:32:41 +00:00
Logger.Warning?.Print(LogClass.Gpu, "Not Implemented!");
}
public void CopyTo(BufferRange range, int layer, int level, int stride)
{
var blitCommandEncoder = _pipeline.GetOrCreateBlitEncoder();
2023-07-29 03:50:00 +00:00
ulong bytesPerRow = (ulong)Info.GetMipStride(level);
ulong bytesPerImage = 0;
if (MTLTexture.TextureType == MTLTextureType.Type3D)
{
bytesPerImage = bytesPerRow * (ulong)Info.Height;
}
var handle = range.Handle;
MTLBuffer mtlBuffer = new(Unsafe.As<BufferHandle, IntPtr>(ref handle));
blitCommandEncoder.CopyFromTexture(
MTLTexture,
(ulong)layer,
(ulong)level,
new MTLOrigin(),
new MTLSize { width = MTLTexture.Width, height = MTLTexture.Height, depth = MTLTexture.Depth },
mtlBuffer,
(ulong)range.Offset,
bytesPerRow,
bytesPerImage);
}
public ITexture CreateView(TextureCreateInfo info, int firstLayer, int firstLevel)
{
2023-08-14 11:17:22 +00:00
Logger.Warning?.Print(LogClass.Gpu, "Not Implemented!");
return this;
}
public PinnedSpan<byte> GetData()
{
throw new NotImplementedException();
}
public PinnedSpan<byte> GetData(int layer, int level)
{
throw new NotImplementedException();
}
2023-08-02 19:57:57 +00:00
// TODO: Handle array formats
public unsafe void SetData(SpanOrArray<byte> data)
{
var blitCommandEncoder = _pipeline.GetOrCreateBlitEncoder();
2023-08-02 19:57:57 +00:00
var dataSpan = data.Span;
var mtlBuffer = _device.NewBuffer((ulong)dataSpan.Length, MTLResourceOptions.ResourceStorageModeShared);
var bufferSpan = new Span<byte>(mtlBuffer.Contents.ToPointer(), dataSpan.Length);
dataSpan.CopyTo(bufferSpan);
int width = Info.Width;
int height = Info.Height;
int depth = Info.Depth;
int levels = Info.GetLevelsClamped();
int offset = 0;
for (int level = 0; level < levels; level++)
{
int mipSize = Info.GetMipSize(level);
int endOffset = offset + mipSize;
if ((uint)endOffset > (uint)dataSpan.Length)
{
return;
}
blitCommandEncoder.CopyFromBuffer(
mtlBuffer,
(ulong)offset,
(ulong)Info.GetMipStride(level),
(ulong)mipSize,
new MTLSize { width = (ulong)width, height = (ulong)height, depth = (ulong)depth },
MTLTexture,
0,
(ulong)level,
new MTLOrigin()
);
offset += mipSize;
width = Math.Max(1, width >> 1);
height = Math.Max(1, height >> 1);
depth = Math.Max(1, depth >> 1);
}
}
public void SetData(SpanOrArray<byte> data, int layer, int level)
{
var blitCommandEncoder = _pipeline.GetOrCreateBlitEncoder();
2023-07-29 03:35:55 +00:00
ulong bytesPerRow = (ulong)Info.GetMipStride(level);
ulong bytesPerImage = 0;
if (MTLTexture.TextureType == MTLTextureType.Type3D)
{
bytesPerImage = bytesPerRow * (ulong)Info.Height;
}
unsafe
{
var dataSpan = data.Span;
var mtlBuffer = _device.NewBuffer((ulong)dataSpan.Length, MTLResourceOptions.ResourceStorageModeShared);
var bufferSpan = new Span<byte>(mtlBuffer.Contents.ToPointer(), dataSpan.Length);
dataSpan.CopyTo(bufferSpan);
blitCommandEncoder.CopyFromBuffer(
mtlBuffer,
0,
bytesPerRow,
bytesPerImage,
2023-07-31 21:44:01 +00:00
new MTLSize { width = MTLTexture.Width, height = MTLTexture.Height, depth = MTLTexture.Depth },
2023-07-29 03:35:55 +00:00
MTLTexture,
(ulong)layer,
(ulong)level,
new MTLOrigin()
);
}
}
2023-07-28 19:04:35 +00:00
public void SetData(SpanOrArray<byte> data, int layer, int level, Rectangle<int> region)
{
var blitCommandEncoder = _pipeline.GetOrCreateBlitEncoder();
2023-07-29 03:33:09 +00:00
2023-07-28 19:09:40 +00:00
ulong bytesPerRow = (ulong)Info.GetMipStride(level);
2023-07-28 19:04:35 +00:00
ulong bytesPerImage = 0;
if (MTLTexture.TextureType == MTLTextureType.Type3D)
{
bytesPerImage = bytesPerRow * (ulong)Info.Height;
}
unsafe
2023-07-28 15:55:52 +00:00
{
2023-07-29 03:33:09 +00:00
var dataSpan = data.Span;
var mtlBuffer = _device.NewBuffer((ulong)dataSpan.Length, MTLResourceOptions.ResourceStorageModeShared);
var bufferSpan = new Span<byte>(mtlBuffer.Contents.ToPointer(), dataSpan.Length);
dataSpan.CopyTo(bufferSpan);
blitCommandEncoder.CopyFromBuffer(
mtlBuffer,
0,
bytesPerRow,
bytesPerImage,
2023-07-29 03:50:00 +00:00
new MTLSize { width = (ulong)region.Width, height = (ulong)region.Height, depth = 1 },
2023-07-29 03:33:09 +00:00
MTLTexture,
(ulong)layer,
(ulong)level,
new MTLOrigin { x = (ulong)region.X, y = (ulong)region.Y }
);
2023-07-28 15:55:52 +00:00
}
}
public void SetStorage(BufferRange buffer)
{
2023-08-02 02:36:07 +00:00
Logger.Warning?.Print(LogClass.Gpu, "Not Implemented!");
}
public void Release()
{
2023-08-02 02:36:07 +00:00
Logger.Warning?.Print(LogClass.Gpu, "Not Implemented!");
}
public void Dispose()
{
2023-08-02 02:36:07 +00:00
Logger.Warning?.Print(LogClass.Gpu, "Not Implemented!");
}
}
}