using System; using System.IO; using Mono.Cecil; using Mono.Cecil.Cil; using CPU = Indy.IL2CPU.Assembler; using CPUx86 = Indy.IL2CPU.Assembler.X86; namespace Indy.IL2CPU.IL.X86 { [OpCode(Code.Unbox)] public class Unbox: Op { private int mTypeId; private string mThisLabel; private string mNextOpLabel; private int mTypeSize; public Unbox(Mono.Cecil.Cil.Instruction aInstruction, MethodInformation aMethodInfo) : base(aInstruction, aMethodInfo) { TypeReference xType = aInstruction.Operand as TypeReference; if(xType==null) { throw new Exception("Unable to determine Type!"); } TypeDefinition xTypeDef = Engine.GetDefinitionFromTypeReference(xType); mTypeSize = Engine.GetFieldStorageSize(xTypeDef); mTypeId = Engine.RegisterType(xTypeDef); mThisLabel = GetInstructionLabel(aInstruction); mNextOpLabel = GetInstructionLabel(aInstruction.Next); } public override void DoAssemble() { string mReturnNullLabel = mThisLabel + "_ReturnNull"; new CPUx86.Pop(CPUx86.Registers.EAX); new CPUx86.Compare(CPUx86.Registers.EAX, "0"); new CPUx86.JumpIfZero(mReturnNullLabel); new CPUx86.Pushd(CPUx86.Registers.AtEAX); Assembler.StackSizes.Push(4); new CPUx86.Pushd("0" + mTypeId + "h"); Assembler.StackSizes.Push(4); MethodDefinition xMethodIsInstance = Engine.GetMethodDefinition(Engine.GetTypeDefinition("", "Indy.IL2CPU.VTablesImpl"), "IsInstance", "System.Int32", "System.Int32"); Engine.QueueMethod(xMethodIsInstance); Op xOp = new Call(xMethodIsInstance); xOp.Assembler = Assembler; xOp.Assemble(); new CPUx86.Pop(CPUx86.Registers.EAX); Assembler.StackSizes.Pop(); new CPUx86.Compare(CPUx86.Registers.EAX, "0"); new CPUx86.JumpIfEquals(mReturnNullLabel); new CPUx86.Pushd(CPUx86.Registers.EAX); Assembler.StackSizes.Push(mTypeSize); new CPUx86.JumpAlways(mNextOpLabel); new CPU.Label(mReturnNullLabel); new CPUx86.Pushd("0"); Assembler.StackSizes.Push(0); } } }