fpc/rtl/linux/x86_64/sighnd.inc
florian 91fe098eb4 * check first for x86 exception and only for sse if no x87 exceptions is thrown, clear all exception afterwards
* throw rte 206 instead of rte 216 for denormal on m68k-linux as well

git-svn-id: trunk@47117 -
2020-10-16 19:44:25 +00:00

144 lines
4.8 KiB
PHP

{
This file is part of the Free Pascal run time library.
Copyright (c) 1999-2000 by Michael Van Canneyt,
member of the Free Pascal development team.
Signal handler is arch dependant due to processor to language
exception conversion.
See the file COPYING.FPC, included in this distribution,
for details about the copyright.
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.
**********************************************************************}
{ $define SYSTEM_DEBUG}
{ use a trampoline which pushes the return address for proper unwinding }
Procedure SignalToHandleErrorAddrFrame (Errno : longint;addr : CodePointer; frame : Pointer); nostackframe; assembler;
asm
pushq addr
jmp HandleErrorAddrFrame
end;
const
FPU_All = $7f;
function GetFPUState(const SigContext : TSigContext) : word;
begin
if assigned(SigContext.fpstate) then
GetfpuState:=SigContext.fpstate^.swd
else
GetFPUState:=0;
{$ifdef SYSTEM_DEBUG}
if assigned(SigContext.fpstate) then
writeln('Tag: ',sigcontext.fpstate^.twd,' Cw: ',sigcontext.fpstate^.cwd);
Writeln(stderr,'FpuState = ',result);
{$endif SYSTEM_DEBUG}
end;
function GetMMState(const SigContext : TSigContext) : dword;
begin
if assigned(SigContext.fpstate) then
Result:=SigContext.fpstate^.mxcsr
else
Result:=0;
{$ifdef SYSTEM_DEBUG}
Writeln(stderr,'MMState = ',result);
{$endif SYSTEM_DEBUG}
end;
procedure SignalToRunerror(sig : longint; SigInfo: PSigInfo; SigContext: PSigContext); public name '_FPC_DEFAULTSIGHANDLER'; cdecl;
var
res,fpustate,MMState : word;
begin
res:=0;
case sig of
SIGFPE :
begin
{ this is not allways necessary but I don't know yet
how to tell if it is or not PM }
res:=200;
if SigInfo^.si_code<>FPE_INTDIV then
begin
fpustate:=GetFPUState(SigContext^);
if (FpuState and FPU_All) <> 0 then
begin
{ first check the more precise options }
if (FpuState and FPU_DivisionByZero)<>0 then
res:=208
else if (FpuState and FPU_Overflow)<>0 then
res:=205
else if (FpuState and FPU_Underflow)<>0 then
res:=206
else if (FpuState and FPU_Denormal)<>0 then
res:=206
else if (FpuState and (FPU_StackOverflow or FPU_StackUnderflow or FPU_Invalid))<>0 Then
res:=207
else
res:=207; {'Coprocessor Error'}
end
else
begin
MMState:=getMMState(SigContext^);
if (MMState and MM_ExceptionMask)<>0 then
begin
{ first check the more precise options }
if (MMState and MM_DivisionByZero)<>0 then
res:=208
else if (MMState and MM_Invalid)<>0 Then
res:=207
else if (MMState and MM_Overflow)<>0 then
res:=205
else if (MMState and MM_Underflow)<>0 then
res:=206
else if (MMState and MM_Denormal)<>0 then
res:=206
else
res:=207; {'Coprocessor Error'}
end;
end;
if assigned(SigContext^.fpstate) then
with SigContext^.fpstate^ do
begin
{$ifdef SYSTEM_DEBUG}
Writeln(stderr,'fpstate^.swd = ',swd);
{$endif SYSTEM_DEBUG}
{ actually, I am not sure if we should really touch the controll word }
cwd:=Default8087CW;
{ found by trial and error that setting to 0 means empty }
twd:=$0;
{ clear top }
swd:=swd and not($3700);
{ exceptions are handled, clear all flags
as we return from SignalToRunerrer, we have to clear the exception flags in the context }
mxcsr:=mxcsr and not(MM_ExceptionMask);
swd:=swd and not($37ff);
end;
end;
end;
SIGILL,
SIGBUS,
SIGSEGV:
res:=216;
SIGINT:
res:=217;
SIGQUIT:
res:=233;
end;
reenable_signal(sig);
if res<>0 then
begin
SigContext^.rdi := res;
SigContext^.rsi := SigContext^.rip;
SigContext^.rdx := SigContext^.rbp;
SigContext^.rip := ptruint(@SignalToHandleErrorAddrFrame);
end;
end;