2019-12-29 18:41:50 +01:00
|
|
|
using Ryujinx.Graphics.GAL;
|
2019-10-18 04:41:18 +02:00
|
|
|
using Ryujinx.Graphics.Gpu.Image;
|
2019-11-14 19:26:40 +01:00
|
|
|
using Ryujinx.Graphics.Gpu.Shader;
|
2019-10-13 08:02:07 +02:00
|
|
|
using Ryujinx.Graphics.Gpu.State;
|
|
|
|
using Ryujinx.Graphics.Shader;
|
|
|
|
using System;
|
|
|
|
|
|
|
|
namespace Ryujinx.Graphics.Gpu.Engine
|
|
|
|
{
|
|
|
|
partial class Methods
|
|
|
|
{
|
2019-12-31 20:19:44 +01:00
|
|
|
/// <summary>
|
|
|
|
/// Dispatches compute work.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="state">Current GPU state</param>
|
|
|
|
/// <param name="argument">Method call argument</param>
|
2019-11-22 03:46:14 +01:00
|
|
|
public void Dispatch(GpuState state, int argument)
|
2019-10-13 08:02:07 +02:00
|
|
|
{
|
2020-02-02 04:25:52 +01:00
|
|
|
uint qmdAddress = (uint)state.Get<int>(MethodOffset.DispatchParamsAddress);
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2020-07-25 08:39:45 +02:00
|
|
|
var qmd = _context.MemoryManager.Read<ComputeQmd>((ulong)qmdAddress << 8);
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2019-11-22 03:46:14 +01:00
|
|
|
GpuVa shaderBaseAddress = state.Get<GpuVa>(MethodOffset.ShaderBaseAddress);
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2020-02-02 04:25:52 +01:00
|
|
|
ulong shaderGpuVa = shaderBaseAddress.Pack() + (uint)qmd.ProgramOffset;
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2020-02-02 04:25:52 +01:00
|
|
|
int localMemorySize = qmd.ShaderLocalMemoryLowSize + qmd.ShaderLocalMemoryHighSize;
|
|
|
|
|
|
|
|
int sharedMemorySize = Math.Min(qmd.SharedMemorySize, _context.Capabilities.MaximumComputeSharedMemorySize);
|
2019-12-09 22:57:49 +01:00
|
|
|
|
2020-04-22 01:35:28 +02:00
|
|
|
uint sbEnableMask = 0;
|
|
|
|
uint ubEnableMask = 0;
|
|
|
|
|
|
|
|
for (int index = 0; index < Constants.TotalCpUniformBuffers; index++)
|
|
|
|
{
|
|
|
|
if (!qmd.ConstantBufferValid(index))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
ubEnableMask |= 1u << index;
|
|
|
|
|
|
|
|
ulong gpuVa = (uint)qmd.ConstantBufferAddrLower(index) | (ulong)qmd.ConstantBufferAddrUpper(index) << 32;
|
|
|
|
ulong size = (ulong)qmd.ConstantBufferSize(index);
|
|
|
|
|
|
|
|
BufferManager.SetComputeUniformBuffer(index, gpuVa, size);
|
|
|
|
}
|
|
|
|
|
2020-05-06 03:02:28 +02:00
|
|
|
ShaderBundle cs = ShaderCache.GetComputeShader(
|
2020-04-22 01:35:28 +02:00
|
|
|
state,
|
2019-10-13 08:02:07 +02:00
|
|
|
shaderGpuVa,
|
2020-02-02 04:25:52 +01:00
|
|
|
qmd.CtaThreadDimension0,
|
|
|
|
qmd.CtaThreadDimension1,
|
|
|
|
qmd.CtaThreadDimension2,
|
|
|
|
localMemorySize,
|
|
|
|
sharedMemorySize);
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2019-12-29 18:41:50 +01:00
|
|
|
_context.Renderer.Pipeline.SetProgram(cs.HostProgram);
|
2019-10-18 04:41:18 +02:00
|
|
|
|
2019-11-22 03:46:14 +01:00
|
|
|
var samplerPool = state.Get<PoolState>(MethodOffset.SamplerPoolState);
|
2019-10-18 04:41:18 +02:00
|
|
|
|
2020-02-02 04:25:52 +01:00
|
|
|
TextureManager.SetComputeSamplerPool(samplerPool.Address.Pack(), samplerPool.MaximumId, qmd.SamplerIndex);
|
2019-10-18 04:41:18 +02:00
|
|
|
|
2019-11-22 03:46:14 +01:00
|
|
|
var texturePool = state.Get<PoolState>(MethodOffset.TexturePoolState);
|
2019-10-18 04:41:18 +02:00
|
|
|
|
2019-12-29 18:41:50 +01:00
|
|
|
TextureManager.SetComputeTexturePool(texturePool.Address.Pack(), texturePool.MaximumId);
|
2019-10-18 04:41:18 +02:00
|
|
|
|
2019-12-29 18:41:50 +01:00
|
|
|
TextureManager.SetComputeTextureBufferIndex(state.Get<int>(MethodOffset.TextureBufferIndex));
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2020-07-12 05:07:01 +02:00
|
|
|
ShaderProgramInfo info = cs.Shaders[0].Program.Info;
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2020-02-11 01:10:05 +01:00
|
|
|
for (int index = 0; index < info.CBuffers.Count; index++)
|
|
|
|
{
|
|
|
|
BufferDescriptor cb = info.CBuffers[index];
|
|
|
|
|
|
|
|
// NVN uses the "hardware" constant buffer for anything that is less than 8,
|
|
|
|
// and those are already bound above.
|
|
|
|
// Anything greater than or equal to 8 uses the emulated constant buffers.
|
|
|
|
// They are emulated using global memory loads.
|
|
|
|
if (cb.Slot < 8)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
ubEnableMask |= 1u << cb.Slot;
|
|
|
|
|
|
|
|
ulong cbDescAddress = BufferManager.GetComputeUniformBufferAddress(0);
|
|
|
|
|
2020-06-22 13:48:32 +02:00
|
|
|
int cbDescOffset = 0x260 + (cb.Slot - 8) * 0x10;
|
2020-02-11 01:10:05 +01:00
|
|
|
|
|
|
|
cbDescAddress += (ulong)cbDescOffset;
|
|
|
|
|
2020-05-27 16:07:10 +02:00
|
|
|
SbDescriptor cbDescriptor = _context.PhysicalMemory.Read<SbDescriptor>(cbDescAddress);
|
2020-02-11 01:10:05 +01:00
|
|
|
|
|
|
|
BufferManager.SetComputeUniformBuffer(cb.Slot, cbDescriptor.PackAddress(), (uint)cbDescriptor.Size);
|
|
|
|
}
|
|
|
|
|
2019-10-13 08:02:07 +02:00
|
|
|
for (int index = 0; index < info.SBuffers.Count; index++)
|
|
|
|
{
|
|
|
|
BufferDescriptor sb = info.SBuffers[index];
|
|
|
|
|
|
|
|
sbEnableMask |= 1u << sb.Slot;
|
|
|
|
|
2019-12-29 18:41:50 +01:00
|
|
|
ulong sbDescAddress = BufferManager.GetComputeUniformBufferAddress(0);
|
2019-10-13 08:02:07 +02:00
|
|
|
|
|
|
|
int sbDescOffset = 0x310 + sb.Slot * 0x10;
|
|
|
|
|
|
|
|
sbDescAddress += (ulong)sbDescOffset;
|
|
|
|
|
2020-05-27 16:07:10 +02:00
|
|
|
SbDescriptor sbDescriptor = _context.PhysicalMemory.Read<SbDescriptor>(sbDescAddress);
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2019-12-29 18:41:50 +01:00
|
|
|
BufferManager.SetComputeStorageBuffer(sb.Slot, sbDescriptor.PackAddress(), (uint)sbDescriptor.Size);
|
2019-10-13 08:02:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ubEnableMask = 0;
|
|
|
|
|
|
|
|
for (int index = 0; index < info.CBuffers.Count; index++)
|
|
|
|
{
|
|
|
|
ubEnableMask |= 1u << info.CBuffers[index].Slot;
|
|
|
|
}
|
|
|
|
|
2019-12-29 18:41:50 +01:00
|
|
|
BufferManager.SetComputeStorageBufferEnableMask(sbEnableMask);
|
|
|
|
BufferManager.SetComputeUniformBufferEnableMask(ubEnableMask);
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2019-11-23 00:37:45 +01:00
|
|
|
var textureBindings = new TextureBindingInfo[info.Textures.Count];
|
|
|
|
|
|
|
|
for (int index = 0; index < info.Textures.Count; index++)
|
|
|
|
{
|
|
|
|
var descriptor = info.Textures[index];
|
|
|
|
|
2020-10-21 00:03:20 +02:00
|
|
|
Target target = ShaderTexture.GetTarget(descriptor.Type);
|
2019-11-23 00:37:45 +01:00
|
|
|
|
2019-12-28 02:16:14 +01:00
|
|
|
if (descriptor.IsBindless)
|
|
|
|
{
|
2020-07-07 04:41:07 +02:00
|
|
|
textureBindings[index] = new TextureBindingInfo(target, descriptor.CbufOffset, descriptor.CbufSlot, descriptor.Flags);
|
2019-12-28 02:16:14 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-07-07 04:41:07 +02:00
|
|
|
textureBindings[index] = new TextureBindingInfo(target, descriptor.HandleIndex, descriptor.Flags);
|
2019-12-28 02:16:14 +01:00
|
|
|
}
|
2019-11-23 00:37:45 +01:00
|
|
|
}
|
|
|
|
|
2019-12-29 18:41:50 +01:00
|
|
|
TextureManager.SetComputeTextures(textureBindings);
|
2019-11-23 00:37:45 +01:00
|
|
|
|
|
|
|
var imageBindings = new TextureBindingInfo[info.Images.Count];
|
|
|
|
|
|
|
|
for (int index = 0; index < info.Images.Count; index++)
|
|
|
|
{
|
|
|
|
var descriptor = info.Images[index];
|
|
|
|
|
2020-10-21 00:03:20 +02:00
|
|
|
Target target = ShaderTexture.GetTarget(descriptor.Type);
|
|
|
|
Format format = ShaderTexture.GetFormat(descriptor.Format);
|
2019-11-23 00:37:45 +01:00
|
|
|
|
2020-10-21 00:03:20 +02:00
|
|
|
imageBindings[index] = new TextureBindingInfo(target, format, descriptor.HandleIndex, descriptor.Flags);
|
2019-11-23 00:37:45 +01:00
|
|
|
}
|
|
|
|
|
2019-12-29 18:41:50 +01:00
|
|
|
TextureManager.SetComputeImages(imageBindings);
|
2019-11-23 00:37:45 +01:00
|
|
|
|
2019-12-29 18:41:50 +01:00
|
|
|
BufferManager.CommitComputeBindings();
|
|
|
|
TextureManager.CommitComputeBindings();
|
2019-10-13 08:02:07 +02:00
|
|
|
|
2019-12-29 18:41:50 +01:00
|
|
|
_context.Renderer.Pipeline.DispatchCompute(
|
2020-02-02 04:25:52 +01:00
|
|
|
qmd.CtaRasterWidth,
|
|
|
|
qmd.CtaRasterHeight,
|
|
|
|
qmd.CtaRasterDepth);
|
2019-10-26 19:50:52 +02:00
|
|
|
|
2020-07-04 00:30:41 +02:00
|
|
|
_forceShaderUpdate = true;
|
2019-10-13 08:02:07 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|