123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991 |
- {
- 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.
- **********************************************************************}
- unit sysutils;
- interface
- {$MODE objfpc}
- {$MODESWITCH OUT}
- { force ansistrings }
- {$H+}
- {$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}
- { Include platform independent interface part }
- {$i sysutilh.inc}
- implementation
- uses
- sysconst, DosCalls;
- 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}
- { 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 in attribute
- specification for DosFindFirst call.}
- 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(PChar (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 (PChar (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): longint;
- var Handle: longint;
- begin
- Handle := FileOpen (FileName, 0);
- if Handle <> -1 then
- begin
- Result := FileGetDate (Handle);
- FileClose (Handle);
- end
- else
- Result := -1;
- end;
- function FileExists (const FileName: RawByteString): 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 TRec = record
- T, D: word;
- end;
- 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 (PChar (SystemEncodedPath), Rslt.FindHandle,
- Attr and FindResvdMask, FStat, SizeOf (FStat^), Count, ilStandardL)
- else
- Err := DosFindFirst (PChar (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;
- TRec (Rslt.Time).T := FStat^.TimeLastWrite;
- TRec (Rslt.Time).D := FStat^.DateLastWrite;
- 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;
- TRec (Rslt.Time).T := FStat^.TimeLastWrite;
- TRec (Rslt.Time).D := FStat^.DateLastWrite;
- 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): longint;
- var
- FStat: TFileStatus3;
- Time: Longint;
- RC: cardinal;
- begin
- RC := DosQueryFileInfo(Handle, ilStandard, @FStat, SizeOf(FStat));
- if RC = 0 then
- begin
- Time := FStat.TimeLastWrite + longint (FStat.DateLastWrite) shl 16;
- if Time = 0 then
- Time := FStat.TimeCreation + longint (FStat.DateCreation) shl 16;
- end else
- begin
- Time:=0;
- OSErrorWatch (RC);
- end;
- FileGetDate:=Time;
- end;
- function FileSetDate (Handle: THandle; Age: longint): 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 (Age);
- FStat^.DateLastWrite := Hi (Age);
- FStat^.TimeLastAccess := Lo (Age);
- FStat^.TimeLastWrite := Lo (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(PChar (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);
- FillChar(FS, SizeOf(FS^), 0);
- FS^.AttrFile:=Attr;
- RC := DosSetPathInfo(PChar (SystemFileName), ilStandard, FS, SizeOf(FS^), 0);
- if RC <> 0 then
- 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 (PChar (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 (PChar (OldSystemFileName), PChar (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): 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
- ****************************************************************************}
- procedure GetLocalTime (var SystemTime: TSystemTime);
- var
- DT: DosCalls.TDateTime;
- begin
- DosGetDateTime(DT);
- with SystemTime do
- begin
- Year:=DT.Year;
- Month:=DT.Month;
- Day:=DT.Day;
- Hour:=DT.Hour;
- Minute:=DT.Minute;
- Second:=DT.Second;
- MilliSecond:=DT.Sec100;
- 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 := PChar (CtryInfo.CurrencyUnit);
- end
- else
- OSErrorWatch (RC);
- InitAnsi;
- InitInternationalGeneric;
- end;
- function SysErrorMessage(ErrorCode: Integer): String;
- const
- SysMsgFile: array [0..10] of char = 'OSO001.MSG'#0;
- var
- OutBuf: array [0..999] of char;
- 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): PChar;
- (* The assembler version is more than three times as fast as Pascal. *)
- var
- P: PChar;
- 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: AnsiString; const ComLine: AnsiString;Flags:TExecuteFlags=[]):
- integer;
- var
- E: EOSError;
- CommandLine: ansistring;
- Args0, Args: DosCalls.PByteArray;
- ObjNameBuf: PChar;
- 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 := PChar (CommandLine);
- if ComLine <> '' then
- SD.PgmInputs := PChar (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 (PChar (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,
- PChar (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: AnsiString;
- const ComLine: array of AnsiString;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));
- Finalization
- DoneExceptions;
- end.
|