mirror of
https://github.com/Ryujinx/ChocolArm64.git
synced 2025-01-03 16:55:43 +00:00
Add FCVT <Hd>, <Sn> and FCVT <Sd>, <Hn> Inst.; add Tests. (#692)
* Update OpCodeTable.cs * Update InstEmitSimdCvt.cs * Update CpuTestSimd.cs * Address PR feedback.
This commit is contained in:
parent
54e780a803
commit
aa2a84998c
|
@ -16,23 +16,10 @@ namespace ChocolArm64.Instructions
|
||||||
{
|
{
|
||||||
OpCodeSimd64 op = (OpCodeSimd64)context.CurrOp;
|
OpCodeSimd64 op = (OpCodeSimd64)context.CurrOp;
|
||||||
|
|
||||||
|
if (op.Size == 0 && op.Opc == 1) // Single -> Double.
|
||||||
|
{
|
||||||
if (Optimizations.UseSse2)
|
if (Optimizations.UseSse2)
|
||||||
{
|
{
|
||||||
if (op.Size == 1 && op.Opc == 0)
|
|
||||||
{
|
|
||||||
//Double -> Single.
|
|
||||||
Type[] typesCvt = new Type[] { typeof(Vector128<float>), typeof(Vector128<double>) };
|
|
||||||
|
|
||||||
VectorHelper.EmitCall(context, nameof(VectorHelper.VectorSingleZero));
|
|
||||||
context.EmitLdvec(op.Rn);
|
|
||||||
|
|
||||||
context.EmitCall(typeof(Sse2).GetMethod(nameof(Sse2.ConvertScalarToVector128Single), typesCvt));
|
|
||||||
|
|
||||||
context.EmitStvec(op.Rd);
|
|
||||||
}
|
|
||||||
else if (op.Size == 0 && op.Opc == 1)
|
|
||||||
{
|
|
||||||
//Single -> Double.
|
|
||||||
Type[] typesCvt = new Type[] { typeof(Vector128<double>), typeof(Vector128<float>) };
|
Type[] typesCvt = new Type[] { typeof(Vector128<double>), typeof(Vector128<float>) };
|
||||||
|
|
||||||
VectorHelper.EmitCall(context, nameof(VectorHelper.VectorSingleZero));
|
VectorHelper.EmitCall(context, nameof(VectorHelper.VectorSingleZero));
|
||||||
|
@ -44,17 +31,68 @@ namespace ChocolArm64.Instructions
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
//Invalid encoding.
|
EmitVectorExtractF(context, op.Rn, 0, 0);
|
||||||
throw new InvalidOperationException();
|
|
||||||
|
EmitFloatCast(context, 1);
|
||||||
|
|
||||||
|
EmitScalarSetF(context, op.Rd, 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else if (op.Size == 1 && op.Opc == 0) // Double -> Single.
|
||||||
|
{
|
||||||
|
if (Optimizations.UseSse2)
|
||||||
|
{
|
||||||
|
Type[] typesCvt = new Type[] { typeof(Vector128<float>), typeof(Vector128<double>) };
|
||||||
|
|
||||||
|
VectorHelper.EmitCall(context, nameof(VectorHelper.VectorSingleZero));
|
||||||
|
context.EmitLdvec(op.Rn);
|
||||||
|
|
||||||
|
context.EmitCall(typeof(Sse2).GetMethod(nameof(Sse2.ConvertScalarToVector128Single), typesCvt));
|
||||||
|
|
||||||
|
context.EmitStvec(op.Rd);
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
EmitVectorExtractF(context, op.Rn, 0, op.Size);
|
EmitVectorExtractF(context, op.Rn, 0, 1);
|
||||||
|
|
||||||
EmitFloatCast(context, op.Opc);
|
EmitFloatCast(context, 0);
|
||||||
|
|
||||||
EmitScalarSetF(context, op.Rd, op.Opc);
|
EmitScalarSetF(context, op.Rd, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (op.Size == 0 && op.Opc == 3) // Single -> Half.
|
||||||
|
{
|
||||||
|
EmitVectorExtractF(context, op.Rn, 0, 0);
|
||||||
|
|
||||||
|
context.EmitLdarg(TranslatedSub.StateArgIdx);
|
||||||
|
|
||||||
|
context.EmitCall(typeof(SoftFloat32_16), nameof(SoftFloat32_16.FPConvert));
|
||||||
|
|
||||||
|
context.Emit(OpCodes.Conv_U8);
|
||||||
|
EmitScalarSet(context, op.Rd, 1);
|
||||||
|
}
|
||||||
|
else if (op.Size == 3 && op.Opc == 0) // Half -> Single.
|
||||||
|
{
|
||||||
|
EmitVectorExtractZx(context, op.Rn, 0, 1);
|
||||||
|
context.Emit(OpCodes.Conv_U2);
|
||||||
|
|
||||||
|
context.EmitLdarg(TranslatedSub.StateArgIdx);
|
||||||
|
|
||||||
|
context.EmitCall(typeof(SoftFloat16_32), nameof(SoftFloat16_32.FPConvert));
|
||||||
|
|
||||||
|
EmitScalarSetF(context, op.Rd, 0);
|
||||||
|
}
|
||||||
|
else if (op.Size == 1 && op.Opc == 3) // Double -> Half.
|
||||||
|
{
|
||||||
|
throw new NotImplementedException("Double-precision to half-precision.");
|
||||||
|
}
|
||||||
|
else if (op.Size == 3 && op.Opc == 1) // Double -> Half.
|
||||||
|
{
|
||||||
|
throw new NotImplementedException("Half-precision to double-precision.");
|
||||||
|
}
|
||||||
|
else // Invalid encoding.
|
||||||
|
{
|
||||||
|
throw new InvalidOperationException($"type == {op.Size} && opc == {op.Opc}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -296,7 +296,7 @@ namespace ChocolArm64
|
||||||
SetA64("000111100x1xxxxx001000xxxxx0x000", InstEmit.Fcmp_S, typeof(OpCodeSimdReg64));
|
SetA64("000111100x1xxxxx001000xxxxx0x000", InstEmit.Fcmp_S, typeof(OpCodeSimdReg64));
|
||||||
SetA64("000111100x1xxxxx001000xxxxx1x000", InstEmit.Fcmpe_S, typeof(OpCodeSimdReg64));
|
SetA64("000111100x1xxxxx001000xxxxx1x000", InstEmit.Fcmpe_S, typeof(OpCodeSimdReg64));
|
||||||
SetA64("000111100x1xxxxxxxxx11xxxxxxxxxx", InstEmit.Fcsel_S, typeof(OpCodeSimdFcond64));
|
SetA64("000111100x1xxxxxxxxx11xxxxxxxxxx", InstEmit.Fcsel_S, typeof(OpCodeSimdFcond64));
|
||||||
SetA64("000111100x10001xx10000xxxxxxxxxx", InstEmit.Fcvt_S, typeof(OpCodeSimd64));
|
SetA64("00011110xx10001xx10000xxxxxxxxxx", InstEmit.Fcvt_S, typeof(OpCodeSimd64));
|
||||||
SetA64("x00111100x100100000000xxxxxxxxxx", InstEmit.Fcvtas_Gp, typeof(OpCodeSimdCvt64));
|
SetA64("x00111100x100100000000xxxxxxxxxx", InstEmit.Fcvtas_Gp, typeof(OpCodeSimdCvt64));
|
||||||
SetA64("x00111100x100101000000xxxxxxxxxx", InstEmit.Fcvtau_Gp, typeof(OpCodeSimdCvt64));
|
SetA64("x00111100x100101000000xxxxxxxxxx", InstEmit.Fcvtau_Gp, typeof(OpCodeSimdCvt64));
|
||||||
SetA64("0x0011100x100001011110xxxxxxxxxx", InstEmit.Fcvtl_V, typeof(OpCodeSimd64));
|
SetA64("0x0011100x100001011110xxxxxxxxxx", InstEmit.Fcvtl_V, typeof(OpCodeSimd64));
|
||||||
|
|
Loading…
Reference in a new issue