fpc/rtl/darwin/aarch64/sighnd.inc
Jonas Maebe ae4c8359aa Darwin/AArch64: detect when SIGILL indicates an FPU exception
Parse the ESR (ESR_ELx, Exception Syndrome Register (ELx)), return run error
as in float_raise
2022-10-23 19:10:55 +02:00

101 lines
3.8 KiB
PHP

{
This file is part of the Free Pascal run time library.
(c) 2008 by Jonas Maebe
member of the Free Pascal development team.
See the file COPYING.FPC, included in this distribution,
for details about the copyright.
Signalhandler for Darwin/arm
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY;without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
}
procedure SignalToRunerror(Sig: cint; info : PSigInfo; SigContext:PSigContext); public name '_FPC_DEFAULTSIGHANDLER'; cdecl;
var
fpuexceptionflags: cardinal;
res : word;
begin
res:=0;
case sig of
SIGFPE :
begin
Case Info^.si_code Of
FPE_FLTDIV : Res:=208; { floating point divide by zero }
FPE_INTDIV : Res:=200; { integer divide by zero }
FPE_FLTOVF : Res:=205; { floating point overflow }
FPE_FLTUND : Res:=206; { floating point underflow }
FPE_FLTRES, { floating point inexact result }
FPE_FLTINV : Res:=207; { invalid floating point operation }
Else
Res:=207; {coprocessor error}
SigContext^.uc_mcontext^.__ns.__fpsr:=SigContext^.uc_mcontext^.__ns.__fpsr and not(fpu_exception_mask shr fpu_exception_mask_to_status_mask_shift);
end;
end;
SIGBUS:
res:=214;
SIGILL:
begin
{ right now, macOS generates SIGILL signals for fpu exceptions on AArch64.
Additionally, fpsr is 0 in the context when this happens. Fortunately,
the esr is valid, so we can decode that one. }
if (Info^.si_code=ILL_ILLTRP) and
{ Trapped AArch64 floating point exception }
((SigContext^.uc_mcontext^.__es.__esr and __ESR_EC_Mask)=__ESR_EC_TrappedAArch64_FloatingPoint) then
begin
{ the FPU status bits in the ESR are valid }
if (SigContext^.uc_mcontext^.__es.__esr and __ESR_ISS_TFV)<>0 then
begin
fpuexceptionflags:=(SigContext^.uc_mcontext^.__es.__esr shl fpu_exception_mask_to_status_mask_shift) and fpu_exception_mask;
if (fpuexceptionflags and fpu_dze)<>0 then
res:=208
else if (fpuexceptionflags and fpu_ofe)<>0 then
res:=205
else if (fpuexceptionflags and fpu_ufe)<>0 then
res:=206
else if (fpuexceptionflags and fpu_ioe)<>0 then
res:=207
else if (fpuexceptionflags and fpu_ixe)<>0 then
res:=207
else if (fpuexceptionflags and fpu_ide)<>0 then
res:=216
else
{ unknown FPU exception }
res:=207
end
else
{ unknown FPU exception }
res:=207;
end
else
res:=216;
{ for safety, always clear in case we had a SIGILL to prevent potential
infinite trap loops, even if it can cause us to miss some FPU
exceptions in case we process an actual illegal instruction }
SigContext^.uc_mcontext^.__ns.__fpsr:=SigContext^.uc_mcontext^.__ns.__fpsr and not(fpu_exception_mask shr fpu_exception_mask_to_status_mask_shift);
end;
SIGSEGV :
res:=216;
SIGINT:
res:=217;
SIGQUIT:
res:=233;
end;
{$ifdef FPC_USE_SIGPROCMASK}
reenable_signal(sig);
{$endif }
{ return to trampoline }
if res <> 0 then
begin
SigContext^.uc_mcontext^.__ss.__r[0] := res;
SigContext^.uc_mcontext^.__ss.__r[1] := SigContext^.uc_mcontext^.__ss.__pc;
SigContext^.uc_mcontext^.__ss.__r[2] := SigContext^.uc_mcontext^.__ss.__fp;
pointer(SigContext^.uc_mcontext^.__ss.__pc) := @HandleErrorAddrFrame;
end;
end;