Cosmos/source/Indy.IL2CPU/IL/x86/Isinst.cs
mterwoord_cp 5928581f2f
2008-11-07 12:27:08 +00:00

81 lines
No EOL
3.9 KiB
C#

using System;
using System.Collections.Generic;
using System.IO;
using CPU = Indy.IL2CPU.Assembler;
using CPUx86 = Indy.IL2CPU.Assembler.X86;
using System.Reflection;
using Indy.IL2CPU.Assembler;
namespace Indy.IL2CPU.IL.X86 {
[OpCode(OpCodeEnum.Isinst)]
public class Isinst : Op {
private int mTypeId;
private string mThisLabel;
private string mNextOpLabel;
private int mCurrentILOffset;
private bool mDebugMode;
public static void ScanOp(ILReader aReader,
MethodInformation aMethodInfo,
SortedList<string, object> aMethodData) {
Type xType = aReader.OperandValueType;
if (xType == null) {
throw new Exception("Unable to determine Type!");
}
Engine.RegisterType(xType);
Call.ScanOp(Engine.GetMethodBase(typeof(VTablesImpl),
"IsInstance",
"System.Int32",
"System.Int32"));
Newobj.ScanOp(typeof(InvalidCastException).GetConstructor(new Type[0]));
}
public Isinst(ILReader aReader,
MethodInformation aMethodInfo)
: base(aReader,
aMethodInfo) {
Type xType = aReader.OperandValueType;
if (xType == null) {
throw new Exception("Unable to determine Type!");
}
Type xTypeDef = xType;
mTypeId = Engine.RegisterType(xTypeDef);
mThisLabel = GetInstructionLabel(aReader.Position);
mNextOpLabel = GetInstructionLabel(aReader.NextPosition);
mCurrentILOffset = (int)aReader.Position;
mDebugMode = aMethodInfo.DebugMode;
}
public override void DoAssemble() {
string mReturnNullLabel = mThisLabel + "_ReturnNull";
new CPUx86.Move { DestinationReg = CPUx86.Registers.EAX, SourceReg = CPUx86.Registers.ESP, SourceIsIndirect = true };
new CPUx86.Compare { DestinationReg = CPUx86.Registers.EAX, SourceValue = 0 };
new CPUx86.JumpIfZero(mReturnNullLabel);
new CPUx86.Push { DestinationReg = CPUx86.Registers.EAX, DestinationIsIndirect = true };
new CPUx86.Push { DestinationValue = (uint)mTypeId };
Assembler.StackContents.Push(new StackContent(4,
typeof(object)));
Assembler.StackContents.Push(new StackContent(4,
typeof(object)));
MethodBase xMethodIsInstance = Engine.GetMethodBase(typeof(VTablesImpl),
"IsInstance",
"System.Int32",
"System.Int32");
Engine.QueueMethod(xMethodIsInstance);
Op xOp = new Call(xMethodIsInstance,
(uint)mCurrentILOffset,
mDebugMode,
mThisLabel+ "_After_IsInstance_Call");
xOp.Assembler = Assembler;
xOp.Assemble();
new Label(mThisLabel + "_After_IsInstance_Call");
Assembler.StackContents.Pop();
new CPUx86.Pop { DestinationReg = CPUx86.Registers.EAX };
new CPUx86.Compare { DestinationReg = CPUx86.Registers.EAX, SourceValue = 0 };
new CPUx86.JumpIfNotEqual(mNextOpLabel);
new CPU.Label(mReturnNullLabel);
new CPUx86.Add { DestinationReg = CPUx86.Registers.ESP, SourceValue = 4 };
new CPUx86.Push { DestinationValue = 0 };
}
}
}