Cosmos/source/Cosmos.IL2CPU/IL/Initobj.cs
José Pedro eff94b0f11 Improved code organization.
Added method tests.
Implemented Rethrow opcode.
2017-01-27 17:01:51 +00:00

122 lines
5.8 KiB
C#

using System;
using Cosmos.IL2CPU.ILOpCodes;
using XSharp.Compiler;
using static XSharp.Compiler.XSRegisters;
namespace Cosmos.IL2CPU.X86.IL
{
[Cosmos.IL2CPU.OpCode(ILOpCode.Code.Initobj)]
public class Initobj : ILOp
{
public Initobj(Cosmos.Assembler.Assembler aAsmblr)
: base( aAsmblr )
{
}
public override void Execute(MethodInfo aMethod, ILOpCode aOpCode)
{
// MtW: for now, disable this instruction. To me, it's unclear in what context it's being used.
uint xObjSize = 0;
Type xType = ((OpType)aOpCode).Value;
xObjSize = SizeOfType(xType);
XS.Pop(EAX);
for(int i = 0; i < (xObjSize / 4); i++)
{
XS.Set(EAX, 0, destinationDisplacement: i * 4, size: RegisterSize.Int32);
}
switch(xObjSize % 4)
{
case 0:
break;
case 1:
//new CPUx86.Mov { DestinationReg = CPUx86.RegistersEnum.EAX, DestinationIsIndirect = true, DestinationDisplacement = ( int )( ( xObjSize / 4 ) * 4 ), SourceValue = 0, Size = 8 };
XS.Set(EAX, 0, destinationDisplacement: (int)((xObjSize / 4) * 4), size: RegisterSize.Byte8);
break;
case 2:
//new CPUx86.Mov { DestinationReg = CPUx86.RegistersEnum.EAX, DestinationIsIndirect = true, DestinationDisplacement = ( int )( ( xObjSize / 4 ) * 4 ), SourceValue = 0, Size = 16 };
XS.Set(EAX, 0, destinationDisplacement: (int)((xObjSize / 4) * 4), size: RegisterSize.Short16);
break;
case 3:
//new CPUx86.Mov { DestinationReg = CPUx86.RegistersEnum.EAX, DestinationIsIndirect = true, DestinationDisplacement = ( int )( ( xObjSize / 4 ) * 4 ), SourceValue = 0, Size = 8 };
//new CPUx86.Mov { DestinationReg = CPUx86.RegistersEnum.EAX, DestinationIsIndirect = true, DestinationDisplacement = ( int )( ( ( xObjSize / 4 ) * 4 ) + 1 ), SourceValue = 0, Size = 16 };
XS.Set(EAX, 0, destinationDisplacement: (int)((xObjSize / 4) * 4), size: RegisterSize.Short16);
XS.Set(EAX, 0, destinationDisplacement: (int)((xObjSize / 4) * 4 + 2), size: RegisterSize.Byte8);
break;
default:
throw new NotImplementedException("Remainder size " + xObjSize % 4 + " not supported yet! (Type = '" + xType.FullName + "')");
}
}
// using System;
// using System.Collections.Generic;
// using CPUx86 = Cosmos.Assembler.x86;
// using Cosmos.IL2CPU.Compiler;
//
// namespace Cosmos.IL2CPU.IL.X86 {
// [Cosmos.Assembler.OpCode(OpCodeEnum.Initobj)]
// public class Initobj: Op {
// private uint mObjSize;
//
// //public static void ScanOp(ILReader aReader, MethodInformation aMethodInfo, SortedList<string, object> aMethodData) {
// // Type xTypeRef = aReader.OperandValueType;
// // if (xTypeRef == null)
// // {
// // throw new Exception("Type not found!");
// // }
// // Engine.RegisterType(xTypeRef);
// //}
// private Type mType;
//
// public Initobj(ILReader aReader, MethodInformation aMethodInfo)
// : base(aReader, aMethodInfo) {
// mType = aReader.OperandValueType;
// if (mType == null)
// {
// throw new Exception("Type not found!");
// }
// mObjSize = 0;
// }
//
// public override void DoAssemble() {
// if (mType.IsValueType)
// {
// GetService<IMetaDataInfoService>().GetTypeFieldInfo(mType, out mObjSize);
// }
// Assembler.Stack.Pop();
// XS.Pop(XSRegisters.EAX);
// for (int i = 0; i < (mObjSize / 4); i++) {
// XS.Mov(XSRegisters.EAX, 0, destinationDisplacement: i * 4, size: RegisterSizes.Int32);
// }
// switch (mObjSize % 4) {
// case 1: {
// new CPUx86.Move { DestinationReg = CPUx86.Registers.EAX, DestinationIsIndirect = true, DestinationDisplacement = (int)((mObjSize / 4) * 4), SourceValue = 0, Size = 8 };
// break;
// }
// case 2: {
// new CPUx86.Move { DestinationReg = CPUx86.Registers.EAX, DestinationIsIndirect = true, DestinationDisplacement = (int)((mObjSize / 4) * 4), SourceValue = 0, Size = 16 };
// break;
// }
// case 3:
// {
// new CPUx86.Move { DestinationReg = CPUx86.Registers.EAX, DestinationIsIndirect = true, DestinationDisplacement = (int)((mObjSize / 4) * 4), SourceValue = 0, Size = 8 };
// new CPUx86.Move { DestinationReg = CPUx86.Registers.EAX, DestinationIsIndirect = true, DestinationDisplacement = (int)(((mObjSize / 4) * 4)+1), SourceValue = 0, Size = 16 };
// break;
// }
// case 0:
// break;
// default: {
// throw new Exception("Remainder size " + mObjSize % 4 + " not supported yet! (Type = '" + mType.FullName + "')");
// }
// }
// }
// }
// }
}
}