mirror of
https://github.com/ryujinx-mirror/ryujinx.git
synced 2024-12-15 17:51:16 +01:00
c1bdf19061
* Implement ARM32 memory instructions: LDM, LDR, LDRB, LDRD, LDRH, LDRSB, LDRSH, STM, STR, STRB, STRD, STRH (immediate and register + immediate variants), implement CMP (immediate and register shifted by immediate variants) * Rename some opcode classes and flag masks for consistency * Fix a few suboptimal ARM32 codegen issues, only loads should be considered on decoder when checking if Rt == PC, and only NZCV flags should be considered for comparison optimizations * Take into account Rt2 for LDRD instructions aswell when checking if the instruction changes PC * Re-align arm32 instructions on the opcode table
325 lines
8.9 KiB
C#
325 lines
8.9 KiB
C#
using ChocolArm64.Decoders;
|
|
using ChocolArm64.State;
|
|
using ChocolArm64.Translation;
|
|
using System;
|
|
using System.Reflection.Emit;
|
|
|
|
using static ChocolArm64.Instructions.InstEmit32Helper;
|
|
using static ChocolArm64.Instructions.InstEmitMemoryHelper;
|
|
|
|
namespace ChocolArm64.Instructions
|
|
{
|
|
static partial class InstEmit32
|
|
{
|
|
private const int ByteSizeLog2 = 0;
|
|
private const int HWordSizeLog2 = 1;
|
|
private const int WordSizeLog2 = 2;
|
|
private const int DWordSizeLog2 = 3;
|
|
|
|
[Flags]
|
|
enum AccessType
|
|
{
|
|
Store = 0,
|
|
Signed = 1,
|
|
Load = 2,
|
|
|
|
LoadZx = Load,
|
|
LoadSx = Load | Signed,
|
|
}
|
|
|
|
public static void Ldm(ILEmitterCtx context)
|
|
{
|
|
OpCode32MemMult op = (OpCode32MemMult)context.CurrOp;
|
|
|
|
EmitLoadFromRegister(context, op.Rn);
|
|
|
|
bool writesToPc = (op.RegisterMask & (1 << RegisterAlias.Aarch32Pc)) != 0;
|
|
|
|
bool writeBack = op.PostOffset != 0 && (op.Rn != RegisterAlias.Aarch32Pc || !writesToPc);
|
|
|
|
if (writeBack)
|
|
{
|
|
context.Emit(OpCodes.Dup);
|
|
}
|
|
|
|
context.EmitLdc_I4(op.Offset);
|
|
|
|
context.Emit(OpCodes.Add);
|
|
|
|
context.EmitSttmp();
|
|
|
|
if (writeBack)
|
|
{
|
|
context.EmitLdc_I4(op.PostOffset);
|
|
|
|
context.Emit(OpCodes.Add);
|
|
|
|
EmitStoreToRegister(context, op.Rn);
|
|
}
|
|
|
|
int mask = op.RegisterMask;
|
|
int offset = 0;
|
|
|
|
for (int register = 0; mask != 0; mask >>= 1, register++)
|
|
{
|
|
if ((mask & 1) != 0)
|
|
{
|
|
context.EmitLdarg(TranslatedSub.MemoryArgIdx);
|
|
context.EmitLdtmp();
|
|
|
|
context.EmitLdc_I4(offset);
|
|
|
|
context.Emit(OpCodes.Add);
|
|
|
|
EmitReadZxCall(context, WordSizeLog2);
|
|
|
|
EmitStoreToRegister(context, register);
|
|
|
|
offset += 4;
|
|
}
|
|
}
|
|
}
|
|
|
|
public static void Ldr(ILEmitterCtx context)
|
|
{
|
|
EmitLoadOrStore(context, WordSizeLog2, AccessType.LoadZx);
|
|
}
|
|
|
|
public static void Ldrb(ILEmitterCtx context)
|
|
{
|
|
EmitLoadOrStore(context, ByteSizeLog2, AccessType.LoadZx);
|
|
}
|
|
|
|
public static void Ldrd(ILEmitterCtx context)
|
|
{
|
|
EmitLoadOrStore(context, DWordSizeLog2, AccessType.LoadZx);
|
|
}
|
|
|
|
public static void Ldrh(ILEmitterCtx context)
|
|
{
|
|
EmitLoadOrStore(context, HWordSizeLog2, AccessType.LoadZx);
|
|
}
|
|
|
|
public static void Ldrsb(ILEmitterCtx context)
|
|
{
|
|
EmitLoadOrStore(context, ByteSizeLog2, AccessType.LoadSx);
|
|
}
|
|
|
|
public static void Ldrsh(ILEmitterCtx context)
|
|
{
|
|
EmitLoadOrStore(context, HWordSizeLog2, AccessType.LoadSx);
|
|
}
|
|
|
|
public static void Stm(ILEmitterCtx context)
|
|
{
|
|
OpCode32MemMult op = (OpCode32MemMult)context.CurrOp;
|
|
|
|
EmitLoadFromRegister(context, op.Rn);
|
|
|
|
context.EmitLdc_I4(op.Offset);
|
|
|
|
context.Emit(OpCodes.Add);
|
|
|
|
context.EmitSttmp();
|
|
|
|
int mask = op.RegisterMask;
|
|
int offset = 0;
|
|
|
|
for (int register = 0; mask != 0; mask >>= 1, register++)
|
|
{
|
|
if ((mask & 1) != 0)
|
|
{
|
|
context.EmitLdarg(TranslatedSub.MemoryArgIdx);
|
|
context.EmitLdtmp();
|
|
|
|
context.EmitLdc_I4(offset);
|
|
|
|
context.Emit(OpCodes.Add);
|
|
|
|
EmitLoadFromRegister(context, register);
|
|
|
|
EmitWriteCall(context, WordSizeLog2);
|
|
|
|
//Note: If Rn is also specified on the register list,
|
|
//and Rn is the first register on this list, then the
|
|
//value that is written to memory is the unmodified value,
|
|
//before the write back. If it is on the list, but it's
|
|
//not the first one, then the value written to memory
|
|
//varies between CPUs.
|
|
if (offset == 0 && op.PostOffset != 0)
|
|
{
|
|
//Emit write back after the first write.
|
|
EmitLoadFromRegister(context, op.Rn);
|
|
|
|
context.EmitLdc_I4(op.PostOffset);
|
|
|
|
context.Emit(OpCodes.Add);
|
|
|
|
EmitStoreToRegister(context, op.Rn);
|
|
}
|
|
|
|
offset += 4;
|
|
}
|
|
}
|
|
}
|
|
|
|
public static void Str(ILEmitterCtx context)
|
|
{
|
|
EmitLoadOrStore(context, WordSizeLog2, AccessType.Store);
|
|
}
|
|
|
|
public static void Strb(ILEmitterCtx context)
|
|
{
|
|
EmitLoadOrStore(context, ByteSizeLog2, AccessType.Store);
|
|
}
|
|
|
|
public static void Strd(ILEmitterCtx context)
|
|
{
|
|
EmitLoadOrStore(context, DWordSizeLog2, AccessType.Store);
|
|
}
|
|
|
|
public static void Strh(ILEmitterCtx context)
|
|
{
|
|
EmitLoadOrStore(context, HWordSizeLog2, AccessType.Store);
|
|
}
|
|
|
|
private static void EmitLoadOrStore(ILEmitterCtx context, int size, AccessType accType)
|
|
{
|
|
OpCode32Mem op = (OpCode32Mem)context.CurrOp;
|
|
|
|
if (op.Index || op.WBack)
|
|
{
|
|
EmitLoadFromRegister(context, op.Rn);
|
|
|
|
context.EmitLdc_I4(op.Imm);
|
|
|
|
context.Emit(op.Add ? OpCodes.Add : OpCodes.Sub);
|
|
|
|
context.EmitSttmp();
|
|
}
|
|
|
|
context.EmitLdarg(TranslatedSub.MemoryArgIdx);
|
|
|
|
if (op.Index)
|
|
{
|
|
context.EmitLdtmp();
|
|
}
|
|
else
|
|
{
|
|
EmitLoadFromRegister(context, op.Rn);
|
|
}
|
|
|
|
if ((accType & AccessType.Load) != 0)
|
|
{
|
|
if ((accType & AccessType.Signed) != 0)
|
|
{
|
|
EmitReadSx32Call(context, size);
|
|
}
|
|
else
|
|
{
|
|
EmitReadZxCall(context, size);
|
|
}
|
|
|
|
if (op.WBack)
|
|
{
|
|
context.EmitLdtmp();
|
|
|
|
EmitStoreToRegister(context, op.Rn);
|
|
}
|
|
|
|
if (size == DWordSizeLog2)
|
|
{
|
|
context.Emit(OpCodes.Dup);
|
|
|
|
context.EmitLdflg((int)PState.EBit);
|
|
|
|
ILLabel lblBigEndian = new ILLabel();
|
|
ILLabel lblEnd = new ILLabel();
|
|
|
|
context.Emit(OpCodes.Brtrue_S, lblBigEndian);
|
|
|
|
//Little endian mode.
|
|
context.Emit(OpCodes.Conv_U4);
|
|
|
|
EmitStoreToRegister(context, op.Rt);
|
|
|
|
context.EmitLsr(32);
|
|
|
|
context.Emit(OpCodes.Conv_U4);
|
|
|
|
EmitStoreToRegister(context, op.Rt | 1);
|
|
|
|
context.Emit(OpCodes.Br_S, lblEnd);
|
|
|
|
//Big endian mode.
|
|
context.MarkLabel(lblBigEndian);
|
|
|
|
context.EmitLsr(32);
|
|
|
|
context.Emit(OpCodes.Conv_U4);
|
|
|
|
EmitStoreToRegister(context, op.Rt);
|
|
|
|
context.Emit(OpCodes.Conv_U4);
|
|
|
|
EmitStoreToRegister(context, op.Rt | 1);
|
|
|
|
context.MarkLabel(lblEnd);
|
|
}
|
|
else
|
|
{
|
|
EmitStoreToRegister(context, op.Rt);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (op.WBack)
|
|
{
|
|
context.EmitLdtmp();
|
|
|
|
EmitStoreToRegister(context, op.Rn);
|
|
}
|
|
|
|
EmitLoadFromRegister(context, op.Rt);
|
|
|
|
if (size == DWordSizeLog2)
|
|
{
|
|
context.Emit(OpCodes.Conv_U8);
|
|
|
|
context.EmitLdflg((int)PState.EBit);
|
|
|
|
ILLabel lblBigEndian = new ILLabel();
|
|
ILLabel lblEnd = new ILLabel();
|
|
|
|
context.Emit(OpCodes.Brtrue_S, lblBigEndian);
|
|
|
|
//Little endian mode.
|
|
EmitLoadFromRegister(context, op.Rt | 1);
|
|
|
|
context.Emit(OpCodes.Conv_U8);
|
|
|
|
context.EmitLsl(32);
|
|
|
|
context.Emit(OpCodes.Or);
|
|
|
|
context.Emit(OpCodes.Br_S, lblEnd);
|
|
|
|
//Big endian mode.
|
|
context.MarkLabel(lblBigEndian);
|
|
|
|
context.EmitLsl(32);
|
|
|
|
EmitLoadFromRegister(context, op.Rt | 1);
|
|
|
|
context.Emit(OpCodes.Conv_U8);
|
|
|
|
context.Emit(OpCodes.Or);
|
|
|
|
context.MarkLabel(lblEnd);
|
|
}
|
|
|
|
EmitWriteCall(context, size);
|
|
}
|
|
}
|
|
}
|
|
} |