fpc/rtl/linux/m68k/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

145 lines
4.0 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.
**********************************************************************}
const
{ Bits in control register }
RoundingMode = $30;
RoundingPrecision = $c0;
{ Exception bits common to status and control registers }
InexactDecimal = $100;
InexactOperation = $200;
DivideByZero = $400;
UnderFlow = $800;
OverFlow = $1000;
OperandError = $2000;
SignalingNaN = $4000;
BranchOnUnordered = $8000;
FPU_ES_Mask = $ff00;
{ Accrued exception bit only in status register }
AE_Inexact = $8;
AE_DivideByZero = $10;
AE_Underflow = $20;
AE_Overflow = $40;
AE_InvalidOperation = $80;
FPU_AE_Mask = $F80;
reset_fpucw : longint = {InexactOperation or }DivideByZero or
OverFlow or OperandError or
SignalingNaN or BranchOnUnordered;
reset_fpust : longint = 0;
{ Bits in psr SigContext field }
PSR_Invalid = $80;
PSR_Denormal = $8;
PSR_DivisionByZero = $10;
PSR_Overflow = $40;
PSR_Underflow = $20;
{ m68k is not stack based }
PSR_StackUnderflow = $0;
PSR_StackOverflow = $0;
FPU_Status_Exception_Mask = FPU_ES_Mask or FPU_AE_Mask;
PSR_Exception_Mask =$f8;
FPU_Control_Exception_Mask = FPU_ES_Mask;
Procedure ResetFPU;
var
l_fpucw : longint;
begin
{$if defined(FPU68881) or defined(FPUCOLDFIRE)}
asm
fmove.l fpcr,l_fpucw
end;
{ Reset only exception based control bits in fpcr }
l_fpucw := (l_fpucw and not (dword(FPU_Control_Exception_Mask)))
or (reset_fpucw and FPU_Control_Exception_Mask);
asm
fmove.l l_fpucw,fpcr
{ Reset fpsr to zero }
fmove.l reset_fpust,fpsr
end;
{$endif}
end;
function GetFPUState(const SigContext : TSigContext) : dword;
begin
GetfpuState:=dword(SigContext.psr);
{$ifdef SYSTEM_DEBUG}
Writeln(stderr,'FpuState = ',GetFpuState);
{$endif SYSTEM_DEBUG}
end;
procedure SignalToRunerror(Sig: longint; Info : pointer; var SigContext: TSigContext); public name '_FPC_DEFAULTSIGHANDLER'; cdecl;
var
res : word;
fpustate : dword;
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;
fpustate:=GetFPUState(SigContext);
if (FpuState and FPU_Status_Exception_Mask) <> 0 then
begin
{ first check the more precise options }
if (FpuState and (DivideByZero or AE_DividebyZero))<>0 then
res:=208
else if (FpuState and (Overflow or AE_Overflow))<>0 then
res:=205
else if (FpuState and (Underflow or AE_Underflow))<>0 then
res:=206
else if (FpuState and PSR_Denormal)<>0 then
res:=206
{ else if (FpuState and (PSR_StackOverflow or PRS_StackUnderflow))<>0 then
res:=207, disabled, as there is no fpu stack }
else if (FpuState and (OperandError or SignalingNan or BranchOnUnordered or AE_InvalidOperation))<>0 then
res:=216
else
res:=207; {'Coprocessor Error'}
end;
ResetFPU;
end;
SIGILL,
SIGBUS,
SIGSEGV :
res:=216;
SIGINT:
res:=217;
SIGQUIT:
res:=233;
end;
reenable_signal(sig);
{ give runtime error at the position where the signal was raised }
if res<>0 then
begin
{ HandleErrorAddrFrame(res,SigContext.sc_pc,SigContext.sc_fp);}
{ fp is not saved in context record :( }
HandleError(res);
HandleError(res);
end;
end;