mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-19 21:59:27 +02:00

* throw rte 206 instead of rte 216 for denormal on m68k-linux as well git-svn-id: trunk@47117 -
144 lines
4.8 KiB
PHP
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;
|
|
|
|
|