123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127 |
- {
- 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.
- **********************************************************************}
- function GetHandleErrorAddrFrameAddr: pointer;
- begin
- result:=@HandleErrorAddrFrame;
- end;
- {$if not(defined(CPUTHUMB)) and not(defined(CPUTHUMB2))}
- Procedure SignalToHandleErrorAddrFrame_ARM(Errno : longint;addr : CodePointer; frame : Pointer); nostackframe; assembler;
- asm
- {$if FPC_FULLVERSION >= 30200}
- .code 32
- {$endif}
- // the address is of the faulting instruction, and sigreturn will
- // skip it -> start with a nop
- nop
- push {r0,r1,r2,r3}
- bl GetHandleErrorAddrFrameAddr
- // overwrite last stack slot with new return address
- str r0, [sp,#12]
- // lr := addr
- ldr lr, [sp,#4]
- pop {r0,r1,r2,pc}
- .text
- end;
- {$endif not(defined(CPUTHUMB)) and not(defined(CPUTHUMB2))}
- {$if FPC_FULLVERSION >= 30200}
- {$if defined(CPU_HAS_THUMB))}
- Procedure SignalToHandleErrorAddrFrame_Thumb(Errno : longint;addr : CodePointer; frame : Pointer); nostackframe; assembler;
- asm
- .thumb_func
- .code 16
- // the address is of the faulting instruction, and sigreturn will
- // skip it -> start with a nop
- nop
- push {r0,r1,r2,r3}
- bl GetHandleErrorAddrFrameAddr
- // overwrite last stack slot with new return address
- str r0, [sp,#12]
- // lr := addr
- ldr r0, [sp,#4]
- mov lr, r0
- pop {r0,r1,r2,pc}
- .text
- {$if not(defined(CPUTHUMB)) and not(defined(CPUTHUMB2))}
- .code 32
- {$endif not(defined(CPUTHUMB)) and not(defined(CPUTHUMB2))}
- end;
- {$endif defined(CPU_HAS_THUMB))}
- {$endif FPC_FULLVERSION >= 30200}
- procedure SignalToRunerror(Sig: longint; { _a2,_a3,_a4 : dword; } SigContext: PSigInfo; uContext : PuContext); public name '_FPC_DEFAULTSIGHANDLER'; cdecl;
- var
- res : word;
- begin
- res:=0;
- case sig of
- SIGFPE :
- begin
- { don't know how to find the different causes, maybe via xer? }
- res := 207;
- SysResetFPU;
- end;
- SIGILL:
- {$ifdef FPC_SYSTEM_FPC_MOVE}
- if in_edsp_test then
- begin
- res:=0;
- cpu_has_edsp:=false;
- inc(uContext^.uc_mcontext.arm_pc,4);
- end
- else
- {$endif FPC_SYSTEM_FPC_MOVE}
- res:=216;
- SIGSEGV :
- res:=216;
- SIGBUS:
- res:=214;
- SIGINT:
- res:=217;
- SIGQUIT:
- res:=233;
- end;
- { give runtime error at the position where the signal was raised }
- if res<>0 then
- begin
- ucontext^.uc_mcontext.arm_r0:=res;
- ucontext^.uc_mcontext.arm_r1:=uContext^.uc_mcontext.arm_pc;
- ucontext^.uc_mcontext.arm_r2:=uContext^.uc_mcontext.arm_fp;
- {$if FPC_FULLVERSION >= 30200}
- {$if not(defined(CPUTHUMB)) and not(defined(CPUTHUMB2))}
- if (ucontext^.uc_mcontext.arm_cpsr and (1 shl 5))=0 then
- begin
- ucontext^.uc_mcontext.arm_pc:=ptruint(@SignalToHandleErrorAddrFrame_ARM);
- end
- else
- {$endif not(defined(CPUTHUMB)) and not(defined(CPUTHUMB2))}
- begin
- {$if defined(CPU_HAS_THUMB))}
- ucontext^.uc_mcontext.arm_pc:=ptruint(@SignalToHandleErrorAddrFrame_Thumb);
- {$else defined(CPU_HAS_THUMB))}
- halt(217);
- {$endif defined(CPU_HAS_THUMB))}
- end;
- {$else}
- ucontext^.uc_mcontext.arm_pc:=ptruint(@SignalToHandleErrorAddrFrame_ARM);
- {$endif}
- end;
- end;
|