mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-10 10:09:22 +02:00
575 lines
13 KiB
ObjectPascal
575 lines
13 KiB
ObjectPascal
{
|
|
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;
|
|
interface
|
|
|
|
uses 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
|
|
{ No wildcards allowed in these things }
|
|
if (pos('?',path)<>0) or (pos('*',path)<>0) then
|
|
begin
|
|
fsearch:='';
|
|
exit;
|
|
end;
|
|
{ check if the file specified exists }
|
|
findfirst(path,anyfile and not(directory),s);
|
|
if doserror=0 then
|
|
begin
|
|
findclose(s);
|
|
fsearch:=path;
|
|
exit;
|
|
end;
|
|
findclose(s);
|
|
{ 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 [DirectorySeparator,DriveSeparator])) then
|
|
newdir:=newdir+DirectorySeparator;
|
|
findfirst(newdir+path,anyfile and not(directory),s);
|
|
if doserror=0 then
|
|
newdir:=newdir+path
|
|
else
|
|
newdir:='';
|
|
findclose(s);
|
|
until (dirlist='') or (newdir<>'');
|
|
fsearch:=newdir;
|
|
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;
|
|
{$ifdef FPC_ANSI_TEXTFILEREC}
|
|
u: unicodestring;
|
|
{$endif FPC_ANSI_TEXTFILEREC}
|
|
begin
|
|
if filerec(f).name[0] = #0 then
|
|
begin
|
|
doserror:=3;
|
|
attr:=0;
|
|
end
|
|
else
|
|
begin
|
|
doserror:=0;
|
|
{$ifdef FPC_ANSI_TEXTFILEREC}
|
|
widestringmanager.Ansi2UnicodeMoveProc(filerec(f).name,DefaultFileSystemCodePage,u,length(filerec(f).name));
|
|
l:=GetFileAttributes(pwidechar(u));
|
|
{$else}
|
|
l:=GetFileAttributes(filerec(f).name);
|
|
{$endif}
|
|
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.
|