| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662 | {    $Id$    This file is part of the Free Pascal run time library.    Copyright (c) 1999-2000 by Florian Klaempfl,    member of the Free Pascal development team.    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 system;interface{ include system-independent routine headers }{$I systemh.inc}{ include heap support headers }{$I heaph.inc}{Platform specific information}const LineEnding = #13#10;{ LFNSupport is a variable here, defined below!!! } DirectorySeparator = '\'; DriveSeparator = ':'; PathSeparator = ';'; FileNameCaseSensitive = false;const{ Default filehandles }  UnusedHandle    = $ffff;  StdInputHandle  = 0;  StdOutputHandle = 1;  StdErrorHandle  = 2;{ Default memory segments (Tp7 compatibility) }  seg0040 = $0040;  segA000 = $A000;  segB000 = $B000;  segB800 = $B800;var{ C-compatible arguments and environment }  argc  : longint;  argv  : ppchar;  envp  : ppchar;type{ Dos Extender info }  p_stub_info   = ^t_stub_info;  t_stub_info = packed record       magic         : array[0..15] of char;       size          : longint;       minstack      : longint;       memory_handle : longint;       initial_size  : longint;       minkeep       : word;       ds_selector   : word;       ds_segment    : word;       psp_selector  : word;       cs_selector   : word;       env_size      : word;       basename      : array[0..7] of char;       argv0         : array [0..15] of char;       dpmi_server   : array [0..15] of char;  end;  t_go32_info_block = packed record       size_of_this_structure_in_bytes    : longint; {offset 0}       linear_address_of_primary_screen   : longint; {offset 4}       linear_address_of_secondary_screen : longint; {offset 8}       linear_address_of_transfer_buffer  : longint; {offset 12}       size_of_transfer_buffer            : longint; {offset 16}       pid                                : longint; {offset 20}       master_interrupt_controller_base   : byte; {offset 24}       slave_interrupt_controller_base    : byte; {offset 25}       selector_for_linear_memory         : word; {offset 26}       linear_address_of_stub_info_structure : longint; {offset 28}       linear_address_of_original_psp     : longint; {offset 32}       run_mode                           : word; {offset 36}       run_mode_info                      : word; {offset 38}  end;var  stub_info       : p_stub_info;  go32_info_block : t_go32_info_block;  LFNSupport : boolean;{ Needed for CRT unit }function do_read(h,addr,len : longint) : longint;implementation{ include system independent routines }{$I system.inc}{$ASMMODE DIRECT}procedure int_stackcheck(stack_size:longint);[public,alias:'FPC_STACKCHECK'];begin{ called when trying to get local stack  if the compiler directive $S is set  this function must preserve esi !!!!  because esi is set by the calling  proc for methods  it must preserve all registers !!  With a 2048 byte safe area used to write to StdIo without crossing  the stack boundary  }  asm            pushl %eax            pushl %ebx            movl stack_size,%ebx            addl $2048,%ebx            movl %esp,%eax            subl %ebx,%eax{$ifdef SYSTEMDEBUG}            movl U_SYSTEM_LOWESTSTACK,%ebx            cmpl %eax,%ebx            jb   _is_not_lowest            movl %eax,U_SYSTEM_LOWESTSTACK            _is_not_lowest:{$endif SYSTEMDEBUG}            movl __stkbottom,%ebx            cmpl %eax,%ebx            jae  __short_on_stack            popl %ebx            popl %eax            leave            ret  $4            __short_on_stack:            { can be usefull for error recovery !! }            popl %ebx            popl %eax  end['EAX','EBX'];  HandleError(202);end;function paramcount : longint;begin  paramcount := argc - 1;end;function paramstr(l : longint) : string;begin  if (l>=0) and (l+1<=argc) then   paramstr:=strpas(argv[l])  else   paramstr:='';end;procedure randomize;Begin asm        movb    $0x2c,%ah        int     $0x21        shll    $16,%ecx        movw    %dx,%cx        movl    %ecx,randseed end;end;{*****************************************************************************                              Heap Management*****************************************************************************}function getheapstart:pointer;assembler;asm        leal    HEAP,%eaxend ['EAX'];function getheapsize:longint;assembler;asm        movl    HEAPSIZE,%eaxend ['EAX'];function Sbrk(size : longint) : longint;assembler;asm        movl    size,%ebx        movl    $0x4a01,%eax        int     $0x21end;{ include standard heap management }{$I heap.inc}{****************************************************************************                          Low Level File Routines ****************************************************************************}procedure AllowSlash(p:pchar);var  i : longint;begin{ allow slash as backslash }  for i:=0 to strlen(p) do   if p[i]='/' then p[i]:='\';end;procedure do_close(h : longint);assembler;asm        movl    h,%ebx        movb    $0x3e,%ah        pushl   %ebp        int     $0x21        popl    %ebp        jnc     .LCLOSE1        movw    %ax,inoutres.LCLOSE1:end;procedure do_erase(p : pchar);begin  AllowSlash(p);  asm        movl    p,%edx        movb    $0x41,%ah        pushl   %ebp        int     $0x21        popl    %ebp        jnc     .LERASE1        movw    %ax,inoutres.LERASE1:  end;end;procedure do_rename(p1,p2 : pchar);begin  AllowSlash(p1);  AllowSlash(p2);  asm        movl    p1,%edx        movl    p2,%edi        movb    $0x56,%ah        pushl   %ebp        int     $0x21        popl    %ebp        jnc     .LRENAME1        movw    %ax,inoutres.LRENAME1:  end;end;function do_write(h,addr,len : longint) : longint;assembler;asm        movl    len,%ecx        movl    addr,%edx        movl    h,%ebx        movb    $0x40,%ah        int     $0x21        jnc     .LDOSWRITE1        movw    %ax,inoutres        xorl    %eax,%eax.LDOSWRITE1:end;function do_read(h,addr,len : longint) : longint;assembler;asm        movl    len,%ecx        movl    addr,%edx        movl    h,%ebx        movb    $0x3f,%ah        int     $0x21        jnc     .LDOSREAD1        movw    %ax,inoutres        xorl    %eax,%eax.LDOSREAD1:end;function do_filepos(handle : longint) : longint;assembler;asm        movl    $0x4201,%eax        movl    handle,%ebx        xorl    %ecx,%ecx        xorl    %edx,%edx        pushl   %ebp        int     $0x21        popl    %ebp        jnc     .LDOSFILEPOS1        movw    %ax,inoutres        xorl    %eax,%eax        jmp     .LDOSFILEPOS2.LDOSFILEPOS1:        shll    $16,%edx        movzwl  %ax,%eax        orl     %edx,%eax.LDOSFILEPOS2:end;procedure do_seek(handle,pos : longint);assembler;asm        movl    $0x4200,%eax        movl    handle,%ebx        movl    pos,%edx        movl    %edx,%ecx        shrl    $16,%ecx        pushl   %ebp        int     $0x21        popl    %ebp        jnc     .LDOSSEEK1        movw    %ax,inoutres.LDOSSEEK1:end;function do_seekend(handle : longint) : longint;assembler;asm        movl    $0x4202,%eax        movl    handle,%ebx        xorl    %ecx,%ecx        xorl    %edx,%edx        pushl   %ebp        int     $0x21        popl    %ebp        jnc     .Lset_at_end1        movw    %ax,inoutres        xorl    %eax,%eax        jmp     .Lset_at_end2.Lset_at_end1:        shll    $16,%edx        movzwl  %ax,%eax        orl     %edx,%eax.Lset_at_end2:end;function do_filesize(handle : longint) : longint;var  aktfilepos : longint;begin  aktfilepos:=do_filepos(handle);  do_filesize:=do_seekend(handle);  do_seek(handle,aktfilepos);end;procedure do_truncate(handle,pos : longint);assembler;asm        movl    $0x4200,%eax        movl    handle,%ebx        movl    pos,%edx        movl    %edx,%ecx        shrl    $16,%ecx        pushl   %ebp        int     $0x21        popl    %ebp        jc      .LTruncate1        movl    handle,%ebx        movl    %ebp,%edx        xorl    %ecx,%ecx        movb    $0x40,%ah        int     $0x21        jnc     .LTruncate2.LTruncate1:        movw    %ax,inoutres.LTruncate2:end;procedure do_open(var f;p:pchar;flags:longint);{  filerec and textrec have both handle and mode as the first items so  they could use the same routine for opening/creating.  when (flags and $100)   the file will be append  when (flags and $1000)  the file will be truncate/rewritten  when (flags and $10000) there is no check for close (needed for textfiles)}var  oflags : longint;begin  AllowSlash(p);{ close first if opened }  if ((flags and $10000)=0) then   begin     case filerec(f).mode of      fminput,fmoutput,fminout : Do_Close(filerec(f).handle);      fmclosed : ;     else      begin        inoutres:=102; {not assigned}        exit;      end;     end;   end;{ reset file handle }  filerec(f).handle:=UnusedHandle;  oflags:=$8404;{ convert filemode to filerec modes }  case (flags and 3) of   0 : begin         filerec(f).mode:=fminput;         oflags:=$8001;       end;   1 : filerec(f).mode:=fmoutput;   2 : filerec(f).mode:=fminout;  end;  if (flags and $1000)<>0 then   begin     filerec(f).mode:=fmoutput;     oflags:=$8302;   end  else   if (flags and $100)<>0 then    begin      filerec(f).mode:=fmoutput;      oflags:=$8404;    end;{ empty name is special }  if p[0]=#0 then   begin     case FileRec(f).mode of       fminput :         FileRec(f).Handle:=StdInputHandle;       fminout, { this is set by rewrite }       fmoutput :         FileRec(f).Handle:=StdOutputHandle;       fmappend :         begin           FileRec(f).Handle:=StdOutputHandle;           FileRec(f).mode:=fmoutput; {fool fmappend}         end;     end;     exit;   end;  asm        movl    $0xff02,%eax        movl    oflags,%ecx        movl    p,%ebx        int     $0x21        jnc     .LOPEN1        movw    %ax,inoutres        movw    $0xffff,%ax.LOPEN1:        movl    f,%edx        movw    %ax,(%edx)  end;  if (flags and $100)<>0 then   do_seekend(filerec(f).handle);end;function do_isdevice(handle : longint):boolean;assembler;asm        movl    $0x4400,%eax        movl    handle,%ebx        pushl   %ebp        int     $0x21        popl    %ebp        jnc     .LDOSDEVICE        movw    %ax,inoutres             xorl       %edx,%edx  .LDOSDEVICE:        movl    %edx,%eax             shrl       $7,%eax        andl    $1,%eaxend;{*****************************************************************************                           UnTyped File Handling*****************************************************************************}{$i file.inc}{*****************************************************************************                           Typed File Handling*****************************************************************************}{$i typefile.inc}{*****************************************************************************                           Text File Handling*****************************************************************************}{$DEFINE EOF_CTRLZ}{$i text.inc}{*****************************************************************************                           Directory Handling*****************************************************************************}procedure DosDir(func:byte;const s:string);var  buffer : array[0..255] of char;begin  move(s[1],buffer,length(s));  buffer[length(s)]:=#0;  AllowSlash(pchar(@buffer));  asm        leal    buffer,%edx        movb    func,%ah        int     $0x21        jnc     .LDOS_DIRS1        movw    %ax,inoutres.LDOS_DIRS1:  end;end;procedure mkdir(const s : string);[IOCheck];begin  If InOutRes <> 0 then exit;  DosDir($39,s);end;procedure rmdir(const s : string);[IOCheck];begin  If InOutRes <> 0 then exit;  DosDir($3a,s);end;procedure chdir(const s : string);[IOCheck];begin  If InOutRes <> 0 then exit;  DosDir($3b,s);end;procedure GetDir (DriveNr: byte; var Dir: ShortString);var  temp : array[0..255] of char;  sof  : pchar;  i    : byte;  Err: boolean;begin  sof:=pchar(@dir[4]);{ dir[1..3] will contain '[drivenr]:\', but is not supplied by DOS,  so we let dos string start at dir[4]  Get dir from drivenr : 0=default, 1=A etc }  asm        movb    drivenr,%dl        movl    sof,%esi        movw    $0x4700,%ax        movb    %al,Err        int     $0x21        jnc .LGetDir        movw %ax, InOutRes        incb Err.LGetDir:  end;  if Err and (DriveNr <> 0) then   begin    Dir := char (DriveNr + 64) + ':\';    Exit;   end;{ Now Dir should be filled with directory in ASCIIZ starting from dir[4] }  dir[0]:=#3;  dir[2]:=':';  dir[3]:='\';  i:=4;{ conversation to Pascal string }  while (dir[i]<>#0) do   begin   { convert path name to DOS }     if dir[i]='/' then      dir[i]:='\';     dir[0]:=chr(i);     inc(i);   end;{ upcase the string }  if drivenr<>0 then   { Drive was supplied. We know it }   dir[1]:=chr(65+drivenr-1)  else   begin   { We need to get the current drive from DOS function 19H  }   { because the drive was the default, which can be unknown }     asm        movb    $0x19,%ah        int     $0x21        addb    $65,%al        movb    %al,i     end;     dir[1]:=chr(i);   end;  dir:=upcase(dir);end;{*****************************************************************************                         System Dependent Exit code*****************************************************************************}Procedure system_exit;var  err : byte;begin  flush(stderr);  err:=exitcode and $ff;  asm        movl    $0x4c00,%eax        movb    err,%al        int     $0x21  end;end;{*****************************************************************************                         SystemUnit Initialization*****************************************************************************}Begin{$ifdef SYSTEMDEBUG}{ to test stack depth }  loweststack:=maxlongint;{$endif}{ Setup heap }  InitHeap;{ Setup stdin, stdout and stderr }  OpenStdIO(Input,fmInput,StdInputHandle);  OpenStdIO(Output,fmOutput,StdOutputHandle);  OpenStdIO(StdOut,fmOutput,StdOutputHandle);  OpenStdIO(StdErr,fmOutput,StdErrorHandle);{ Reset IO Error }  InOutRes:=0;End.{  $Log$  Revision 1.8  2001-07-29 13:50:44  peter    * merged updates from v10  Revision 1.7  2001/06/30 18:55:49  hajny    * GetDir fix for inaccessible drives  Revision 1.6  2001/06/19 20:46:07  hajny    * platform specific constants moved after systemh.inc, BeOS omission corrected  Revision 1.5  2001/06/13 22:22:59  hajny    + platform specific information  Revision 1.4  2001/03/21 21:08:20  hajny    * GetDir fixed  Revision 1.3  2001/03/10 09:57:51  hajny    * FExpand without IOResult change, remaining direct asm removed  Revision 1.2  2000/07/13 11:33:38  michael  + removed logs}
 |