Compare commits
5 commits
a285136c1b
...
f6f4c3829a
Author | SHA1 | Date | |
---|---|---|---|
|
f6f4c3829a | ||
|
e194f91245 | ||
|
564e82edd7 | ||
|
6ce49a2dc7 | ||
|
ccd330ba0f |
6 changed files with 119 additions and 168 deletions
|
@ -54,10 +54,10 @@ namespace Ryujinx.Graphics.Metal
|
||||||
record struct TextureRef
|
record struct TextureRef
|
||||||
{
|
{
|
||||||
public ShaderStage Stage;
|
public ShaderStage Stage;
|
||||||
public Texture Storage;
|
public TextureBase Storage;
|
||||||
public Sampler Sampler;
|
public Sampler Sampler;
|
||||||
|
|
||||||
public TextureRef(ShaderStage stage, Texture storage, Sampler sampler)
|
public TextureRef(ShaderStage stage, TextureBase storage, Sampler sampler)
|
||||||
{
|
{
|
||||||
Stage = stage;
|
Stage = stage;
|
||||||
Storage = storage;
|
Storage = storage;
|
||||||
|
|
|
@ -805,13 +805,9 @@ namespace Ryujinx.Graphics.Metal
|
||||||
|
|
||||||
public readonly void UpdateTextureAndSampler(ShaderStage stage, ulong binding, TextureBase texture, Sampler sampler)
|
public readonly void UpdateTextureAndSampler(ShaderStage stage, ulong binding, TextureBase texture, Sampler sampler)
|
||||||
{
|
{
|
||||||
if (texture is TextureBuffer)
|
if (texture != null)
|
||||||
{
|
{
|
||||||
// TODO: Texture buffers
|
_currentState.TextureRefs[binding] = new(stage, texture, sampler);
|
||||||
}
|
|
||||||
else if (texture is Texture view)
|
|
||||||
{
|
|
||||||
_currentState.TextureRefs[binding] = new(stage, view, sampler);
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -1124,59 +1120,52 @@ namespace Ryujinx.Graphics.Metal
|
||||||
case MetalRenderer.TextureSetIndex:
|
case MetalRenderer.TextureSetIndex:
|
||||||
if (!segment.IsArray)
|
if (!segment.IsArray)
|
||||||
{
|
{
|
||||||
if (segment.Type != ResourceType.BufferTexture)
|
for (int i = 0; i < count; i++)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < count; i++)
|
int index = binding + i;
|
||||||
|
|
||||||
|
ref var texture = ref _currentState.TextureRefs[index];
|
||||||
|
|
||||||
|
var storage = texture.Storage;
|
||||||
|
|
||||||
|
if (storage == null)
|
||||||
{
|
{
|
||||||
int index = binding + i;
|
continue;
|
||||||
|
|
||||||
ref var texture = ref _currentState.TextureRefs[index];
|
|
||||||
|
|
||||||
var storage = texture.Storage;
|
|
||||||
|
|
||||||
if (storage == null)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
var mtlTexture = storage.GetHandle();
|
|
||||||
|
|
||||||
MTLRenderStages renderStages = 0;
|
|
||||||
|
|
||||||
if ((segment.Stages & ResourceStages.Vertex) != 0)
|
|
||||||
{
|
|
||||||
vertResourceIds[vertResourceIdIndex] = mtlTexture.GpuResourceID._impl;
|
|
||||||
vertResourceIdIndex++;
|
|
||||||
|
|
||||||
if (texture.Sampler != null)
|
|
||||||
{
|
|
||||||
vertResourceIds[vertResourceIdIndex] = texture.Sampler.GetSampler().GpuResourceID._impl;
|
|
||||||
vertResourceIdIndex++;
|
|
||||||
}
|
|
||||||
|
|
||||||
renderStages |= MTLRenderStages.RenderStageVertex;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((segment.Stages & ResourceStages.Fragment) != 0)
|
|
||||||
{
|
|
||||||
fragResourceIds[fragResourceIdIndex] = mtlTexture.GpuResourceID._impl;
|
|
||||||
fragResourceIdIndex++;
|
|
||||||
|
|
||||||
if (texture.Sampler != null)
|
|
||||||
{
|
|
||||||
fragResourceIds[fragResourceIdIndex] = texture.Sampler.GetSampler().GpuResourceID._impl;
|
|
||||||
fragResourceIdIndex++;
|
|
||||||
}
|
|
||||||
|
|
||||||
renderStages |= MTLRenderStages.RenderStageFragment;
|
|
||||||
}
|
|
||||||
|
|
||||||
renderCommandEncoder.UseResource(new MTLResource(mtlTexture.NativePtr), MTLResourceUsage.Read, renderStages);
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
else
|
var mtlTexture = storage.GetHandle();
|
||||||
{
|
|
||||||
// TODO: Buffer textures
|
MTLRenderStages renderStages = 0;
|
||||||
|
|
||||||
|
if ((segment.Stages & ResourceStages.Vertex) != 0)
|
||||||
|
{
|
||||||
|
vertResourceIds[vertResourceIdIndex] = mtlTexture.GpuResourceID._impl;
|
||||||
|
vertResourceIdIndex++;
|
||||||
|
|
||||||
|
if (texture.Sampler != null)
|
||||||
|
{
|
||||||
|
vertResourceIds[vertResourceIdIndex] = texture.Sampler.GetSampler().GpuResourceID._impl;
|
||||||
|
vertResourceIdIndex++;
|
||||||
|
}
|
||||||
|
|
||||||
|
renderStages |= MTLRenderStages.RenderStageVertex;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((segment.Stages & ResourceStages.Fragment) != 0)
|
||||||
|
{
|
||||||
|
fragResourceIds[fragResourceIdIndex] = mtlTexture.GpuResourceID._impl;
|
||||||
|
fragResourceIdIndex++;
|
||||||
|
|
||||||
|
if (texture.Sampler != null)
|
||||||
|
{
|
||||||
|
fragResourceIds[fragResourceIdIndex] = texture.Sampler.GetSampler().GpuResourceID._impl;
|
||||||
|
fragResourceIdIndex++;
|
||||||
|
}
|
||||||
|
|
||||||
|
renderStages |= MTLRenderStages.RenderStageFragment;
|
||||||
|
}
|
||||||
|
|
||||||
|
renderCommandEncoder.UseResource(new MTLResource(mtlTexture.NativePtr), MTLResourceUsage.Read, renderStages);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -1343,41 +1332,34 @@ namespace Ryujinx.Graphics.Metal
|
||||||
case MetalRenderer.TextureSetIndex:
|
case MetalRenderer.TextureSetIndex:
|
||||||
if (!segment.IsArray)
|
if (!segment.IsArray)
|
||||||
{
|
{
|
||||||
if (segment.Type != ResourceType.BufferTexture)
|
for (int i = 0; i < count; i++)
|
||||||
{
|
{
|
||||||
for (int i = 0; i < count; i++)
|
int index = binding + i;
|
||||||
|
|
||||||
|
ref var texture = ref _currentState.TextureRefs[index];
|
||||||
|
|
||||||
|
var storage = texture.Storage;
|
||||||
|
|
||||||
|
if (storage == null)
|
||||||
{
|
{
|
||||||
int index = binding + i;
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
ref var texture = ref _currentState.TextureRefs[index];
|
var mtlTexture = storage.GetHandle();
|
||||||
|
|
||||||
var storage = texture.Storage;
|
if (segment.Stages.HasFlag(ResourceStages.Compute))
|
||||||
|
{
|
||||||
|
computeCommandEncoder.UseResource(new MTLResource(mtlTexture.NativePtr), MTLResourceUsage.Read);
|
||||||
|
resourceIds[resourceIdIndex] = mtlTexture.GpuResourceID._impl;
|
||||||
|
resourceIdIndex++;
|
||||||
|
|
||||||
if (storage == null)
|
if (texture.Sampler != null)
|
||||||
{
|
{
|
||||||
continue;
|
resourceIds[resourceIdIndex] = texture.Sampler.GetSampler().GpuResourceID._impl;
|
||||||
}
|
|
||||||
|
|
||||||
var mtlTexture = storage.GetHandle();
|
|
||||||
|
|
||||||
if (segment.Stages.HasFlag(ResourceStages.Compute))
|
|
||||||
{
|
|
||||||
computeCommandEncoder.UseResource(new MTLResource(mtlTexture.NativePtr), MTLResourceUsage.Read);
|
|
||||||
resourceIds[resourceIdIndex] = mtlTexture.GpuResourceID._impl;
|
|
||||||
resourceIdIndex++;
|
resourceIdIndex++;
|
||||||
|
|
||||||
if (texture.Sampler != null)
|
|
||||||
{
|
|
||||||
resourceIds[resourceIdIndex] = texture.Sampler.GetSampler().GpuResourceID._impl;
|
|
||||||
resourceIdIndex++;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
|
||||||
// TODO: Buffer textures
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
|
|
@ -28,7 +28,6 @@ namespace Ryujinx.Graphics.Metal
|
||||||
private MTLComputePipelineState? _computePipelineCache;
|
private MTLComputePipelineState? _computePipelineCache;
|
||||||
private bool _firstBackgroundUse;
|
private bool _firstBackgroundUse;
|
||||||
|
|
||||||
public ResourceBindingSegment[][] ClearSegments { get; }
|
|
||||||
public ResourceBindingSegment[][] BindingSegments { get; }
|
public ResourceBindingSegment[][] BindingSegments { get; }
|
||||||
// Argument buffer sizes for Vertex or Compute stages
|
// Argument buffer sizes for Vertex or Compute stages
|
||||||
public int[] ArgumentBufferSizes { get; }
|
public int[] ArgumentBufferSizes { get; }
|
||||||
|
@ -53,7 +52,6 @@ namespace Ryujinx.Graphics.Metal
|
||||||
_handles[i] = device.NewLibrary(StringHelper.NSString(shader.Code), compileOptions, (library, error) => CompilationResultHandler(library, error, index));
|
_handles[i] = device.NewLibrary(StringHelper.NSString(shader.Code), compileOptions, (library, error) => CompilationResultHandler(library, error, index));
|
||||||
}
|
}
|
||||||
|
|
||||||
ClearSegments = BuildClearSegments(resourceLayout.Sets);
|
|
||||||
(BindingSegments, ArgumentBufferSizes, FragArgumentBufferSizes) = BuildBindingSegments(resourceLayout.SetUsages);
|
(BindingSegments, ArgumentBufferSizes, FragArgumentBufferSizes) = BuildBindingSegments(resourceLayout.SetUsages);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -98,62 +96,6 @@ namespace Ryujinx.Graphics.Metal
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static ResourceBindingSegment[][] BuildClearSegments(ReadOnlyCollection<ResourceDescriptorCollection> sets)
|
|
||||||
{
|
|
||||||
ResourceBindingSegment[][] segments = new ResourceBindingSegment[sets.Count][];
|
|
||||||
|
|
||||||
for (int setIndex = 0; setIndex < sets.Count; setIndex++)
|
|
||||||
{
|
|
||||||
List<ResourceBindingSegment> currentSegments = new();
|
|
||||||
|
|
||||||
ResourceDescriptor currentDescriptor = default;
|
|
||||||
int currentCount = 0;
|
|
||||||
|
|
||||||
for (int index = 0; index < sets[setIndex].Descriptors.Count; index++)
|
|
||||||
{
|
|
||||||
ResourceDescriptor descriptor = sets[setIndex].Descriptors[index];
|
|
||||||
|
|
||||||
if (currentDescriptor.Binding + currentCount != descriptor.Binding ||
|
|
||||||
currentDescriptor.Type != descriptor.Type ||
|
|
||||||
currentDescriptor.Stages != descriptor.Stages ||
|
|
||||||
currentDescriptor.Count > 1 ||
|
|
||||||
descriptor.Count > 1)
|
|
||||||
{
|
|
||||||
if (currentCount != 0)
|
|
||||||
{
|
|
||||||
currentSegments.Add(new ResourceBindingSegment(
|
|
||||||
currentDescriptor.Binding,
|
|
||||||
currentCount,
|
|
||||||
currentDescriptor.Type,
|
|
||||||
currentDescriptor.Stages,
|
|
||||||
currentDescriptor.Count > 1));
|
|
||||||
}
|
|
||||||
|
|
||||||
currentDescriptor = descriptor;
|
|
||||||
currentCount = descriptor.Count;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
currentCount += descriptor.Count;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (currentCount != 0)
|
|
||||||
{
|
|
||||||
currentSegments.Add(new ResourceBindingSegment(
|
|
||||||
currentDescriptor.Binding,
|
|
||||||
currentCount,
|
|
||||||
currentDescriptor.Type,
|
|
||||||
currentDescriptor.Stages,
|
|
||||||
currentDescriptor.Count > 1));
|
|
||||||
}
|
|
||||||
|
|
||||||
segments[setIndex] = currentSegments.ToArray();
|
|
||||||
}
|
|
||||||
|
|
||||||
return segments;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static (ResourceBindingSegment[][], int[], int[]) BuildBindingSegments(ReadOnlyCollection<ResourceUsageCollection> setUsages)
|
private static (ResourceBindingSegment[][], int[], int[]) BuildBindingSegments(ReadOnlyCollection<ResourceUsageCollection> setUsages)
|
||||||
{
|
{
|
||||||
ResourceBindingSegment[][] segments = new ResourceBindingSegment[setUsages.Count][];
|
ResourceBindingSegment[][] segments = new ResourceBindingSegment[setUsages.Count][];
|
||||||
|
|
|
@ -667,8 +667,36 @@ namespace Ryujinx.Graphics.Vulkan
|
||||||
|
|
||||||
if (PrepareOutputBuffer(cbs, hostSize, buffer, out VkBuffer copyToBuffer, out BufferHolder tempCopyHolder))
|
if (PrepareOutputBuffer(cbs, hostSize, buffer, out VkBuffer copyToBuffer, out BufferHolder tempCopyHolder))
|
||||||
{
|
{
|
||||||
|
// No barrier necessary, as this is a temporary copy buffer.
|
||||||
offset = 0;
|
offset = 0;
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
BufferHolder.InsertBufferBarrier(
|
||||||
|
_gd,
|
||||||
|
cbs.CommandBuffer,
|
||||||
|
copyToBuffer,
|
||||||
|
BufferHolder.DefaultAccessFlags,
|
||||||
|
AccessFlags.TransferWriteBit,
|
||||||
|
PipelineStageFlags.AllCommandsBit,
|
||||||
|
PipelineStageFlags.TransferBit,
|
||||||
|
offset,
|
||||||
|
outSize);
|
||||||
|
}
|
||||||
|
|
||||||
|
InsertImageBarrier(
|
||||||
|
_gd.Api,
|
||||||
|
cbs.CommandBuffer,
|
||||||
|
image,
|
||||||
|
TextureStorage.DefaultAccessMask,
|
||||||
|
AccessFlags.TransferReadBit,
|
||||||
|
PipelineStageFlags.AllCommandsBit,
|
||||||
|
PipelineStageFlags.TransferBit,
|
||||||
|
Info.Format.ConvertAspectFlags(),
|
||||||
|
FirstLayer + layer,
|
||||||
|
FirstLevel + level,
|
||||||
|
1,
|
||||||
|
1);
|
||||||
|
|
||||||
CopyFromOrToBuffer(cbs.CommandBuffer, copyToBuffer, image, hostSize, true, layer, level, 1, 1, singleSlice: true, offset, stride);
|
CopyFromOrToBuffer(cbs.CommandBuffer, copyToBuffer, image, hostSize, true, layer, level, 1, 1, singleSlice: true, offset, stride);
|
||||||
|
|
||||||
|
@ -677,6 +705,19 @@ namespace Ryujinx.Graphics.Vulkan
|
||||||
CopyDataToOutputBuffer(cbs, tempCopyHolder, autoBuffer, hostSize, range.Offset);
|
CopyDataToOutputBuffer(cbs, tempCopyHolder, autoBuffer, hostSize, range.Offset);
|
||||||
tempCopyHolder.Dispose();
|
tempCopyHolder.Dispose();
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
BufferHolder.InsertBufferBarrier(
|
||||||
|
_gd,
|
||||||
|
cbs.CommandBuffer,
|
||||||
|
copyToBuffer,
|
||||||
|
AccessFlags.TransferWriteBit,
|
||||||
|
BufferHolder.DefaultAccessFlags,
|
||||||
|
PipelineStageFlags.TransferBit,
|
||||||
|
PipelineStageFlags.AllCommandsBit,
|
||||||
|
offset,
|
||||||
|
outSize);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private ReadOnlySpan<byte> GetData(CommandBufferPool cbp, PersistentFlushBuffer flushBuffer)
|
private ReadOnlySpan<byte> GetData(CommandBufferPool cbp, PersistentFlushBuffer flushBuffer)
|
||||||
|
|
|
@ -1,21 +1,20 @@
|
||||||
using LibHac.Ns;
|
|
||||||
using Ryujinx.Ava.Common.Locale;
|
using Ryujinx.Ava.Common.Locale;
|
||||||
|
|
||||||
namespace Ryujinx.Ava.UI.Models
|
namespace Ryujinx.Ava.UI.Models
|
||||||
{
|
{
|
||||||
public class TitleUpdateModel
|
public class TitleUpdateModel
|
||||||
{
|
{
|
||||||
public ApplicationControlProperty Control { get; }
|
public uint Version { get; }
|
||||||
public string Path { get; }
|
public string Path { get; }
|
||||||
|
public string Label { get; }
|
||||||
|
|
||||||
public string Label => LocaleManager.Instance.UpdateAndGetDynamicValue(
|
public TitleUpdateModel(uint version, string displayVersion, string path)
|
||||||
System.IO.Path.GetExtension(Path)?.ToLower() == ".xci" ? LocaleKeys.TitleBundledUpdateVersionLabel : LocaleKeys.TitleUpdateVersionLabel,
|
|
||||||
Control.DisplayVersionString.ToString()
|
|
||||||
);
|
|
||||||
|
|
||||||
public TitleUpdateModel(ApplicationControlProperty control, string path)
|
|
||||||
{
|
{
|
||||||
Control = control;
|
Version = version;
|
||||||
|
Label = LocaleManager.Instance.UpdateAndGetDynamicValue(
|
||||||
|
System.IO.Path.GetExtension(path)?.ToLower() == ".xci" ? LocaleKeys.TitleBundledUpdateVersionLabel : LocaleKeys.TitleUpdateVersionLabel,
|
||||||
|
displayVersion
|
||||||
|
);
|
||||||
Path = path;
|
Path = path;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -131,26 +131,11 @@ namespace Ryujinx.Ava.UI.ViewModels
|
||||||
|
|
||||||
public void SortUpdates()
|
public void SortUpdates()
|
||||||
{
|
{
|
||||||
var list = TitleUpdates.ToList();
|
var sortedUpdates = TitleUpdates.OrderByDescending(update => update.Version);
|
||||||
|
|
||||||
list.Sort((first, second) =>
|
|
||||||
{
|
|
||||||
if (string.IsNullOrEmpty(first.Control.DisplayVersionString.ToString()))
|
|
||||||
{
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(second.Control.DisplayVersionString.ToString()))
|
|
||||||
{
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
return Version.Parse(first.Control.DisplayVersionString.ToString()).CompareTo(Version.Parse(second.Control.DisplayVersionString.ToString())) * -1;
|
|
||||||
});
|
|
||||||
|
|
||||||
Views.Clear();
|
Views.Clear();
|
||||||
Views.Add(new BaseModel());
|
Views.Add(new BaseModel());
|
||||||
Views.AddRange(list);
|
Views.AddRange(sortedUpdates);
|
||||||
|
|
||||||
if (SelectedUpdate == null)
|
if (SelectedUpdate == null)
|
||||||
{
|
{
|
||||||
|
@ -204,7 +189,9 @@ namespace Ryujinx.Ava.UI.ViewModels
|
||||||
controlNca.OpenFileSystem(NcaSectionType.Data, IntegrityCheckLevel.None).OpenFile(ref nacpFile.Ref, "/control.nacp".ToU8Span(), OpenMode.Read).ThrowIfFailure();
|
controlNca.OpenFileSystem(NcaSectionType.Data, IntegrityCheckLevel.None).OpenFile(ref nacpFile.Ref, "/control.nacp".ToU8Span(), OpenMode.Read).ThrowIfFailure();
|
||||||
nacpFile.Get.Read(out _, 0, SpanHelpers.AsByteSpan(ref controlData), ReadOption.None).ThrowIfFailure();
|
nacpFile.Get.Read(out _, 0, SpanHelpers.AsByteSpan(ref controlData), ReadOption.None).ThrowIfFailure();
|
||||||
|
|
||||||
var update = new TitleUpdateModel(controlData, path);
|
var displayVersion = controlData.DisplayVersionString.ToString();
|
||||||
|
var update = new TitleUpdateModel(content.Version.Version, displayVersion, path);
|
||||||
|
|
||||||
TitleUpdates.Add(update);
|
TitleUpdates.Add(update);
|
||||||
|
|
||||||
if (selected)
|
if (selected)
|
||||||
|
|
Loading…
Add table
Reference in a new issue