1
0
Fork 0
mirror of https://github.com/hawkeye-stan/msfs-popout-panel-manager.git synced 2024-11-30 09:40:12 +00:00
msfs-popout-panel-manager/Orchestration/DynamicLodOrchestrator.cs

315 lines
11 KiB
C#
Raw Normal View History

2024-03-14 22:50:32 +00:00
using MSFSPopoutPanelManager.DomainModel.DynamicLod;
using MSFSPopoutPanelManager.DomainModel.Setting;
using MSFSPopoutPanelManager.Shared;
using MSFSPopoutPanelManager.WindowsAgent;
using System;
using System.Diagnostics;
using System.Runtime.InteropServices;
namespace MSFSPopoutPanelManager.Orchestration
{
public class DynamicLodOrchestrator : BaseOrchestrator
{
private const string SIMMODULE_NAME = "WwiseLibPCx64P.dll";
private const int PROCESS_VM_OPERATION = 0x0008;
private const int PROCESS_VM_READ = 0x0010;
private const int PROCESS_VM_WRITE = 0x0020;
private const long OFFSET_MODULE_BASE =0x004B2368;
private const long OFFSET_POINTER_MAIN = 0x3D0;
private const long OFFSET_POINTER_TLOD = 0xC;
private const long OFFSET_POINTER_OLOD = 0xC;
2024-09-06 02:58:59 +00:00
private const long OFFSET_POINTER_CLOUD_QUALITY = 0x44;
2024-03-14 22:50:32 +00:00
private const long OFFSET_POINTER_FG_MODE = 0x4A;
private const long OFFSET_POINTER_ANSIO_FILTER = -0x18;
private const long OFFSET_POINTER_WATER_WAVES = 0x3C;
private bool _isActive;
private WindowProcess _process;
private IntPtr _processHandle;
private long _processModuleAddress;
private long _addressTlod;
private long _addressOlod;
private long _addressCloudQ;
private long _addressFgMode;
private DynamicLodSetting DynamicLodSetting => AppSettingData.ApplicationSetting.DynamicLodSetting;
2024-09-06 02:58:59 +00:00
private DynamicLodSimData DynamicLodSimData => FlightSimData.DynamicLodSimData;
2024-03-14 22:50:32 +00:00
private DateTime _lastLodUpdateTime = DateTime.Now;
2024-09-06 02:58:59 +00:00
private bool _isDecreasedCloudQualityActive;
2024-03-14 22:50:32 +00:00
public DynamicLodOrchestrator(SharedStorage sharedStorage) : base(sharedStorage) {}
public void Attach()
{
if (AppSettingData == null || _isActive)
return;
_process = WindowProcessManager.SimulatorProcess;
_processModuleAddress = GetSimModuleAddress();
if (_process == null || _processModuleAddress == IntPtr.Zero)
return;
_processHandle = PInvoke.OpenProcess(PROCESS_VM_OPERATION | PROCESS_VM_READ | PROCESS_VM_WRITE, false, _process.ProcessId);
if (_processHandle == IntPtr.Zero)
return;
_addressTlod = ReadMemory<long>(_processModuleAddress + OFFSET_MODULE_BASE) + OFFSET_POINTER_MAIN;
if (_addressTlod > 0)
{
_addressTlod = ReadMemory<long>(_addressTlod) + OFFSET_POINTER_TLOD;
_addressOlod = _addressTlod + OFFSET_POINTER_OLOD;
2024-09-06 02:58:59 +00:00
_addressCloudQ = _addressTlod + OFFSET_POINTER_CLOUD_QUALITY;
2024-03-14 22:50:32 +00:00
_addressFgMode = _addressTlod - OFFSET_POINTER_FG_MODE;
if (!MemoryBoundaryTest())
{
FileLogger.WriteLog("Unable to validate memory space for Dynamic LOD", StatusMessageType.Error);
return;
}
}
_isActive = true;
_lastLodUpdateTime = DateTime.Now;
_isDecreasedCloudQualityActive = false;
}
public void Detach()
{
if (DynamicLodSetting == null)
return;
if (DynamicLodSetting.ResetEnabled)
{
2024-09-06 02:58:59 +00:00
WriteMemory(_addressTlod, DynamicLodSetting.ResetTlod / 100.0f);
WriteMemory(_addressOlod, DynamicLodSetting.ResetOlod / 100.0f);
2024-03-14 22:50:32 +00:00
}
_isActive = false;
Debug.WriteLine($"Reset to custom LOD: TLOD: {DynamicLodSetting.ResetTlod}, OLOD: {DynamicLodSetting.ResetOlod}");
}
2024-09-06 02:58:59 +00:00
public int ReadTlod()
2024-03-14 22:50:32 +00:00
{
2024-09-06 02:58:59 +00:00
return Convert.ToInt32(ReadMemory<float>(_addressTlod) * 100.0f);
2024-03-14 22:50:32 +00:00
}
2024-09-06 02:58:59 +00:00
public int ReadOlod()
2024-03-14 22:50:32 +00:00
{
2024-09-06 02:58:59 +00:00
return Convert.ToInt32(ReadMemory<float>(_addressOlod) * 100.0f);
2024-03-14 22:50:32 +00:00
}
2024-09-06 02:58:59 +00:00
public string ReadCloudQuality()
2024-03-14 22:50:32 +00:00
{
2024-09-06 02:58:59 +00:00
return ReadCloudQualitySimValue() switch
2024-03-14 22:50:32 +00:00
{
0 => "Low",
1 => "Medium",
2 => "High",
3 => "Ultra",
_ => "N/A"
};
}
2024-09-06 02:58:59 +00:00
public int ReadCloudQualitySimValue()
2024-03-14 22:50:32 +00:00
{
2024-09-06 02:58:59 +00:00
return Convert.ToInt32(ReadMemory<int>(_addressCloudQ));
2024-03-14 22:50:32 +00:00
}
2024-09-06 02:58:59 +00:00
public bool ReadIsFg()
2024-03-14 22:50:32 +00:00
{
return ReadMemory<byte>(_addressFgMode) == 1;
}
2024-09-06 02:58:59 +00:00
public void UpdateLod()
2024-03-14 22:50:32 +00:00
{
2024-09-06 02:58:59 +00:00
if (!FlightSimData.IsFlightStarted || !FlightSimData.IsInCockpit)
2024-03-14 22:50:32 +00:00
return;
2024-09-06 02:58:59 +00:00
2024-09-17 14:42:07 +00:00
if (DateTime.Now - _lastLodUpdateTime <= TimeSpan.FromSeconds(3))
2024-03-14 22:50:32 +00:00
return;
2024-09-06 02:58:59 +00:00
var deltaFps = DynamicLodSimData.Fps - DynamicLodSetting.TargetedFps;
2024-03-14 22:50:32 +00:00
if (Math.Abs(deltaFps) < DynamicLodSetting.TargetedFps * DynamicLodSetting.FpsTolerance / 100.0) // within FPS tolerance
return;
_lastLodUpdateTime = DateTime.Now;
SetTlod(deltaFps);
SetOlod();
}
private long GetSimModuleAddress()
{
if (_process == null)
return -1;
foreach (ProcessModule processModule in _process.Modules)
{
if (processModule.ModuleName == SIMMODULE_NAME)
return processModule.BaseAddress;
}
return -1;
}
private void WriteMemory(long address, object value)
{
try
{
var buffer = StructureToByteArray(value);
PInvoke.NtWriteVirtualMemory(checked((int)_processHandle), address, buffer, buffer.Length, out _);
}
catch
{
// ignored
}
}
private T ReadMemory<T>(long address) where T : struct
{
try
{
var byteSize = Marshal.SizeOf(typeof(T));
var buffer = new byte[byteSize];
PInvoke.NtReadVirtualMemory(checked((int)_processHandle), address, buffer, buffer.Length, out _);
return ByteArrayToStructure<T>(buffer);
}
catch
{
// ignored
}
return default(T);
}
private byte[] StructureToByteArray(object obj)
{
var length = Marshal.SizeOf(obj);
var array = new byte[length];
var pointer = Marshal.AllocHGlobal(length);
Marshal.StructureToPtr(obj, pointer, true);
Marshal.Copy(pointer, array, 0, length);
Marshal.FreeHGlobal(pointer);
return array;
}
private T ByteArrayToStructure<T>(byte[] bytes) where T : struct
{
var handle = GCHandle.Alloc(bytes, GCHandleType.Pinned);
try
{
var result = Marshal.PtrToStructure(handle.AddrOfPinnedObject(), typeof(T));
if(result != null)
return (T)result;
return default(T);
}
finally
{
handle.Free();
}
}
private bool MemoryBoundaryTest()
{
// Boundary check a few known setting memory addresses to see if any fail which likely indicates MSFS memory map has changed
2024-09-06 02:58:59 +00:00
if (ReadTlod() < 10 || ReadTlod() > 1000 || ReadTlod() < 10 || ReadTlod() > 1000
|| ReadOlod() < 10 || ReadOlod() > 1000 || ReadOlod() < 10 || ReadOlod() > 1000
|| ReadCloudQuality() == "N/A" || ReadCloudQuality() == "N/A"
2024-03-14 22:50:32 +00:00
|| ReadMemory<int>(_addressTlod + OFFSET_POINTER_ANSIO_FILTER) < 1 || ReadMemory<int>(_addressTlod + OFFSET_POINTER_ANSIO_FILTER) > 16
|| !(ReadMemory<int>(_addressTlod + OFFSET_POINTER_WATER_WAVES) == 128 || ReadMemory<int>(_addressTlod + OFFSET_POINTER_WATER_WAVES) == 256 || ReadMemory<int>(_addressTlod + OFFSET_POINTER_WATER_WAVES) == 512))
{
return false;
}
return true;
}
2024-09-06 02:58:59 +00:00
private void SetTlod(int deltaFps)
2024-03-14 22:50:32 +00:00
{
2024-09-17 14:42:07 +00:00
var tlodStep = Math.Max(10, Math.Abs(deltaFps / 2));
2024-09-06 02:58:59 +00:00
var newTlod = DynamicLodSimData.Tlod + Math.Sign(deltaFps) * tlodStep;
2024-03-14 22:50:32 +00:00
2024-09-06 02:58:59 +00:00
if (DynamicLodSetting.TlodMinOnGround && DynamicLodSimData.AltAboveGround <= DynamicLodSetting.AltTlodBase)
2024-03-14 22:50:32 +00:00
{
newTlod = DynamicLodSetting.TlodMin;
}
else if (newTlod < DynamicLodSetting.TlodMin)
{
newTlod = DynamicLodSetting.TlodMin;
}
else if (newTlod > DynamicLodSetting.TlodMax)
{
newTlod = DynamicLodSetting.TlodMax;
}
2024-09-06 02:58:59 +00:00
if (ReadTlod() == newTlod)
2024-03-14 22:50:32 +00:00
return;
// Adjust cloud quality if applicable
2024-09-06 02:58:59 +00:00
if (DynamicLodSetting.DecreaseCloudQuality && (!DynamicLodSetting.TlodMinOnGround && !(DynamicLodSimData.AltAboveGround <= DynamicLodSetting.AltTlodBase)))
2024-03-14 22:50:32 +00:00
{
2024-03-15 12:55:55 +00:00
switch (deltaFps)
{
case < 0 when newTlod < DynamicLodSetting.CloudRecoveryTlod && !_isDecreasedCloudQualityActive:
_isDecreasedCloudQualityActive = true;
2024-09-06 02:58:59 +00:00
WriteMemory(_addressCloudQ, ReadCloudQualitySimValue() - 1);
2024-03-14 22:50:32 +00:00
2024-03-15 12:55:55 +00:00
_lastLodUpdateTime =
_lastLodUpdateTime.AddSeconds(2); // Add extra delay for cloud setting to take effect
Debug.WriteLine("New Cloud Quality written - 2.");
2024-03-14 22:50:32 +00:00
2024-03-15 12:55:55 +00:00
return;
case > 0 when newTlod >= DynamicLodSetting.CloudRecoveryTlod && _isDecreasedCloudQualityActive:
_isDecreasedCloudQualityActive = false;
2024-09-06 02:58:59 +00:00
WriteMemory(_addressCloudQ, ReadCloudQualitySimValue() + 1);
2024-03-14 22:50:32 +00:00
2024-03-15 12:55:55 +00:00
_lastLodUpdateTime = _lastLodUpdateTime.AddSeconds(2);
Debug.WriteLine("New Cloud Quality written - 3.");
2024-03-14 22:50:32 +00:00
2024-03-15 12:55:55 +00:00
return;
}
2024-03-14 22:50:32 +00:00
}
2024-03-15 12:55:55 +00:00
2024-03-14 22:50:32 +00:00
Debug.WriteLine($"New TLOD written - {newTlod}.");
2024-09-06 02:58:59 +00:00
WriteMemory(_addressTlod, newTlod / 100.0f);
2024-03-14 22:50:32 +00:00
}
2024-09-06 02:58:59 +00:00
private void SetOlod()
2024-03-14 22:50:32 +00:00
{
int newOlod;
2024-09-06 02:58:59 +00:00
if (DynamicLodSimData.AltAboveGround < DynamicLodSetting.AltOlodBase)
2024-03-14 22:50:32 +00:00
{
newOlod = DynamicLodSetting.OlodBase;
}
2024-09-06 02:58:59 +00:00
else if (DynamicLodSimData.AltAboveGround > DynamicLodSetting.AltOlodTop)
2024-03-14 22:50:32 +00:00
{
newOlod = DynamicLodSetting.OlodTop;
}
else
{
2024-09-06 02:58:59 +00:00
newOlod = Convert.ToInt32(DynamicLodSetting.OlodBase - (DynamicLodSetting.OlodBase - DynamicLodSetting.OlodTop) * (DynamicLodSimData.AltAboveGround - DynamicLodSetting.AltOlodBase) / (DynamicLodSetting.AltOlodTop - DynamicLodSetting.AltOlodBase));
2024-03-14 22:50:32 +00:00
}
2024-09-06 02:58:59 +00:00
if (ReadOlod() == newOlod)
2024-03-14 22:50:32 +00:00
return;
Debug.WriteLine($"New OLOD written - {newOlod}.");
2024-09-06 02:58:59 +00:00
WriteMemory(_addressOlod, newOlod / 100.0f);
2024-03-14 22:50:32 +00:00
}
}
}