using ARMeilleure.Memory;
using ARMeilleure.State;
using ARMeilleure.Translation;
using System;
using System.Runtime.InteropServices;
namespace ARMeilleure.Instructions
{
static class NativeInterface
private class ThreadContext
public ExecutionContext Context { get; }
public IMemoryManager Memory { get; }
public Translator Translator { get; }
public ThreadContext(ExecutionContext context, IMemoryManager memory, Translator translator)
Context = context;
Memory = memory;
Translator = translator;
}
[ThreadStatic]
private static ThreadContext _context;
public static void RegisterThread(ExecutionContext context, IMemoryManager memory, Translator translator)
_context = new ThreadContext(context, memory, translator);
public static void UnregisterThread()
_context = null;
public static void Break(ulong address, int imm)
Statistics.PauseTimer();
GetContext().OnBreak(address, imm);
Statistics.ResumeTimer();
public static void SupervisorCall(ulong address, int imm)
GetContext().OnSupervisorCall(address, imm);
public static void Undefined(ulong address, int opCode)
GetContext().OnUndefined(address, opCode);
#region "System registers"
public static ulong GetCtrEl0()
return (ulong)GetContext().CtrEl0;
public static ulong GetDczidEl0()
return (ulong)GetContext().DczidEl0;
public static ulong GetFpcr()
return (ulong)GetContext().Fpcr;
public static ulong GetFpsr()
return (ulong)GetContext().Fpsr;
public static uint GetFpscr()
var context = GetContext();
uint result = (uint)(context.Fpsr & FPSR.A32Mask) | (uint)(context.Fpcr & FPCR.A32Mask);
result |= context.GetFPstateFlag(FPState.NFlag) ? (1u << 31) : 0;
result |= context.GetFPstateFlag(FPState.ZFlag) ? (1u << 30) : 0;
result |= context.GetFPstateFlag(FPState.CFlag) ? (1u << 29) : 0;
result |= context.GetFPstateFlag(FPState.VFlag) ? (1u << 28) : 0;
return result;
public static ulong GetTpidrEl0()
return (ulong)GetContext().TpidrEl0;
public static uint GetTpidrEl032()
return (uint)GetContext().TpidrEl0;
public static ulong GetTpidr()
return (ulong)GetContext().Tpidr;
public static uint GetTpidr32()
return (uint)GetContext().Tpidr;
public static ulong GetCntfrqEl0()
return GetContext().CntfrqEl0;
public static ulong GetCntpctEl0()
return GetContext().CntpctEl0;
public static ulong GetCntvctEl0()
return GetContext().CntvctEl0;
public static void SetFpcr(ulong value)
GetContext().Fpcr = (FPCR)value;
public static void SetFpsr(ulong value)
GetContext().Fpsr = (FPSR)value;
public static void SetFpscr(uint value)
context.SetFPstateFlag(FPState.NFlag, (value & (1u << 31)) != 0);
context.SetFPstateFlag(FPState.ZFlag, (value & (1u << 30)) != 0);
context.SetFPstateFlag(FPState.CFlag, (value & (1u << 29)) != 0);
context.SetFPstateFlag(FPState.VFlag, (value & (1u << 28)) != 0);
context.Fpsr = FPSR.A32Mask & (FPSR)value;
context.Fpcr = FPCR.A32Mask & (FPCR)value;
public static void SetTpidrEl0(ulong value)
GetContext().TpidrEl0 = (long)value;
public static void SetTpidrEl032(uint value)
#endregion
#region "Read"
public static byte ReadByte(ulong address)
return GetMemoryManager().Read<byte>(address);
public static ushort ReadUInt16(ulong address)
return GetMemoryManager().Read<ushort>(address);
public static uint ReadUInt32(ulong address)
return GetMemoryManager().Read<uint>(address);
public static ulong ReadUInt64(ulong address)
return GetMemoryManager().Read<ulong>(address);
public static V128 ReadVector128(ulong address)
return GetMemoryManager().Read<V128>(address);
#region "Write"
public static void WriteByte(ulong address, byte value)
GetMemoryManager().Write(address, value);
public static void WriteUInt16(ulong address, ushort value)
public static void WriteUInt32(ulong address, uint value)
public static void WriteUInt64(ulong address, ulong value)
public static void WriteVector128(ulong address, V128 value)
public static void MarkRegionAsModified(ulong address, ulong size)
GetMemoryManager().MarkRegionAsModified(address, size);
public static void ThrowInvalidMemoryAccess(ulong address)
throw new InvalidAccessException(address);
public static ulong GetFunctionAddress(ulong address)
TranslatedFunction function = _context.Translator.GetOrTranslate(address, GetContext().ExecutionMode);
return (ulong)function.FuncPtr.ToInt64();
public static ulong GetIndirectFunctionAddress(ulong address, ulong entryAddress)
ulong ptr = (ulong)function.FuncPtr.ToInt64();
if (function.HighCq)
// Rewrite the host function address in the table to point to the highCq function.
Marshal.WriteInt64((IntPtr)entryAddress, 8, (long)ptr);
return ptr;
public static bool CheckSynchronization()
context.CheckInterrupt();
return context.Running;
public static ExecutionContext GetContext()
return _context.Context;
public static IMemoryManager GetMemoryManager()
return _context.Memory;