| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135 | {    This file is part of the Free Pascal run time library.    Copyright (c) 2016 by the Free Pascal development team    System Entry point for Amiga/68k, Pascal only programs    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. **********************************************************************}unit si_prc;interfaceimplementationvar  AOS_ExecBase: Pointer; public name '_ExecBase';  AOS_DosBase: Pointer; external name '_DOSBase';  realExecBase: Pointer absolute $4;  StkLen: LongInt; external name '__stklen';  sysinit_jmpbuf: jmp_buf;  ExitCode: LongInt;{ the definitions in there need AOS_Execbase }{$include execd.inc}{$include execf.inc}{$include timerd.inc}{$include doslibd.inc}{$include doslibf.inc}{$if defined(AMIGA_V1_0_ONLY) or defined(AMIGA_V1_2_ONLY)}{$define AMIGA_LEGACY}{$include legacyexech.inc}{$endif}{$ifdef AMIGA_LEGACY}var  args: pointer; public name '__fpc_args';  arglen: dword; public name '__fpc_arglen';{$endif}var  sst: TStackSwapStruct;const{$if defined(AMIGA_V1_0_ONLY)}  NEEDS_NEWER_OS = 'This program needs newer OS.'+LineEnding;{$else}{$if defined(AMIGA_V1_2_ONLY)}  NEEDS_NEWER_OS = 'This program needs OS 1.2 or newer.'+LineEnding;{$else}{$if defined(AMIGA_V2_0_ONLY)}  NEEDS_NEWER_OS = 'This program needs OS 2.04 or newer.'+LineEnding;{$else}  NEEDS_NEWER_OS = 'This program needs OS 3.0 or newer.'+LineEnding;{$endif}{$endif}{$endif}procedure PascalMain; external name 'PASCALMAIN';{ this function must be the first in this unit which contains code }function _FPC_proc_start: longint; cdecl; public name '_start';var  newStack: Pointer;  task: PTask;begin{$IFDEF AMIGA_LEGACY}  asm    move.l d0, arglen    move.l a0, args  end;{$ENDIF}  AOS_ExecBase:=realExecBase;  if PLibrary(AOS_ExecBase)^.lib_Version < AMIGA_OS_MINVERSION then    begin      AOS_DOSBase:=OpenLibrary('dos.library',0);      if AOS_DOSBase <> nil then        begin          dosWrite(dosOutput,PAnsiChar(NEEDS_NEWER_OS),length(NEEDS_NEWER_OS));          CloseLibrary(AOS_DOSBase);        end;      exit(20);    end;  newStack:=nil;  task:=FindTask(nil);  if (task^.tc_SPUpper-task^.tc_SPLower < StkLen) then    begin      newStack:=AllocVec(StkLen,MEMF_ANY);      sst.stk_Lower:=newStack;      sst.stk_Upper:=newStack+StkLen;      sst.stk_Pointer:=newStack+StkLen;      StackSwap(@sst);    end;  { Note: code between the two stackswaps only works because of the    nature of the generated code. We're accessing globals which is    safe, and the locals are either kept in reg, or accessed via    the base pointer (A5), and because we don't use the stack for    call arguments, only regs. If this CG behavior changes, this    code might break. In that case an asm-written StackSwap+call    wrapper code is the solution. (Basically the reimplementation    of AROS' NewStackSwap or MorphOS' NewPPCStackSwap.) (KB) }  if setjmp(sysinit_jmpbuf) = 0 then    PascalMain;  if newStack <> nil then    begin      StackSwap(@sst);      FreeVec(newStack);    end;  _FPC_proc_start:=ExitCode;end;procedure _FPC_proc_halt(_ExitCode: longint); cdecl; public name '_haltproc';begin  ExitCode:=_ExitCode;  longjmp(sysinit_jmpbuf,1);end;end.
 |