您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
343 行
10 KiB
343 行
10 KiB
using UnityEngine;
|
|
using System.Collections.Generic;
|
|
using System.Collections.ObjectModel;
|
|
using System.IO;
|
|
using UnityEngine.Profiling;
|
|
using System.Runtime.Serialization;
|
|
using System.Runtime.Serialization.Json;
|
|
#if UNITY_EDITOR
|
|
using UnityEditor;
|
|
|
|
#endif
|
|
|
|
|
|
namespace MLAgents
|
|
{
|
|
[DataContract]
|
|
public class TimerNode
|
|
{
|
|
static string s_Separator = ".";
|
|
static double s_TicksToSeconds = 1e-7; // 100 ns per tick
|
|
|
|
/// <summary>
|
|
/// Full name of the node. This is the node's parents full name concatenated with this node's name
|
|
/// </summary>
|
|
string m_FullName;
|
|
|
|
/// <summary>
|
|
/// Child nodes, indexed by name.
|
|
/// </summary>
|
|
[DataMember(Name = "children", Order = 999)]
|
|
Dictionary<string, TimerNode> m_Children;
|
|
|
|
/// <summary>
|
|
/// Custom sampler used to add timings to the profiler.
|
|
/// </summary>
|
|
private CustomSampler m_Sampler;
|
|
|
|
/// <summary>
|
|
/// Number of total ticks elapsed for this node.
|
|
/// </summary>
|
|
long m_TotalTicks = 0;
|
|
|
|
/// <summary>
|
|
/// If the node is currently running, the time (in ticks) when the node was started.
|
|
/// If the node is not running, is set to 0.
|
|
/// </summary>
|
|
long m_TickStart = 0;
|
|
|
|
/// <summary>
|
|
/// Number of times the corresponding code block has been called.
|
|
/// </summary>
|
|
[DataMember(Name = "count")]
|
|
int m_NumCalls = 0;
|
|
|
|
/// <summary>
|
|
/// The total recorded ticks for the timer node, plus the currently elapsed ticks
|
|
/// if the timer is still running (i.e. if m_TickStart is non-zero).
|
|
/// </summary>
|
|
public long CurrentTicks
|
|
{
|
|
get
|
|
{
|
|
long currentTicks = m_TotalTicks;
|
|
if (m_TickStart != 0)
|
|
{
|
|
currentTicks += (System.DateTime.Now.Ticks - m_TickStart);
|
|
}
|
|
|
|
return currentTicks;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Total elapsed seconds.
|
|
/// </summary>
|
|
[DataMember(Name = "total")]
|
|
public double TotalSeconds
|
|
{
|
|
get { return CurrentTicks * s_TicksToSeconds; }
|
|
set {} // Serialization needs this, but unused.
|
|
}
|
|
|
|
/// <summary>
|
|
/// Total seconds spent in this block, excluding it's children.
|
|
/// </summary>
|
|
[DataMember(Name = "self")]
|
|
public double SelfSeconds
|
|
{
|
|
get
|
|
{
|
|
long totalChildTicks = 0;
|
|
if (m_Children != null)
|
|
{
|
|
foreach (var child in m_Children.Values)
|
|
{
|
|
totalChildTicks += child.m_TotalTicks;
|
|
}
|
|
}
|
|
|
|
var selfTicks = Mathf.Max(0, CurrentTicks - totalChildTicks);
|
|
return selfTicks * s_TicksToSeconds;
|
|
}
|
|
set {} // Serialization needs this, but unused.
|
|
}
|
|
|
|
public IReadOnlyDictionary<string, TimerNode> Children
|
|
{
|
|
get { return m_Children; }
|
|
}
|
|
|
|
public int NumCalls
|
|
{
|
|
get { return m_NumCalls; }
|
|
}
|
|
|
|
public TimerNode(string name, bool isRoot = false)
|
|
{
|
|
m_FullName = name;
|
|
if (isRoot)
|
|
{
|
|
// The root node is considered always running. This means that when we output stats, it'll
|
|
// have a sensible value for total time (the running time since reset).
|
|
// The root node doesn't have a sampler since that could interfere with the profiler.
|
|
m_NumCalls = 1;
|
|
m_TickStart = System.DateTime.Now.Ticks;
|
|
}
|
|
else
|
|
{
|
|
m_Sampler = CustomSampler.Create(m_FullName);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Start timing a block of code.
|
|
/// </summary>
|
|
public void Begin()
|
|
{
|
|
m_Sampler?.Begin();
|
|
m_TickStart = System.DateTime.Now.Ticks;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Stop timing a block of code, and increment internal counts.
|
|
/// </summary>
|
|
public void End()
|
|
{
|
|
var elapsed = System.DateTime.Now.Ticks - m_TickStart;
|
|
m_TotalTicks += elapsed;
|
|
m_TickStart = 0;
|
|
m_NumCalls++;
|
|
m_Sampler?.End();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Return a child node for the given name.
|
|
/// The children dictionary will be created if it does not already exist, and
|
|
/// a new Node will be created if it's not already in the dictionary.
|
|
/// Note that these allocations only happen once for a given timed block.
|
|
/// </summary>
|
|
/// <param name="name"></param>
|
|
/// <returns></returns>
|
|
public TimerNode GetChild(string name)
|
|
{
|
|
// Lazily create the children dictionary.
|
|
if (m_Children == null)
|
|
{
|
|
m_Children = new Dictionary<string, TimerNode>();
|
|
}
|
|
|
|
if (!m_Children.ContainsKey(name))
|
|
{
|
|
var childFullName = m_FullName + s_Separator + name;
|
|
var newChild = new TimerNode(childFullName);
|
|
m_Children[name] = newChild;
|
|
return newChild;
|
|
}
|
|
|
|
return m_Children[name];
|
|
}
|
|
|
|
/// <summary>
|
|
/// Recursively form a string representing the current timer information.
|
|
/// </summary>
|
|
/// <param name="parentName"></param>
|
|
/// <param name="level"></param>
|
|
/// <returns></returns>
|
|
public string DebugGetTimerString(string parentName = "", int level = 0)
|
|
{
|
|
string indent = new string(' ', 2 * level); // TODO generalize
|
|
string shortName = (level == 0) ? m_FullName : m_FullName.Replace(parentName + s_Separator, "");
|
|
string timerString = "";
|
|
if (level == 0)
|
|
{
|
|
timerString = $"{shortName}(root)\n";
|
|
}
|
|
else
|
|
{
|
|
timerString = $"{indent}{shortName}\t\traw={TotalSeconds} rawCount={m_NumCalls}\n";
|
|
}
|
|
|
|
// TODO use stringbuilder? might be overkill since this is only debugging code?
|
|
if (m_Children != null)
|
|
{
|
|
foreach (TimerNode c in m_Children.Values)
|
|
{
|
|
timerString += c.DebugGetTimerString(m_FullName, level + 1);
|
|
}
|
|
}
|
|
return timerString;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// A "stack" of timers that allows for lightweight hierarchical profiling of long-running processes.
|
|
/// Example usage:
|
|
///
|
|
/// using(TimerStack.Instance.Scoped("foo"))
|
|
/// {
|
|
/// doSomeWork();
|
|
/// for (int i=0; i<5; i++)
|
|
/// {
|
|
/// using(myTimer.Scoped("bar"))
|
|
/// {
|
|
/// doSomeMoreWork();
|
|
/// }
|
|
/// }
|
|
/// }
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// This implements the Singleton pattern (solution 4) as described in
|
|
/// https://csharpindepth.com/articles/singleton
|
|
/// </remarks>
|
|
public class TimerStack : System.IDisposable
|
|
{
|
|
private static readonly TimerStack instance = new TimerStack();
|
|
|
|
Stack<TimerNode> m_Stack;
|
|
TimerNode m_RootNode;
|
|
|
|
// Explicit static constructor to tell C# compiler
|
|
// not to mark type as beforefieldinit
|
|
static TimerStack()
|
|
{
|
|
}
|
|
|
|
private TimerStack()
|
|
{
|
|
Reset();
|
|
}
|
|
|
|
public void Reset(string name = "root")
|
|
{
|
|
m_Stack = new Stack<TimerNode>();
|
|
m_RootNode = new TimerNode(name, true);
|
|
m_Stack.Push(m_RootNode);
|
|
}
|
|
|
|
public static TimerStack Instance
|
|
{
|
|
get { return instance; }
|
|
}
|
|
|
|
public TimerNode RootNode
|
|
{
|
|
get { return m_RootNode; }
|
|
}
|
|
|
|
private void Push(string name)
|
|
{
|
|
TimerNode current = m_Stack.Peek();
|
|
TimerNode next = current.GetChild(name);
|
|
m_Stack.Push(next);
|
|
next.Begin();
|
|
}
|
|
|
|
private void Pop()
|
|
{
|
|
var node = m_Stack.Pop();
|
|
node.End();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Start a scoped timer. This should be used with the "using" statement.
|
|
/// </summary>
|
|
/// <param name="name"></param>
|
|
/// <returns></returns>
|
|
public TimerStack Scoped(string name)
|
|
{
|
|
Push(name);
|
|
return this;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Closes the current scoped timer. This should never be called directly, only
|
|
/// at the end of a "using" statement.
|
|
/// Note that the instance is not actually disposed of; this is just to allow it to be used
|
|
/// conveniently with "using".
|
|
/// </summary>
|
|
public void Dispose()
|
|
{
|
|
Pop();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get a string representation of the timers.
|
|
/// Potentially slow so call sparingly.
|
|
/// </summary>
|
|
/// <returns></returns>
|
|
public string DebugGetTimerString()
|
|
{
|
|
return m_RootNode.DebugGetTimerString();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Save the timers in JSON format to the provided filename.
|
|
/// If the filename is null, a default one will be used.
|
|
/// </summary>
|
|
/// <param name="filename"></param>
|
|
public void SaveJsonTimers(string filename = null)
|
|
{
|
|
if (filename == null)
|
|
{
|
|
var fullpath = Path.GetFullPath(".");
|
|
filename = $"{fullpath}/csharp_timers.json";
|
|
}
|
|
var fs = new FileStream(filename, FileMode.Create, FileAccess.Write);
|
|
SaveJsonTimers(fs);
|
|
fs.Close();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Write the timers in JSON format to the provided stream.
|
|
/// </summary>
|
|
/// <param name="stream"></param>
|
|
public void SaveJsonTimers(Stream stream)
|
|
{
|
|
var jsonSettings = new DataContractJsonSerializerSettings();
|
|
jsonSettings.UseSimpleDictionaryFormat = true;
|
|
var ser = new DataContractJsonSerializer(typeof(TimerNode), jsonSettings);
|
|
ser.WriteObject(stream, m_RootNode);
|
|
}
|
|
}
|
|
}
|