| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586 | {    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    Sysutils unit for netware    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 sysutils;interface{$MODE objfpc}{ force ansistrings }{$H+}uses DOS;{$I nwsys.inc}{$I errno.inc}{$DEFINE HAS_SLEEP}TYPE  TNetwareFindData =  RECORD    DirP  : PNWDirEnt;               { used for opendir }    EntryP: PNWDirEnt;               { and readdir }    Magic : WORD;                    { to avoid abends with uninitialized TSearchRec }  END;{ Include platform independent interface part }{$i sysutilh.inc}{ additional NetWare file flags}CONST  faSHARE              = $00000080;  { Sharable file                   }  faNO_SUBALLOC        = $00000800;  { Don't sub alloc. this file      }  faTRANS              = $00001000;  { Transactional file (TTS usable) }  faREADAUD            = $00004000;  { Read audit                      }  faWRITAUD            = $00008000;  { Write audit                     }  faIMMPURG            = $00010000;  { Immediate purge                 }  faNORENAM            = $00020000;  { Rename inhibit                  }  faNODELET            = $00040000;  { Delete inhibit                  }  faNOCOPY             = $00080000;  { Copy inhibit                    }  faFILE_MIGRATED      = $00400000;  { File has been migrated          }  faDONT_MIGRATE       = $00800000;  { Don't migrate this file         }  faIMMEDIATE_COMPRESS = $02000000;  { Compress this file immediately  }  faFILE_COMPRESSED    = $04000000;  { File is compressed              }  faDONT_COMPRESS      = $08000000;  { Don't compress this file        }  faCANT_COMPRESS      = $20000000;  { Can't compress this file        }  faATTR_ARCHIVE       = $40000000;  { Entry has had an EA modified,   }                                     { an ownerID changed, or trustee  }                                     { info changed, etc.              }implementation  uses    sysconst;{$define FPC_FEXPAND_DRIVES}{$define FPC_FEXPAND_VOLUMES}{$define FPC_FEXPAND_NO_DEFAULT_PATHS}{ Include platform independent implementation part }{$i sysutils.inc}{****************************************************************************                              File Functions****************************************************************************}Function FileOpen (Const FileName : string; Mode : Integer) : THandle;VAR NWOpenFlags : longint;BEGIN  NWOpenFlags:=0;  Case (Mode and 3) of    0 : NWOpenFlags:=NWOpenFlags or O_RDONLY;    1 : NWOpenFlags:=NWOpenFlags or O_WRONLY;    2 : NWOpenFlags:=NWOpenFlags or O_RDWR;  end;  FileOpen := _open (pchar(FileName),NWOpenFlags,0);  //!! We need to set locking based on Mode !!end;Function FileCreate (Const FileName : String) : THandle;begin  FileCreate:=_open(Pchar(FileName),O_RdWr or O_Creat or O_Trunc,0);end;Function FileCreate (Const FileName : String; mode:longint) : THandle;begin  FileCreate:=FileCreate (FileName);end;Function FileRead (Handle : THandle; Var Buffer; Count : longint) : longint;begin  FileRead:=_read (Handle,@Buffer,Count);end;Function FileWrite (Handle : THandle; const Buffer; Count : Longint) : longint;begin  FileWrite:=_write (Handle,@Buffer,Count);end;Function FileSeek (Handle : THandle; FOffset,Origin : Longint) : Longint;begin  FileSeek:=_lseek (Handle,FOffset,Origin);end;Function FileSeek (Handle : THandle; FOffset: Int64; Origin: Longint) : Int64;begin  {$warning need to add 64bit FileSeek }  FileSeek:=FileSeek(Handle,Longint(FOffset),Longint(Origin));end;Procedure FileClose (Handle : THandle);begin  _close(Handle);end;Function FileTruncate (Handle : THandle; Size: Longint) : boolean;begin  FileTruncate:=(_chsize(Handle,Size) = 0);end;Function FileLock (Handle,FOffset,FLen : Longint) : Longint;begin  FileLock := _lock (Handle,FOffset,FLen);end;Function FileLock (Handle : Longint; FOffset,FLen : Int64) : Longint;begin  {$warning need to add 64bit FileLock call }  FileLock := FileLock (Handle, longint(FOffset),longint(FLen));end;Function FileUnlock (Handle,FOffset,FLen : Longint) : Longint;begin  FileUnlock := _unlock (Handle,FOffset,FLen);end;Function FileUnlock (Handle : Longint; FOffset,FLen : Int64) : Longint;begin  {$warning need to add 64bit FileUnlock call }  FileUnlock := FileUnlock (Handle, longint(FOffset),longint(FLen));end;Function FileAge (Const FileName : String): Longint;VAR Info : NWStatBufT;    PTM  : PNWTM;begin  If _stat (pchar(FileName),Info) <> 0 then    exit(-1)  else    begin      PTM := _localtime (Info.st_mtime);      IF PTM = NIL THEN        exit(-1)      else        WITH PTM^ DO          Result:=DateTimeToFileDate(EncodeDate(tm_year+1900,tm_mon+1,tm_mday)+EncodeTime(tm_hour,tm_min,tm_sec,0));    end;end;Function FileExists (Const FileName : String) : Boolean;VAR Info : NWStatBufT;begin  FileExists:=(_stat(pchar(filename),Info) = 0);end;PROCEDURE find_setfields (VAR f : TsearchRec);VAR T : Dos.DateTime;BEGIN  WITH F DO  BEGIN    IF FindData.Magic = $AD01 THEN    BEGIN      {attr := FindData.EntryP^.d_attr AND $FF;}  // lowest 8 bit -> same as dos      attr := FindData.EntryP^.d_attr;   { return complete netware attributes }      UnpackTime(FindData.EntryP^.d_time + (LONGINT (FindData.EntryP^.d_date) SHL 16), T);      time := DateTimeToFileDate(EncodeDate(T.Year,T.Month,T.day)+EncodeTime(T.Hour,T.Min,T.Sec,0));      size := FindData.EntryP^.d_size;      name := strpas (FindData.EntryP^.d_nameDOS);    END ELSE    BEGIN      FillChar (f,SIZEOF(f),0);    END;  END;END;Function FindFirst (Const Path : String; Attr : Longint; out Rslt : TSearchRec) : Longint;begin  IF path = '' then    exit (18);  Rslt.FindData.DirP := _opendir (pchar(Path));  IF Rslt.FindData.DirP = NIL THEN    exit (18);  IF attr <> faAnyFile THEN    _SetReaddirAttribute (Rslt.FindData.DirP, attr);  Rslt.FindData.Magic := $AD01;  Rslt.FindData.EntryP := _readdir (Rslt.FindData.DirP);  if Rslt.FindData.EntryP = nil then  begin    _closedir (Rslt.FindData.DirP);    Rslt.FindData.DirP := NIL;    result := 18;  end else  begin    find_setfields (Rslt);    result := 0;  end;end;Function FindNext (Var Rslt : TSearchRec) : Longint;begin  IF Rslt.FindData.Magic <> $AD01 THEN    exit (18);  Rslt.FindData.EntryP := _readdir (Rslt.FindData.DirP);  IF Rslt.FindData.EntryP = NIL THEN    exit (18);  find_setfields (Rslt);  result := 0;end;Procedure FindClose (Var F : TSearchrec);begin  IF F.FindData.Magic = $AD01 THEN  BEGIN    IF F.FindData.DirP <> NIL THEN      _closedir (F.FindData.DirP);    F.FindData.Magic := 0;    F.FindData.DirP := NIL;    F.FindData.EntryP := NIL;  END;end;Function FileGetDate (Handle : THandle) : Longint;Var Info : NWStatBufT;    PTM  : PNWTM;begin  If _fstat(Handle,Info) <> 0 then    Result:=-1  else    begin      PTM := _localtime (Info.st_mtime);      IF PTM = NIL THEN        exit(-1)      else        WITH PTM^ DO          Result:=DateTimeToFileDate(EncodeDate(tm_year+1900,tm_mon+1,tm_mday)+EncodeTime(tm_hour,tm_min,tm_sec,0));    end;end;Function FileSetDate (Handle : THandle; Age : Longint) : Longint;begin  { i think its impossible under netware from FileHandle. I dident found a way to get the    complete pathname of a filehandle, that would be needed for ChangeDirectoryEntry }  FileSetDate:=-1;  ConsolePrintf ('warning: fpc sysutils.FileSetDate not implemented'#13#10,0);  {$warning FileSetDate not implemented (i think is impossible) }end;Function FileGetAttr (Const FileName : String) : Longint;Var Info : NWStatBufT;begin  If _stat (pchar(FileName),Info) <> 0 then    Result:=-1  Else    Result := Info.st_attr AND $FFFF;end;Function FileSetAttr (Const Filename : String; Attr: longint) : Longint;VAR MS : NWModifyStructure;begin  FillChar (MS, SIZEOF (MS), 0);  if _ChangeDirectoryEntry (PChar (Filename), MS, MFileAtrributesBit, 0) <> 0 then    result := -1  else    result := 0;end;Function DeleteFile (Const FileName : String) : Boolean;begin  Result:= (_UnLink (pchar(FileName)) = 0);end;Function RenameFile (Const OldName, NewName : String) : Boolean;begin  RenameFile:=(_rename(pchar(OldName),pchar(NewName)) = 0);end;{****************************************************************************                              Disk Functions****************************************************************************}{  The Diskfree and Disksize functions need a file on the specified drive, since this  is required for the statfs system call.  These filenames are set in drivestr[0..26], and have been preset to :   0 - '.'      (default drive - hence current dir is ok.)   1 - '/fd0/.'  (floppy drive 1 - should be adapted to local system )   2 - '/fd1/.'  (floppy drive 2 - should be adapted to local system )   3 - '/'       (C: equivalent of dos is the root partition)   4..26          (can be set by you're own applications)  ! Use AddDisk() to Add new drives !  They both return -1 when a failure occurs.}Const  FixDriveStr : array[0..3] of pchar=(    '.',    'a:.',    'b:.',    'sys:/'    );var  Drives   : byte;  DriveStr : array[4..26] of pchar;Procedure AddDisk(const path:string);begin  if not (DriveStr[Drives]=nil) then   FreeMem(DriveStr[Drives],StrLen(DriveStr[Drives])+1);  GetMem(DriveStr[Drives],length(Path)+1);  StrPCopy(DriveStr[Drives],path);  inc(Drives);  if Drives>26 then   Drives:=4;end;Function DiskFree(Drive: Byte): int64;//var fs : statfs;Begin{  if ((Drive<4) and (not (fixdrivestr[Drive]=nil)) and fsstat(StrPas(fixdrivestr[drive]),fs)) or     ((not (drivestr[Drive]=nil)) and fsstat(StrPas(drivestr[drive]),fs)) then   Diskfree:=int64(fs.bavail)*int64(fs.bsize)  else   Diskfree:=-1;}  DiskFree := -1;  ConsolePrintf ('warning: fpc sysutils.diskfree not implemented'#13#10,0);  {$warning DiskFree not implemented (does it make sense ?) }End;Function DiskSize(Drive: Byte): int64;//var fs : statfs;Begin{  if ((Drive<4) and (not (fixdrivestr[Drive]=nil)) and fsstat(StrPas(fixdrivestr[drive]),fs)) or     ((not (drivestr[Drive]=nil)) and fsstat(StrPas(drivestr[drive]),fs)) then   DiskSize:=int64(fs.blocks)*int64(fs.bsize)  else   DiskSize:=-1;}  DiskSize := -1;  ConsolePrintf ('warning: fpc sysutils.disksize not implemented'#13#10,0);  {$warning DiskSize not implemented (does it make sense ?) }End;Function GetCurrentDir : String;begin  GetDir (0,Result);end;Function SetCurrentDir (Const NewDir : String) : Boolean;begin  {$I-}   ChDir(NewDir);  {$I+}  result := (IOResult = 0);end;Function CreateDir (Const NewDir : String) : Boolean;begin  {$I-}   MkDir(NewDir);  {$I+}  result := (IOResult = 0);end;Function RemoveDir (Const Dir : String) : Boolean;begin  {$I-}   RmDir(Dir);  {$I+}  result := (IOResult = 0);end;function DirectoryExists (const Directory: string): boolean;VAR Info : NWStatBufT;begin  If _stat (pchar(Directory),Info) <> 0 then    exit(false)  else    Exit ((Info.st_attr and faDirectory) <> 0);end;{****************************************************************************                              Misc Functions****************************************************************************}procedure Beep;begin  _RingTheBell;end;{****************************************************************************                              Locale Functions****************************************************************************}Procedure GetLocalTime(var SystemTime: TSystemTime);var xx : word;begin  Dos.GetTime(SystemTime.Hour, SystemTime.Minute, SystemTime.Second, xx);  Dos.GetDate(SystemTime.Year, SystemTime.Month, SystemTime.Day, xx);  SystemTime.MilliSecond := 0;end;Procedure InitAnsi;Var i : longint;begin  {  Fill table entries 0 to 127  }  for i := 0 to 96 do    UpperCaseTable[i] := chr(i);  for i := 97 to 122 do    UpperCaseTable[i] := chr(i - 32);  for i := 123 to 191 do    UpperCaseTable[i] := chr(i);  Move (CPISO88591UCT,UpperCaseTable[192],SizeOf(CPISO88591UCT));  for i := 0 to 64 do    LowerCaseTable[i] := chr(i);  for i := 65 to 90 do    LowerCaseTable[i] := chr(i + 32);  for i := 91 to 191 do    LowerCaseTable[i] := chr(i);  Move (CPISO88591LCT,UpperCaseTable[192],SizeOf(CPISO88591UCT));end;Procedure InitInternational;begin  InitInternationalGeneric;  InitAnsi;end;function SysErrorMessage(ErrorCode: Integer): String;begin  Result:='';  // StrError(ErrorCode);end;{****************************************************************************                              OS utility functions****************************************************************************}Function GetEnvironmentVariable(Const EnvVar : String) : String;begin  Result:=StrPas(_getenv(PChar(EnvVar)));end;Function GetEnvironmentVariableCount : Integer;begin  // Result:=FPCCountEnvVar(EnvP);  Result:=0;end;Function GetEnvironmentString(Index : Integer) : String;begin  // Result:=FPCGetEnvStrFromP(Envp,Index);  Result:='';end;function ExecuteProcess(Const Path: AnsiString; Const ComLine: AnsiString):integer;var  e : EOSError;  CommandLine: AnsiString;begin  dos.exec(path,comline);  if (Dos.DosError <> 0) then    begin      if ComLine <> '' then       CommandLine := Path + ' ' + ComLine      else       CommandLine := Path;      e:=EOSError.CreateFmt(SExecuteProcessFailed,[CommandLine,Dos.DosError]);      e.ErrorCode:=Dos.DosError;      raise e;    end;  Result := DosExitCode;end;function ExecuteProcess (const Path: AnsiString;                                  const ComLine: array of AnsiString): integer;var  CommandLine: AnsiString;  I: integer;begin  Commandline := '';  for I := 0 to High (ComLine) do   if Pos (' ', ComLine [I]) <> 0 then    CommandLine := CommandLine + ' ' + '"' + ComLine [I] + '"'   else    CommandLine := CommandLine + ' ' + Comline [I];  ExecuteProcess := ExecuteProcess (Path, CommandLine);end;procedure Sleep(milliseconds: Cardinal);begin  _delay (milliseconds);end;{****************************************************************************                              Initialization code****************************************************************************}Initialization  InitExceptions;       { Initialize exceptions. OS independent }  InitInternational;    { Initialize internationalization settings }Finalization  DoneExceptions;end.
 |