| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532 | {****************************************************************************                         Free Pascal Runtime-Library                              DOS unit for OS/2                   Copyright (c) 1997,1999-2000 by Daniel Mantione,                   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 dos;{$ASMMODE ATT}{***************************************************************************}interface{***************************************************************************}{$PACKRECORDS 1}uses    Strings, DosCalls;Type   {Search record which is used by findfirst and findnext:}   SearchRec = record            case boolean of             false: (Handle: THandle;     {Used in os_OS2 mode}                     FStat: PFileFindBuf3;                     Fill: array [1..21 - SizeOf (THandle) - SizeOf (pointer)]                                                                       of byte;                     Attr: byte;                     Time: longint;                     Size: longint;                     Name: string);      {Filenames can be long in OS/2!}             true:  (Fill2: array [1..21] of byte;                     Attr2: byte;                     Time2: longint;                     Size2: longint;                     Name2: string);       {Filenames can be long in OS/2!}        end;        {Flags for the exec procedure:        }threadvar(* For compatibility with VP/2, used for runflags in Exec procedure. *)    ExecFlags: cardinal;{$i dosh.inc}{OS/2 specific functions}function GetEnvPChar (EnvVar: string): PChar;function DosErrorModuleName: string;(* In case of an error in Dos.Exec returns the name of the module *)(* causing the problem - e.g. name of a missing or corrupted DLL. *)implementation{$DEFINE HAS_GETMSCOUNT}{$DEFINE FPC_FEXPAND_UNC} (* UNC paths are supported *){$DEFINE FPC_FEXPAND_DRIVES} (* Full paths begin with drive specification *){$DEFINE FPC_FEXPAND_GETENV_PCHAR}{$I dos.inc}threadvar  LastDosErrorModuleName: string;const   FindResvdMask = $00003737; {Allowed bits in attribute                                    specification for DosFindFirst call.}function GetMsCount: int64;var  L: cardinal;begin  DosQuerySysInfo (svMsCount, svMsCount, L, 4);  GetMsCount := L;end;function fsearch(path:pathstr;dirlist:string):pathstr;Var  A: array [0..255] of char;  D, P: AnsiString;begin  P:=Path;  D:=DirList;  DosError := DosSearchPath (dsIgnoreNetErrs, PChar(D), PChar(P), @A, 255);  fsearch := StrPas (@A);end;procedure getftime(var f;var time:longint);var  FStat: TFileStatus3;begin  DosError := DosQueryFileInfo (FileRec (F).Handle, ilStandard, @FStat,                                                               SizeOf (FStat));  if DosError=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    Time:=0;end;procedure SetFTime (var F; Time: longint);var FStat: TFileStatus3;    RC: cardinal;begin  RC := DosQueryFileInfo (FileRec (F).Handle, ilStandard, @FStat,                                                               SizeOf (FStat));  if RC = 0 then  begin    FStat.DateLastAccess := Hi (Time);    FStat.DateLastWrite := Hi (Time);    FStat.TimeLastAccess := Lo (Time);    FStat.TimeLastWrite := Lo (Time);    RC := DosSetFileInfo (FileRec (F).Handle, ilStandard, @FStat,                                                               SizeOf (FStat));  end;  DosError := integer (RC);end;procedure Exec (const Path: PathStr; const ComLine: ComStr);{Execute a program.}var Args0, Args: PByteArray;    ArgSize: word;    Res: TResultCodes;    ObjName: string;    RC: longint;    HQ: THandle;    SPID, STID, QName: string;    SD: TStartData;    SID, PID: cardinal;    RD: TRequestData;    PCI: PChildInfo;    CISize: cardinal;    Prio: byte;const    MaxArgsSize = 3072; (* Amount of memory reserved for arguments in bytes. *)begin{  LastDosExitCode := Exec (Path, ExecRunFlags (ExecFlags), efDefault, ComLine);}  QName := FExpand (Path);  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. *)    if ((PtrUInt (Args) + 1024) and $FFFF) < 1024 then     Inc (pointer (Args), 1024);    ArgSize := 0;    Move (QName [1], Args^ [ArgSize], Length (QName));    Inc (ArgSize, Length (QName));    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;  RC := DosExecPgm (ObjName, cardinal (ExecFlags), Args, nil, Res, Path);  if RC = 0 then   begin    LastDosExitCode := Res.ExitCode;    LastDosErrorModuleName := '';   end  else   if (RC = 190) or (RC = 191) then    begin     FillChar (SD, SizeOf (SD), 0);     SD.Length := 24;     SD.Related := ssf_Related_Child;     if Args = nil then(* No parameters passed, Args not allocated for DosExecPgm, so allocate now. *)      begin       GetMem (Args, MaxArgsSize);       Move (QName [1], Args^ [0], Length (QName));       Args^ [Length (QName)] := 0;      end     else      SD.PgmInputs := PChar (@Args^ [Length (QName) + 1]);     SD.PgmName := PChar (Args);     SD.InheritOpt := ssf_InhertOpt_Parent;     Str (GetProcessID, SPID);     Str (ThreadID, STID);     QName := '\QUEUES\FPC_Dos_Exec_p' + SPID + 't' + STID + '.QUE'#0;     SD.TermQ := @QName [1];     RC := DosCreateQueue (HQ, quFIFO or quConvert_Address, @QName [1]);     if RC = 0 then      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           LastDosExitCode := PCI^.Return;           DosCloseQueue (HQ);           DosFreeMem (PCI);          end         else          DosCloseQueue (HQ);        end       else        DosCloseQueue (HQ);      end;    end   else    LastDosErrorModuleName := ObjName;  if RC <> 0 then   LastDosExitCode := 0; (* Needed for TP/BP compatibility *)  DosError := RC;  if Args0 <> nil then   FreeMem (Args0, MaxArgsSize);end;function DosErrorModuleName: string;begin  DosErrorModuleName := LastDosErrorModuleName;end;function dosversion:word;{Returns OS/2 version}var  Minor, Major: Cardinal;begin  DosQuerySysInfo(svMajorVersion, svMajorVersion, Major, 4);  DosQuerySysInfo(svMinorVersion, svMinorVersion, Minor, 4);  DosVersion:=Major or Minor shl 8;end;procedure GetDate (var Year, Month, MDay, WDay: word);Var  dt: TDateTime;begin  DosGetDateTime(dt);  Year:=dt.year;  Month:=dt.month;  MDay:=dt.Day;  WDay:=dt.Weekday;end;procedure SetDate (Year, Month, Day: word);var  DT: TDateTime;begin  DosGetDateTime (DT);  DT.Year := Year;  DT.Month := byte (Month);  DT.Day := byte (Day);  DosSetDateTime (DT);end;procedure GetTime (var Hour, Minute, Second, Sec100: word);var  dt: TDateTime;begin  DosGetDateTime(dt);  Hour:=dt.Hour;  Minute:=dt.Minute;  Second:=dt.Second;  Sec100:=dt.Hundredths;end;procedure SetTime (Hour, Minute, Second, Sec100: word);var  DT: TDateTime;begin  DosGetDateTime (DT);  DT.Hour := byte (Hour);  DT.Minute := byte (Minute);  DT.Second := byte (Second);  DT.Sec100 := byte (Sec100);  DosSetDateTime (DT);end;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      DiskFree := -1;end;function DiskSize (Drive: byte): int64;var FI: TFSinfo;    RC: cardinal;begin  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      DiskSize := -1;end;procedure DosSearchRec2SearchRec (var F: SearchRec);type  TRec = record    T, D: word;  end;begin with F do    begin        Name := FStat^.Name;        Size := FStat^.FileSize;        Attr := byte(FStat^.AttrFile and $FF);        TRec (Time).T := FStat^.TimeLastWrite;        TRec (Time).D := FStat^.DateLastWrite;    end;end;procedure FindFirst (const Path: PathStr; Attr: word; var F: SearchRec);var Count: cardinal;begin  {No error.}  DosError := 0;  New (F.FStat);  F.Handle := THandle ($FFFFFFFF);  Count := 1;  DosError := integer (DosFindFirst (Path, F.Handle,                     Attr and FindResvdMask, F.FStat, SizeOf (F.FStat^),                                                           Count, ilStandard));  if (DosError = 0) and (Count = 0) then DosError := 18;  DosSearchRec2SearchRec (F);end;procedure FindNext (var F: SearchRec);var  Count: cardinal;begin    {No error}    DosError := 0;    Count := 1;    DosError := integer (DosFindNext (F.Handle, F.FStat, SizeOf (F.FStat^),                                                                       Count));    if (DosError = 0) and (Count = 0) then DosError := 18;    DosSearchRec2SearchRec (F);end;procedure FindClose (var F: SearchRec);begin  if F.Handle <> THandle ($FFFFFFFF) then DosError := DosFindClose (F.Handle);  Dispose (F.FStat);end;function envcount:longint;begin  envcount:=envc;end;function envstr (index : longint) : string;var hp:Pchar;begin    if (index<=0) or (index>envcount) then        begin            envstr:='';            exit;        end;    hp:=EnvP[index-1];    envstr:=strpas(hp);end;function GetEnvPChar (EnvVar: string): 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 GetEnv(envvar: string): string;(* The assembler version is more than three times as fast as Pascal. *)begin GetEnv := StrPas (GetEnvPChar (EnvVar));end;procedure GetFAttr (var F; var Attr: word);var  PathInfo: TFileStatus3;  RC: cardinal;begin  Attr := 0;  RC := DosQueryPathInfo (@FileRec (F).Name, ilStandard,                                                 @PathInfo, SizeOf (PathInfo));  DosError := integer (RC);  if RC = 0 then    Attr := PathInfo.AttrFile;end;procedure SetFAttr (var F; Attr: word);var  PathInfo: TFileStatus3;  RC: cardinal;begin  RC := DosQueryPathInfo (@FileRec (F).Name, ilStandard,                                                 @PathInfo, SizeOf (PathInfo));  if RC = 0 then  begin    PathInfo.AttrFile := Attr;    RC := DosSetPathInfo (@FileRec (F).Name, ilStandard, @PathInfo,                                               SizeOf (PathInfo), doWriteThru);  end;  DosError := integer (RC);end;{function  GetShortName(var p : String) : boolean;begin  GetShortName:=true;}{$WARNING EA .shortname support (see FAT32 driver) should be probably added here!}{end;function  GetLongName(var p : String) : boolean;begin  GetLongName:=true;}{$WARNING EA .longname support should be probably added here!}{end;}begin LastDosExitCode := 0; LastDosErrorModuleName := ''; ExecFlags := 0;end.
 |