| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567 | {    This file is part of the Free Pascal run time library.    Copyright (c) 1999-2004 by the Free Pascal development team.    Dos unit for BP7 compatible RTL    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;interfaceuses windows;Const  Max_Path = MaxPathLen;Type  Searchrec = Packed Record    FindHandle  : THandle;    W32FindData : TWin32FindData;    ExcludeAttr : longint;    time : longint;    size : longint;    attr : longint;    name : string;  end;{$i dosh.inc}Function WinToDosTime (Const Wtime : TFileTime; var DTime:longint):longbool;Function DosToWinTime (DTime:longint; var Wtime : TFileTime):longbool;implementation{$DEFINE HAS_GETMSCOUNT}{$DEFINE FPC_FEXPAND_NO_DEFAULT_PATHS}{$DEFINE FPC_FEXPAND_UNC} (* UNC paths are supported *){$I dos.inc}{******************************************************************************                           --- Conversion ---******************************************************************************}function GetMsCount: int64;begin  GetMsCount := cardinal (GetTickCount);end;function Last2DosError(d:dword):integer;begin  case d of    87 : { Parameter invalid -> Data invalid }      Last2DosError:=13;    else      Last2DosError:=integer(d);  end;end;Function DosToWinAttr (Const Attr : Longint) : longint;begin  DosToWinAttr:=Attr;end;Function WinToDosAttr (Const Attr : Longint) : longint;begin  WinToDosAttr:=Attr;end;type  Longrec=packed record    lo,hi : word;  end;Function DosToWinTime (DTime:longint; var Wtime : TFileTime):longbool;var  FatDate, FatTime: WORD;  lft: TFileTime;  st: SYSTEMTIME;begin  FatDate:=Longrec(Dtime).Hi;  FatTime:=Longrec(Dtime).Lo;  with st do  begin    wDay:=FatDate and $1F;    wMonth:=(FatDate shr 5) and $F;    wYear:=(FatDate shr 9) + 1980;    wSecond:=(FatTime and $1F)*2;    wMinute:=(FatTime shr 5) and $1F;    wHour:=FatTime shr 11;    wMilliseconds:=0;    wDayOfWeek:=0;  end;  DosToWinTime:=SystemTimeToFileTime(@st, @lft) and LocalFileTimeToFileTime(@lft, @Wtime);end;Function WinToDosTime (Const Wtime : TFileTime; var DTime:longint):longbool;var  FatDate, FatTime: WORD;  lft: TFileTime;  st: SYSTEMTIME;  res: longbool;begin  res:=FileTimeToLocalFileTime(@WTime, @lft) and FileTimeToSystemTime(@lft, @st);  if res then  begin    FatDate:=st.wDay or (st.wMonth shl 5) or (word(st.wYear - 1980) shl 9);    FatTime:=word(st.wSecond div 2) or (st.wMinute shl 5) or (st.wHour shl 11);    Longrec(Dtime).Hi:=FatDate;    Longrec(Dtime).Lo:=FatTime;  end;  WinToDosTime:=res;end;{******************************************************************************                        --- Info / Date / Time ---******************************************************************************}function dosversion : word;var  versioninfo : OSVERSIONINFO;begin  versioninfo.dwOSVersionInfoSize:=sizeof(versioninfo);  GetVersionEx(versioninfo);  dosversion:=versioninfo.dwMajorVersion and $FF or versioninfo.dwMinorVersion and $FF shl 8;end;procedure getdate(var year,month,mday,wday : word);var  t : TSystemTime;begin  GetLocalTime(t);  year:=t.wYear;  month:=t.wMonth;  mday:=t.wDay;  wday:=t.wDayOfWeek;end;procedure setdate(year,month,day : word);var  t : TSystemTime;begin  GetLocalTime(t);  t.wYear:=year;  t.wMonth:=month;  t.wDay:=day;  { only a quite good solution, we can loose some ms }  SetLocalTime(t);end;procedure gettime(var hour,minute,second,sec100 : word);var  t : TSystemTime;begin   GetLocalTime(t);   hour:=t.wHour;   minute:=t.wMinute;   second:=t.wSecond;   sec100:=t.wMilliSeconds div 10;end;procedure settime(hour,minute,second,sec100 : word);var   t : TSystemTime;begin   GetLocalTime(t);   t.wHour:=hour;   t.wMinute:=minute;   t.wSecond:=second;   t.wMilliSeconds:=sec100*10;   SetLocalTime(t);end;{******************************************************************************                               --- Exec ---******************************************************************************}procedure exec(const path : pathstr;const comline : comstr);var  PI: TProcessInformation;  Proc : THandle;  l    : LongInt;  PathW : array[0..FileNameLen] of WideChar;  CmdLineW : array[0..FileNameLen] of WideChar;begin  DosError := 0;  AnsiToWideBuf(@path[1], Length(path), PathW, SizeOf(PathW));  AnsiToWideBuf(@comline[1], Length(comline), CmdLineW, SizeOf(CmdLineW));  if not CreateProcess(PathW, CmdLineW,           nil, nil, FALSE, 0, nil, nil, nil, PI) then   begin     DosError:=Last2DosError(GetLastError);     exit;   end;  Proc:=PI.hProcess;  CloseHandle(PI.hThread);  if WaitForSingleObject(Proc, dword($ffffffff)) <> $ffffffff then    GetExitCodeProcess(Proc, @l)  else    l:=-1;  CloseHandle(Proc);  LastDosExitCode:=l;end;{******************************************************************************                               --- Disk ---******************************************************************************}var  DriveNames: array[1..24] of PWideChar;function GetDriveName(drive: byte): PWideChar;const  dev_attr = FILE_ATTRIBUTE_TEMPORARY or FILE_ATTRIBUTE_DIRECTORY;var  h: THandle;  fd: TWin32FindData;  i, len: LongInt;begin  GetDriveName:=nil;  // Current drive is C: drive always  if drive = 0 then    drive:=2;  if (drive < 3) or (drive > 26) then    exit;  if DriveNames[1] = nil then  begin    // Drive C: is filesystem root always    GetMem(DriveNames[1], 2*SizeOf(WideChar));    DriveNames[1][0]:='\';    DriveNames[1][1]:=#0;    // Other drives are found dinamically    h:=FindFirstFile('\*', @fd);    if h <> 0 then    begin      i:=2;      repeat        if fd.dwFileAttributes and dev_attr = dev_attr then begin          len:=0;          while fd.cFileName[len] <> #0 do            Inc(len);          len:=(len + 2)*SizeOf(WideChar);          GetMem(DriveNames[i], len);          DriveNames[i]^:='\';          Move(fd.cFileName, DriveNames[i][1], len - SizeOf(WideChar));          Inc(i);        end;      until (i > 24) or not FindNextFile(h, fd);      Windows.FindClose(h);    end;  end;  GetDriveName:=DriveNames[drive - 2];end;function diskfree(drive : byte) : int64;var  disk: PWideChar;  qwtotal,qwfree,qwcaller : int64;begin  disk:=GetDriveName(drive);  if (disk <> nil) and GetDiskFreeSpaceEx(disk, @qwcaller, @qwtotal, @qwfree) then    diskfree:=qwfree  else    diskfree:=-1;end;function disksize(drive : byte) : int64;var  disk : PWideChar;  qwtotal,qwfree,qwcaller : int64;begin  disk:=GetDriveName(drive);  if (disk <> nil) and GetDiskFreeSpaceEx(disk, @qwcaller, @qwtotal, @qwfree) then    disksize:=qwtotal  else    disksize:=-1;end;{******************************************************************************                         --- Findfirst FindNext ---******************************************************************************}Procedure StringToPchar (Var S : String);Var L : Longint;begin  L:=ord(S[0]);  Move (S[1],S[0],L);  S[L]:=#0;end;Procedure PCharToString (Var S : String);Var L : Longint;begin  L:=strlen(pchar(@S[0]));  Move (S[0],S[1],L);  S[0]:=char(l);end;procedure FindMatch(var f:searchrec);var  buf: array[0..MaxPathLen] of char;begin  { Find file with correct attribute }  While (F.W32FindData.dwFileAttributes and cardinal(F.ExcludeAttr))<>0 do   begin     if not FindNextFile (F.FindHandle, F.W32FindData) then      begin        DosError:=Last2DosError(GetLastError);        if DosError=2 then         DosError:=18;        exit;      end;   end;  { Convert some attributes back }  f.size:=F.W32FindData.NFileSizeLow;  f.attr:=WinToDosAttr(F.W32FindData.dwFileAttributes);  WinToDosTime(F.W32FindData.ftLastWriteTime,f.Time);  WideToAnsiBuf(@F.W32FindData.cFileName, -1, buf, SizeOf(buf));  f.Name:=StrPas(@buf);end;procedure findfirst(const path : pathstr;attr : word;var f : searchRec);var  buf: array[0..MaxPathLen] of WideChar;begin  if path = ''then    begin      DosError:=3;      exit;    end;  fillchar(f,sizeof(f),0);  { no error }  doserror:=0;  F.Name:=Path;  F.Attr:=attr;  F.ExcludeAttr:=(not Attr) and ($1e); {hidden,sys,dir,volume}  StringToPchar(f.name);  { FindFirstFile is a WinCE Call }  F.W32FindData.dwFileAttributes:=DosToWinAttr(f.attr);  AnsiToWideBuf(@f.Name, -1, buf, SizeOf(buf));  F.FindHandle:=FindFirstFile (buf, F.W32FindData);  If F.FindHandle = Invalid_Handle_value then   begin     DosError:=Last2DosError(GetLastError);     if DosError=2 then      DosError:=18;     exit;   end;  { Find file with correct attribute }  FindMatch(f);end;procedure findnext(var f : searchRec);begin{ no error }  doserror:=0;  if not FindNextFile (F.FindHandle, F.W32FindData) then   begin     DosError:=Last2DosError(GetLastError);     if DosError=2 then      DosError:=18;     exit;   end;{ Find file with correct attribute }  FindMatch(f);end;Procedure FindClose(Var f: SearchRec);begin  If F.FindHandle <> Invalid_Handle_value then    Windows.FindClose(F.FindHandle);end;{******************************************************************************                               --- File ---******************************************************************************}Function FSearch(path: pathstr; dirlist: string): pathstr;var  p1     : longint;  s      : searchrec;  newdir : pathstr;begin  { check if the file specified exists }  findfirst(path,anyfile and not(directory),s);  if doserror=0 then   begin     findclose(s);     fsearch:=path;     exit;   end;  { No wildcards allowed in these things }  if (pos('?',path)<>0) or (pos('*',path)<>0) then    fsearch:=''  else    begin       { allow slash as backslash }       DoDirSeparators(dirlist);       repeat         p1:=pos(';',dirlist);         if p1<>0 then          begin            newdir:=copy(dirlist,1,p1-1);            delete(dirlist,1,p1);          end         else          begin            newdir:=dirlist;            dirlist:='';          end;         if (newdir<>'') and (not (newdir[length(newdir)] in ['\',':'])) then          newdir:=newdir+'\';         findfirst(newdir+path,anyfile and not(directory),s);         if doserror=0 then          newdir:=newdir+path         else          newdir:='';       until (dirlist='') or (newdir<>'');       fsearch:=newdir;    end;  findclose(s);end;{ </immobilizer> }procedure getftime(var f;var time : longint);var   ft : TFileTime;begin  doserror:=0;  if GetFileTime(filerec(f).Handle,nil,nil,@ft) and     WinToDosTime(ft,time) then    exit  else    begin      DosError:=Last2DosError(GetLastError);      time:=0;    end;end;procedure setftime(var f;time : longint);var  ft : TFileTime;begin  doserror:=0;  if DosToWinTime(time,ft) and     SetFileTime(filerec(f).Handle,nil,nil,@ft) then   exit  else   DosError:=Last2DosError(GetLastError);end;procedure getfattr(var f;var attr : word);var  l : cardinal;  buf: array[0..MaxPathLen] of WideChar;begin  if filerec(f).name[1] = #0 then    begin      doserror:=3;      attr:=0;    end  else    begin      doserror:=0;      AnsiToWideBuf(@filerec(f).name, -1, buf, SizeOf(buf));      l:=GetFileAttributes(buf);      if l = $ffffffff then       begin         doserror:=Last2DosError(GetLastError);         attr:=0;       end      else       attr:=l and $ffff;    end;end;procedure setfattr(var f;attr : word);var  buf: array[0..MaxPathLen] of WideChar;begin  { Fail for setting VolumeId }  if (attr and VolumeID)<>0 then    doserror:=5  else    begin      AnsiToWideBuf(@filerec(f).name, -1, buf, SizeOf(buf));      if SetFileAttributes(buf,attr) then        doserror:=0      else        doserror:=Last2DosError(GetLastError);    end;end;{******************************************************************************                             --- Environment ---******************************************************************************}// WinCE does not have environment. It can be emulated via registry or file. (YS)function envcount : longint;begin  envcount:=0;end;Function EnvStr (Index: longint): string;begin  EnvStr:='';end;Function  GetEnv(envvar: string): string;begin  GetEnv:='';end;var  oldexitproc : pointer;procedure dosexitproc;var  i: LongInt;begin  exitproc:=oldexitproc;  if DriveNames[1] <> nil then    for i:=1 to 24 do      if DriveNames[i] <> nil then        FreeMem(DriveNames[i])      else        break;end;begin  oldexitproc:=exitproc;  exitproc:=@dosexitproc;end.
 |