mirror of
https://github.com/danbulant/Cosmos
synced 2026-05-30 21:00:30 +00:00
change StackContents to uint, able to use now mnemoric with 3 operands, shl IL near 64 bit (unknown error), add asm line to nasm error
76 lines
No EOL
3.4 KiB
C#
76 lines
No EOL
3.4 KiB
C#
using System;
|
|
using CPUx86 = Cosmos.Compiler.Assembler.X86;
|
|
using Cosmos.Compiler.Assembler;
|
|
using Cosmos.IL2CPU.IL.CustomImplementations.System;
|
|
namespace Cosmos.IL2CPU.X86.IL
|
|
{
|
|
[Cosmos.IL2CPU.OpCode( ILOpCode.Code.Ldelem_Ref )]
|
|
public class Ldelem_Ref : ILOp
|
|
{
|
|
public Ldelem_Ref( Cosmos.Compiler.Assembler.Assembler aAsmblr )
|
|
: base( aAsmblr )
|
|
{
|
|
}
|
|
public static void Assemble( Assembler aAssembler, uint aElementSize )
|
|
{
|
|
new CPUx86.Pop { DestinationReg = CPUx86.Registers.EAX };
|
|
new CPUx86.Move { DestinationReg = CPUx86.Registers.EDX, SourceValue = aElementSize };
|
|
new CPUx86.Multiply { DestinationReg = CPUx86.Registers.EDX };
|
|
|
|
//TODO: implement ObjectImpl first
|
|
new CPUx86.Add { DestinationReg = CPUx86.Registers.EAX, SourceValue = ( ObjectImpl.FieldDataOffset + 4 ) };
|
|
|
|
new CPUx86.Pop { DestinationReg = CPUx86.Registers.EDX };
|
|
new CPUx86.Add { DestinationReg = CPUx86.Registers.EDX, SourceReg = CPUx86.Registers.EAX };
|
|
new CPUx86.Move { DestinationReg = CPUx86.Registers.EAX, SourceReg = CPUx86.Registers.EDX };
|
|
uint xSizeLeft = aElementSize;
|
|
while( xSizeLeft > 0 )
|
|
{
|
|
if( xSizeLeft >= 4 )
|
|
{
|
|
new CPUx86.Push { DestinationReg = CPUx86.Registers.EAX, DestinationIsIndirect = true };
|
|
new CPUx86.Add { DestinationReg = CPUx86.Registers.EAX, SourceValue = 4 };
|
|
xSizeLeft -= 4;
|
|
}
|
|
else
|
|
{
|
|
if( xSizeLeft >= 2 )
|
|
{
|
|
new CPUx86.Move { DestinationReg = CPUx86.Registers.ECX, SourceValue = 0 };
|
|
new CPUx86.Move { DestinationReg = CPUx86.Registers.CX, SourceReg = CPUx86.Registers.EAX, SourceIsIndirect = true };
|
|
new CPUx86.Push { DestinationReg = CPUx86.Registers.ECX };
|
|
new CPUx86.Add { DestinationReg = CPUx86.Registers.EAX, SourceValue = 2 };
|
|
xSizeLeft -= 2;
|
|
}
|
|
else
|
|
{
|
|
if( xSizeLeft >= 1 )
|
|
{
|
|
new CPUx86.Move { DestinationReg = CPUx86.Registers.ECX, SourceValue = 0 };
|
|
new CPUx86.Move { DestinationReg = CPUx86.Registers.CL, SourceReg = CPUx86.Registers.EAX, SourceIsIndirect = true };
|
|
new CPUx86.Push { DestinationReg = CPUx86.Registers.ECX };
|
|
new CPUx86.Add { DestinationReg = CPUx86.Registers.EAX, SourceValue = 1 };
|
|
xSizeLeft -= 1;
|
|
}
|
|
else
|
|
{
|
|
throw new Exception( "Size left: " + xSizeLeft );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
aAssembler.Stack.Pop();
|
|
aAssembler.Stack.Pop();
|
|
#if DOTNETCOMPATIBLE
|
|
aAssembler.Stack.Push(ILOp.Align(aElementSize, 4), typeof(uint)); //TODO typeof type if aElementsize is other then 4
|
|
#else
|
|
aAssembler.Stack.Push(aElementSize, typeof(uint));
|
|
#endif
|
|
}
|
|
|
|
public override void Execute( MethodInfo aMethod, ILOpCode aOpCode )
|
|
{
|
|
Assemble( Assembler, 4 );
|
|
}
|
|
}
|
|
} |