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.Castclass, false)] public class Castclass: Op { private int mTypeId; private string mThisLabel; private string mNextOpLabel; public Castclass(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); mTypeId = Engine.RegisterType(xTypeDef); mThisLabel = GetInstructionLabel(aInstruction); mNextOpLabel = GetInstructionLabel(aInstruction.Next); } public override void DoAssemble() { // todo: throw an exception when the class does not support the cast! string mReturnNullLabel = mThisLabel + "_ReturnNull"; Pop("eax"); Assembler.StackSizes.Pop(); Compare("eax", "0"); JumpIfZero(mReturnNullLabel); Pushd(4, "[eax]"); Pushd(4, "0" + mTypeId + "h"); 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(); Pop("eax"); Assembler.StackSizes.Pop(); Compare("eax", "0"); JumpIfEquals(mReturnNullLabel); Assembler.Add(new CPUx86.Pushd("eax")); JumpAlways(mNextOpLabel); Label(mReturnNullLabel); Assembler.Add(new CPUx86.Pushd("0")); Assembler.StackSizes.Push(4); } } }