123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143 |
- {
- 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;
|