Core: Update R4300iOp::COP1_S_MUL to handle exceptions

This commit is contained in:
zilmar 2023-03-21 10:49:49 +10:30
parent cbf67cede4
commit ce69324dbe
2 changed files with 68 additions and 45 deletions

View File

@ -2151,8 +2151,20 @@ void R4300iOp::COP1_S_MUL()
{ {
return; return;
} }
_FPCR[31] &= ~0x0003F000;
fesetround(*_RoundingModel); fesetround(*_RoundingModel);
*(float *)_FPR_S[m_Opcode.fd] = (*(float *)_FPR_S[m_Opcode.fs] * *(float *)_FPR_S[m_Opcode.ft]); feclearexcept(FE_ALL_EXCEPT);
if (!CheckFPUInput32(*(float *)_FPR_S[m_Opcode.fs]) || !CheckFPUInput32(*(float *)_FPR_S[m_Opcode.ft]))
{
return;
}
float Result = (*(float *)_FPR_S[m_Opcode.fs] * *(float *)_FPR_S[m_Opcode.ft]);
if (CheckFPUException() || CheckFPUResult32(Result))
{
return;
}
*(uint32_t *)_FPR_S[m_Opcode.fd] = *(uint32_t *)&Result;
} }
void R4300iOp::COP1_S_DIV() void R4300iOp::COP1_S_DIV()
@ -2493,7 +2505,6 @@ void R4300iOp::COP1_D_SUB()
return; return;
} }
*(uint64_t *)_FPR_D[m_Opcode.fd] = *(uint64_t *)&Result; *(uint64_t *)_FPR_D[m_Opcode.fd] = *(uint64_t *)&Result;
} }
void R4300iOp::COP1_D_MUL() void R4300iOp::COP1_D_MUL()
@ -3040,64 +3051,76 @@ bool R4300iOp::CheckFPUException(void)
bool Res = false; bool Res = false;
FPStatusReg & StatusReg = (FPStatusReg &)_FPCR[31]; FPStatusReg & StatusReg = (FPStatusReg &)_FPCR[31];
if ((Except & FE_INEXACT) != 0)
{
StatusReg.Cause.Inexact = 1;
if (StatusReg.Enable.Inexact)
{
Res = true;
}
else
{
StatusReg.Flags.Inexact = 1;
}
}
if ((Except & FE_UNDERFLOW) != 0) if ((Except & FE_UNDERFLOW) != 0)
{ {
StatusReg.Cause.Underflow = 1; if (StatusReg.FlushSubnormals == 0 || StatusReg.Enable.Underflow || StatusReg.Enable.Inexact)
if (StatusReg.Enable.Underflow)
{ {
StatusReg.Cause.UnimplementedOperation = 1;
Res = true; Res = true;
} }
else
{
StatusReg.Flags.Underflow = 1;
}
} }
if ((Except & FE_OVERFLOW) != 0)
if (!Res)
{ {
StatusReg.Cause.Overflow = 1; if ((Except & FE_INEXACT) != 0)
if (StatusReg.Enable.Overflow)
{ {
Res = true; StatusReg.Cause.Inexact = 1;
if (StatusReg.Enable.Inexact)
{
Res = true;
}
else
{
StatusReg.Flags.Inexact = 1;
}
} }
else if ((Except & FE_UNDERFLOW) != 0)
{ {
StatusReg.Flags.Overflow = 1; StatusReg.Cause.Underflow = 1;
if (StatusReg.Enable.Underflow)
{
Res = true;
}
else
{
StatusReg.Flags.Underflow = 1;
}
} }
} if ((Except & FE_OVERFLOW) != 0)
if ((Except & FE_DIVBYZERO) != 0)
{
StatusReg.Cause.DivisionByZero = 1;
if (StatusReg.Enable.DivisionByZero)
{ {
Res = true; StatusReg.Cause.Overflow = 1;
if (StatusReg.Enable.Overflow)
{
Res = true;
}
else
{
StatusReg.Flags.Overflow = 1;
}
} }
else if ((Except & FE_DIVBYZERO) != 0)
{ {
StatusReg.Flags.DivisionByZero = 1; StatusReg.Cause.DivisionByZero = 1;
if (StatusReg.Enable.DivisionByZero)
{
Res = true;
}
else
{
StatusReg.Flags.DivisionByZero = 1;
}
} }
} if ((Except & FE_INVALID) != 0)
if ((Except & FE_INVALID) != 0)
{
StatusReg.Cause.InvalidOperation = 1;
if (StatusReg.Enable.InvalidOperation)
{ {
Res = true; StatusReg.Cause.InvalidOperation = 1;
} if (StatusReg.Enable.InvalidOperation)
else {
{ Res = true;
StatusReg.Flags.InvalidOperation = 1; }
else
{
StatusReg.Flags.InvalidOperation = 1;
}
} }
} }
if (Res) if (Res)

View File

@ -29,7 +29,7 @@ union COP0Cause
unsigned ExceptionCode : 5; unsigned ExceptionCode : 5;
unsigned : 1; unsigned : 1;
unsigned PendingInterrupts : 8; unsigned PendingInterrupts : 8;
unsigned : 12; unsigned : 12;
unsigned CoprocessorUnitNumber : 2; unsigned CoprocessorUnitNumber : 2;
unsigned : 1; unsigned : 1;
unsigned BranchDelay : 1; unsigned BranchDelay : 1;