mirror of
https://github.com/Ryujinx/Ryujinx.git
synced 2025-01-07 16:46:02 +00:00
a33dc2f491
* Logger class changes only Now compile-time checking is possible with the help of Nullable Value types. * Misc formatting * Manual optimizations PrintGuestLog PrintGuestStackTrace Surfaceflinger DequeueBuffer * Reduce SendVibrationXX log level to Debug * Add Notice log level This level is always enabled and used to print system info, etc... Also, rewrite LogColor to switch expression as colors are static * Unify unhandled exception event handlers * Print enabled LogLevels during init * Re-add App Exit disposes in proper order nit: switch case spacing * Revert PrintGuestStackTrace to Info logs due to #1407 PrintGuestStackTrace is now called in some critical error handlers so revert to old behavior as KThread isn't part of Guest. * Batch replace Logger statements
105 lines
3 KiB
C#
105 lines
3 KiB
C#
using OpenTK.Graphics.OpenGL;
|
|
using Ryujinx.Common.Logging;
|
|
using System;
|
|
using System.Runtime.InteropServices;
|
|
using System.Threading;
|
|
|
|
namespace Ryujinx.Graphics.OpenGL.Queries
|
|
{
|
|
class BufferedQuery : IDisposable
|
|
{
|
|
private const int MaxQueryRetries = 5000;
|
|
private const long DefaultValue = -1;
|
|
|
|
public int Query { get; }
|
|
|
|
private int _buffer;
|
|
private IntPtr _bufferMap;
|
|
private QueryTarget _type;
|
|
|
|
public BufferedQuery(QueryTarget type)
|
|
{
|
|
_buffer = GL.GenBuffer();
|
|
Query = GL.GenQuery();
|
|
_type = type;
|
|
|
|
GL.BindBuffer(BufferTarget.QueryBuffer, _buffer);
|
|
|
|
unsafe
|
|
{
|
|
long defaultValue = DefaultValue;
|
|
GL.BufferStorage(BufferTarget.QueryBuffer, sizeof(long), (IntPtr)(&defaultValue), BufferStorageFlags.MapReadBit | BufferStorageFlags.MapWriteBit | BufferStorageFlags.MapPersistentBit);
|
|
}
|
|
_bufferMap = GL.MapBufferRange(BufferTarget.QueryBuffer, IntPtr.Zero, sizeof(long), BufferAccessMask.MapReadBit | BufferAccessMask.MapWriteBit | BufferAccessMask.MapPersistentBit);
|
|
}
|
|
|
|
public void Reset()
|
|
{
|
|
GL.EndQuery(_type);
|
|
GL.BeginQuery(_type, Query);
|
|
}
|
|
|
|
public void Begin()
|
|
{
|
|
GL.BeginQuery(_type, Query);
|
|
}
|
|
|
|
public unsafe void End()
|
|
{
|
|
GL.Flush();
|
|
GL.EndQuery(_type);
|
|
|
|
GL.BindBuffer(BufferTarget.QueryBuffer, _buffer);
|
|
|
|
Marshal.WriteInt64(_bufferMap, -1L);
|
|
GL.GetQueryObject(Query, GetQueryObjectParam.QueryResult, (long*)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 timed out. Took more than {MaxQueryRetries} tries.");
|
|
}
|
|
}
|
|
|
|
return data;
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
GL.BindBuffer(BufferTarget.QueryBuffer, _buffer);
|
|
GL.UnmapBuffer(BufferTarget.QueryBuffer);
|
|
GL.DeleteBuffer(_buffer);
|
|
GL.DeleteQuery(Query);
|
|
}
|
|
}
|
|
}
|