2019-10-17 22:41:18 -04:00
|
|
|
using Ryujinx.Graphics.GAL.Texture;
|
|
|
|
using Ryujinx.Graphics.Gpu.Image;
|
2019-11-14 13:26:40 -05:00
|
|
|
using Ryujinx.Graphics.Gpu.Shader;
|
2019-10-13 02:02:07 -04:00
|
|
|
using Ryujinx.Graphics.Gpu.State;
|
|
|
|
using Ryujinx.Graphics.Shader;
|
|
|
|
using System;
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
|
|
|
|
namespace Ryujinx.Graphics.Gpu.Engine
|
|
|
|
{
|
|
|
|
partial class Methods
|
|
|
|
{
|
2019-11-21 21:46:14 -05:00
|
|
|
public void Dispatch(GpuState state, int argument)
|
2019-10-13 02:02:07 -04:00
|
|
|
{
|
2019-11-21 21:46:14 -05:00
|
|
|
uint dispatchParamsAddress = (uint)state.Get<int>(MethodOffset.DispatchParamsAddress);
|
2019-10-13 02:02:07 -04:00
|
|
|
|
|
|
|
var dispatchParams = _context.MemoryAccessor.Read<ComputeParams>((ulong)dispatchParamsAddress << 8);
|
|
|
|
|
2019-11-21 21:46:14 -05:00
|
|
|
GpuVa shaderBaseAddress = state.Get<GpuVa>(MethodOffset.ShaderBaseAddress);
|
2019-10-13 02:02:07 -04:00
|
|
|
|
|
|
|
ulong shaderGpuVa = shaderBaseAddress.Pack() + (uint)dispatchParams.ShaderOffset;
|
|
|
|
|
2019-12-09 16:57:49 -05:00
|
|
|
// Note: A size of 0 is also invalid, the size must be at least 1.
|
2019-12-15 23:59:46 -05:00
|
|
|
int sharedMemorySize = Math.Clamp(dispatchParams.SharedMemorySize & 0xffff, 1, _context.Capabilities.MaximumComputeSharedMemorySize);
|
2019-12-09 16:57:49 -05:00
|
|
|
|
2019-10-13 02:02:07 -04:00
|
|
|
ComputeShader cs = _shaderCache.GetComputeShader(
|
|
|
|
shaderGpuVa,
|
2019-12-09 16:57:49 -05:00
|
|
|
sharedMemorySize,
|
2019-10-13 02:02:07 -04:00
|
|
|
dispatchParams.UnpackBlockSizeX(),
|
|
|
|
dispatchParams.UnpackBlockSizeY(),
|
|
|
|
dispatchParams.UnpackBlockSizeZ());
|
|
|
|
|
2019-11-14 13:26:40 -05:00
|
|
|
_context.Renderer.Pipeline.BindProgram(cs.HostProgram);
|
2019-10-17 22:41:18 -04:00
|
|
|
|
2019-11-21 21:46:14 -05:00
|
|
|
var samplerPool = state.Get<PoolState>(MethodOffset.SamplerPoolState);
|
2019-10-17 22:41:18 -04:00
|
|
|
|
2019-12-05 15:34:47 -05:00
|
|
|
_textureManager.SetComputeSamplerPool(samplerPool.Address.Pack(), samplerPool.MaximumId, dispatchParams.SamplerIndex);
|
2019-10-17 22:41:18 -04:00
|
|
|
|
2019-11-21 21:46:14 -05:00
|
|
|
var texturePool = state.Get<PoolState>(MethodOffset.TexturePoolState);
|
2019-10-17 22:41:18 -04:00
|
|
|
|
|
|
|
_textureManager.SetComputeTexturePool(texturePool.Address.Pack(), texturePool.MaximumId);
|
|
|
|
|
2019-11-21 21:46:14 -05:00
|
|
|
_textureManager.SetComputeTextureBufferIndex(state.Get<int>(MethodOffset.TextureBufferIndex));
|
2019-10-13 02:02:07 -04:00
|
|
|
|
2019-11-14 13:26:40 -05:00
|
|
|
ShaderProgramInfo info = cs.Shader.Program.Info;
|
2019-10-13 02:02:07 -04:00
|
|
|
|
|
|
|
uint sbEnableMask = 0;
|
|
|
|
uint ubEnableMask = dispatchParams.UnpackUniformBuffersEnableMask();
|
|
|
|
|
|
|
|
for (int index = 0; index < dispatchParams.UniformBuffers.Length; index++)
|
|
|
|
{
|
|
|
|
if ((ubEnableMask & (1 << index)) == 0)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
ulong gpuVa = dispatchParams.UniformBuffers[index].PackAddress();
|
|
|
|
ulong size = dispatchParams.UniformBuffers[index].UnpackSize();
|
|
|
|
|
|
|
|
_bufferManager.SetComputeUniformBuffer(index, gpuVa, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int index = 0; index < info.SBuffers.Count; index++)
|
|
|
|
{
|
|
|
|
BufferDescriptor sb = info.SBuffers[index];
|
|
|
|
|
|
|
|
sbEnableMask |= 1u << sb.Slot;
|
|
|
|
|
|
|
|
ulong sbDescAddress = _bufferManager.GetComputeUniformBufferAddress(0);
|
|
|
|
|
|
|
|
int sbDescOffset = 0x310 + sb.Slot * 0x10;
|
|
|
|
|
|
|
|
sbDescAddress += (ulong)sbDescOffset;
|
|
|
|
|
|
|
|
Span<byte> sbDescriptorData = _context.PhysicalMemory.Read(sbDescAddress, 0x10);
|
|
|
|
|
|
|
|
SbDescriptor sbDescriptor = MemoryMarshal.Cast<byte, SbDescriptor>(sbDescriptorData)[0];
|
|
|
|
|
|
|
|
_bufferManager.SetComputeStorageBuffer(sb.Slot, sbDescriptor.PackAddress(), (uint)sbDescriptor.Size);
|
|
|
|
}
|
|
|
|
|
|
|
|
ubEnableMask = 0;
|
|
|
|
|
|
|
|
for (int index = 0; index < info.CBuffers.Count; index++)
|
|
|
|
{
|
|
|
|
ubEnableMask |= 1u << info.CBuffers[index].Slot;
|
|
|
|
}
|
|
|
|
|
|
|
|
_bufferManager.SetComputeStorageBufferEnableMask(sbEnableMask);
|
|
|
|
_bufferManager.SetComputeUniformBufferEnableMask(ubEnableMask);
|
|
|
|
|
2019-11-22 18:37:45 -05:00
|
|
|
var textureBindings = new TextureBindingInfo[info.Textures.Count];
|
|
|
|
|
|
|
|
for (int index = 0; index < info.Textures.Count; index++)
|
|
|
|
{
|
|
|
|
var descriptor = info.Textures[index];
|
|
|
|
|
|
|
|
Target target = GetTarget(descriptor.Type);
|
|
|
|
|
2019-12-27 20:16:14 -05:00
|
|
|
if (descriptor.IsBindless)
|
|
|
|
{
|
|
|
|
textureBindings[index] = new TextureBindingInfo(target, descriptor.CbufOffset, descriptor.CbufSlot);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
textureBindings[index] = new TextureBindingInfo(target, descriptor.HandleIndex);
|
|
|
|
}
|
2019-11-22 18:37:45 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
_textureManager.SetComputeTextures(textureBindings);
|
|
|
|
|
|
|
|
var imageBindings = new TextureBindingInfo[info.Images.Count];
|
|
|
|
|
|
|
|
for (int index = 0; index < info.Images.Count; index++)
|
|
|
|
{
|
|
|
|
var descriptor = info.Images[index];
|
|
|
|
|
|
|
|
Target target = GetTarget(descriptor.Type);
|
|
|
|
|
|
|
|
imageBindings[index] = new TextureBindingInfo(target, descriptor.HandleIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
_textureManager.SetComputeImages(imageBindings);
|
|
|
|
|
2019-10-13 02:02:07 -04:00
|
|
|
_bufferManager.CommitComputeBindings();
|
2019-10-17 22:41:18 -04:00
|
|
|
_textureManager.CommitComputeBindings();
|
2019-10-13 02:02:07 -04:00
|
|
|
|
2019-10-17 22:41:18 -04:00
|
|
|
_context.Renderer.Pipeline.Dispatch(
|
2019-10-13 02:02:07 -04:00
|
|
|
dispatchParams.UnpackGridSizeX(),
|
|
|
|
dispatchParams.UnpackGridSizeY(),
|
|
|
|
dispatchParams.UnpackGridSizeZ());
|
2019-10-26 13:50:52 -04:00
|
|
|
|
2019-11-21 21:46:14 -05:00
|
|
|
UpdateShaderState(state);
|
2019-10-13 02:02:07 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|