| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094 | {    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 OS/2    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. **********************************************************************}{$IFNDEF FPC_DOTTEDUNITS}unit sysutils;{$ENDIF FPC_DOTTEDUNITS}interface{$MODE objfpc}{$MODESWITCH OUT}{$IFDEF UNICODERTL}{$MODESWITCH UNICODESTRINGS}{$ELSE}{$H+}{$ENDIF}{$modeswitch typehelpers}{$modeswitch advancedrecords}{$DEFINE HAS_SLEEP}{$DEFINE HAS_OSERROR}{ used OS file system APIs use ansistring }{$define SYSUTILS_HAS_ANSISTR_FILEUTIL_IMPL}{ OS has an ansistring/single byte environment variable API }{$define SYSUTILS_HAS_ANSISTR_ENVVAR_IMPL}{ OS has an ansistring/single byte API for executing other processes }{$DEFINE EXECUTEPROCUNI}{ Include platform independent interface part }{$i sysutilh.inc}implementation{$IFDEF FPC_DOTTEDUNITS}  uses    System.SysConst, OS2Api.doscalls;{$ELSE FPC_DOTTEDUNITS}  uses    sysconst, DosCalls;{$ENDIF FPC_DOTTEDUNITS}type(* Necessary here due to a different definition of TDateTime in DosCalls. *)  TDateTime = System.TDateTime;threadvar  LastOSError: cardinal;{$DEFINE FPC_FEXPAND_UNC} (* UNC paths are supported *){$DEFINE FPC_FEXPAND_DRIVES} (* Full paths begin with drive specification *){$DEFINE FPC_FEXPAND_GETENV_PCHAR}{$DEFINE HAS_GETTICKCOUNT}{$DEFINE HAS_GETTICKCOUNT64}{$DEFINE HAS_LOCALTIMEZONEOFFSET}{ Include platform independent implementation part }{$i sysutils.inc}{****************************************************************************                              File Functions****************************************************************************}const ofRead        = $0000;     {Open for reading} ofWrite       = $0001;     {Open for writing} ofReadWrite   = $0002;     {Open for reading/writing} doDenyRW      = $0010;     {DenyAll (no sharing)} faCreateNew   = $00010000; {Create if file does not exist} faOpenReplace = $00040000; {Truncate if file exists} faCreate      = $00050000; {Create if file does not exist, truncate otherwise} FindResvdMask = $00003737  {Allowed bits for DosFindFirst parameter Attribute}             and $000000FF; {combined with a mask for allowed attributes only}function FileOpen (const FileName: rawbytestring; Mode: integer): THandle;Var  SystemFileName: RawByteString;  Handle: THandle;  Rc, Action: cardinal;begin  SystemFileName:=ToSingleByteFileSystemEncodedFileName(FileName);(* DenyReadWrite if sharing not specified. *)  if (Mode and 112 = 0) or (Mode and 112 > 64) then   Mode := Mode or doDenyRW;  Rc:=Sys_DosOpenL(PAnsiChar (SystemFileName), Handle, Action, 0, 0, 1, Mode, nil);  If Rc=0 then    FileOpen:=Handle  else   begin    FileOpen:=feInvalidHandle; //FileOpen:=-RC;    //should return feInvalidHandle(=-1) if fail, other negative returned value are no more errors    OSErrorWatch (RC);   end;end;function FileCreate (const FileName: RawByteString): THandle;begin  FileCreate := FileCreate (FileName, doDenyRW, 777); (* Sharing to DenyAll *)end;function FileCreate (const FileName: RawByteString; Rights: integer): THandle;begin  FileCreate := FileCreate (FileName, doDenyRW, Rights);                                      (* Sharing to DenyAll *)end;function FileCreate (const FileName: RawByteString; ShareMode: integer;                                                     Rights: integer): THandle;var  SystemFileName: RawByteString;  Handle: THandle;  RC, Action: cardinal;begin  SystemFileName:=ToSingleByteFileSystemEncodedFileName(FileName);  ShareMode := ShareMode and 112;  (* Sharing to DenyAll as default in case of values not allowed by OS/2. *)  if (ShareMode = 0) or (ShareMode > 64) then   ShareMode := doDenyRW;  RC := Sys_DosOpenL (PAnsiChar (SystemFileName), Handle, Action, 0, 0, $12,                                    faCreate or ofReadWrite or ShareMode, nil);  if RC = 0 then   FileCreate := Handle  else   begin    FileCreate := feInvalidHandle;    OSErrorWatch (RC);   end;End;function FileRead (Handle: THandle; Out Buffer; Count: longint): longint;Var  T: cardinal;  RC: cardinal;begin  RC := DosRead (Handle, Buffer, Count, T);  if RC = 0 then   FileRead := longint (T)  else   begin    FileRead := -1;    OSErrorWatch (RC);   end;end;function FileWrite (Handle: THandle; const Buffer; Count: longint): longint;Var  T: cardinal;  RC: cardinal;begin  RC := DosWrite (Handle, Buffer, Count, T);  if RC = 0 then   FileWrite := longint (T)  else   begin    FileWrite := -1;    OSErrorWatch (RC);   end;end;function FileSeek (Handle: THandle; FOffset, Origin: longint): longint;var  NPos: int64;  RC: cardinal;begin  RC := Sys_DosSetFilePtrL (Handle, FOffset, Origin, NPos);  if (RC = 0) and (NPos < high (longint)) then    FileSeek:= longint (NPos)  else   begin    FileSeek:=-1;    OSErrorWatch (RC);   end;end;function FileSeek (Handle: THandle; FOffset: Int64; Origin: Longint): Int64;var  NPos: int64;  RC: cardinal;begin  RC := Sys_DosSetFilePtrL (Handle, FOffset, Origin, NPos);  if RC = 0 then    FileSeek:= NPos  else   begin    FileSeek:=-1;    OSErrorWatch (RC);   end;end;procedure FileClose (Handle: THandle);var  RC: cardinal;begin  RC := DosClose (Handle);  if RC <> 0 then   OSErrorWatch (RC);end;function FileTruncate (Handle: THandle; Size: Int64): boolean;var  RC: cardinal;begin  RC := Sys_DosSetFileSizeL(Handle, Size);  FileTruncate := RC = 0;  if RC = 0 then   FileSeek(Handle, 0, 2)  else   OSErrorWatch (RC);end;function FileAge (const FileName: RawByteString): Int64;var Handle: longint;begin    Handle := FileOpen (FileName, 0);    if Handle <> -1 then        begin            Result := FileGetDate (Handle);            FileClose (Handle);        end    else        Result := -1;end;function FileGetSymLinkTarget(const FileName: RawByteString; out SymLinkRec: TRawbyteSymLinkRec): Boolean;begin  Result := False;end;function FileExists (const FileName: RawByteString; FollowLink : Boolean): boolean;var  L: longint;begin  { no need to convert to DefaultFileSystemEncoding, FileGetAttr will do that }  if FileName = '' then    Result := false  else   begin     L := FileGetAttr (FileName);     Result := (L >= 0) and (L and (faDirectory or faVolumeID) = 0);(* Neither VolumeIDs nor directories are files. *)   end;end;type    PSearchRec = ^TSearchRec;Function InternalFindFirst (Const Path : RawByteString; Attr : Longint; out Rslt : TAbstractSearchRec; var Name: RawByteString) : Longint;var SR: PSearchRec;    FStat: PFileFindBuf3L;    Count: cardinal;    Err: cardinal;    I: cardinal;    SystemEncodedPath: RawByteString;begin  SystemEncodedPath := ToSingleByteFileSystemEncodedFileName(Path);  New (FStat);  Rslt.FindHandle := THandle ($FFFFFFFF);  Count := 1;  if FSApi64 then   Err := DosFindFirst (PAnsiChar (SystemEncodedPath), Rslt.FindHandle,            Attr and FindResvdMask, FStat, SizeOf (FStat^), Count, ilStandardL)  else   Err := DosFindFirst (PAnsiChar (SystemEncodedPath), Rslt.FindHandle,            Attr and FindResvdMask, FStat, SizeOf (FStat^), Count, ilStandard);  if Err <> 0 then   OSErrorWatch (Err)  else if Count = 0 then   Err := 18;  InternalFindFirst := -Err;  if Err = 0 then   begin    Rslt.ExcludeAttr := 0;    Rslt.Time := cardinal (FStat^.DateLastWrite) shl 16 + FStat^.TimeLastWrite;    if FSApi64 then     begin      Rslt.Size := FStat^.FileSize;      Name := FStat^.Name;      Rslt.Attr := FStat^.AttrFile;     end    else     begin      Rslt.Size := PFileFindBuf3 (FStat)^.FileSize;      Name := PFileFindBuf3 (FStat)^.Name;      Rslt.Attr := PFileFindBuf3 (FStat)^.AttrFile;     end;    SetCodePage (Name, DefaultFileSystemCodePage, false);   end  else   InternalFindClose(Rslt.FindHandle);  Dispose (FStat);end;Function InternalFindNext (var Rslt : TAbstractSearchRec; var Name : RawByteString) : Longint;var  SR: PSearchRec;  FStat: PFileFindBuf3L;  Count: cardinal;  Err: cardinal;begin  New (FStat);  Count := 1;  Err := DosFindNext (Rslt.FindHandle, FStat, SizeOf (FStat^), Count);  if Err <> 0 then   OSErrorWatch (Err)  else if Count = 0 then   Err := 18;  InternalFindNext := -Err;  if Err = 0 then  begin    Rslt.ExcludeAttr := 0;    Rslt.Time := cardinal (FStat^.DateLastWrite) shl 16 + FStat^.TimeLastWrite;    if FSApi64 then     begin      Rslt.Size := FStat^.FileSize;      Name := FStat^.Name;      Rslt.Attr := FStat^.AttrFile;     end    else     begin      Rslt.Size := PFileFindBuf3 (FStat)^.FileSize;      Name := PFileFindBuf3 (FStat)^.Name;      Rslt.Attr := PFileFindBuf3 (FStat)^.AttrFile;     end;    SetCodePage (Name, DefaultFileSystemCodePage, false);  end;  Dispose (FStat);end;Procedure InternalFindClose(var Handle: THandle);var  SR: PSearchRec;  RC: cardinal;begin  RC := DosFindClose (Handle);  Handle := 0;  if RC <> 0 then   OSErrorWatch (RC);end;function FileGetDate (Handle: THandle): Int64;var  FStat: TFileStatus3;  Time: Longint;  RC: cardinal;begin  RC := DosQueryFileInfo(Handle, ilStandard, @FStat, SizeOf(FStat));  if RC = 0 then  begin    Time := FStat.TimeLastWrite + dword (FStat.DateLastWrite) shl 16;    if Time = 0 then      Time := FStat.TimeCreation + dword (FStat.DateCreation) shl 16;  end else   begin    Time:=0;    OSErrorWatch (RC);   end;  FileGetDate:=Time;end;function FileSetDate (Handle: THandle; Age: Int64): longint;var  FStat: PFileStatus3;  RC: cardinal;begin  New (FStat);  RC := DosQueryFileInfo (Handle, ilStandard, FStat, SizeOf (FStat^));  if RC <> 0 then   begin    FileSetDate := -1;    OSErrorWatch (RC);   end  else   begin    FStat^.DateLastAccess := Hi (dword (Age));    FStat^.DateLastWrite := Hi (dword (Age));    FStat^.TimeLastAccess := Lo (dword (Age));    FStat^.TimeLastWrite := Lo (dword (Age));    RC := DosSetFileInfo (Handle, ilStandard, FStat, SizeOf (FStat^));    if RC <> 0 then     begin      FileSetDate := -1;      OSErrorWatch (RC);     end    else     FileSetDate := 0;   end;  Dispose (FStat);end;function FileGetAttr (const FileName: RawByteString): longint;var  FS: PFileStatus3;  SystemFileName: RawByteString;  RC: cardinal;begin  SystemFileName:=ToSingleByteFileSystemEncodedFileName(Filename);  New(FS);  RC := DosQueryPathInfo(PAnsiChar (SystemFileName), ilStandard, FS, SizeOf(FS^));  if RC = 0 then   Result := FS^.AttrFile  else   begin    Result := - longint (RC);    OSErrorWatch (RC);   end;  Dispose(FS);end;function FileSetAttr (const Filename: RawByteString; Attr: longint): longint;Var  FS: PFileStatus3;  SystemFileName: RawByteString;  RC: cardinal;Begin  SystemFileName:=ToSingleByteFileSystemEncodedFileName(Filename);  New(FS);  RC := DosQueryPathInfo (PAnsiChar (SystemFileName), ilStandard, FS, SizeOf (FS^));  if RC = 0 then   begin    FS^.AttrFile:=Attr;    RC := DosSetPathInfo(PAnsiChar (SystemFileName), ilStandard, FS, SizeOf(FS^), 0);    if RC <> 0 then     OSErrorWatch (RC);   end  else   OSErrorWatch (RC);  Result := - longint (RC);  Dispose(FS);end;function DeleteFile (const FileName: RawByteString): boolean;var  SystemFileName: RawByteString;  RC: cardinal;Begin  SystemFileName:=ToSingleByteFileSystemEncodedFileName(Filename);  RC := DosDelete (PAnsiChar (SystemFileName));  if RC <> 0 then   begin    Result := false;    OSErrorWatch (RC);   end  else   Result := true;End;function RenameFile (const OldName, NewName: RawByteString): boolean;var  OldSystemFileName, NewSystemFileName: RawByteString;  RC: cardinal;Begin  OldSystemFileName:=ToSingleByteFileSystemEncodedFileName(OldName);  NewSystemFileName:=ToSingleByteFileSystemEncodedFileName(NewName);  RC := DosMove (PAnsiChar (OldSystemFileName), PAnsiChar (NewSystemFileName));  if RC <> 0 then   begin    Result := false;    OSErrorWatch (RC);   end  else   Result := true;End;{****************************************************************************                              Disk Functions****************************************************************************}function DiskFree (Drive: byte): int64;var FI: TFSinfo;    RC: cardinal;begin  {In OS/2, we use the filesystem information.}  RC := DosQueryFSInfo (Drive, 1, FI, SizeOf (FI));  if RC = 0 then   DiskFree := int64 (FI.Free_Clusters) *                   int64 (FI.Sectors_Per_Cluster) * int64 (FI.Bytes_Per_Sector)  else   begin    DiskFree := -1;    OSErrorWatch (RC);   end;end;function DiskSize (Drive: byte): int64;var FI: TFSinfo;    RC: cardinal;begin  {In OS/2, we use the filesystem information.}  RC := DosQueryFSinfo (Drive, 1, FI, SizeOf (FI));  if RC = 0 then   DiskSize := int64 (FI.Total_Clusters) *                   int64 (FI.Sectors_Per_Cluster) * int64 (FI.Bytes_Per_Sector)  else   begin    DiskSize := -1;    OSErrorWatch (RC);   end;end;function DirectoryExists (const Directory: RawByteString; FollowLink : Boolean): boolean;var  L: longint;begin  { no need to convert to DefaultFileSystemEncoding, FileGetAttr will do that }  if Directory = '' then   Result := false  else   begin    if ((Length (Directory) = 2) or        (Length (Directory) = 3) and        (Directory [3] in AllowDirectorySeparators)) and       (Directory [2] in AllowDriveSeparators) and       (UpCase (Directory [1]) in ['A'..'Z']) then(* Checking attributes for 'x:' is not possible but for 'x:.' it is. *)     L := FileGetAttr (Directory + '.')    else if (Directory [Length (Directory)] in AllowDirectorySeparators) and                                              (Length (Directory) > 1) and(* Do not remove '\' in '\\' (invalid path, possibly broken UNC path). *)      not (Directory [Length (Directory) - 1] in AllowDirectorySeparators) then     L := FileGetAttr (Copy (Directory, 1, Length (Directory) - 1))    else     L := FileGetAttr (Directory);    Result := (L > 0) and (L and faDirectory = faDirectory);   end;end;{****************************************************************************                              Time Functions****************************************************************************}{$DEFINE HAS_DUAL_TZHANDLING}{$I tzenv.inc}var  TZAlwaysFromEnv: boolean;procedure InitTZ2; inline;var  DT: {$IFDEF FPC_DOTTEDUNITS}OS2Api.{$endif}DosCalls.TDateTime;begin  DosGetDateTime (DT);  TZAlwaysFromEnv := DT.TimeZone = -1;end;procedure GetLocalTime (var SystemTime: TSystemTime);var  DT: {$IFDEF FPC_DOTTEDUNITS}OS2Api.{$endif}DosCalls.TDateTime;begin  DosGetDateTime(DT);  with SystemTime do  begin    Year:=DT.Year;    Month:=DT.Month;    Day:=DT.Day;    DayOfWeek:=DT.WeekDay;    Hour:=DT.Hour;    Minute:=DT.Minute;    Second:=DT.Second;    MilliSecond:=DT.Sec100 * 10;  end;end;function GetUniversalTime (var SystemTime: TSystemTime): boolean;var  DT: {$IFDEF FPC_DOTTEDUNITS}OS2Api.{$endif}DosCalls.TDateTime;  Offset: longint;begin  if TZAlwaysFromEnv then   begin    GetLocalTime (SystemTime);    Offset := GetLocalTimeOffset;   end  else   begin    DosGetDateTime (DT);    with SystemTime do     begin      Year := DT.Year;      Month := DT.Month;      Day := DT.Day;      DayOfWeek := DT.WeekDay;      Hour := DT.Hour;      Minute := DT.Minute;      Second := DT.Second;      MilliSecond := DT.Sec100 * 10;     end;    if DT.TimeZone = -1 then     Offset := GetLocalTimeOffset    else     Offset := DT.TimeZone;   end;  UpdateTimeWithOffset (SystemTime, Offset);  GetUniversalTime := true;end;function GetLocalTimeOffset: integer;var  DT: {$IFDEF FPC_DOTTEDUNITS}OS2Api.{$endif}DosCalls.TDateTime;begin  if TZAlwaysFromEnv then   begin    if InDST then     GetLocalTimeOffset := DSTOffsetMin    else     GetLocalTimeOffset := TZOffsetMin;   end  else   begin    DosGetDateTime (DT);    if DT.TimeZone <> -1 then     GetLocalTimeOffset := DT.TimeZone    else     begin      if InDST then       GetLocalTimeOffset := DSTOffsetMin      else       GetLocalTimeOffset := TZOffsetMin;     end;   end;end;{****************************************************************************                              Misc Functions****************************************************************************}procedure sysbeep;begin  DosBeep (800, 250);end;{****************************************************************************                              Locale Functions****************************************************************************}var  Country: TCountryCode;  CtryInfo: TCountryInfo;procedure InitAnsi;var  I: byte;  RC: cardinal;begin    for I := 0 to 255 do        UpperCaseTable [I] := Chr (I);    Move (UpperCaseTable, LowerCaseTable, SizeOf (UpperCaseTable));            FillChar (Country, SizeOf (Country), 0);            DosMapCase (SizeOf (UpperCaseTable), Country, @UpperCaseTable);    for I := 0 to 255 do        if UpperCaseTable [I] <> Chr (I) then            LowerCaseTable [Ord (UpperCaseTable [I])] := Chr (I);end;procedure InitInternational;var  Size: cardinal;  RC: cardinal;begin  Size := 0;  FillChar (Country, SizeOf (Country), 0);  FillChar (CtryInfo, SizeOf (CtryInfo), 0);  RC := DosQueryCtryInfo (SizeOf (CtryInfo), Country, CtryInfo, Size);  if RC = 0 then   begin    DateSeparator := CtryInfo.DateSeparator;    case CtryInfo.DateFormat of     1: begin         ShortDateFormat := 'd/m/y';         LongDateFormat := 'dd" "mmmm" "yyyy';        end;     2: begin         ShortDateFormat := 'y/m/d';         LongDateFormat := 'yyyy" "mmmm" "dd';        end;     3: begin         ShortDateFormat := 'm/d/y';         LongDateFormat := 'mmmm" "dd" "yyyy';        end;    end;    TimeSeparator := CtryInfo.TimeSeparator;    DecimalSeparator := CtryInfo.DecimalSeparator;    ThousandSeparator := CtryInfo.ThousandSeparator;    CurrencyFormat := CtryInfo.CurrencyFormat;    CurrencyString := PAnsiChar (CtryInfo.CurrencyUnit);   end  else   OSErrorWatch (RC);  InitAnsi;  InitInternationalGeneric;end;function SysErrorMessage(ErrorCode: Integer): String;const  SysMsgFile: array [0..10] of AnsiChar = 'OSO001.MSG'#0;var  OutBuf: array [0..999] of AnsiChar;  RetMsgSize: cardinal;  RC: cardinal;begin  RC := DosGetMessage (nil, 0, @OutBuf [0], SizeOf (OutBuf),                                       ErrorCode, @SysMsgFile [0], RetMsgSize);  if RC = 0 then   begin    SetLength (Result, RetMsgSize);    Move (OutBuf [0], Result [1], RetMsgSize);   end  else   begin    Result:=Format(SUnknownErrorCode,[ErrorCode]);    OSErrorWatch (RC);   end;end;{****************************************************************************                              OS Utils****************************************************************************}function GetEnvPChar (EnvVar: shortstring): PAnsiChar;(* The assembler version is more than three times as fast as Pascal. *)var P: PAnsiChar;begin EnvVar := UpCase (EnvVar);{$ASMMODE INTEL} asm  cld  mov edi, Environment  lea esi, EnvVar  xor eax, eax  lodsb@NewVar:  cmp byte ptr [edi], 0  jz @Stop  push eax        { eax contains length of searched variable name }  push esi        { esi points to the beginning of the variable name }  mov ecx, -1     { our character ('=' - see below) _must_ be found }  mov edx, edi    { pointer to beginning of variable name saved in edx }  mov al, '='     { searching until '=' (end of variable name) }  repne  scasb           { scan until '=' not found }  neg ecx         { what was the name length? }  dec ecx         { corrected }  dec ecx         { exclude the '=' character }  pop esi         { restore pointer to beginning of variable name }  pop eax         { restore length of searched variable name }  push eax        { and save both of them again for later use }  push esi  cmp ecx, eax    { compare length of searched variable name with name }  jnz @NotEqual   { ... of currently found variable, jump if different }  xchg edx, edi   { pointer to current variable name restored in edi }  repe  cmpsb           { compare till the end of variable name }  xchg edx, edi   { pointer to beginning of variable contents in edi }  jz @Equal       { finish if they're equal }@NotEqual:  xor eax, eax    { look for 00h }  mov ecx, -1     { it _must_ be found }  repne  scasb           { scan until found }  pop esi         { restore pointer to beginning of variable name }  pop eax         { restore length of searched variable name }  jmp @NewVar     { ... or continue with new variable otherwise }@Stop:  xor eax, eax  mov P, eax      { Not found - return nil }  jmp @End@Equal:  pop esi         { restore the stack position }  pop eax  mov P, edi      { place pointer to variable contents in P }@End: end ['eax','ecx','edx','esi','edi']; GetEnvPChar := P;end;{$ASMMODE ATT}Function GetEnvironmentVariable(Const EnvVar : String) : String;begin    GetEnvironmentVariable := GetEnvPChar (EnvVar);end;Function GetEnvironmentVariableCount : Integer;begin(*  Result:=FPCCountEnvVar(EnvP); - the amount is already known... *)  GetEnvironmentVariableCount := EnvC;end;Function GetEnvironmentString(Index : Integer) : {$ifdef FPC_RTL_UNICODE}UnicodeString{$else}AnsiString{$endif};begin  Result:=FPCGetEnvStrFromP (EnvP, Index);end;procedure Sleep (Milliseconds: cardinal);begin DosSleep (Milliseconds);end;function SysTimerTick: QWord;var  L: cardinal;begin  DosQuerySysInfo (svMsCount, svMsCount, L, 4);  SysTimerTick := L;end;function ExecuteProcess (const Path: RawByteString;                 const ComLine: RawByteString;Flags:TExecuteFlags=[]): integer;var E: EOSError; CommandLine: RawByteString; Args0, Args: {$IFDEF FPC_DOTTEDUNITS}OS2Api.{$endif}DosCalls.PByteArray; ObjNameBuf: PAnsiChar; ArgSize: word; Res: TResultCodes; ObjName: shortstring; RC: cardinal; ExecAppType: cardinal; MaxArgsSize: PtrUInt; (* Amount of memory reserved for arguments in bytes. *) MaxArgsSizeInc: word;const ObjBufSize = 512;function StartSession: cardinal;var HQ: THandle; SPID, STID, QName: shortstring; SID, PID: cardinal; SD: TStartData; RD: TRequestData; PCI: PChildInfo; CISize: cardinal; Prio: byte;begin Result := $FFFFFFFF; FillChar (SD, SizeOf (SD), 0); SD.Length := SizeOf (SD); SD.Related := ssf_Related_Child; if FileExists (Path) then(* Full path necessary for starting different executable files from current *)(* directory. *)  CommandLine := ExpandFileName (Path) else  CommandLine := Path; SD.PgmName := PAnsiChar (CommandLine); if ComLine <> '' then  SD.PgmInputs := PAnsiChar (ComLine); if ExecInheritsHandles in Flags then   SD.InheritOpt := ssf_InhertOpt_Parent; Str (GetProcessID, SPID); Str (ThreadID, STID); QName := '\QUEUES\FPC_ExecuteProcess_p' + SPID + 't' + STID + '.QUE'#0; SD.TermQ := @QName [1]; SD.ObjectBuffer := ObjNameBuf; SD.ObjectBuffLen := ObjBufSize; RC := DosCreateQueue (HQ, quFIFO or quConvert_Address, @QName [1]); if RC <> 0 then  begin   Move (QName [1], ObjNameBuf^, Length (QName));   OSErrorWatch (RC);  end else  begin   RC := DosStartSession (SD, SID, PID);   if (RC = 0) or (RC = 457) then    begin     RC := DosReadQueue (HQ, RD, CISize, PCI, 0, 0, Prio, 0);     if RC = 0 then      begin       Result := PCI^.Return;       RC := DosCloseQueue (HQ);       if RC <> 0 then        OSErrorWatch (RC);       RC := DosFreeMem (PCI);       if RC <> 0 then        OSErrorWatch (RC);       FreeMem (ObjNameBuf, ObjBufSize);      end     else      begin       OSErrorWatch (RC);       RC := DosCloseQueue (HQ);       OSErrorWatch (RC);      end;    end   else    begin     OSErrorWatch (RC);     RC := DosCloseQueue (HQ);     if RC <> 0 then      OSErrorWatch (RC);    end;  end;end;begin Result := integer ($FFFFFFFF); ObjName := ''; GetMem (ObjNameBuf, ObjBufSize); FillChar (ObjNameBuf^, ObjBufSize, 0); RC := DosQueryAppType (PAnsiChar (Path), ExecAppType); if RC <> 0 then  begin   OSErrorWatch (RC);   if (RC = 190) or (RC = 191) then    Result := StartSession;  end else  begin   if (ApplicationType and 3 = ExecAppType and 3) then(* DosExecPgm should work... *)    begin     MaxArgsSize := Length (ComLine) + Length (Path) + 256; (* More than enough *)     if MaxArgsSize > high (word) then      Exit;     if ComLine = '' then      begin       Args0 := nil;       Args := nil;      end     else      begin       GetMem (Args0, MaxArgsSize);       Args := Args0;(* Work around a bug in OS/2 - argument to DosExecPgm *)(* should not cross 64K boundary. *)       while ((PtrUInt (Args) + MaxArgsSize) and $FFFF) < MaxArgsSize do        begin         MaxArgsSizeInc := MaxArgsSize -                                    ((PtrUInt (Args) + MaxArgsSize) and $FFFF);         Inc (MaxArgsSize, MaxArgsSizeInc);         if MaxArgsSize > high (word) then          Exit;         ReallocMem (Args0, MaxArgsSize);         Inc (pointer (Args), MaxArgsSizeInc);        end;       ArgSize := 0;       Move (Path [1], Args^ [ArgSize], Length (Path));       Inc (ArgSize, Length (Path));       Args^ [ArgSize] := 0;       Inc (ArgSize);       {Now do the real arguments.}       Move (ComLine [1], Args^ [ArgSize], Length (ComLine));       Inc (ArgSize, Length (ComLine));       Args^ [ArgSize] := 0;       Inc (ArgSize);       Args^ [ArgSize] := 0;      end;     Res.ExitCode := $FFFFFFFF;     RC := DosExecPgm (ObjNameBuf, ObjBufSize, 0, Args, nil, Res,                                                                 PAnsiChar (Path));     if RC <> 0 then      OSErrorWatch (RC);     if Args0 <> nil then      FreeMem (Args0, MaxArgsSize);     if RC = 0 then      begin       Result := Res.ExitCode;       FreeMem (ObjNameBuf, ObjBufSize);      end    end  end; if RC <> 0 then  begin   ObjName := StrPas (ObjNameBuf);   FreeMem (ObjNameBuf, ObjBufSize);   if ComLine = '' then    CommandLine := Path   else    CommandLine := Path + ' ' + ComLine;   if ObjName = '' then    E := EOSError.CreateFmt (SExecuteProcessFailed, [CommandLine, RC])   else    E := EOSError.CreateFmt (SExecuteProcessFailed + ' (' + ObjName + ')', [CommandLine, RC]);   E.ErrorCode := Result;   raise E;  end;end;function ExecuteProcess (const Path: RawByteString;        const ComLine: array of RawByteString;Flags:TExecuteFlags=[]): 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;function GetTickCount: LongWord;var  L: cardinal;begin  DosQuerySysInfo (svMsCount, svMsCount, L, 4);  GetTickCount := L;end;function GetTickCount64: QWord;var  Freq2: cardinal;  T: QWord;begin  DosTmrQueryFreq (Freq2);  DosTmrQueryTime (T);  GetTickCount64 := T div (QWord (Freq2) div 1000);{$NOTE GetTickCount64 takes 20 microseconds on 1GHz CPU, GetTickCount not measurable}end;const  OrigOSErrorWatch: TOSErrorWatch = nil;procedure TrackLastOSError (Error: cardinal);begin  LastOSError := Error;  OrigOSErrorWatch (Error);end;function GetLastOSError: Integer;begin  GetLastOSError := Integer (LastOSError);end;{****************************************************************************                              Initialization code****************************************************************************}Initialization  InitExceptions;       { Initialize exceptions. OS independent }  InitInternational;    { Initialize internationalization settings }  OnBeep:=@SysBeep;  LastOSError := 0;  OrigOSErrorWatch := TOSErrorWatch (SetOSErrorTracking (@TrackLastOSError));  InitTZ;  InitTZ2;Finalization  FreeTerminateProcs;  DoneExceptions;end.
 |