267 lines
7.9 KiB
C#
267 lines
7.9 KiB
C#
|
using ARMeilleure.State;
|
||
|
using System;
|
||
|
using System.Collections.Generic;
|
||
|
using System.Diagnostics;
|
||
|
using System.IO;
|
||
|
using System.IO.Compression;
|
||
|
using System.Runtime.Serialization.Formatters.Binary;
|
||
|
using System.Security.Cryptography;
|
||
|
using System.Threading;
|
||
|
|
||
|
namespace ARMeilleure.Translation.PTC
|
||
|
{
|
||
|
public static class PtcProfiler
|
||
|
{
|
||
|
private const int SaveInterval = 30; // Seconds.
|
||
|
|
||
|
private const CompressionLevel SaveCompressionLevel = CompressionLevel.Fastest;
|
||
|
|
||
|
private static readonly BinaryFormatter _binaryFormatter;
|
||
|
|
||
|
private static readonly System.Timers.Timer _timer;
|
||
|
|
||
|
private static readonly ManualResetEvent _waitEvent;
|
||
|
|
||
|
private static readonly object _lock;
|
||
|
|
||
|
private static bool _disposed;
|
||
|
|
||
|
internal static Dictionary<ulong, (ExecutionMode mode, bool highCq)> ProfiledFuncs { get; private set; } //! Not to be modified.
|
||
|
|
||
|
internal static bool Enabled { get; private set; }
|
||
|
|
||
|
public static ulong StaticCodeStart { internal get; set; }
|
||
|
public static int StaticCodeSize { internal get; set; }
|
||
|
|
||
|
static PtcProfiler()
|
||
|
{
|
||
|
_binaryFormatter = new BinaryFormatter();
|
||
|
|
||
|
_timer = new System.Timers.Timer((double)SaveInterval * 1000d);
|
||
|
_timer.Elapsed += PreSave;
|
||
|
|
||
|
_waitEvent = new ManualResetEvent(true);
|
||
|
|
||
|
_lock = new object();
|
||
|
|
||
|
_disposed = false;
|
||
|
|
||
|
ProfiledFuncs = new Dictionary<ulong, (ExecutionMode, bool)>();
|
||
|
|
||
|
Enabled = false;
|
||
|
}
|
||
|
|
||
|
internal static void AddEntry(ulong address, ExecutionMode mode, bool highCq)
|
||
|
{
|
||
|
if (IsAddressInStaticCodeRange(address))
|
||
|
{
|
||
|
lock (_lock)
|
||
|
{
|
||
|
Debug.Assert(!highCq && !ProfiledFuncs.ContainsKey(address));
|
||
|
|
||
|
ProfiledFuncs.TryAdd(address, (mode, highCq));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal static void UpdateEntry(ulong address, ExecutionMode mode, bool highCq)
|
||
|
{
|
||
|
if (IsAddressInStaticCodeRange(address))
|
||
|
{
|
||
|
lock (_lock)
|
||
|
{
|
||
|
Debug.Assert(highCq && ProfiledFuncs.ContainsKey(address));
|
||
|
|
||
|
ProfiledFuncs[address] = (mode, highCq);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal static bool IsAddressInStaticCodeRange(ulong address)
|
||
|
{
|
||
|
return address >= StaticCodeStart && address < StaticCodeStart + (ulong)StaticCodeSize;
|
||
|
}
|
||
|
|
||
|
internal static void ClearEntries()
|
||
|
{
|
||
|
ProfiledFuncs.Clear();
|
||
|
}
|
||
|
|
||
|
internal static void PreLoad()
|
||
|
{
|
||
|
string fileNameActual = String.Concat(Ptc.CachePathActual, ".info");
|
||
|
string fileNameBackup = String.Concat(Ptc.CachePathBackup, ".info");
|
||
|
|
||
|
FileInfo fileInfoActual = new FileInfo(fileNameActual);
|
||
|
FileInfo fileInfoBackup = new FileInfo(fileNameBackup);
|
||
|
|
||
|
if (fileInfoActual.Exists && fileInfoActual.Length != 0L)
|
||
|
{
|
||
|
if (!Load(fileNameActual))
|
||
|
{
|
||
|
if (fileInfoBackup.Exists && fileInfoBackup.Length != 0L)
|
||
|
{
|
||
|
Load(fileNameBackup);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else if (fileInfoBackup.Exists && fileInfoBackup.Length != 0L)
|
||
|
{
|
||
|
Load(fileNameBackup);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private static bool Load(string fileName)
|
||
|
{
|
||
|
using (FileStream compressedStream = new FileStream(fileName, FileMode.Open))
|
||
|
using (DeflateStream deflateStream = new DeflateStream(compressedStream, CompressionMode.Decompress, true))
|
||
|
using (MemoryStream stream = new MemoryStream())
|
||
|
using (MD5 md5 = MD5.Create())
|
||
|
{
|
||
|
int hashSize = md5.HashSize / 8;
|
||
|
|
||
|
deflateStream.CopyTo(stream);
|
||
|
|
||
|
stream.Seek(0L, SeekOrigin.Begin);
|
||
|
|
||
|
byte[] currentHash = new byte[hashSize];
|
||
|
stream.Read(currentHash, 0, hashSize);
|
||
|
|
||
|
byte[] expectedHash = md5.ComputeHash(stream);
|
||
|
|
||
|
if (!CompareHash(currentHash, expectedHash))
|
||
|
{
|
||
|
InvalidateCompressedStream(compressedStream);
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
stream.Seek((long)hashSize, SeekOrigin.Begin);
|
||
|
|
||
|
try
|
||
|
{
|
||
|
ProfiledFuncs = (Dictionary<ulong, (ExecutionMode, bool)>)_binaryFormatter.Deserialize(stream);
|
||
|
}
|
||
|
catch
|
||
|
{
|
||
|
ProfiledFuncs = new Dictionary<ulong, (ExecutionMode, bool)>();
|
||
|
|
||
|
InvalidateCompressedStream(compressedStream);
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private static bool CompareHash(ReadOnlySpan<byte> currentHash, ReadOnlySpan<byte> expectedHash)
|
||
|
{
|
||
|
return currentHash.SequenceEqual(expectedHash);
|
||
|
}
|
||
|
|
||
|
private static void InvalidateCompressedStream(FileStream compressedStream)
|
||
|
{
|
||
|
compressedStream.SetLength(0L);
|
||
|
}
|
||
|
|
||
|
private static void PreSave(object source, System.Timers.ElapsedEventArgs e)
|
||
|
{
|
||
|
_waitEvent.Reset();
|
||
|
|
||
|
string fileNameActual = String.Concat(Ptc.CachePathActual, ".info");
|
||
|
string fileNameBackup = String.Concat(Ptc.CachePathBackup, ".info");
|
||
|
|
||
|
FileInfo fileInfoActual = new FileInfo(fileNameActual);
|
||
|
|
||
|
if (fileInfoActual.Exists && fileInfoActual.Length != 0L)
|
||
|
{
|
||
|
File.Copy(fileNameActual, fileNameBackup, true);
|
||
|
}
|
||
|
|
||
|
Save(fileNameActual);
|
||
|
|
||
|
_waitEvent.Set();
|
||
|
}
|
||
|
|
||
|
private static void Save(string fileName)
|
||
|
{
|
||
|
using (MemoryStream stream = new MemoryStream())
|
||
|
using (MD5 md5 = MD5.Create())
|
||
|
{
|
||
|
int hashSize = md5.HashSize / 8;
|
||
|
|
||
|
stream.Seek((long)hashSize, SeekOrigin.Begin);
|
||
|
|
||
|
lock (_lock)
|
||
|
{
|
||
|
_binaryFormatter.Serialize(stream, ProfiledFuncs);
|
||
|
}
|
||
|
|
||
|
stream.Seek((long)hashSize, SeekOrigin.Begin);
|
||
|
byte[] hash = md5.ComputeHash(stream);
|
||
|
|
||
|
stream.Seek(0L, SeekOrigin.Begin);
|
||
|
stream.Write(hash, 0, hashSize);
|
||
|
|
||
|
using (FileStream compressedStream = new FileStream(fileName, FileMode.OpenOrCreate))
|
||
|
using (DeflateStream deflateStream = new DeflateStream(compressedStream, SaveCompressionLevel, true))
|
||
|
{
|
||
|
try
|
||
|
{
|
||
|
stream.WriteTo(deflateStream);
|
||
|
}
|
||
|
catch
|
||
|
{
|
||
|
compressedStream.Position = 0L;
|
||
|
}
|
||
|
|
||
|
if (compressedStream.Position < compressedStream.Length)
|
||
|
{
|
||
|
compressedStream.SetLength(compressedStream.Position);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal static void Start()
|
||
|
{
|
||
|
if (Ptc.State == PtcState.Enabled ||
|
||
|
Ptc.State == PtcState.Continuing)
|
||
|
{
|
||
|
Enabled = true;
|
||
|
|
||
|
_timer.Enabled = true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public static void Stop()
|
||
|
{
|
||
|
Enabled = false;
|
||
|
|
||
|
if (!_disposed)
|
||
|
{
|
||
|
_timer.Enabled = false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal static void Wait()
|
||
|
{
|
||
|
_waitEvent.WaitOne();
|
||
|
}
|
||
|
|
||
|
public static void Dispose()
|
||
|
{
|
||
|
if (!_disposed)
|
||
|
{
|
||
|
_disposed = true;
|
||
|
|
||
|
_timer.Elapsed -= PreSave;
|
||
|
_timer.Dispose();
|
||
|
|
||
|
Wait();
|
||
|
_waitEvent.Dispose();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|