Fix Fcvtl_V and Fcvtn_V; fix half to float conv. and add float to half conv. (full FP emu.). Add 4 FP Tests. (#468)

* Update CpuTest.cs

* Update CpuTestSimd.cs

* Superseded.

* Update AInstEmitSimdCvt.cs

* Update ASoftFloat.cs

* Nit.

* Update PackageReferences.

* Update AInstEmitSimdArithmetic.cs

* Update AVectorHelper.cs

* Update ASoftFloat.cs

* Update ASoftFallback.cs

* Update AThreadState.cs

* Create FPType.cs

* Create FPExc.cs

* Create FPCR.cs

* Create FPSR.cs

* Update ARoundMode.cs

* Update APState.cs

* Avoid an unwanted implicit cast of the operator >= to long, continuing to check for negative values. Remove a leftover.

* Nits.
This commit is contained in:
LDj3SNuD 2018-10-23 16:12:45 +02:00 committed by gdkchan
parent 374e660b78
commit 28e6b93634
14 changed files with 653 additions and 154 deletions

View file

@ -835,8 +835,6 @@ namespace ChocolArm64.Instruction
{ {
Context.EmitLdarg(ATranslatedSub.StateArgIdx); Context.EmitLdarg(ATranslatedSub.StateArgIdx);
Context.EmitCallPropGet(typeof(AThreadState), nameof(AThreadState.Fpcr));
if (Op.Size == 0) if (Op.Size == 0)
{ {
AVectorHelper.EmitCall(Context, nameof(AVectorHelper.RoundF)); AVectorHelper.EmitCall(Context, nameof(AVectorHelper.RoundF));
@ -862,8 +860,6 @@ namespace ChocolArm64.Instruction
{ {
Context.EmitLdarg(ATranslatedSub.StateArgIdx); Context.EmitLdarg(ATranslatedSub.StateArgIdx);
Context.EmitCallPropGet(typeof(AThreadState), nameof(AThreadState.Fpcr));
if (SizeF == 0) if (SizeF == 0)
{ {
AVectorHelper.EmitCall(Context, nameof(AVectorHelper.RoundF)); AVectorHelper.EmitCall(Context, nameof(AVectorHelper.RoundF));
@ -938,8 +934,6 @@ namespace ChocolArm64.Instruction
{ {
Context.EmitLdarg(ATranslatedSub.StateArgIdx); Context.EmitLdarg(ATranslatedSub.StateArgIdx);
Context.EmitCallPropGet(typeof(AThreadState), nameof(AThreadState.Fpcr));
if (Op.Size == 0) if (Op.Size == 0)
{ {
AVectorHelper.EmitCall(Context, nameof(AVectorHelper.RoundF)); AVectorHelper.EmitCall(Context, nameof(AVectorHelper.RoundF));
@ -963,8 +957,6 @@ namespace ChocolArm64.Instruction
{ {
Context.EmitLdarg(ATranslatedSub.StateArgIdx); Context.EmitLdarg(ATranslatedSub.StateArgIdx);
Context.EmitCallPropGet(typeof(AThreadState), nameof(AThreadState.Fpcr));
if (Op.Size == 0) if (Op.Size == 0)
{ {
AVectorHelper.EmitCall(Context, nameof(AVectorHelper.RoundF)); AVectorHelper.EmitCall(Context, nameof(AVectorHelper.RoundF));

View file

@ -284,11 +284,11 @@ namespace ChocolArm64.Instruction
{ {
if (Op.Size == 0) if (Op.Size == 0)
{ {
Context.EmitLdc_R4(0); Context.EmitLdc_R4(0f);
} }
else /* if (SizeF == 1) */ else /* if (Op.Size == 1) */
{ {
Context.EmitLdc_R8(0); Context.EmitLdc_R8(0d);
} }
} }
else else
@ -378,7 +378,7 @@ namespace ChocolArm64.Instruction
} }
else else
{ {
Context.EmitLdc_I8(0); Context.EmitLdc_I8(0L);
} }
AILLabel LblTrue = new AILLabel(); AILLabel LblTrue = new AILLabel();
@ -422,7 +422,7 @@ namespace ChocolArm64.Instruction
Context.Emit(OpCodes.And); Context.Emit(OpCodes.And);
Context.EmitLdc_I8(0); Context.EmitLdc_I8(0L);
Context.Emit(OpCodes.Bne_Un_S, LblTrue); Context.Emit(OpCodes.Bne_Un_S, LblTrue);
@ -455,8 +455,9 @@ namespace ChocolArm64.Instruction
int SizeF = Op.Size & 1; int SizeF = Op.Size & 1;
int Bytes = Op.GetBitsCount() >> 3; int Bytes = Op.GetBitsCount() >> 3;
int Elems = Bytes >> SizeF + 2;
for (int Index = 0; Index < Bytes >> SizeF + 2; Index++) for (int Index = 0; Index < Elems; Index++)
{ {
EmitFcmp(Context, ILOp, Index, Scalar: false); EmitFcmp(Context, ILOp, Index, Scalar: false);
} }
@ -483,11 +484,11 @@ namespace ChocolArm64.Instruction
} }
else if (SizeF == 0) else if (SizeF == 0)
{ {
Context.EmitLdc_R4(0); Context.EmitLdc_R4(0f);
} }
else /* if (SizeF == 1) */ else /* if (SizeF == 1) */
{ {
Context.EmitLdc_R8(0); Context.EmitLdc_R8(0d);
} }
AILLabel LblTrue = new AILLabel(); AILLabel LblTrue = new AILLabel();

View file

@ -78,7 +78,7 @@ namespace ChocolArm64.Instruction
int Elems = 4 >> SizeF; int Elems = 4 >> SizeF;
int Part = Context.CurrOp.RegisterSize == ARegisterSize.SIMD128 ? Elems : 0; int Part = Op.RegisterSize == ARegisterSize.SIMD128 ? Elems : 0;
for (int Index = 0; Index < Elems; Index++) for (int Index = 0; Index < Elems; Index++)
{ {
@ -87,7 +87,9 @@ namespace ChocolArm64.Instruction
EmitVectorExtractZx(Context, Op.Rn, Part + Index, 1); EmitVectorExtractZx(Context, Op.Rn, Part + Index, 1);
Context.Emit(OpCodes.Conv_U2); Context.Emit(OpCodes.Conv_U2);
Context.EmitCall(typeof(ASoftFloat), nameof(ASoftFloat.ConvertHalfToSingle)); Context.EmitLdarg(ATranslatedSub.StateArgIdx);
Context.EmitCall(typeof(ASoftFloat16_32), nameof(ASoftFloat16_32.FPConvert));
} }
else /* if (SizeF == 1) */ else /* if (SizeF == 1) */
{ {
@ -96,8 +98,11 @@ namespace ChocolArm64.Instruction
Context.Emit(OpCodes.Conv_R8); Context.Emit(OpCodes.Conv_R8);
} }
EmitVectorInsertF(Context, Op.Rd, Index, SizeF); EmitVectorInsertTmpF(Context, Index, SizeF);
} }
Context.EmitLdvectmp();
Context.EmitStvec(Op.Rd);
} }
public static void Fcvtms_Gp(AILEmitterCtx Context) public static void Fcvtms_Gp(AILEmitterCtx Context)
@ -118,28 +123,39 @@ namespace ChocolArm64.Instruction
int Elems = 4 >> SizeF; int Elems = 4 >> SizeF;
int Part = Context.CurrOp.RegisterSize == ARegisterSize.SIMD128 ? Elems : 0; int Part = Op.RegisterSize == ARegisterSize.SIMD128 ? Elems : 0;
if (Part != 0)
{
Context.EmitLdvec(Op.Rd);
Context.EmitStvectmp();
}
for (int Index = 0; Index < Elems; Index++) for (int Index = 0; Index < Elems; Index++)
{ {
EmitVectorExtractF(Context, Op.Rd, Index, SizeF); EmitVectorExtractF(Context, Op.Rn, Index, SizeF);
if (SizeF == 0) if (SizeF == 0)
{ {
//TODO: This need the half precision floating point type, Context.EmitLdarg(ATranslatedSub.StateArgIdx);
//that is not yet supported on .NET. We should probably
//do our own implementation on the meantime. Context.EmitCall(typeof(ASoftFloat32_16), nameof(ASoftFloat32_16.FPConvert));
throw new NotImplementedException();
Context.Emit(OpCodes.Conv_U8);
EmitVectorInsertTmp(Context, Part + Index, 1);
} }
else /* if (SizeF == 1) */ else /* if (SizeF == 1) */
{ {
Context.Emit(OpCodes.Conv_R4); Context.Emit(OpCodes.Conv_R4);
EmitVectorInsertF(Context, Op.Rd, Part + Index, 0); EmitVectorInsertTmpF(Context, Part + Index, 0);
} }
} }
if (Op.RegisterSize == ARegisterSize.SIMD64) Context.EmitLdvectmp();
Context.EmitStvec(Op.Rd);
if (Part == 0)
{ {
EmitVectorZeroUpper(Context, Op.Rd); EmitVectorZeroUpper(Context, Op.Rd);
} }
@ -445,8 +461,9 @@ namespace ChocolArm64.Instruction
int FBits = GetFBits(Context); int FBits = GetFBits(Context);
int Bytes = Op.GetBitsCount() >> 3; int Bytes = Op.GetBitsCount() >> 3;
int Elems = Bytes >> SizeI;
for (int Index = 0; Index < (Bytes >> SizeI); Index++) for (int Index = 0; Index < Elems; Index++)
{ {
EmitVectorExtract(Context, Op.Rn, Index, SizeI, Signed); EmitVectorExtract(Context, Op.Rn, Index, SizeI, Signed);
@ -534,8 +551,9 @@ namespace ChocolArm64.Instruction
int FBits = GetFBits(Context); int FBits = GetFBits(Context);
int Bytes = Op.GetBitsCount() >> 3; int Bytes = Op.GetBitsCount() >> 3;
int Elems = Bytes >> SizeI;
for (int Index = 0; Index < (Bytes >> SizeI); Index++) for (int Index = 0; Index < Elems; Index++)
{ {
EmitVectorExtractF(Context, Op.Rn, Index, SizeF); EmitVectorExtractF(Context, Op.Rn, Index, SizeF);
@ -640,11 +658,11 @@ namespace ChocolArm64.Instruction
{ {
if (Size == 0) if (Size == 0)
{ {
Context.EmitLdc_R4(MathF.Pow(2, FBits)); Context.EmitLdc_R4(MathF.Pow(2f, FBits));
} }
else if (Size == 1) else if (Size == 1)
{ {
Context.EmitLdc_R8(Math.Pow(2, FBits)); Context.EmitLdc_R8(Math.Pow(2d, FBits));
} }
else else
{ {
@ -661,11 +679,11 @@ namespace ChocolArm64.Instruction
{ {
if (Size == 0) if (Size == 0)
{ {
Context.EmitLdc_R4(1f / MathF.Pow(2, FBits)); Context.EmitLdc_R4(1f / MathF.Pow(2f, FBits));
} }
else if (Size == 1) else if (Size == 1)
{ {
Context.EmitLdc_R8(1 / Math.Pow(2, FBits)); Context.EmitLdc_R8(1d / Math.Pow(2d, FBits));
} }
else else
{ {

View file

@ -1274,8 +1274,6 @@ namespace ChocolArm64.Instruction
{ {
ThrowIfInvalid(Index, Size); ThrowIfInvalid(Index, Size);
IAOpCodeSimd Op = (IAOpCodeSimd)Context.CurrOp;
Context.EmitLdvec(Reg); Context.EmitLdvec(Reg);
Context.EmitLdc_I4(Index); Context.EmitLdc_I4(Index);
Context.EmitLdc_I4(Size); Context.EmitLdc_I4(Size);
@ -1470,12 +1468,12 @@ namespace ChocolArm64.Instruction
private static void ThrowIfInvalid(int Index, int Size) private static void ThrowIfInvalid(int Index, int Size)
{ {
if ((uint)Size > 3) if ((uint)Size > 3u)
{ {
throw new ArgumentOutOfRangeException(nameof(Size)); throw new ArgumentOutOfRangeException(nameof(Size));
} }
if ((uint)Index >= 16 >> Size) if ((uint)Index >= 16u >> Size)
{ {
throw new ArgumentOutOfRangeException(nameof(Index)); throw new ArgumentOutOfRangeException(nameof(Index));
} }
@ -1483,12 +1481,12 @@ namespace ChocolArm64.Instruction
private static void ThrowIfInvalidF(int Index, int Size) private static void ThrowIfInvalidF(int Index, int Size)
{ {
if ((uint)Size > 1) if ((uint)Size > 1u)
{ {
throw new ArgumentOutOfRangeException(nameof(Size)); throw new ArgumentOutOfRangeException(nameof(Size));
} }
if ((uint)Index >= 4 >> Size) if ((uint)Index >= 4u >> Size)
{ {
throw new ArgumentOutOfRangeException(nameof(Index)); throw new ArgumentOutOfRangeException(nameof(Index));
} }

View file

@ -112,13 +112,13 @@ namespace ChocolArm64.Instruction
if (op > TMaxValue) if (op > TMaxValue)
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
return TMaxValue; return TMaxValue;
} }
else if (op < TMinValue) else if (op < TMinValue)
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
return TMinValue; return TMinValue;
} }
@ -137,13 +137,13 @@ namespace ChocolArm64.Instruction
if (op > (long)TMaxValue) if (op > (long)TMaxValue)
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
return TMaxValue; return TMaxValue;
} }
else if (op < (long)TMinValue) else if (op < (long)TMinValue)
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
return TMinValue; return TMinValue;
} }
@ -161,7 +161,7 @@ namespace ChocolArm64.Instruction
if (op > (ulong)TMaxValue) if (op > (ulong)TMaxValue)
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
return TMaxValue; return TMaxValue;
} }
@ -179,7 +179,7 @@ namespace ChocolArm64.Instruction
if (op > TMaxValue) if (op > TMaxValue)
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
return TMaxValue; return TMaxValue;
} }
@ -193,7 +193,7 @@ namespace ChocolArm64.Instruction
{ {
if (op == long.MinValue) if (op == long.MinValue)
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
return long.MaxValue; return long.MaxValue;
} }
@ -209,7 +209,7 @@ namespace ChocolArm64.Instruction
if ((~(op1 ^ op2) & (op1 ^ Add)) < 0L) if ((~(op1 ^ op2) & (op1 ^ Add)) < 0L)
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
if (op1 < 0L) if (op1 < 0L)
{ {
@ -232,7 +232,7 @@ namespace ChocolArm64.Instruction
if ((Add < op1) && (Add < op2)) if ((Add < op1) && (Add < op2))
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
return ulong.MaxValue; return ulong.MaxValue;
} }
@ -248,7 +248,7 @@ namespace ChocolArm64.Instruction
if (((op1 ^ op2) & (op1 ^ Sub)) < 0L) if (((op1 ^ op2) & (op1 ^ Sub)) < 0L)
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
if (op1 < 0L) if (op1 < 0L)
{ {
@ -271,7 +271,7 @@ namespace ChocolArm64.Instruction
if (op1 < op2) if (op1 < op2)
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
return ulong.MinValue; return ulong.MinValue;
} }
@ -292,7 +292,7 @@ namespace ChocolArm64.Instruction
if ((~op2 & Add) < 0L) if ((~op2 & Add) < 0L)
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
return long.MaxValue; return long.MaxValue;
} }
@ -306,7 +306,7 @@ namespace ChocolArm64.Instruction
// op1 from (ulong)long.MaxValue + 1UL to ulong.MaxValue // op1 from (ulong)long.MaxValue + 1UL to ulong.MaxValue
// op2 from (long)ulong.MinValue to long.MaxValue // op2 from (long)ulong.MinValue to long.MaxValue
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
return long.MaxValue; return long.MaxValue;
} }
@ -319,7 +319,7 @@ namespace ChocolArm64.Instruction
if (Add > (ulong)long.MaxValue) if (Add > (ulong)long.MaxValue)
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
return long.MaxValue; return long.MaxValue;
} }
@ -341,7 +341,7 @@ namespace ChocolArm64.Instruction
if ((Add < (ulong)op1) && (Add < op2)) if ((Add < (ulong)op1) && (Add < op2))
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
return ulong.MaxValue; return ulong.MaxValue;
} }
@ -366,7 +366,7 @@ namespace ChocolArm64.Instruction
if (Add < (long)ulong.MinValue) if (Add < (long)ulong.MinValue)
{ {
SetFpsrQCFlag(State); State.SetFpsrFlag(FPSR.QC);
return ulong.MinValue; return ulong.MinValue;
} }
@ -376,13 +376,6 @@ namespace ChocolArm64.Instruction
} }
} }
} }
private static void SetFpsrQCFlag(AThreadState State)
{
const int QCFlagBit = 27;
State.Fpsr |= 1 << QCFlagBit;
}
#endregion #endregion
#region "Count" #region "Count"

View file

@ -195,41 +195,535 @@ namespace ChocolArm64.Instruction
ulong result = x_sign | (result_exp << 52) | fraction; ulong result = x_sign | (result_exp << 52) | fraction;
return BitConverter.Int64BitsToDouble((long)result); return BitConverter.Int64BitsToDouble((long)result);
} }
public static float ConvertHalfToSingle(ushort x)
{
uint x_sign = (uint)(x >> 15) & 0x0001;
uint x_exp = (uint)(x >> 10) & 0x001F;
uint x_mantissa = (uint)x & 0x03FF;
if (x_exp == 0 && x_mantissa == 0)
{
// Zero
return BitConverter.Int32BitsToSingle((int)(x_sign << 31));
} }
if (x_exp == 0x1F) static class ASoftFloat16_32
{ {
// NaN or Infinity public static float FPConvert(ushort ValueBits, AThreadState State)
return BitConverter.Int32BitsToSingle((int)((x_sign << 31) | 0x7F800000 | (x_mantissa << 13))); {
Debug.WriteLineIf(State.Fpcr != 0, $"ASoftFloat16_32.FPConvert: State.Fpcr = 0x{State.Fpcr:X8}");
double Real = ValueBits.FPUnpackCV(out FPType Type, out bool Sign, State);
float Result;
if (Type == FPType.SNaN || Type == FPType.QNaN)
{
if (State.GetFpcrFlag(FPCR.DN))
{
Result = FPDefaultNaN();
}
else
{
Result = FPConvertNaN(ValueBits);
} }
int exponent = (int)x_exp - 15; if (Type == FPType.SNaN)
if (x_exp == 0)
{ {
// Denormal FPProcessException(FPExc.InvalidOp, State);
x_mantissa <<= 1;
while ((x_mantissa & 0x0400) == 0)
{
x_mantissa <<= 1;
exponent--;
} }
x_mantissa &= 0x03FF; }
else if (Type == FPType.Infinity)
{
Result = FPInfinity(Sign);
}
else if (Type == FPType.Zero)
{
Result = FPZero(Sign);
}
else
{
Result = FPRoundCV(Real, State);
} }
uint new_exp = (uint)((exponent + 127) & 0xFF) << 23; return Result;
return BitConverter.Int32BitsToSingle((int)((x_sign << 31) | new_exp | (x_mantissa << 13))); }
private static float FPDefaultNaN()
{
return -float.NaN;
}
private static float FPInfinity(bool Sign)
{
return Sign ? float.NegativeInfinity : float.PositiveInfinity;
}
private static float FPZero(bool Sign)
{
return Sign ? -0f : +0f;
}
private static float FPMaxNormal(bool Sign)
{
return Sign ? float.MinValue : float.MaxValue;
}
private static double FPUnpackCV(this ushort ValueBits, out FPType Type, out bool Sign, AThreadState State)
{
Sign = (~(uint)ValueBits & 0x8000u) == 0u;
uint Exp16 = ((uint)ValueBits & 0x7C00u) >> 10;
uint Frac16 = (uint)ValueBits & 0x03FFu;
double Real;
if (Exp16 == 0u)
{
if (Frac16 == 0u)
{
Type = FPType.Zero;
Real = 0d;
}
else
{
Type = FPType.Nonzero; // Subnormal.
Real = Math.Pow(2d, -14) * ((double)Frac16 * Math.Pow(2d, -10));
}
}
else if (Exp16 == 0x1Fu && !State.GetFpcrFlag(FPCR.AHP))
{
if (Frac16 == 0u)
{
Type = FPType.Infinity;
Real = Math.Pow(2d, 1000);
}
else
{
Type = (~Frac16 & 0x0200u) == 0u ? FPType.QNaN : FPType.SNaN;
Real = 0d;
}
}
else
{
Type = FPType.Nonzero; // Normal.
Real = Math.Pow(2d, (int)Exp16 - 15) * (1d + (double)Frac16 * Math.Pow(2d, -10));
}
return Sign ? -Real : Real;
}
private static float FPRoundCV(double Real, AThreadState State)
{
const int MinimumExp = -126;
const int E = 8;
const int F = 23;
bool Sign;
double Mantissa;
if (Real < 0d)
{
Sign = true;
Mantissa = -Real;
}
else
{
Sign = false;
Mantissa = Real;
}
int Exponent = 0;
while (Mantissa < 1d)
{
Mantissa *= 2d;
Exponent--;
}
while (Mantissa >= 2d)
{
Mantissa /= 2d;
Exponent++;
}
if (State.GetFpcrFlag(FPCR.FZ) && Exponent < MinimumExp)
{
State.SetFpsrFlag(FPSR.UFC);
return FPZero(Sign);
}
uint BiasedExp = (uint)Math.Max(Exponent - MinimumExp + 1, 0);
if (BiasedExp == 0u)
{
Mantissa /= Math.Pow(2d, MinimumExp - Exponent);
}
uint IntMant = (uint)Math.Floor(Mantissa * Math.Pow(2d, F));
double Error = Mantissa * Math.Pow(2d, F) - (double)IntMant;
if (BiasedExp == 0u && (Error != 0d || State.GetFpcrFlag(FPCR.UFE)))
{
FPProcessException(FPExc.Underflow, State);
}
bool OverflowToInf;
bool RoundUp;
switch (State.FPRoundingMode())
{
default:
case ARoundMode.ToNearest:
RoundUp = (Error > 0.5d || (Error == 0.5d && (IntMant & 1u) == 1u));
OverflowToInf = true;
break;
case ARoundMode.TowardsPlusInfinity:
RoundUp = (Error != 0d && !Sign);
OverflowToInf = !Sign;
break;
case ARoundMode.TowardsMinusInfinity:
RoundUp = (Error != 0d && Sign);
OverflowToInf = Sign;
break;
case ARoundMode.TowardsZero:
RoundUp = false;
OverflowToInf = false;
break;
}
if (RoundUp)
{
IntMant++;
if (IntMant == (uint)Math.Pow(2d, F))
{
BiasedExp = 1u;
}
if (IntMant == (uint)Math.Pow(2d, F + 1))
{
BiasedExp++;
IntMant >>= 1;
}
}
float Result;
if (BiasedExp >= (uint)Math.Pow(2d, E) - 1u)
{
Result = OverflowToInf ? FPInfinity(Sign) : FPMaxNormal(Sign);
FPProcessException(FPExc.Overflow, State);
Error = 1d;
}
else
{
Result = BitConverter.Int32BitsToSingle(
(int)((Sign ? 1u : 0u) << 31 | (BiasedExp & 0xFFu) << 23 | (IntMant & 0x007FFFFFu)));
}
if (Error != 0d)
{
FPProcessException(FPExc.Inexact, State);
}
return Result;
}
private static float FPConvertNaN(ushort ValueBits)
{
return BitConverter.Int32BitsToSingle(
(int)(((uint)ValueBits & 0x8000u) << 16 | 0x7FC00000u | ((uint)ValueBits & 0x01FFu) << 13));
}
private static void FPProcessException(FPExc Exc, AThreadState State)
{
int Enable = (int)Exc + 8;
if ((State.Fpcr & (1 << Enable)) != 0)
{
throw new NotImplementedException("floating-point trap handling");
}
else
{
State.Fpsr |= 1 << (int)Exc;
}
}
}
static class ASoftFloat32_16
{
public static ushort FPConvert(float Value, AThreadState State)
{
Debug.WriteLineIf(State.Fpcr != 0, $"ASoftFloat32_16.FPConvert: State.Fpcr = 0x{State.Fpcr:X8}");
double Real = Value.FPUnpackCV(out FPType Type, out bool Sign, State, out uint ValueBits);
bool AltHp = State.GetFpcrFlag(FPCR.AHP);
ushort ResultBits;
if (Type == FPType.SNaN || Type == FPType.QNaN)
{
if (AltHp)
{
ResultBits = FPZero(Sign);
}
else if (State.GetFpcrFlag(FPCR.DN))
{
ResultBits = FPDefaultNaN();
}
else
{
ResultBits = FPConvertNaN(ValueBits);
}
if (Type == FPType.SNaN || AltHp)
{
FPProcessException(FPExc.InvalidOp, State);
}
}
else if (Type == FPType.Infinity)
{
if (AltHp)
{
ResultBits = (ushort)((Sign ? 1u : 0u) << 15 | 0x7FFFu);
FPProcessException(FPExc.InvalidOp, State);
}
else
{
ResultBits = FPInfinity(Sign);
}
}
else if (Type == FPType.Zero)
{
ResultBits = FPZero(Sign);
}
else
{
ResultBits = FPRoundCV(Real, State);
}
return ResultBits;
}
private static ushort FPDefaultNaN()
{
return (ushort)0x7E00u;
}
private static ushort FPInfinity(bool Sign)
{
return Sign ? (ushort)0xFC00u : (ushort)0x7C00u;
}
private static ushort FPZero(bool Sign)
{
return Sign ? (ushort)0x8000u : (ushort)0x0000u;
}
private static ushort FPMaxNormal(bool Sign)
{
return Sign ? (ushort)0xFBFFu : (ushort)0x7BFFu;
}
private static double FPUnpackCV(this float Value, out FPType Type, out bool Sign, AThreadState State, out uint ValueBits)
{
ValueBits = (uint)BitConverter.SingleToInt32Bits(Value);
Sign = (~ValueBits & 0x80000000u) == 0u;
uint Exp32 = (ValueBits & 0x7F800000u) >> 23;
uint Frac32 = ValueBits & 0x007FFFFFu;
double Real;
if (Exp32 == 0u)
{
if (Frac32 == 0u || State.GetFpcrFlag(FPCR.FZ))
{
Type = FPType.Zero;
Real = 0d;
if (Frac32 != 0u) FPProcessException(FPExc.InputDenorm, State);
}
else
{
Type = FPType.Nonzero; // Subnormal.
Real = Math.Pow(2d, -126) * ((double)Frac32 * Math.Pow(2d, -23));
}
}
else if (Exp32 == 0xFFu)
{
if (Frac32 == 0u)
{
Type = FPType.Infinity;
Real = Math.Pow(2d, 1000);
}
else
{
Type = (~Frac32 & 0x00400000u) == 0u ? FPType.QNaN : FPType.SNaN;
Real = 0d;
}
}
else
{
Type = FPType.Nonzero; // Normal.
Real = Math.Pow(2d, (int)Exp32 - 127) * (1d + (double)Frac32 * Math.Pow(2d, -23));
}
return Sign ? -Real : Real;
}
private static ushort FPRoundCV(double Real, AThreadState State)
{
const int MinimumExp = -14;
const int E = 5;
const int F = 10;
bool Sign;
double Mantissa;
if (Real < 0d)
{
Sign = true;
Mantissa = -Real;
}
else
{
Sign = false;
Mantissa = Real;
}
int Exponent = 0;
while (Mantissa < 1d)
{
Mantissa *= 2d;
Exponent--;
}
while (Mantissa >= 2d)
{
Mantissa /= 2d;
Exponent++;
}
uint BiasedExp = (uint)Math.Max(Exponent - MinimumExp + 1, 0);
if (BiasedExp == 0u)
{
Mantissa /= Math.Pow(2d, MinimumExp - Exponent);
}
uint IntMant = (uint)Math.Floor(Mantissa * Math.Pow(2d, F));
double Error = Mantissa * Math.Pow(2d, F) - (double)IntMant;
if (BiasedExp == 0u && (Error != 0d || State.GetFpcrFlag(FPCR.UFE)))
{
FPProcessException(FPExc.Underflow, State);
}
bool OverflowToInf;
bool RoundUp;
switch (State.FPRoundingMode())
{
default:
case ARoundMode.ToNearest:
RoundUp = (Error > 0.5d || (Error == 0.5d && (IntMant & 1u) == 1u));
OverflowToInf = true;
break;
case ARoundMode.TowardsPlusInfinity:
RoundUp = (Error != 0d && !Sign);
OverflowToInf = !Sign;
break;
case ARoundMode.TowardsMinusInfinity:
RoundUp = (Error != 0d && Sign);
OverflowToInf = Sign;
break;
case ARoundMode.TowardsZero:
RoundUp = false;
OverflowToInf = false;
break;
}
if (RoundUp)
{
IntMant++;
if (IntMant == (uint)Math.Pow(2d, F))
{
BiasedExp = 1u;
}
if (IntMant == (uint)Math.Pow(2d, F + 1))
{
BiasedExp++;
IntMant >>= 1;
}
}
ushort ResultBits;
if (!State.GetFpcrFlag(FPCR.AHP))
{
if (BiasedExp >= (uint)Math.Pow(2d, E) - 1u)
{
ResultBits = OverflowToInf ? FPInfinity(Sign) : FPMaxNormal(Sign);
FPProcessException(FPExc.Overflow, State);
Error = 1d;
}
else
{
ResultBits = (ushort)((Sign ? 1u : 0u) << 15 | (BiasedExp & 0x1Fu) << 10 | (IntMant & 0x03FFu));
}
}
else
{
if (BiasedExp >= (uint)Math.Pow(2d, E))
{
ResultBits = (ushort)((Sign ? 1u : 0u) << 15 | 0x7FFFu);
FPProcessException(FPExc.InvalidOp, State);
Error = 0d;
}
else
{
ResultBits = (ushort)((Sign ? 1u : 0u) << 15 | (BiasedExp & 0x1Fu) << 10 | (IntMant & 0x03FFu));
}
}
if (Error != 0d)
{
FPProcessException(FPExc.Inexact, State);
}
return ResultBits;
}
private static ushort FPConvertNaN(uint ValueBits)
{
return (ushort)((ValueBits & 0x80000000u) >> 16 | 0x7E00u | (ValueBits & 0x003FE000u) >> 13);
}
private static void FPProcessException(FPExc Exc, AThreadState State)
{
int Enable = (int)Exc + 8;
if ((State.Fpcr & (1 << Enable)) != 0)
{
throw new NotImplementedException("floating-point trap handling");
}
else
{
State.Fpsr |= 1 << (int)Exc;
}
} }
} }
@ -756,56 +1250,31 @@ namespace ChocolArm64.Instruction
return Result; return Result;
} }
private enum FPType
{
Nonzero,
Zero,
Infinity,
QNaN,
SNaN
}
private enum FPExc
{
InvalidOp,
DivideByZero,
Overflow,
Underflow,
Inexact,
InputDenorm = 7
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static float FPDefaultNaN() private static float FPDefaultNaN()
{ {
return -float.NaN; return -float.NaN;
} }
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static float FPInfinity(bool Sign) private static float FPInfinity(bool Sign)
{ {
return Sign ? float.NegativeInfinity : float.PositiveInfinity; return Sign ? float.NegativeInfinity : float.PositiveInfinity;
} }
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static float FPZero(bool Sign) private static float FPZero(bool Sign)
{ {
return Sign ? -0f : +0f; return Sign ? -0f : +0f;
} }
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static float FPTwo(bool Sign) private static float FPTwo(bool Sign)
{ {
return Sign ? -2f : +2f; return Sign ? -2f : +2f;
} }
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static float FPOnePointFive(bool Sign) private static float FPOnePointFive(bool Sign)
{ {
return Sign ? -1.5f : +1.5f; return Sign ? -1.5f : +1.5f;
} }
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static float FPNeg(this float Value) private static float FPNeg(this float Value)
{ {
return -Value; return -Value;
@ -927,8 +1396,6 @@ namespace ChocolArm64.Instruction
private static float FPProcessNaN(FPType Type, uint Op, AThreadState State) private static float FPProcessNaN(FPType Type, uint Op, AThreadState State)
{ {
const int DNBit = 25; // Default NaN mode control bit.
if (Type == FPType.SNaN) if (Type == FPType.SNaN)
{ {
Op |= 1u << 22; Op |= 1u << 22;
@ -936,7 +1403,7 @@ namespace ChocolArm64.Instruction
FPProcessException(FPExc.InvalidOp, State); FPProcessException(FPExc.InvalidOp, State);
} }
if ((State.Fpcr & (1 << DNBit)) != 0) if (State.GetFpcrFlag(FPCR.DN))
{ {
return FPDefaultNaN(); return FPDefaultNaN();
} }
@ -1482,56 +1949,31 @@ namespace ChocolArm64.Instruction
return Result; return Result;
} }
private enum FPType
{
Nonzero,
Zero,
Infinity,
QNaN,
SNaN
}
private enum FPExc
{
InvalidOp,
DivideByZero,
Overflow,
Underflow,
Inexact,
InputDenorm = 7
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static double FPDefaultNaN() private static double FPDefaultNaN()
{ {
return -double.NaN; return -double.NaN;
} }
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static double FPInfinity(bool Sign) private static double FPInfinity(bool Sign)
{ {
return Sign ? double.NegativeInfinity : double.PositiveInfinity; return Sign ? double.NegativeInfinity : double.PositiveInfinity;
} }
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static double FPZero(bool Sign) private static double FPZero(bool Sign)
{ {
return Sign ? -0d : +0d; return Sign ? -0d : +0d;
} }
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static double FPTwo(bool Sign) private static double FPTwo(bool Sign)
{ {
return Sign ? -2d : +2d; return Sign ? -2d : +2d;
} }
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static double FPOnePointFive(bool Sign) private static double FPOnePointFive(bool Sign)
{ {
return Sign ? -1.5d : +1.5d; return Sign ? -1.5d : +1.5d;
} }
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private static double FPNeg(this double Value) private static double FPNeg(this double Value)
{ {
return -Value; return -Value;
@ -1653,8 +2095,6 @@ namespace ChocolArm64.Instruction
private static double FPProcessNaN(FPType Type, ulong Op, AThreadState State) private static double FPProcessNaN(FPType Type, ulong Op, AThreadState State)
{ {
const int DNBit = 25; // Default NaN mode control bit.
if (Type == FPType.SNaN) if (Type == FPType.SNaN)
{ {
Op |= 1ul << 51; Op |= 1ul << 51;
@ -1662,7 +2102,7 @@ namespace ChocolArm64.Instruction
FPProcessException(FPExc.InvalidOp, State); FPProcessException(FPExc.InvalidOp, State);
} }
if ((State.Fpcr & (1 << DNBit)) != 0) if (State.GetFpcrFlag(FPCR.DN))
{ {
return FPDefaultNaN(); return FPDefaultNaN();
} }

View file

@ -105,9 +105,9 @@ namespace ChocolArm64.Instruction
Value < ulong.MinValue ? ulong.MinValue : (ulong)Value; Value < ulong.MinValue ? ulong.MinValue : (ulong)Value;
} }
public static double Round(double Value, int Fpcr) public static double Round(double Value, AThreadState State)
{ {
switch ((ARoundMode)((Fpcr >> 22) & 3)) switch (State.FPRoundingMode())
{ {
case ARoundMode.ToNearest: return Math.Round (Value); case ARoundMode.ToNearest: return Math.Round (Value);
case ARoundMode.TowardsPlusInfinity: return Math.Ceiling (Value); case ARoundMode.TowardsPlusInfinity: return Math.Ceiling (Value);
@ -118,9 +118,9 @@ namespace ChocolArm64.Instruction
throw new InvalidOperationException(); throw new InvalidOperationException();
} }
public static float RoundF(float Value, int Fpcr) public static float RoundF(float Value, AThreadState State)
{ {
switch ((ARoundMode)((Fpcr >> 22) & 3)) switch (State.FPRoundingMode())
{ {
case ARoundMode.ToNearest: return MathF.Round (Value); case ARoundMode.ToNearest: return MathF.Round (Value);
case ARoundMode.TowardsPlusInfinity: return MathF.Ceiling (Value); case ARoundMode.TowardsPlusInfinity: return MathF.Ceiling (Value);

View file

@ -3,7 +3,7 @@ using System;
namespace ChocolArm64.State namespace ChocolArm64.State
{ {
[Flags] [Flags]
public enum APState enum APState
{ {
VBit = 28, VBit = 28,
CBit = 29, CBit = 29,

View file

@ -1,6 +1,6 @@
namespace ChocolArm64.State namespace ChocolArm64.State
{ {
public enum ARoundMode enum ARoundMode
{ {
ToNearest = 0, ToNearest = 0,
TowardsPlusInfinity = 1, TowardsPlusInfinity = 1,

View file

@ -145,5 +145,20 @@ namespace ChocolArm64.State
{ {
Undefined?.Invoke(this, new AInstUndefinedEventArgs(Position, RawOpCode)); Undefined?.Invoke(this, new AInstUndefinedEventArgs(Position, RawOpCode));
} }
internal bool GetFpcrFlag(FPCR Flag)
{
return (Fpcr & (1 << (int)Flag)) != 0;
}
internal void SetFpsrFlag(FPSR Flag)
{
Fpsr |= 1 << (int)Flag;
}
internal ARoundMode FPRoundingMode()
{
return (ARoundMode)((Fpcr >> (int)FPCR.RMode) & 3);
}
} }
} }

11
State/FPCR.cs Normal file
View file

@ -0,0 +1,11 @@
namespace ChocolArm64.State
{
enum FPCR
{
UFE = 11,
RMode = 22,
FZ = 24,
DN = 25,
AHP = 26
}
}

12
State/FPExc.cs Normal file
View file

@ -0,0 +1,12 @@
namespace ChocolArm64.State
{
enum FPExc
{
InvalidOp = 0,
DivideByZero = 1,
Overflow = 2,
Underflow = 3,
Inexact = 4,
InputDenorm = 7
}
}

8
State/FPSR.cs Normal file
View file

@ -0,0 +1,8 @@
namespace ChocolArm64.State
{
enum FPSR
{
UFC = 3,
QC = 27
}
}

11
State/FPType.cs Normal file
View file

@ -0,0 +1,11 @@
namespace ChocolArm64.State
{
enum FPType
{
Nonzero,
Zero,
Infinity,
QNaN,
SNaN
}
}