using System; using System.Collections.Concurrent; using System.Collections.Generic; using System.Runtime.InteropServices; namespace ARMeilleure.Translation.PTC { [Serializable] class PtcJumpTable { private readonly List> _jumpTable; private readonly List> _dynamicTable; private readonly List _targets; private readonly Dictionary> _dependants; public int TableEnd => _jumpTable.Count; public int DynTableEnd => _dynamicTable.Count; public List Targets => _targets; public Dictionary> Dependants => _dependants; public PtcJumpTable() { _jumpTable = new List>(); _dynamicTable = new List>(); _targets = new List(); _dependants = new Dictionary>(); } public void Initialize(JumpTable jumpTable) { _targets.Clear(); foreach (ulong guestAddress in jumpTable.Targets.Keys) { _targets.Add(guestAddress); } _dependants.Clear(); foreach (var item in jumpTable.Dependants) { _dependants.Add(item.Key, new LinkedList(item.Value)); } } public void Clear() { _jumpTable.Clear(); _dynamicTable.Clear(); _targets.Clear(); _dependants.Clear(); } public void WriteJumpTable(JumpTable jumpTable, ConcurrentDictionary funcs) { jumpTable.ExpandIfNeededJumpTable(TableEnd); int entry = 0; foreach (var item in _jumpTable) { entry += 1; long guestAddress = item.Key; DirectHostAddress directHostAddress = item.Value; long hostAddress; if (directHostAddress == DirectHostAddress.CallStub) { hostAddress = DirectCallStubs.DirectCallStub(false).ToInt64(); } else if (directHostAddress == DirectHostAddress.TailCallStub) { hostAddress = DirectCallStubs.DirectCallStub(true).ToInt64(); } else if (directHostAddress == DirectHostAddress.Host) { if (funcs.TryGetValue((ulong)guestAddress, out TranslatedFunction func)) { hostAddress = func.FuncPtr.ToInt64(); } else { throw new KeyNotFoundException($"({nameof(guestAddress)} = 0x{(ulong)guestAddress:X16})"); } } else { throw new InvalidOperationException(nameof(directHostAddress)); } IntPtr addr = jumpTable.GetEntryAddressJumpTable(entry); Marshal.WriteInt64(addr, 0, guestAddress); Marshal.WriteInt64(addr, 8, hostAddress); } } public void WriteDynamicTable(JumpTable jumpTable) { if (JumpTable.DynamicTableElems > 1) { throw new NotSupportedException(); } jumpTable.ExpandIfNeededDynamicTable(DynTableEnd); int entry = 0; foreach (var item in _dynamicTable) { entry += 1; long guestAddress = item.Key; IndirectHostAddress indirectHostAddress = item.Value; long hostAddress; if (indirectHostAddress == IndirectHostAddress.CallStub) { hostAddress = DirectCallStubs.IndirectCallStub(false).ToInt64(); } else if (indirectHostAddress == IndirectHostAddress.TailCallStub) { hostAddress = DirectCallStubs.IndirectCallStub(true).ToInt64(); } else { throw new InvalidOperationException(nameof(indirectHostAddress)); } IntPtr addr = jumpTable.GetEntryAddressDynamicTable(entry); Marshal.WriteInt64(addr, 0, guestAddress); Marshal.WriteInt64(addr, 8, hostAddress); } } public void ReadJumpTable(JumpTable jumpTable) { _jumpTable.Clear(); for (int entry = 1; entry <= jumpTable.TableEnd; entry++) { IntPtr addr = jumpTable.GetEntryAddressJumpTable(entry); long guestAddress = Marshal.ReadInt64(addr, 0); long hostAddress = Marshal.ReadInt64(addr, 8); DirectHostAddress directHostAddress; if (hostAddress == DirectCallStubs.DirectCallStub(false).ToInt64()) { directHostAddress = DirectHostAddress.CallStub; } else if (hostAddress == DirectCallStubs.DirectCallStub(true).ToInt64()) { directHostAddress = DirectHostAddress.TailCallStub; } else { directHostAddress = DirectHostAddress.Host; } _jumpTable.Add(new KeyValuePair(guestAddress, directHostAddress)); } } public void ReadDynamicTable(JumpTable jumpTable) { if (JumpTable.DynamicTableElems > 1) { throw new NotSupportedException(); } _dynamicTable.Clear(); for (int entry = 1; entry <= jumpTable.DynTableEnd; entry++) { IntPtr addr = jumpTable.GetEntryAddressDynamicTable(entry); long guestAddress = Marshal.ReadInt64(addr, 0); long hostAddress = Marshal.ReadInt64(addr, 8); IndirectHostAddress indirectHostAddress; if (hostAddress == DirectCallStubs.IndirectCallStub(false).ToInt64()) { indirectHostAddress = IndirectHostAddress.CallStub; } else if (hostAddress == DirectCallStubs.IndirectCallStub(true).ToInt64()) { indirectHostAddress = IndirectHostAddress.TailCallStub; } else { throw new InvalidOperationException($"({nameof(hostAddress)} = 0x{hostAddress:X16})"); } _dynamicTable.Add(new KeyValuePair(guestAddress, indirectHostAddress)); } } private enum DirectHostAddress { CallStub, TailCallStub, Host } private enum IndirectHostAddress { CallStub, TailCallStub } } }