911 lines
40 KiB
C#
911 lines
40 KiB
C#
// Copyright (c) Microsoft Corporation.
|
|
// Licensed under the MIT License.
|
|
|
|
using Microsoft.MixedReality.Toolkit.Utilities;
|
|
using System.Text;
|
|
using Unity.Profiling;
|
|
using UnityEngine;
|
|
|
|
#if WINDOWS_UWP
|
|
using Windows.Media.Capture;
|
|
using Windows.System;
|
|
#else
|
|
using UnityEngine.Profiling;
|
|
#endif
|
|
|
|
namespace Microsoft.MixedReality.Toolkit.Diagnostics
|
|
{
|
|
/// <summary>
|
|
/// The VisualProfiler provides a drop in, single file, solution for viewing
|
|
/// your Windows Mixed Reality Unity application's frame rate and memory usage. Missed
|
|
/// frames are displayed over time to visually find problem areas. Memory is reported
|
|
/// as current, peak and max usage in a bar graph.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// <para>To use this profiler simply add this script as a component of any GameObject in
|
|
/// your Unity scene. The profiler is initially enabled (toggle-able via the initiallyActive
|
|
/// property), but can be toggled via the enabled/disable voice commands keywords.</para>
|
|
/// </remarks>
|
|
[AddComponentMenu("Scripts/MRTK/Services/MixedRealityToolkitVisualProfiler")]
|
|
public class MixedRealityToolkitVisualProfiler : MonoBehaviour
|
|
{
|
|
private static readonly int maxStringLength = 32;
|
|
private static readonly int maxTargetFrameRate = 120;
|
|
private static readonly int maxFrameTimings = 128;
|
|
private static readonly int frameRange = 30;
|
|
private static readonly Vector2 defaultWindowRotation = new Vector2(10.0f, 20.0f);
|
|
private static readonly Vector3 defaultWindowScale = new Vector3(0.2f, 0.04f, 1.0f);
|
|
private static readonly Vector3[] backgroundScales = { new Vector3(1.05f, 1.2f, 1.2f), new Vector3(1.0f, 0.5f, 1.0f), new Vector3(1.0f, 0.25f, 1.0f) };
|
|
private static readonly Vector3[] backgroundOffsets = { new Vector3(0.0f, 0.0f, 0.0f), new Vector3(0.0f, 0.25f, 0.0f), new Vector3(0.0f, 0.375f, 0.0f) };
|
|
private static readonly string usedMemoryString = "Used: ";
|
|
private static readonly string peakMemoryString = "Peak: ";
|
|
private static readonly string limitMemoryString = "Limit: ";
|
|
private static readonly string voiceCommandString = "Say \"Toggle Profiler\" to show/hide";
|
|
private static readonly string visualProfilerTitleString = "MRTK Visual Profiler";
|
|
|
|
public Transform WindowParent { get; set; } = null;
|
|
|
|
[Header("Profiler Settings")]
|
|
[SerializeField, Tooltip("Is the profiler currently visible.")]
|
|
private bool isVisible = false;
|
|
|
|
public bool IsVisible
|
|
{
|
|
get { return isVisible; }
|
|
set { isVisible = value; }
|
|
}
|
|
|
|
private bool ShouldShowProfiler =>
|
|
#if WINDOWS_UWP
|
|
(!appCaptureIsCapturingVideo || showProfilerDuringMRC) &&
|
|
#endif // WINDOWS_UWP
|
|
isVisible;
|
|
|
|
[SerializeField, Tooltip("Should the frame info (colored bars) be displayed.")]
|
|
private bool frameInfoVisible = true;
|
|
|
|
public bool FrameInfoVisible
|
|
{
|
|
get { return frameInfoVisible; }
|
|
set { frameInfoVisible = value; }
|
|
}
|
|
|
|
[SerializeField, Tooltip("Should memory stats (used, peak, and limit) be displayed.")]
|
|
private bool memoryStatsVisible = true;
|
|
|
|
public bool MemoryStatsVisible
|
|
{
|
|
get { return memoryStatsVisible; }
|
|
set { memoryStatsVisible = value; }
|
|
}
|
|
|
|
[SerializeField, Tooltip("The amount of time, in seconds, to collect frames for frame rate calculation.")]
|
|
private float frameSampleRate = 0.1f;
|
|
|
|
public float FrameSampleRate
|
|
{
|
|
get { return frameSampleRate; }
|
|
set { frameSampleRate = value; }
|
|
}
|
|
|
|
[Header("Window Settings")]
|
|
[SerializeField, Tooltip("What part of the view port to anchor the window to.")]
|
|
private TextAnchor windowAnchor = TextAnchor.LowerCenter;
|
|
|
|
public TextAnchor WindowAnchor
|
|
{
|
|
get { return windowAnchor; }
|
|
set { windowAnchor = value; }
|
|
}
|
|
|
|
[SerializeField, Tooltip("The offset from the view port center applied based on the window anchor selection.")]
|
|
private Vector2 windowOffset = new Vector2(0.1f, 0.1f);
|
|
|
|
public Vector2 WindowOffset
|
|
{
|
|
get { return windowOffset; }
|
|
set { windowOffset = value; }
|
|
}
|
|
|
|
[SerializeField, Range(0.5f, 5.0f), Tooltip("Use to scale the window size up or down, can simulate a zooming effect.")]
|
|
private float windowScale = 1.0f;
|
|
|
|
public float WindowScale
|
|
{
|
|
get { return windowScale; }
|
|
set { windowScale = Mathf.Clamp(value, 0.5f, 5.0f); }
|
|
}
|
|
|
|
[SerializeField, Range(0.0f, 100.0f), Tooltip("How quickly to interpolate the window towards its target position and rotation.")]
|
|
private float windowFollowSpeed = 5.0f;
|
|
|
|
public float WindowFollowSpeed
|
|
{
|
|
get { return windowFollowSpeed; }
|
|
set { windowFollowSpeed = Mathf.Abs(value); }
|
|
}
|
|
|
|
[SerializeField]
|
|
[Tooltip("If the diagnostics profiler should be visible while a mixed reality capture is happening on HoloLens.")]
|
|
private bool showProfilerDuringMRC = false;
|
|
|
|
/// <summary>
|
|
/// If the diagnostics profiler should be visible while a mixed reality capture is happening on HoloLens.
|
|
/// </summary>
|
|
/// <remarks>This is not usually recommended, as MRC can have an effect on an app's frame rate.</remarks>
|
|
public bool ShowProfilerDuringMRC
|
|
{
|
|
get { return showProfilerDuringMRC; }
|
|
set { showProfilerDuringMRC = value; }
|
|
}
|
|
|
|
[Header("UI Settings")]
|
|
[SerializeField, Range(0, 3), Tooltip("How many decimal places to display on numeric strings.")]
|
|
private int displayedDecimalDigits = 1;
|
|
|
|
[System.Serializable]
|
|
private struct FrameRateColor
|
|
{
|
|
[Range(0.0f, 1.0f), Tooltip("The percentage of the target frame rate.")]
|
|
public float percentageOfTarget;
|
|
[Tooltip("The color to display for frames which meet or exceed the percentage of the target frame rate.")]
|
|
public Color color;
|
|
}
|
|
|
|
[SerializeField, Tooltip("A list of colors to display for different percentage of target frame rates.")]
|
|
private FrameRateColor[] frameRateColors = new FrameRateColor[]
|
|
{
|
|
// Green
|
|
new FrameRateColor() { percentageOfTarget = 0.95f, color = new Color(127 / 256.0f, 186 / 256.0f, 0 / 256.0f, 1.0f) },
|
|
// Yellow
|
|
new FrameRateColor() { percentageOfTarget = 0.75f, color = new Color(255 / 256.0f, 185 / 256.0f, 0 / 256.0f, 1.0f) },
|
|
// Red
|
|
new FrameRateColor() { percentageOfTarget = 0.0f, color = new Color(255 / 256.0f, 0 / 256.0f, 0 / 256.0f, 1.0f) },
|
|
};
|
|
|
|
[SerializeField, Tooltip("The color of the window backplate.")]
|
|
private Color baseColor = new Color(80 / 256.0f, 80 / 256.0f, 80 / 256.0f, 1.0f);
|
|
[SerializeField, Tooltip("The color to display for current memory usage values.")]
|
|
private Color memoryUsedColor = new Color(0 / 256.0f, 164 / 256.0f, 239 / 256.0f, 1.0f);
|
|
[SerializeField, Tooltip("The color to display for peak (aka max) memory usage values.")]
|
|
private Color memoryPeakColor = new Color(255 / 256.0f, 185 / 256.0f, 0 / 256.0f, 1.0f);
|
|
[SerializeField, Tooltip("The color to display for the platforms memory usage limit.")]
|
|
private Color memoryLimitColor = new Color(150 / 256.0f, 150 / 256.0f, 150 / 256.0f, 1.0f);
|
|
|
|
private Transform window;
|
|
private Transform background;
|
|
private TextMesh cpuFrameRateText;
|
|
private TextMesh gpuFrameRateText;
|
|
private Transform memoryStats;
|
|
private TextMesh usedMemoryText;
|
|
private TextMesh peakMemoryText;
|
|
private TextMesh limitMemoryText;
|
|
private TextMesh voiceCommandText;
|
|
private TextMesh mrtkText;
|
|
private Transform usedAnchor;
|
|
private Transform peakAnchor;
|
|
private Quaternion windowHorizontalRotation;
|
|
private Quaternion windowHorizontalRotationInverse;
|
|
private Quaternion windowVerticalRotation;
|
|
private Quaternion windowVerticalRotationInverse;
|
|
|
|
private Matrix4x4[] frameInfoMatrices;
|
|
private Vector4[] frameInfoColors;
|
|
private MaterialPropertyBlock frameInfoPropertyBlock;
|
|
private int colorID;
|
|
private int parentMatrixID;
|
|
private int frameCount;
|
|
private System.Diagnostics.Stopwatch stopwatch = new System.Diagnostics.Stopwatch();
|
|
private FrameTiming[] frameTimings = new FrameTiming[maxFrameTimings];
|
|
private string[] cpuFrameRateStrings;
|
|
private string[] gpuFrameRateStrings;
|
|
private char[] stringBuffer = new char[maxStringLength];
|
|
|
|
private ulong memoryUsage;
|
|
private ulong peakMemoryUsage;
|
|
private ulong limitMemoryUsage;
|
|
|
|
// Rendering resources.
|
|
[SerializeField, HideInInspector]
|
|
private Material defaultMaterial;
|
|
[SerializeField, HideInInspector]
|
|
private Material defaultInstancedMaterial;
|
|
private Material backgroundMaterial;
|
|
private Material foregroundMaterial;
|
|
private Material textMaterial;
|
|
private Mesh quadMesh;
|
|
|
|
#if WINDOWS_UWP
|
|
private bool appCaptureIsCapturingVideo = false;
|
|
private AppCapture appCapture;
|
|
#endif // WINDOWS_UWP
|
|
|
|
private void Reset()
|
|
{
|
|
if (defaultMaterial == null)
|
|
{
|
|
defaultMaterial = new Material(Shader.Find("Hidden/Internal-Colored"));
|
|
defaultMaterial.SetFloat("_ZWrite", 1.0f);
|
|
defaultMaterial.SetFloat("_ZTest", (float)UnityEngine.Rendering.CompareFunction.Disabled);
|
|
defaultMaterial.renderQueue = 5000;
|
|
}
|
|
|
|
if (defaultInstancedMaterial == null)
|
|
{
|
|
Shader defaultInstancedShader = Shader.Find("Hidden/Instanced-Colored");
|
|
|
|
if (defaultInstancedShader != null)
|
|
{
|
|
defaultInstancedMaterial = new Material(defaultInstancedShader);
|
|
defaultInstancedMaterial.enableInstancing = true;
|
|
defaultInstancedMaterial.SetFloat("_ZWrite", 1.0f);
|
|
defaultInstancedMaterial.SetFloat("_ZTest", (float)UnityEngine.Rendering.CompareFunction.Disabled);
|
|
defaultInstancedMaterial.renderQueue = 5000;
|
|
}
|
|
else
|
|
{
|
|
Debug.LogWarning("A shader supporting instancing could not be found for the VisualProfiler, falling back to traditional rendering. This may impact performance.");
|
|
}
|
|
}
|
|
|
|
if (Application.isPlaying)
|
|
{
|
|
backgroundMaterial = new Material(defaultMaterial);
|
|
foregroundMaterial = new Material(defaultMaterial);
|
|
defaultMaterial.renderQueue = foregroundMaterial.renderQueue - 1;
|
|
backgroundMaterial.renderQueue = defaultMaterial.renderQueue - 1;
|
|
|
|
MeshRenderer meshRenderer = new GameObject().AddComponent<TextMesh>().GetComponent<MeshRenderer>();
|
|
textMaterial = new Material(meshRenderer.sharedMaterial);
|
|
textMaterial.renderQueue = defaultMaterial.renderQueue;
|
|
Destroy(meshRenderer.gameObject);
|
|
|
|
MeshFilter quadMeshFilter = GameObject.CreatePrimitive(PrimitiveType.Quad).GetComponent<MeshFilter>();
|
|
|
|
if (defaultInstancedMaterial != null)
|
|
{
|
|
// Create a quad mesh with artificially large bounds to disable culling for instanced rendering.
|
|
// TODO: Use shared mesh with normal bounds once Unity allows for more control over instance culling.
|
|
quadMesh = quadMeshFilter.mesh;
|
|
quadMesh.bounds = new Bounds(Vector3.zero, Vector3.one * float.MaxValue);
|
|
}
|
|
else
|
|
{
|
|
quadMesh = quadMeshFilter.sharedMesh;
|
|
}
|
|
|
|
Destroy(quadMeshFilter.gameObject);
|
|
}
|
|
|
|
stopwatch.Reset();
|
|
stopwatch.Start();
|
|
}
|
|
|
|
private void Start()
|
|
{
|
|
Reset();
|
|
BuildWindow();
|
|
BuildFrameRateStrings();
|
|
|
|
#if WINDOWS_UWP
|
|
appCapture = AppCapture.GetForCurrentView();
|
|
if (appCapture != null)
|
|
{
|
|
appCaptureIsCapturingVideo = appCapture.IsCapturingVideo;
|
|
appCapture.CapturingChanged += AppCapture_CapturingChanged;
|
|
}
|
|
#endif // WINDOWS_UWP
|
|
}
|
|
|
|
private void OnDestroy()
|
|
{
|
|
#if WINDOWS_UWP
|
|
if (appCapture != null)
|
|
{
|
|
appCapture.CapturingChanged -= AppCapture_CapturingChanged;
|
|
}
|
|
#endif // WINDOWS_UWP
|
|
|
|
if (window != null)
|
|
{
|
|
Destroy(window.gameObject);
|
|
}
|
|
}
|
|
|
|
private static readonly ProfilerMarker LateUpdatePerfMarker = new ProfilerMarker("[MRTK] MixedRealityToolkitVisualProfiler.LateUpdate");
|
|
|
|
private void LateUpdate()
|
|
{
|
|
if (window == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
using (LateUpdatePerfMarker.Auto())
|
|
{
|
|
// Update window transformation.
|
|
Transform cameraTransform = CameraCache.Main ? CameraCache.Main.transform : null;
|
|
|
|
if (ShouldShowProfiler && cameraTransform != null)
|
|
{
|
|
float t = Time.deltaTime * windowFollowSpeed;
|
|
window.position = Vector3.Lerp(window.position, CalculateWindowPosition(cameraTransform), t);
|
|
window.rotation = Quaternion.Slerp(window.rotation, CalculateWindowRotation(cameraTransform), t);
|
|
window.localScale = defaultWindowScale * windowScale;
|
|
CalculateBackgroundSize();
|
|
}
|
|
|
|
// Capture frame timings every frame and read from it depending on the frameSampleRate.
|
|
FrameTimingManager.CaptureFrameTimings();
|
|
|
|
++frameCount;
|
|
float elapsedSeconds = stopwatch.ElapsedMilliseconds * 0.001f;
|
|
|
|
if (elapsedSeconds >= frameSampleRate)
|
|
{
|
|
int cpuFrameRate = (int)(1.0f / (elapsedSeconds / frameCount));
|
|
int gpuFrameRate = 0;
|
|
|
|
// Many platforms do not yet support the FrameTimingManager. When timing data is returned from the FrameTimingManager we will use
|
|
// its timing data, else we will depend on the stopwatch.
|
|
uint frameTimingsCount = FrameTimingManager.GetLatestTimings((uint)Mathf.Min(frameCount, maxFrameTimings), frameTimings);
|
|
|
|
if (frameTimingsCount != 0)
|
|
{
|
|
float cpuFrameTime, gpuFrameTime;
|
|
AverageFrameTiming(frameTimings, frameTimingsCount, out cpuFrameTime, out gpuFrameTime);
|
|
cpuFrameRate = (int)(1.0f / (cpuFrameTime / frameCount));
|
|
gpuFrameRate = (int)(1.0f / (gpuFrameTime / frameCount));
|
|
}
|
|
|
|
// Update frame rate text.
|
|
cpuFrameRateText.text = cpuFrameRateStrings[Mathf.Clamp(cpuFrameRate, 0, maxTargetFrameRate)];
|
|
|
|
if (gpuFrameRate != 0)
|
|
{
|
|
gpuFrameRateText.gameObject.SetActive(true);
|
|
gpuFrameRateText.text = gpuFrameRateStrings[Mathf.Clamp(gpuFrameRate, 0, maxTargetFrameRate)];
|
|
}
|
|
|
|
// Update frame colors.
|
|
if (frameInfoVisible)
|
|
{
|
|
for (int i = frameRange - 1; i > 0; --i)
|
|
{
|
|
frameInfoColors[i] = frameInfoColors[i - 1];
|
|
}
|
|
|
|
frameInfoColors[0] = CalculateFrameColor(cpuFrameRate);
|
|
frameInfoPropertyBlock.SetVectorArray(colorID, frameInfoColors);
|
|
}
|
|
|
|
// Reset timers.
|
|
frameCount = 0;
|
|
stopwatch.Reset();
|
|
stopwatch.Start();
|
|
}
|
|
|
|
// Draw frame info.
|
|
if (ShouldShowProfiler && frameInfoVisible)
|
|
{
|
|
Matrix4x4 parentLocalToWorldMatrix = window.localToWorldMatrix;
|
|
|
|
if (defaultInstancedMaterial != null && SystemInfo.supportsInstancing)
|
|
{
|
|
frameInfoPropertyBlock.SetMatrix(parentMatrixID, parentLocalToWorldMatrix);
|
|
Graphics.DrawMeshInstanced(quadMesh, 0, defaultInstancedMaterial, frameInfoMatrices, frameInfoMatrices.Length, frameInfoPropertyBlock, UnityEngine.Rendering.ShadowCastingMode.Off, false);
|
|
}
|
|
else
|
|
{
|
|
// If a instanced material is not available or instancing isn't supported, fall back to non-instanced rendering.
|
|
for (int i = 0; i < frameInfoMatrices.Length; ++i)
|
|
{
|
|
frameInfoPropertyBlock.SetColor(colorID, frameInfoColors[i]);
|
|
Graphics.DrawMesh(quadMesh, parentLocalToWorldMatrix * frameInfoMatrices[i], defaultMaterial, 0, null, 0, frameInfoPropertyBlock, false, false, false);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Update memory statistics.
|
|
if (ShouldShowProfiler && memoryStatsVisible)
|
|
{
|
|
ulong limit = AppMemoryUsageLimit;
|
|
|
|
if (limit != limitMemoryUsage)
|
|
{
|
|
if (WillDisplayedMemoryUsageDiffer(limitMemoryUsage, limit, displayedDecimalDigits))
|
|
{
|
|
MemoryUsageToString(stringBuffer, displayedDecimalDigits, limitMemoryText, limitMemoryString, limit);
|
|
}
|
|
|
|
limitMemoryUsage = limit;
|
|
}
|
|
|
|
ulong usage = AppMemoryUsage;
|
|
|
|
if (usage != memoryUsage)
|
|
{
|
|
usedAnchor.localScale = new Vector3((float)usage / limitMemoryUsage, usedAnchor.localScale.y, usedAnchor.localScale.z);
|
|
|
|
if (WillDisplayedMemoryUsageDiffer(memoryUsage, usage, displayedDecimalDigits))
|
|
{
|
|
MemoryUsageToString(stringBuffer, displayedDecimalDigits, usedMemoryText, usedMemoryString, usage);
|
|
}
|
|
|
|
memoryUsage = usage;
|
|
}
|
|
|
|
if (memoryUsage > peakMemoryUsage)
|
|
{
|
|
peakAnchor.localScale = new Vector3((float)memoryUsage / limitMemoryUsage, peakAnchor.localScale.y, peakAnchor.localScale.z);
|
|
|
|
if (WillDisplayedMemoryUsageDiffer(peakMemoryUsage, memoryUsage, displayedDecimalDigits))
|
|
{
|
|
MemoryUsageToString(stringBuffer, displayedDecimalDigits, peakMemoryText, peakMemoryString, memoryUsage);
|
|
}
|
|
|
|
peakMemoryUsage = memoryUsage;
|
|
}
|
|
}
|
|
|
|
// Update visibility state.
|
|
if (window.gameObject.activeSelf != ShouldShowProfiler)
|
|
{
|
|
window.gameObject.SetActive(ShouldShowProfiler);
|
|
}
|
|
|
|
if (memoryStats.gameObject.activeSelf != memoryStatsVisible)
|
|
{
|
|
memoryStats.gameObject.SetActive(memoryStatsVisible);
|
|
}
|
|
}
|
|
}
|
|
|
|
#if WINDOWS_UWP
|
|
private void AppCapture_CapturingChanged(AppCapture sender, object args) => appCaptureIsCapturingVideo = sender.IsCapturingVideo;
|
|
private float previousFieldOfView = -1.0f;
|
|
#endif // WINDOWS_UWP
|
|
|
|
private static readonly ProfilerMarker CalculateWindowPositionPerfMarker = new ProfilerMarker("[MRTK] MixedRealityToolkitVisualProfiler.CalculateWindowPosition");
|
|
|
|
private Vector3 CalculateWindowPosition(Transform cameraTransform)
|
|
{
|
|
using (CalculateWindowPositionPerfMarker.Auto())
|
|
{
|
|
float windowDistance =
|
|
#if WINDOWS_UWP
|
|
Mathf.Max(16.0f / (appCaptureIsCapturingVideo ? previousFieldOfView : previousFieldOfView = CameraCache.Main.fieldOfView), Mathf.Max(CameraCache.Main.nearClipPlane, 0.5f));
|
|
#else
|
|
Mathf.Max(16.0f / CameraCache.Main.fieldOfView, Mathf.Max(CameraCache.Main.nearClipPlane, 0.5f));
|
|
#endif // WINDOWS_UWP
|
|
|
|
Vector3 position = cameraTransform.position + (cameraTransform.forward * windowDistance);
|
|
Vector3 horizontalOffset = cameraTransform.right * windowOffset.x;
|
|
Vector3 verticalOffset = cameraTransform.up * windowOffset.y;
|
|
|
|
switch (windowAnchor)
|
|
{
|
|
case TextAnchor.UpperLeft: position += verticalOffset - horizontalOffset; break;
|
|
case TextAnchor.UpperCenter: position += verticalOffset; break;
|
|
case TextAnchor.UpperRight: position += verticalOffset + horizontalOffset; break;
|
|
case TextAnchor.MiddleLeft: position -= horizontalOffset; break;
|
|
case TextAnchor.MiddleRight: position += horizontalOffset; break;
|
|
case TextAnchor.LowerLeft: position -= verticalOffset + horizontalOffset; break;
|
|
case TextAnchor.LowerCenter: position -= verticalOffset; break;
|
|
case TextAnchor.LowerRight: position -= verticalOffset - horizontalOffset; break;
|
|
}
|
|
|
|
return position;
|
|
}
|
|
}
|
|
|
|
private static readonly ProfilerMarker CalculateWindowRotationPerfMarker = new ProfilerMarker("[MRTK] MixedRealityToolkitVisualProfiler.CalculateWindowRotation");
|
|
|
|
private Quaternion CalculateWindowRotation(Transform cameraTransform)
|
|
{
|
|
using (CalculateWindowRotationPerfMarker.Auto())
|
|
{
|
|
Quaternion rotation = cameraTransform.rotation;
|
|
|
|
switch (windowAnchor)
|
|
{
|
|
case TextAnchor.UpperLeft: rotation *= windowHorizontalRotationInverse * windowVerticalRotationInverse; break;
|
|
case TextAnchor.UpperCenter: rotation *= windowHorizontalRotationInverse; break;
|
|
case TextAnchor.UpperRight: rotation *= windowHorizontalRotationInverse * windowVerticalRotation; break;
|
|
case TextAnchor.MiddleLeft: rotation *= windowVerticalRotationInverse; break;
|
|
case TextAnchor.MiddleRight: rotation *= windowVerticalRotation; break;
|
|
case TextAnchor.LowerLeft: rotation *= windowHorizontalRotation * windowVerticalRotationInverse; break;
|
|
case TextAnchor.LowerCenter: rotation *= windowHorizontalRotation; break;
|
|
case TextAnchor.LowerRight: rotation *= windowHorizontalRotation * windowVerticalRotation; break;
|
|
}
|
|
|
|
return rotation;
|
|
}
|
|
}
|
|
|
|
private static readonly ProfilerMarker CalculateFrameColorPerfMarker = new ProfilerMarker("[MRTK] MixedRealityToolkitVisualProfiler.CalculateFrameColor");
|
|
|
|
private Color CalculateFrameColor(int frameRate)
|
|
{
|
|
using (CalculateFrameColorPerfMarker.Auto())
|
|
{
|
|
// Ideally we would query a device specific API (like the HolographicFramePresentationReport) to detect missed frames.
|
|
// But, many of these APIs are inaccessible in Unity. Currently missed frames are assumed when the average cpuFrameRate
|
|
// is under the target frame rate.
|
|
|
|
int colorCount = frameRateColors.Length;
|
|
|
|
if (colorCount == 0)
|
|
{
|
|
return baseColor;
|
|
}
|
|
|
|
float percentageOfTarget = frameRate / AppTargetFrameRate;
|
|
int lastColor = colorCount - 1;
|
|
|
|
for (int i = 0; i < lastColor; ++i)
|
|
{
|
|
if (percentageOfTarget >= frameRateColors[i].percentageOfTarget)
|
|
{
|
|
return frameRateColors[i].color;
|
|
}
|
|
}
|
|
|
|
return frameRateColors[lastColor].color;
|
|
}
|
|
}
|
|
|
|
private static readonly ProfilerMarker CalculateBackgroundSizePerfMarker = new ProfilerMarker("[MRTK] MixedRealityToolkitVisualProfiler.CalculateBackgroundSize");
|
|
|
|
private void CalculateBackgroundSize()
|
|
{
|
|
using (CalculateBackgroundSizePerfMarker.Auto())
|
|
{
|
|
if (memoryStatsVisible)
|
|
{
|
|
background.localPosition = backgroundOffsets[0];
|
|
background.localScale = backgroundScales[0];
|
|
}
|
|
else if (frameInfoVisible)
|
|
{
|
|
background.localPosition = backgroundOffsets[1];
|
|
background.localScale = backgroundScales[1];
|
|
}
|
|
else
|
|
{
|
|
background.localPosition = backgroundOffsets[2];
|
|
background.localScale = backgroundScales[2];
|
|
}
|
|
}
|
|
}
|
|
|
|
private void BuildWindow()
|
|
{
|
|
// Initialize property block state.
|
|
colorID = Shader.PropertyToID("_Color");
|
|
parentMatrixID = Shader.PropertyToID("_ParentLocalToWorldMatrix");
|
|
|
|
// Build the window root.
|
|
{
|
|
window = new GameObject("VisualProfiler").transform;
|
|
window.parent = WindowParent;
|
|
window.localScale = defaultWindowScale;
|
|
windowHorizontalRotation = Quaternion.AngleAxis(defaultWindowRotation.y, Vector3.right);
|
|
windowHorizontalRotationInverse = Quaternion.Inverse(windowHorizontalRotation);
|
|
windowVerticalRotation = Quaternion.AngleAxis(defaultWindowRotation.x, Vector3.up);
|
|
windowVerticalRotationInverse = Quaternion.Inverse(windowVerticalRotation);
|
|
}
|
|
|
|
// Build the window background.
|
|
{
|
|
background = CreateQuad("Background", window).transform;
|
|
InitializeRenderer(background.gameObject, backgroundMaterial, colorID, baseColor);
|
|
CalculateBackgroundSize();
|
|
}
|
|
|
|
// Add frame rate text and frame indicators.
|
|
{
|
|
cpuFrameRateText = CreateText("CPUFrameRateText", new Vector3(-0.495f, 0.5f, 0.0f), window, TextAnchor.UpperLeft, textMaterial, Color.white, string.Empty);
|
|
gpuFrameRateText = CreateText("GPUFrameRateText", new Vector3(0.495f, 0.5f, 0.0f), window, TextAnchor.UpperRight, textMaterial, Color.white, string.Empty);
|
|
gpuFrameRateText.gameObject.SetActive(false);
|
|
|
|
frameInfoMatrices = new Matrix4x4[frameRange];
|
|
frameInfoColors = new Vector4[frameRange];
|
|
Vector3 scale = new Vector3(1.0f / frameRange, 0.2f, 1.0f);
|
|
Vector3 position = new Vector3(0.5f - (scale.x * 0.5f), 0.15f, 0.0f);
|
|
|
|
for (int i = 0; i < frameRange; ++i)
|
|
{
|
|
frameInfoMatrices[i] = Matrix4x4.TRS(position, Quaternion.identity, new Vector3(scale.x * 0.8f, scale.y, scale.z));
|
|
position.x -= scale.x;
|
|
frameInfoColors[i] = CalculateFrameColor((int)AppTargetFrameRate);
|
|
}
|
|
|
|
frameInfoPropertyBlock = new MaterialPropertyBlock();
|
|
frameInfoPropertyBlock.SetVectorArray(colorID, frameInfoColors);
|
|
}
|
|
|
|
// Add memory usage text and bars.
|
|
{
|
|
memoryStats = new GameObject("MemoryStats").transform;
|
|
memoryStats.parent = window;
|
|
memoryStats.localScale = Vector3.one;
|
|
|
|
usedMemoryText = CreateText("UsedMemoryText", new Vector3(-0.495f, 0.0f, 0.0f), memoryStats, TextAnchor.UpperLeft, textMaterial, memoryUsedColor, usedMemoryString);
|
|
peakMemoryText = CreateText("PeakMemoryText", new Vector3(0.0f, 0.0f, 0.0f), memoryStats, TextAnchor.UpperCenter, textMaterial, memoryPeakColor, peakMemoryString);
|
|
limitMemoryText = CreateText("LimitMemoryText", new Vector3(0.495f, 0.0f, 0.0f), memoryStats, TextAnchor.UpperRight, textMaterial, Color.white, limitMemoryString);
|
|
voiceCommandText = CreateText("VoiceCommandText", new Vector3(-0.525f, -0.7f, 0.0f), memoryStats, TextAnchor.UpperLeft, textMaterial, Color.white, voiceCommandString);
|
|
mrtkText = CreateText("MRTKText", new Vector3(0.52f, -0.7f, 0.0f), memoryStats, TextAnchor.UpperRight, textMaterial, Color.white, visualProfilerTitleString);
|
|
voiceCommandText.fontSize = 32;
|
|
mrtkText.fontSize = 32;
|
|
|
|
GameObject limitBar = CreateQuad("LimitBar", memoryStats);
|
|
InitializeRenderer(limitBar, defaultMaterial, colorID, memoryLimitColor);
|
|
limitBar.transform.localScale = new Vector3(0.99f, 0.2f, 1.0f);
|
|
limitBar.transform.localPosition = new Vector3(0.0f, -0.37f, 0.0f);
|
|
|
|
{
|
|
usedAnchor = CreateAnchor("UsedAnchor", limitBar.transform);
|
|
GameObject bar = CreateQuad("UsedBar", usedAnchor);
|
|
Material material = new Material(foregroundMaterial);
|
|
material.renderQueue += 1;
|
|
InitializeRenderer(bar, material, colorID, memoryUsedColor);
|
|
bar.transform.localScale = Vector3.one;
|
|
bar.transform.localPosition = new Vector3(0.5f, 0.0f, 0.0f);
|
|
}
|
|
{
|
|
peakAnchor = CreateAnchor("PeakAnchor", limitBar.transform);
|
|
GameObject bar = CreateQuad("PeakBar", peakAnchor);
|
|
InitializeRenderer(bar, foregroundMaterial, colorID, memoryPeakColor);
|
|
bar.transform.localScale = Vector3.one;
|
|
bar.transform.localPosition = new Vector3(0.5f, 0.0f, 0.0f);
|
|
}
|
|
}
|
|
|
|
if (window.gameObject.activeSelf != ShouldShowProfiler)
|
|
{
|
|
window.gameObject.SetActive(ShouldShowProfiler);
|
|
}
|
|
|
|
if (memoryStats.gameObject.activeSelf != memoryStatsVisible)
|
|
{
|
|
memoryStats.gameObject.SetActive(memoryStatsVisible);
|
|
}
|
|
}
|
|
|
|
private void BuildFrameRateStrings()
|
|
{
|
|
cpuFrameRateStrings = new string[maxTargetFrameRate + 1];
|
|
gpuFrameRateStrings = new string[maxTargetFrameRate + 1];
|
|
string displayedDecimalFormat = string.Format("{{0:F{0}}}", displayedDecimalDigits);
|
|
|
|
StringBuilder stringBuilder = new StringBuilder(32);
|
|
StringBuilder milisecondStringBuilder = new StringBuilder(16);
|
|
|
|
for (int i = 0; i < cpuFrameRateStrings.Length; ++i)
|
|
{
|
|
float miliseconds = (i == 0) ? 0.0f : (1.0f / i) * 1000.0f;
|
|
milisecondStringBuilder.AppendFormat(displayedDecimalFormat, miliseconds);
|
|
stringBuilder.AppendFormat("CPU: {0} fps ({1} ms)", i.ToString(), milisecondStringBuilder.ToString());
|
|
cpuFrameRateStrings[i] = stringBuilder.ToString();
|
|
stringBuilder.Length = 0;
|
|
stringBuilder.AppendFormat("GPU: {0} fps ({1} ms)", i.ToString(), milisecondStringBuilder.ToString());
|
|
gpuFrameRateStrings[i] = stringBuilder.ToString();
|
|
milisecondStringBuilder.Length = 0;
|
|
stringBuilder.Length = 0;
|
|
}
|
|
}
|
|
|
|
private static Transform CreateAnchor(string name, Transform parent)
|
|
{
|
|
Transform anchor = new GameObject(name).transform;
|
|
anchor.parent = parent;
|
|
anchor.localScale = Vector3.one;
|
|
anchor.localPosition = new Vector3(-0.5f, 0.0f, 0.0f);
|
|
|
|
return anchor;
|
|
}
|
|
|
|
private static GameObject CreateQuad(string name, Transform parent)
|
|
{
|
|
GameObject quad = GameObject.CreatePrimitive(PrimitiveType.Quad);
|
|
Destroy(quad.GetComponent<Collider>());
|
|
quad.name = name;
|
|
quad.transform.parent = parent;
|
|
|
|
return quad;
|
|
}
|
|
|
|
private static TextMesh CreateText(string name, Vector3 position, Transform parent, TextAnchor anchor, Material material, Color color, string text)
|
|
{
|
|
GameObject obj = new GameObject(name);
|
|
obj.transform.localScale = Vector3.one * 0.0016f;
|
|
obj.transform.parent = parent;
|
|
obj.transform.localPosition = position;
|
|
TextMesh textMesh = obj.AddComponent<TextMesh>();
|
|
textMesh.fontSize = 48;
|
|
textMesh.anchor = anchor;
|
|
textMesh.color = color;
|
|
textMesh.text = text;
|
|
textMesh.richText = false;
|
|
|
|
Renderer renderer = obj.GetComponent<Renderer>();
|
|
renderer.sharedMaterial = material;
|
|
|
|
OptimizeRenderer(renderer);
|
|
|
|
return textMesh;
|
|
}
|
|
|
|
private static Renderer InitializeRenderer(GameObject obj, Material material, int colorID, Color color)
|
|
{
|
|
Renderer renderer = obj.GetComponent<Renderer>();
|
|
renderer.sharedMaterial = material;
|
|
|
|
MaterialPropertyBlock propertyBlock = new MaterialPropertyBlock();
|
|
renderer.GetPropertyBlock(propertyBlock);
|
|
propertyBlock.SetColor(colorID, color);
|
|
renderer.SetPropertyBlock(propertyBlock);
|
|
|
|
OptimizeRenderer(renderer);
|
|
|
|
return renderer;
|
|
}
|
|
|
|
private static void OptimizeRenderer(Renderer renderer)
|
|
{
|
|
renderer.shadowCastingMode = UnityEngine.Rendering.ShadowCastingMode.Off;
|
|
renderer.receiveShadows = false;
|
|
renderer.motionVectorGenerationMode = MotionVectorGenerationMode.ForceNoMotion;
|
|
renderer.lightProbeUsage = UnityEngine.Rendering.LightProbeUsage.Off;
|
|
renderer.reflectionProbeUsage = UnityEngine.Rendering.ReflectionProbeUsage.Off;
|
|
renderer.allowOcclusionWhenDynamic = false;
|
|
}
|
|
|
|
private static readonly ProfilerMarker MemoryUsageToStringPerfMarker = new ProfilerMarker("[MRTK] MixedRealityToolkitVisualProfiler.MemoryUsageToString");
|
|
|
|
private static void MemoryUsageToString(char[] stringBuffer, int displayedDecimalDigits, TextMesh textMesh, string prefixString, ulong memoryUsage)
|
|
{
|
|
using (MemoryUsageToStringPerfMarker.Auto())
|
|
{
|
|
// Using a custom number to string method to avoid the overhead, and allocations, of built in string.Format/StringBuilder methods.
|
|
// We can also make some assumptions since the domain of the input number (memoryUsage) is known.
|
|
float memoryUsageMB = ConvertBytesToMegabytes(memoryUsage);
|
|
int memoryUsageIntegerDigits = (int)memoryUsageMB;
|
|
int memoryUsageFractionalDigits = (int)((memoryUsageMB - memoryUsageIntegerDigits) * Mathf.Pow(10.0f, displayedDecimalDigits));
|
|
int bufferIndex = 0;
|
|
|
|
for (int i = 0; i < prefixString.Length; ++i)
|
|
{
|
|
stringBuffer[bufferIndex++] = prefixString[i];
|
|
}
|
|
|
|
bufferIndex = MemoryItoA(memoryUsageIntegerDigits, stringBuffer, bufferIndex);
|
|
stringBuffer[bufferIndex++] = '.';
|
|
|
|
if (memoryUsageFractionalDigits != 0)
|
|
{
|
|
bufferIndex = MemoryItoA(memoryUsageFractionalDigits, stringBuffer, bufferIndex);
|
|
}
|
|
else
|
|
{
|
|
for (int i = 0; i < displayedDecimalDigits; ++i)
|
|
{
|
|
stringBuffer[bufferIndex++] = '0';
|
|
}
|
|
}
|
|
|
|
stringBuffer[bufferIndex++] = 'M';
|
|
stringBuffer[bufferIndex++] = 'B';
|
|
textMesh.text = new string(stringBuffer, 0, bufferIndex);
|
|
}
|
|
}
|
|
|
|
private static readonly ProfilerMarker MemoryItoAPerfMarker = new ProfilerMarker("[MRTK] MixedRealityToolkitVisualProfiler.MemoryItoA");
|
|
|
|
private static int MemoryItoA(int value, char[] stringBuffer, int bufferIndex)
|
|
{
|
|
using (MemoryItoAPerfMarker.Auto())
|
|
{
|
|
int startIndex = bufferIndex;
|
|
|
|
for (; value != 0; value /= 10)
|
|
{
|
|
stringBuffer[bufferIndex++] = (char)((char)(value % 10) + '0');
|
|
}
|
|
|
|
char temp;
|
|
for (int endIndex = bufferIndex - 1; startIndex < endIndex; ++startIndex, --endIndex)
|
|
{
|
|
temp = stringBuffer[startIndex];
|
|
stringBuffer[startIndex] = stringBuffer[endIndex];
|
|
stringBuffer[endIndex] = temp;
|
|
}
|
|
|
|
return bufferIndex;
|
|
}
|
|
}
|
|
|
|
private static float AppTargetFrameRate
|
|
{
|
|
get
|
|
{
|
|
// If the current XR SDK does not report refresh rate information, assume 60Hz.
|
|
float refreshRate = UnityEngine.XR.XRDevice.refreshRate;
|
|
return ((int)refreshRate == 0) ? 60.0f : refreshRate;
|
|
}
|
|
}
|
|
|
|
private static readonly ProfilerMarker AverageFrameTimingPerfMarker = new ProfilerMarker("[MRTK] MixedRealityToolkitVisualProfiler.AverageFrameTiming");
|
|
|
|
private static void AverageFrameTiming(FrameTiming[] frameTimings, uint frameTimingsCount, out float cpuFrameTime, out float gpuFrameTime)
|
|
{
|
|
using (AverageFrameTimingPerfMarker.Auto())
|
|
{
|
|
double cpuTime = 0.0f;
|
|
double gpuTime = 0.0f;
|
|
|
|
for (int i = 0; i < frameTimingsCount; ++i)
|
|
{
|
|
cpuTime += frameTimings[i].cpuFrameTime;
|
|
gpuTime += frameTimings[i].gpuFrameTime;
|
|
}
|
|
|
|
cpuTime /= frameTimingsCount;
|
|
gpuTime /= frameTimingsCount;
|
|
|
|
cpuFrameTime = (float)(cpuTime * 0.001);
|
|
gpuFrameTime = (float)(gpuTime * 0.001);
|
|
}
|
|
}
|
|
|
|
private static ulong AppMemoryUsage
|
|
{
|
|
get
|
|
{
|
|
#if WINDOWS_UWP
|
|
return MemoryManager.AppMemoryUsage;
|
|
#else
|
|
return (ulong)Profiler.GetTotalAllocatedMemoryLong();
|
|
#endif
|
|
}
|
|
}
|
|
|
|
private static ulong AppMemoryUsageLimit
|
|
{
|
|
get
|
|
{
|
|
#if WINDOWS_UWP
|
|
return MemoryManager.AppMemoryUsageLimit;
|
|
#else
|
|
return ConvertMegabytesToBytes(SystemInfo.systemMemorySize);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
private static readonly ProfilerMarker WillDisplayedMemoryUsageDifferPerfMarker = new ProfilerMarker("[MRTK] MixedRealityToolkitVisualProfiler.WillDisplayedMemoryUsageDiffer");
|
|
|
|
private static bool WillDisplayedMemoryUsageDiffer(ulong oldUsage, ulong newUsage, int displayedDecimalDigits)
|
|
{
|
|
using (WillDisplayedMemoryUsageDifferPerfMarker.Auto())
|
|
{
|
|
float oldUsageMBs = ConvertBytesToMegabytes(oldUsage);
|
|
float newUsageMBs = ConvertBytesToMegabytes(newUsage);
|
|
float decimalPower = Mathf.Pow(10.0f, displayedDecimalDigits);
|
|
|
|
return (int)(oldUsageMBs * decimalPower) != (int)(newUsageMBs * decimalPower);
|
|
}
|
|
}
|
|
|
|
private static ulong ConvertMegabytesToBytes(int megabytes)
|
|
{
|
|
return ((ulong)megabytes * 1024UL) * 1024UL;
|
|
}
|
|
|
|
private static float ConvertBytesToMegabytes(ulong bytes)
|
|
{
|
|
return (bytes / 1024.0f) / 1024.0f;
|
|
}
|
|
}
|
|
}
|