mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-10-24 06:44:38 +02:00

types in SignalToRunError. I keep the old thing for now. This help pass a few more tests that need proper support for floating point exceptions. git-svn-id: trunk@20702 -
480 lines
12 KiB
ObjectPascal
480 lines
12 KiB
ObjectPascal
Unit System;
|
|
|
|
interface
|
|
|
|
// Was needed to bootstrap with our old 2.1 fpc for BeOS
|
|
// to define real
|
|
{ $define VER2_0}
|
|
|
|
{$define FPC_IS_SYSTEM}
|
|
|
|
{$I sysunixh.inc}
|
|
|
|
|
|
type
|
|
THeapPointer = ^pointer;
|
|
var
|
|
heapstartpointer : THeapPointer;
|
|
heapstart : pointer;//external;//external name 'HEAP';
|
|
myheapsize : longint; //external;//external name 'HEAPSIZE';
|
|
myheaprealsize : longint;
|
|
heap_handle : longint;
|
|
implementation
|
|
|
|
procedure debugger(s : PChar); cdecl; external 'root' name 'debugger';
|
|
|
|
function disable_debugger(state : integer): integer; cdecl; external 'root' name 'disable_debugger';
|
|
//begin
|
|
//end;
|
|
|
|
{ OS independant parts}
|
|
|
|
{$I system.inc}
|
|
|
|
{*****************************************************************************
|
|
System Dependent Exit code
|
|
*****************************************************************************}
|
|
procedure prthaltproc;external name '_haltproc';
|
|
|
|
procedure system_exit;
|
|
begin
|
|
asm
|
|
jmp prthaltproc
|
|
end;
|
|
End;
|
|
|
|
|
|
{ OS dependant parts }
|
|
|
|
{*****************************************************************************
|
|
Heap Management
|
|
*****************************************************************************}
|
|
|
|
(*var myheapstart:pointer;
|
|
myheapsize:longint;
|
|
myheaprealsize:longint;
|
|
heap_handle:longint;
|
|
zero:longint;
|
|
|
|
|
|
{ first address of heap }
|
|
function getheapstart:pointer;
|
|
begin
|
|
getheapstart:=myheapstart;
|
|
end;
|
|
|
|
{ current length of heap }
|
|
function getheapsize:longint;
|
|
begin
|
|
getheapsize:=myheapsize;
|
|
end;
|
|
*)
|
|
|
|
|
|
(*function getheapstart:pointer;
|
|
assembler;
|
|
asm
|
|
leal HEAP,%eax
|
|
end ['EAX'];
|
|
|
|
|
|
function getheapsize:longint;
|
|
assembler;
|
|
asm
|
|
movl intern_HEAPSIZE,%eax
|
|
end ['EAX'];*)
|
|
|
|
{ function to allocate size bytes more for the program }
|
|
{ must return the first address of new data space or nil if fail }
|
|
(*function Sbrk(size : longint):pointer;
|
|
var newsize,newrealsize:longint;
|
|
s : string;
|
|
begin
|
|
WriteLn('SBRK');
|
|
Str(size, s);
|
|
WriteLn('size : ' + s);
|
|
if (myheapsize+size)<=myheaprealsize then
|
|
begin
|
|
Sbrk:=pointer(heapstart+myheapsize);
|
|
myheapsize:=myheapsize+size;
|
|
exit;
|
|
end;
|
|
newsize:=myheapsize+size;
|
|
newrealsize:=(newsize and $FFFFF000)+$1000;
|
|
case resize_area(heap_handle,newrealsize) of
|
|
B_OK :
|
|
begin
|
|
WriteLn('B_OK');
|
|
Sbrk:=pointer(heapstart+myheapsize);
|
|
myheapsize:=newsize;
|
|
myheaprealsize:=newrealsize;
|
|
exit;
|
|
end;
|
|
B_BAD_VALUE : WriteLn('B_BAD_VALUE');
|
|
B_NO_MEMORY : WriteLn('B_NO_MEMORY');
|
|
B_ERROR : WriteLn('B_ERROR');
|
|
else
|
|
begin
|
|
Sbrk:=pointer(heapstart+myheapsize);
|
|
myheapsize:=newsize;
|
|
myheaprealsize:=newrealsize;
|
|
exit;
|
|
end;
|
|
end;
|
|
|
|
// Sbrk:=nil;
|
|
end;*)
|
|
|
|
function sys_resize_area (handle:cardinal; size:longint):longint; cdecl; external name 'sys_resize_area';
|
|
|
|
//function sbrk2 (size : longint):pointer; cdecl; external name 'sbrk';
|
|
|
|
{ function to allocate size bytes more for the program }
|
|
{ must return the first address of new data space or nil if fail }
|
|
//function Sbrk(size : longint):pointer;
|
|
//var newsize,newrealsize:longint;
|
|
// s : string;
|
|
//begin
|
|
// sbrk := sbrk2(size);
|
|
(* sbrk := nil;
|
|
WriteLn('sbrk');
|
|
Str(size, s);
|
|
WriteLn('size : ' + s);
|
|
if (myheapsize+size)<=myheaprealsize then
|
|
begin
|
|
Sbrk:=heapstart+myheapsize;
|
|
myheapsize:=myheapsize+size;
|
|
exit;
|
|
end;
|
|
newsize:=myheapsize+size;
|
|
newrealsize:=(newsize and $FFFFF000)+$1000;
|
|
if sys_resize_area(heap_handle,newrealsize+$1000)=0 then
|
|
begin
|
|
WriteLn('sys_resize_area OK');
|
|
Str(longint(newrealsize), s);
|
|
WriteLn('newrealsize : $' + Hexstr(longint(newrealsize), 8));
|
|
Str(longint(heapstartpointer), s);
|
|
WriteLn('heapstart : $' + Hexstr(longint(heapstart), 8));
|
|
Str(myheapsize, s);
|
|
WriteLn('myheapsize : ' + s);
|
|
Str(myheapsize, s);
|
|
WriteLn('Total : ' + s);
|
|
WriteLn('Before fillchar');
|
|
WriteLn('sbrk : $' + Hexstr(longint(heapstart+myheapsize), 8));
|
|
Sbrk:=heapstart+myheapsize;
|
|
FillChar(sbrk^, size, #0);
|
|
WriteLn('EndFillChar');
|
|
WriteLn('sbrk : $' + Hexstr(longint(sbrk), 8));
|
|
// ReadLn(s);
|
|
myheapsize:=newsize;
|
|
Str({longint(heapstartpointer) +} myheapsize, s);
|
|
WriteLn('Total : ' + s);
|
|
myheaprealsize:=newrealsize;
|
|
exit;
|
|
end
|
|
else
|
|
begin
|
|
debugger('Bad resize_area');
|
|
WriteLn('Bad resize_area');
|
|
end;
|
|
Sbrk:=nil;
|
|
*)
|
|
//end;
|
|
|
|
{ $I text.inc}
|
|
|
|
{*****************************************************************************
|
|
UnTyped File Handling
|
|
*****************************************************************************}
|
|
|
|
|
|
{ $i file.inc}
|
|
|
|
{*****************************************************************************
|
|
Typed File Handling
|
|
*****************************************************************************}
|
|
|
|
{ $i typefile.inc}
|
|
|
|
{*****************************************************************************
|
|
Misc. System Dependent Functions
|
|
*****************************************************************************}
|
|
|
|
Function ParamCount: Longint;
|
|
var
|
|
s : string;
|
|
Begin
|
|
ParamCount := 0;
|
|
Paramcount:=argc - 1;
|
|
End;
|
|
|
|
{ variable where full path and filename and executable is stored }
|
|
{ is setup by the startup of the system unit. }
|
|
var
|
|
execpathstr : shortstring;
|
|
|
|
{$ifdef FPC_USE_LIBC}
|
|
|
|
// private; use the macros, below
|
|
function _get_image_info(image : image_id; var info : image_info; size : size_t)
|
|
: status_t; cdecl; external 'root' name '_get_image_info';
|
|
|
|
function _get_next_image_info(team : team_id; var cookie : Longint; var info : image_info; size : size_t)
|
|
: status_t; cdecl; external 'root' name '_get_next_image_info';
|
|
|
|
function get_image_info(image : image_id; var info : image_info) : status_t;
|
|
begin
|
|
Result := _get_image_info(image, info, SizeOf(info));
|
|
end;
|
|
|
|
function get_next_image_info(team : team_id; var cookie : Longint; var info : image_info) : status_t;
|
|
begin
|
|
Result := _get_next_image_info(team, cookie, info, SizeOf(info));
|
|
end;
|
|
|
|
{$endif}
|
|
|
|
{ this routine sets up the paramstr(0) string at startup }
|
|
procedure setupexecname;
|
|
var
|
|
cookie: longint;
|
|
image : image_info;
|
|
index : byte;
|
|
s : string;
|
|
begin
|
|
cookie:=0;
|
|
fillchar(image, sizeof(image_info), 0);
|
|
if get_next_image_info(0, cookie, image) = B_OK then
|
|
begin
|
|
execpathstr := strpas(@image.name);
|
|
end
|
|
else
|
|
execpathstr := '';
|
|
{ problem with Be 4.5 noted... path contains . character }
|
|
{ if file is directly executed in CWD }
|
|
index:=pos('/./',execpathstr);
|
|
if index <> 0 then
|
|
begin
|
|
{ remove the /. characters }
|
|
Delete(execpathstr,index, 2);
|
|
end;
|
|
end;
|
|
|
|
function paramstr(l: longint) : string;
|
|
var
|
|
s: string;
|
|
s1: string;
|
|
begin
|
|
|
|
{ stricly conforming POSIX applications }
|
|
{ have the executing filename as argv[0] }
|
|
if l = 0 then
|
|
begin
|
|
paramstr := execpathstr;
|
|
end
|
|
else if (l < argc) then
|
|
begin
|
|
paramstr:=strpas(argv[l]);
|
|
end
|
|
else
|
|
paramstr := '';
|
|
end;
|
|
|
|
Procedure Randomize;
|
|
Begin
|
|
randseed:=longint(Fptime(nil));
|
|
End;
|
|
|
|
function GetProcessID: SizeUInt;
|
|
begin
|
|
GetProcessID := SizeUInt (fpGetPID);
|
|
end;
|
|
|
|
{*****************************************************************************
|
|
SystemUnit Initialization
|
|
*****************************************************************************}
|
|
|
|
function reenable_signal(sig : longint) : boolean;
|
|
var
|
|
e : TSigSet;
|
|
i,j : byte;
|
|
olderrno: cint;
|
|
begin
|
|
fillchar(e,sizeof(e),#0);
|
|
{ set is 1 based PM }
|
|
dec(sig);
|
|
i:=sig mod (sizeof(cuLong) * 8);
|
|
j:=sig div (sizeof(cuLong) * 8);
|
|
e[j]:=1 shl i;
|
|
{ this routine is called from a signal handler, so must not change errno }
|
|
olderrno:=geterrno;
|
|
fpsigprocmask(SIG_UNBLOCK,@e,nil);
|
|
reenable_signal:=geterrno=0;
|
|
seterrno(olderrno);
|
|
end;
|
|
|
|
// signal handler is arch dependant due to processorexception to language
|
|
// exception translation
|
|
|
|
{$i sighnd.inc}
|
|
|
|
//void set_signal_stack(void *ptr, size_t size);
|
|
//int sigaltstack(const stack_t *ss, stack_t *oss);
|
|
|
|
procedure set_signal_stack(ptr : pointer; size : size_t); cdecl; external 'root' name 'set_signal_stack';
|
|
function sigaltstack(const stack : pstack_t; oldStack : pstack_t) : integer; cdecl; external 'root' name 'sigaltstack';
|
|
|
|
type
|
|
{$PACKRECORDS C}
|
|
TAlternateSignalStack = packed record
|
|
case Integer of
|
|
0 : (buffer : array[0..SIGSTKSZ * 4] of Char);
|
|
1 : (ld : clonglong);
|
|
2 : (l : integer);
|
|
3 : (p : pointer);
|
|
end;
|
|
|
|
var
|
|
alternate_signal_stack : TAlternateSignalStack;
|
|
|
|
procedure InstallDefaultSignalHandler(signum: longint; out oldact: SigActionRec); public name '_FPC_INSTALLDEFAULTSIGHANDLER';
|
|
var
|
|
r : integer;
|
|
st : stack_t;
|
|
act : SigActionRec;
|
|
begin
|
|
st.ss_flags := 0;
|
|
st.ss_sp := @alternate_signal_stack.buffer;
|
|
st.ss_size := SizeOf(alternate_signal_stack.buffer);
|
|
|
|
r := sigaltstack(@st, nil);
|
|
|
|
if (r <> 0) then
|
|
begin
|
|
debugger('sigaltstack error');
|
|
end;
|
|
|
|
{ Initialize the sigaction structure }
|
|
{ all flags and information set to zero }
|
|
FillChar(act, sizeof(SigActionRec), #0);
|
|
{ initialize handler }
|
|
act.sa_mask[0] := 0;
|
|
act.sa_handler := SigActionHandler(@SignalToRunError);
|
|
act.sa_flags := SA_ONSTACK;
|
|
FpSigAction(signum,@act,@oldact);
|
|
end;
|
|
|
|
var
|
|
oldsigfpe: SigActionRec; public name '_FPC_OLDSIGFPE';
|
|
oldsigsegv: SigActionRec; public name '_FPC_OLDSIGSEGV';
|
|
oldsigbus: SigActionRec; public name '_FPC_OLDSIGBUS';
|
|
oldsigill: SigActionRec; public name '_FPC_OLDSIGILL';
|
|
|
|
Procedure InstallSignals;
|
|
begin
|
|
InstallDefaultSignalHandler(SIGFPE,oldsigfpe);
|
|
InstallDefaultSignalHandler(SIGSEGV,oldsigsegv);
|
|
InstallDefaultSignalHandler(SIGBUS,oldsigbus);
|
|
InstallDefaultSignalHandler(SIGILL,oldsigill);
|
|
end;
|
|
|
|
Procedure RestoreOldSignalHandlers;
|
|
begin
|
|
FpSigAction(SIGFPE,@oldsigfpe,nil);
|
|
FpSigAction(SIGSEGV,@oldsigsegv,nil);
|
|
FpSigAction(SIGBUS,@oldsigbus,nil);
|
|
FpSigAction(SIGILL,@oldsigill,nil);
|
|
end;
|
|
|
|
procedure SysInitStdIO;
|
|
begin
|
|
{ Setup stdin, stdout and stderr, for GUI apps redirect stderr,stdout to be
|
|
displayed in and messagebox }
|
|
OpenStdIO(Input,fmInput,StdInputHandle);
|
|
OpenStdIO(Output,fmOutput,StdOutputHandle);
|
|
OpenStdIO(StdOut,fmOutput,StdOutputHandle);
|
|
OpenStdIO(StdErr,fmOutput,StdErrorHandle);
|
|
end;
|
|
|
|
function CheckInitialStkLen(stklen : SizeUInt) : SizeUInt;
|
|
begin
|
|
result := stklen;
|
|
end;
|
|
|
|
var
|
|
s : string;
|
|
begin
|
|
IsConsole := TRUE;
|
|
StackLength := CheckInitialStkLen(InitialStkLen);
|
|
StackBottom := Sptr - StackLength;
|
|
ReturnNilIfGrowHeapFails := False;
|
|
|
|
SysResetFPU;
|
|
if not(IsLibrary) then
|
|
SysInitFPU;
|
|
|
|
{ Set up signals handlers }
|
|
InstallSignals;
|
|
|
|
{ Setup heap }
|
|
myheapsize:=4096*100;// $ 20000;
|
|
myheaprealsize:=4096*100;// $ 20000;
|
|
heapstart:=nil;
|
|
heapstartpointer := nil;
|
|
// heapstartpointer := Sbrk2(4096*1);
|
|
heapstartpointer := SysOSAlloc(4096*100);
|
|
{$IFDEF FPC_USE_LIBC}
|
|
// heap_handle := create_area('fpcheap',heapstart,0,myheaprealsize,0,3);//!!
|
|
{$ELSE}
|
|
// debugger('tata'#0);
|
|
// heap_handle := create_area('fpcheap',longint(heapstartpointer),0,myheaprealsize,0,3);//!!
|
|
// case heap_handle of
|
|
// B_BAD_VALUE : WriteLn('B_BAD_VALUE');
|
|
// B_PAGE_SIZE : WriteLn('B_PAGE_SIZE');
|
|
// B_NO_MEMORY : WriteLn('B_NO_MEMORY');
|
|
// B_ERROR : WriteLn('B_ERROR');
|
|
// end;
|
|
|
|
FillChar(heapstartpointer^, myheaprealsize, #0);
|
|
// WriteLn('EndFillChar');
|
|
// WriteLn('P : $' + Hexstr(longint(heapstartpointer), 8));
|
|
// WriteLn('heapstart : $' + Hexstr(longint(heapstartpointer^), 8));
|
|
heapstart := heapstartpointer;
|
|
{$ENDIF}
|
|
// WriteLn('before InitHeap');
|
|
// case heap_handle of
|
|
// B_BAD_VALUE : WriteLn('B_BAD_VALUE');
|
|
// B_PAGE_SIZE : WriteLn('B_PAGE_SIZE');
|
|
// B_NO_MEMORY : WriteLn('B_NO_MEMORY');
|
|
// B_ERROR : WriteLn('B_ERROR');
|
|
// else
|
|
// begin
|
|
// WriteLn('ok');
|
|
// WriteLn('P : $' + Hexstr(longint(heapstartpointer), 8));
|
|
// WriteLn('heapstart : $' + Hexstr(longint(heapstartpointer^), 8));
|
|
// if heap_handle>0 then
|
|
// begin
|
|
InitHeap;
|
|
// end;
|
|
// end;
|
|
// end;
|
|
// WriteLn('after InitHeap');
|
|
// end else system_exit;
|
|
SysInitExceptions;
|
|
// WriteLn('after SysInitException');
|
|
|
|
initunicodestringmanager;
|
|
{ Setup IO }
|
|
SysInitStdIO;
|
|
{ Reset IO Error }
|
|
InOutRes:=0;
|
|
InitSystemThreads;
|
|
{$ifdef HASVARIANT}
|
|
initvariantmanager;
|
|
{$endif HASVARIANT}
|
|
setupexecname;
|
|
{ restore original signal handlers in case this is a library }
|
|
if IsLibrary then
|
|
RestoreOldSignalHandlers;
|
|
end.
|