Cosmos/source/Cosmos.IL2CPU/IL/Ldflda.cs
2017-03-19 09:29:46 -05:00

68 lines
2.4 KiB
C#

using System;
using System.Reflection;
using Cosmos.IL2CPU.Extensions;
using XSharp.Common;
using static XSharp.Common.XSRegisters;
namespace Cosmos.IL2CPU.X86.IL
{
[Cosmos.IL2CPU.OpCode(ILOpCode.Code.Ldflda)]
public class Ldflda : ILOp
{
public Ldflda(Cosmos.Assembler.Assembler aAsmblr)
: base(aAsmblr)
{
}
public override void Execute(_MethodInfo aMethod, ILOpCode aOpCode)
{
var xOpCode = (ILOpCodes.OpField) aOpCode;
DoExecute(Assembler, aMethod, xOpCode.Value.DeclaringType, xOpCode.Value.GetFullName(), true, DebugEnabled, aOpCode.StackPopTypes[0]);
}
public static void DoExecute(Cosmos.Assembler.Assembler Assembler, _MethodInfo aMethod, Type aDeclaringType, string aField, bool aDerefValue, bool aDebugEnabled, Type aTypeOnStack)
{
var xFieldInfo = ResolveField(aDeclaringType, aField, true);
DoExecute(Assembler, aMethod, aDeclaringType, xFieldInfo, aDerefValue, aDebugEnabled, aTypeOnStack);
}
public static void DoExecute(Cosmos.Assembler.Assembler Assembler, _MethodInfo aMethod, Type aDeclaringType, _FieldInfo aField, bool aDerefValue, bool aDebugEnabled, Type aTypeOnStack)
{
XS.Comment("Field: " + aField.Id);
int xExtraOffset = 0;
var xType = aMethod.MethodBase.DeclaringType;
bool xNeedsGC = aDeclaringType.GetTypeInfo().IsClass && !aDeclaringType.GetTypeInfo().IsValueType;
if (xNeedsGC)
{
xExtraOffset = 12;
}
var xActualOffset = aField.Offset + xExtraOffset;
var xSize = aField.Size;
if ((!aTypeOnStack.IsPointer)
&& (aDeclaringType.GetTypeInfo().IsClass))
{
DoNullReferenceCheck(Assembler, aDebugEnabled, 4);
XS.Add(ESP, 4);
}
else
{
DoNullReferenceCheck(Assembler, aDebugEnabled, 0);
}
if (aDerefValue && aField.IsExternalValue)
{
XS.Set(ESP, EAX, destinationIsIndirect: true);
}
else
{
XS.Pop(EAX);
XS.Add(EAX, (uint)(xActualOffset));
XS.Push(EAX);
}
}
}
}