2018-12-11 01:58:52 +01:00
|
|
|
using ChocolArm64.State;
|
|
|
|
using System.Reflection.Emit;
|
|
|
|
|
|
|
|
namespace ChocolArm64.Translation
|
|
|
|
{
|
|
|
|
struct ILOpCodeLoadState : IILEmit
|
|
|
|
{
|
|
|
|
private ILBlock _block;
|
|
|
|
|
2019-02-28 03:03:31 +01:00
|
|
|
private bool _isSubEntry;
|
|
|
|
|
|
|
|
public ILOpCodeLoadState(ILBlock block, bool isSubEntry = false)
|
2018-12-11 01:58:52 +01:00
|
|
|
{
|
2019-02-28 03:03:31 +01:00
|
|
|
_block = block;
|
|
|
|
_isSubEntry = isSubEntry;
|
2018-12-11 01:58:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public void Emit(ILMethodBuilder context)
|
|
|
|
{
|
2019-02-28 03:03:31 +01:00
|
|
|
long intInputs = context.RegUsage.GetIntInputs(_block);
|
|
|
|
long vecInputs = context.RegUsage.GetVecInputs(_block);
|
|
|
|
|
|
|
|
if (Optimizations.AssumeStrictAbiCompliance && context.IsSubComplete)
|
|
|
|
{
|
|
|
|
intInputs = RegisterUsage.ClearCallerSavedIntRegs(intInputs, context.IsAarch64);
|
|
|
|
vecInputs = RegisterUsage.ClearCallerSavedVecRegs(vecInputs, context.IsAarch64);
|
|
|
|
}
|
2018-12-11 01:58:52 +01:00
|
|
|
|
|
|
|
LoadLocals(context, intInputs, RegisterType.Int);
|
|
|
|
LoadLocals(context, vecInputs, RegisterType.Vector);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void LoadLocals(ILMethodBuilder context, long inputs, RegisterType baseType)
|
|
|
|
{
|
|
|
|
for (int bit = 0; bit < 64; bit++)
|
|
|
|
{
|
|
|
|
long mask = 1L << bit;
|
|
|
|
|
|
|
|
if ((inputs & mask) != 0)
|
|
|
|
{
|
|
|
|
Register reg = ILMethodBuilder.GetRegFromBit(bit, baseType);
|
|
|
|
|
|
|
|
context.Generator.EmitLdarg(TranslatedSub.StateArgIdx);
|
|
|
|
context.Generator.Emit(OpCodes.Ldfld, reg.GetField());
|
|
|
|
|
|
|
|
context.Generator.EmitStloc(context.GetLocalIndex(reg));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|