2019-10-13 06:02:07 +00:00
|
|
|
using Ryujinx.Common;
|
|
|
|
using Ryujinx.Graphics.GAL;
|
2020-04-22 06:00:11 +00:00
|
|
|
using Ryujinx.Graphics.Gpu.Memory;
|
2019-10-13 06:02:07 +00:00
|
|
|
using Ryujinx.Graphics.Gpu.State;
|
|
|
|
using System;
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
|
|
|
|
namespace Ryujinx.Graphics.Gpu.Engine
|
|
|
|
{
|
|
|
|
partial class Methods
|
|
|
|
{
|
2020-01-06 22:27:50 +00:00
|
|
|
private const int NsToTicksFractionNumerator = 384;
|
2020-01-01 15:39:09 +00:00
|
|
|
private const int NsToTicksFractionDenominator = 625;
|
|
|
|
|
2020-04-22 06:00:11 +00:00
|
|
|
private readonly CounterCache _counterCache = new CounterCache();
|
|
|
|
|
2019-12-31 19:19:44 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Writes a GPU counter to guest memory.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="state">Current GPU state</param>
|
|
|
|
/// <param name="argument">Method call argument</param>
|
2019-11-22 02:46:14 +00:00
|
|
|
private void Report(GpuState state, int argument)
|
2019-10-13 06:02:07 +00:00
|
|
|
{
|
|
|
|
ReportMode mode = (ReportMode)(argument & 3);
|
|
|
|
|
|
|
|
ReportCounterType type = (ReportCounterType)((argument >> 23) & 0x1f);
|
|
|
|
|
|
|
|
switch (mode)
|
|
|
|
{
|
2020-04-19 01:25:57 +00:00
|
|
|
case ReportMode.Release: ReleaseSemaphore(state); break;
|
|
|
|
case ReportMode.Counter: ReportCounter(state, type); break;
|
2019-10-13 06:02:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-31 19:19:44 +00:00
|
|
|
/// <summary>
|
2020-04-19 01:25:57 +00:00
|
|
|
/// Writes (or Releases) a GPU semaphore value to guest memory.
|
2019-12-31 19:19:44 +00:00
|
|
|
/// </summary>
|
|
|
|
/// <param name="state">Current GPU state</param>
|
2020-04-19 01:25:57 +00:00
|
|
|
private void ReleaseSemaphore(GpuState state)
|
2019-10-13 06:02:07 +00:00
|
|
|
{
|
2019-11-22 02:46:14 +00:00
|
|
|
var rs = state.Get<ReportState>(MethodOffset.ReportState);
|
2019-10-13 06:02:07 +00:00
|
|
|
|
2019-11-22 02:46:14 +00:00
|
|
|
_context.MemoryAccessor.Write(rs.Address.Pack(), rs.Payload);
|
2019-10-13 06:02:07 +00:00
|
|
|
|
|
|
|
_context.AdvanceSequence();
|
|
|
|
}
|
|
|
|
|
2019-12-31 19:19:44 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Packed GPU counter data (including GPU timestamp) in memory.
|
|
|
|
/// </summary>
|
2019-10-13 06:02:07 +00:00
|
|
|
private struct CounterData
|
|
|
|
{
|
|
|
|
public ulong Counter;
|
|
|
|
public ulong Timestamp;
|
|
|
|
}
|
|
|
|
|
2019-12-31 19:19:44 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Writes a GPU counter to guest memory.
|
|
|
|
/// This also writes the current timestamp value.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="state">Current GPU state</param>
|
|
|
|
/// <param name="type">Counter to be written to memory</param>
|
2019-11-22 02:46:14 +00:00
|
|
|
private void ReportCounter(GpuState state, ReportCounterType type)
|
2019-10-13 06:02:07 +00:00
|
|
|
{
|
|
|
|
CounterData counterData = new CounterData();
|
|
|
|
|
|
|
|
ulong counter = 0;
|
|
|
|
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case ReportCounterType.Zero:
|
|
|
|
counter = 0;
|
|
|
|
break;
|
|
|
|
case ReportCounterType.SamplesPassed:
|
|
|
|
counter = _context.Renderer.GetCounter(CounterType.SamplesPassed);
|
|
|
|
break;
|
|
|
|
case ReportCounterType.PrimitivesGenerated:
|
|
|
|
counter = _context.Renderer.GetCounter(CounterType.PrimitivesGenerated);
|
|
|
|
break;
|
|
|
|
case ReportCounterType.TransformFeedbackPrimitivesWritten:
|
|
|
|
counter = _context.Renderer.GetCounter(CounterType.TransformFeedbackPrimitivesWritten);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-04-20 12:41:07 +00:00
|
|
|
ulong ticks = ConvertNanosecondsToTicks((ulong)PerformanceCounter.ElapsedNanoseconds);
|
2019-10-13 06:02:07 +00:00
|
|
|
|
|
|
|
if (GraphicsConfig.FastGpuTime)
|
|
|
|
{
|
2020-04-20 12:41:07 +00:00
|
|
|
// Divide by some amount to report time as if operations were performed faster than they really are.
|
|
|
|
// This can prevent some games from switching to a lower resolution because rendering is too slow.
|
|
|
|
ticks /= 256;
|
2019-10-13 06:02:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
counterData.Counter = counter;
|
|
|
|
counterData.Timestamp = ticks;
|
|
|
|
|
|
|
|
Span<CounterData> counterDataSpan = MemoryMarshal.CreateSpan(ref counterData, 1);
|
|
|
|
|
|
|
|
Span<byte> data = MemoryMarshal.Cast<CounterData, byte>(counterDataSpan);
|
|
|
|
|
2019-11-22 02:46:14 +00:00
|
|
|
var rs = state.Get<ReportState>(MethodOffset.ReportState);
|
2019-10-13 06:02:07 +00:00
|
|
|
|
2019-11-22 02:46:14 +00:00
|
|
|
_context.MemoryAccessor.Write(rs.Address.Pack(), data);
|
2020-04-22 06:00:11 +00:00
|
|
|
|
|
|
|
_counterCache.AddOrUpdate(rs.Address.Pack());
|
2019-10-13 06:02:07 +00:00
|
|
|
}
|
|
|
|
|
2019-12-31 19:19:44 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Converts a nanoseconds timestamp value to Maxwell time ticks.
|
|
|
|
/// </summary>
|
2020-01-01 15:39:09 +00:00
|
|
|
/// <remarks>
|
|
|
|
/// The frequency is 614400000 Hz.
|
|
|
|
/// </remarks>
|
2019-12-31 19:19:44 +00:00
|
|
|
/// <param name="nanoseconds">Timestamp in nanoseconds</param>
|
|
|
|
/// <returns>Maxwell ticks</returns>
|
2019-10-13 06:02:07 +00:00
|
|
|
private static ulong ConvertNanosecondsToTicks(ulong nanoseconds)
|
|
|
|
{
|
|
|
|
// We need to divide first to avoid overflows.
|
|
|
|
// We fix up the result later by calculating the difference and adding
|
|
|
|
// that to the result.
|
2020-01-01 15:39:09 +00:00
|
|
|
ulong divided = nanoseconds / NsToTicksFractionDenominator;
|
2019-10-13 06:02:07 +00:00
|
|
|
|
2020-01-01 15:39:09 +00:00
|
|
|
ulong rounded = divided * NsToTicksFractionDenominator;
|
2019-10-13 06:02:07 +00:00
|
|
|
|
2020-01-01 15:39:09 +00:00
|
|
|
ulong errorBias = (nanoseconds - rounded) * NsToTicksFractionNumerator / NsToTicksFractionDenominator;
|
2019-10-13 06:02:07 +00:00
|
|
|
|
2020-01-01 15:39:09 +00:00
|
|
|
return divided * NsToTicksFractionNumerator + errorBias;
|
2019-10-13 06:02:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|