| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206 |
- using Ryujinx.Common.Logging;
- using Ryujinx.Graphics.GAL;
- using Silk.NET.Vulkan;
- using System;
- using System.Runtime.InteropServices;
- using System.Threading;
- namespace Ryujinx.Graphics.Vulkan.Queries
- {
- class BufferedQuery : IDisposable
- {
- private const int MaxQueryRetries = 5000;
- private const long DefaultValue = -1;
- private const long DefaultValueInt = 0xFFFFFFFF;
- private readonly Vk _api;
- private readonly Device _device;
- private readonly PipelineFull _pipeline;
- private QueryPool _queryPool;
- private bool _isReset;
- private readonly BufferHolder _buffer;
- private readonly IntPtr _bufferMap;
- private readonly CounterType _type;
- private bool _result32Bit;
- private bool _isSupported;
- private long _defaultValue;
- public unsafe BufferedQuery(VulkanRenderer gd, Device device, PipelineFull pipeline, CounterType type, bool result32Bit)
- {
- _api = gd.Api;
- _device = device;
- _pipeline = pipeline;
- _type = type;
- _result32Bit = result32Bit;
- _isSupported = QueryTypeSupported(gd, type);
- if (_isSupported)
- {
- QueryPipelineStatisticFlags flags = type == CounterType.PrimitivesGenerated ?
- QueryPipelineStatisticFlags.GeometryShaderPrimitivesBit : 0;
- var queryPoolCreateInfo = new QueryPoolCreateInfo()
- {
- SType = StructureType.QueryPoolCreateInfo,
- QueryCount = 1,
- QueryType = GetQueryType(type),
- PipelineStatistics = flags
- };
- gd.Api.CreateQueryPool(device, queryPoolCreateInfo, null, out _queryPool).ThrowOnError();
- }
- var buffer = gd.BufferManager.Create(gd, sizeof(long), forConditionalRendering: true);
- _bufferMap = buffer.Map(0, sizeof(long));
- _defaultValue = result32Bit ? DefaultValueInt : DefaultValue;
- Marshal.WriteInt64(_bufferMap, _defaultValue);
- _buffer = buffer;
- }
- private bool QueryTypeSupported(VulkanRenderer gd, CounterType type)
- {
- return type switch
- {
- CounterType.SamplesPassed => true,
- CounterType.TransformFeedbackPrimitivesWritten => gd.Capabilities.SupportsTransformFeedbackQueries,
- CounterType.PrimitivesGenerated => gd.Capabilities.SupportsGeometryShader,
- _ => false
- };
- }
- private static QueryType GetQueryType(CounterType type)
- {
- return type switch
- {
- CounterType.SamplesPassed => QueryType.Occlusion,
- CounterType.PrimitivesGenerated => QueryType.PipelineStatistics,
- CounterType.TransformFeedbackPrimitivesWritten => QueryType.TransformFeedbackStreamExt,
- _ => QueryType.Occlusion
- };
- }
- public Auto<DisposableBuffer> GetBuffer()
- {
- return _buffer.GetBuffer();
- }
- public void Reset()
- {
- End(false);
- Begin();
- }
- public void Begin()
- {
- if (_isSupported)
- {
- _pipeline.BeginQuery(this, _queryPool, !_isReset);
- }
- _isReset = false;
- }
- public unsafe void End(bool withResult)
- {
- if (_isSupported)
- {
- _pipeline.EndQuery(_queryPool);
- }
- if (withResult && _isSupported)
- {
- Marshal.WriteInt64(_bufferMap, _defaultValue);
- _pipeline.CopyQueryResults(this);
- }
- else
- {
- // Dummy result, just return 0.
- Marshal.WriteInt64(_bufferMap, 0);
- }
- }
- public bool TryGetResult(out long result)
- {
- result = Marshal.ReadInt64(_bufferMap);
- return result != _defaultValue;
- }
- public long AwaitResult(AutoResetEvent wakeSignal = null)
- {
- long data = _defaultValue;
- if (wakeSignal == null)
- {
- while (data == _defaultValue)
- {
- data = Marshal.ReadInt64(_bufferMap);
- }
- }
- else
- {
- int iterations = 0;
- while (data == _defaultValue && iterations++ < MaxQueryRetries)
- {
- data = Marshal.ReadInt64(_bufferMap);
- if (data == _defaultValue)
- {
- wakeSignal.WaitOne(1);
- }
- }
- if (iterations >= MaxQueryRetries)
- {
- Logger.Error?.Print(LogClass.Gpu, $"Error: Query result {_type} timed out. Took more than {MaxQueryRetries} tries.");
- }
- }
- return data;
- }
- public void PoolReset(CommandBuffer cmd)
- {
- if (_isSupported)
- {
- _api.CmdResetQueryPool(cmd, _queryPool, 0, 1);
- }
- _isReset = true;
- }
- public void PoolCopy(CommandBufferScoped cbs)
- {
- var buffer = _buffer.GetBuffer(cbs.CommandBuffer, true).Get(cbs, 0, sizeof(long)).Value;
- QueryResultFlags flags = QueryResultFlags.ResultWaitBit;
- if (!_result32Bit)
- {
- flags |= QueryResultFlags.Result64Bit;
- }
- _api.CmdCopyQueryPoolResults(
- cbs.CommandBuffer,
- _queryPool,
- 0,
- 1,
- buffer,
- 0,
- (ulong)(_result32Bit ? sizeof(int) : sizeof(long)),
- flags);
- }
- public unsafe void Dispose()
- {
- _buffer.Dispose();
- if (_isSupported)
- {
- _api.DestroyQueryPool(_device, _queryPool, null);
- }
- _queryPool = default;
- }
- }
- }
|