Cosmos/source/Indy.IL2CPU.IL.X86/Box.cs
2007-11-18 16:51:28 +00:00

50 lines
No EOL
1.8 KiB
C#

using System;
using System.Linq;
using Indy.IL2CPU.Assembler;
using Mono.Cecil;
using Mono.Cecil.Cil;
using CPU = Indy.IL2CPU.Assembler;
using CPUx86 = Indy.IL2CPU.Assembler.X86;
using Instruction = Mono.Cecil.Cil.Instruction;
namespace Indy.IL2CPU.IL.X86 {
[OpCode(Code.Box, false)]
public class Box: Op {
private int mTheSize;
private int mTypeId;
public Box(Instruction aInstruction, MethodInformation aMethodInfo)
: base(aInstruction, aMethodInfo) {
TypeReference xTypeRef = aInstruction.Operand as TypeReference;
if (xTypeRef == null) {
throw new Exception("Couldn't determine Type!");
}
if (xTypeRef is GenericParameter) {
// todo: implement support for generics
mTheSize = 4;
} else {
mTheSize = Engine.GetFieldStorageSize(xTypeRef);
}
if (((mTheSize / 4) * 4) != mTheSize) {
throw new Exception("Incorrect Datasize. ( ((mTheSize / 4) * 4) === mTheSize should evaluate to true!");
}
//if (!(xTypeRef is GenericParameter)) {
mTypeId = Engine.RegisterType(Engine.GetDefinitionFromTypeReference(xTypeRef));
//}
}
public override void DoAssemble() {
new CPUx86.Pushd("0x" + (mTheSize + ObjectImpl.FieldDataOffset).ToString("X").ToUpper());
new CPUx86.Call(CPU.Label.GenerateLabelName(RuntimeEngineRefs.Heap_AllocNewObjectRef));
new CPUx86.Move("dword", CPUx86.Registers.AtEAX, "0x" + mTypeId.ToString("X"));
new CPUx86.Move("dword", "[eax + 4]", "0x" + InstanceTypeEnum.BoxedValueType.ToString("X"));
for (int i = 0; i < (mTheSize / 4); i++) {
new CPUx86.Pop(CPUx86.Registers.EDX);
new CPUx86.Move("dword", "[eax + 0x" + (ObjectImpl.FieldDataOffset + (i * 4)).ToString("X") + "]", "edx");
}
Assembler.StackSizes.Pop();
new CPUx86.Pushd(CPUx86.Registers.EAX);
Assembler.StackSizes.Push(4);
}
}
}