mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-10-24 04:51:32 +02:00
859 lines
21 KiB
ObjectPascal
859 lines
21 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
|
|
|
|
Const
|
|
Max_Path = 260;
|
|
|
|
Type
|
|
PWinFileTime = ^TWinFileTime;
|
|
TWinFileTime = record
|
|
dwLowDateTime,
|
|
dwHighDateTime : DWORD;
|
|
end;
|
|
|
|
PWinFindData = ^TWinFindData;
|
|
TWinFindData = record
|
|
dwFileAttributes: DWORD;
|
|
ftCreationTime: TWinFileTime;
|
|
ftLastAccessTime: TWinFileTime;
|
|
ftLastWriteTime: TWinFileTime;
|
|
nFileSizeHigh: DWORD;
|
|
nFileSizeLow: DWORD;
|
|
dwReserved0: DWORD;
|
|
dwReserved1: DWORD;
|
|
cFileName: array[0..MAX_PATH-1] of Char;
|
|
cAlternateFileName: array[0..15] of Char;
|
|
// The structure should be 320 bytes long...
|
|
pad : system.integer;
|
|
end;
|
|
|
|
Searchrec = Record
|
|
FindHandle : THandle;
|
|
WinFindData : TWinFindData;
|
|
ExcludeAttr : longint;
|
|
time : longint;
|
|
size : longint;
|
|
attr : longint;
|
|
name : string;
|
|
end;
|
|
|
|
{$i dosh.inc}
|
|
|
|
Const
|
|
{ allow EXEC to inherited handles from calling process,
|
|
needed for FPREDIR in ide/text
|
|
now set to true by default because
|
|
other OS also pass open handles to childs
|
|
finally reset to false after Florian's response PM }
|
|
ExecInheritsHandles : Longbool = false;
|
|
|
|
|
|
implementation
|
|
|
|
uses
|
|
strings;
|
|
|
|
{$DEFINE HAS_GETMSCOUNT}
|
|
{$DEFINE HAS_GETSHORTNAME}
|
|
{$DEFINE HAS_GETLONGNAME}
|
|
|
|
{$DEFINE FPC_FEXPAND_UNC} (* UNC paths are supported *)
|
|
{$DEFINE FPC_FEXPAND_DRIVES} (* Full paths begin with drive specification *)
|
|
|
|
{$I dos.inc}
|
|
|
|
const
|
|
INVALID_HANDLE_VALUE = THandle(-1);
|
|
|
|
VER_PLATFORM_WIN32s = 0;
|
|
VER_PLATFORM_WIN32_WINDOWS = 1;
|
|
VER_PLATFORM_WIN32_NT = 2;
|
|
|
|
type
|
|
OSVERSIONINFO = record
|
|
dwOSVersionInfoSize : DWORD;
|
|
dwMajorVersion : DWORD;
|
|
dwMinorVersion : DWORD;
|
|
dwBuildNumber : DWORD;
|
|
dwPlatformId : DWORD;
|
|
szCSDVersion : array[0..127] of char;
|
|
end;
|
|
|
|
var
|
|
versioninfo : OSVERSIONINFO;
|
|
kernel32dll : THandle;
|
|
|
|
{******************************************************************************
|
|
--- Conversion ---
|
|
******************************************************************************}
|
|
|
|
function GetLastError : DWORD;
|
|
stdcall; external 'kernel32' name 'GetLastError';
|
|
function FileTimeToDosDateTime(const ft :TWinFileTime;var data,time : word) : longbool;
|
|
stdcall; external 'kernel32' name 'FileTimeToDosDateTime';
|
|
function DosDateTimeToFileTime(date,time : word;var ft :TWinFileTime) : longbool;
|
|
stdcall; external 'kernel32' name 'DosDateTimeToFileTime';
|
|
function FileTimeToLocalFileTime(const ft : TWinFileTime;var lft : TWinFileTime) : longbool;
|
|
stdcall; external 'kernel32' name 'FileTimeToLocalFileTime';
|
|
function LocalFileTimeToFileTime(const lft : TWinFileTime;var ft : TWinFileTime) : longbool;
|
|
stdcall; external 'kernel32' name 'LocalFileTimeToFileTime';
|
|
function GetTickCount : longint;
|
|
stdcall;external 'kernel32' name 'GetTickCount';
|
|
|
|
function GetMsCount: int64;
|
|
begin
|
|
GetMsCount := cardinal (GetTickCount);
|
|
end;
|
|
|
|
type
|
|
Longrec=packed record
|
|
lo,hi : word;
|
|
end;
|
|
|
|
function Last2DosError(d:dword):integer;
|
|
begin
|
|
case d of
|
|
87 : { Parameter invalid -> Data invalid }
|
|
Last2DosError:=13;
|
|
else
|
|
Last2DosError:=d;
|
|
end;
|
|
end;
|
|
|
|
|
|
Function DosToWinAttr (Const Attr : Longint) : longint;
|
|
begin
|
|
DosToWinAttr:=Attr;
|
|
end;
|
|
|
|
|
|
Function WinToDosAttr (Const Attr : Longint) : longint;
|
|
begin
|
|
WinToDosAttr:=Attr;
|
|
end;
|
|
|
|
|
|
Function DosToWinTime (DTime:longint;Var Wtime : TWinFileTime):longbool;
|
|
var
|
|
lft : TWinFileTime;
|
|
begin
|
|
DosToWinTime:=DosDateTimeToFileTime(longrec(dtime).hi,longrec(dtime).lo,lft) and
|
|
LocalFileTimeToFileTime(lft,Wtime);
|
|
end;
|
|
|
|
|
|
Function WinToDosTime (Const Wtime : TWinFileTime;var DTime:longint):longbool;
|
|
var
|
|
lft : TWinFileTime;
|
|
begin
|
|
WinToDosTime:=FileTimeToLocalFileTime(WTime,lft) and
|
|
FileTimeToDosDateTime(lft,longrec(dtime).hi,longrec(dtime).lo);
|
|
end;
|
|
|
|
|
|
{******************************************************************************
|
|
--- Info / Date / Time ---
|
|
******************************************************************************}
|
|
|
|
type
|
|
TSystemTime = record
|
|
wYear,
|
|
wMonth,
|
|
wDayOfWeek,
|
|
wDay,
|
|
wHour,
|
|
wMinute,
|
|
wSecond,
|
|
wMilliseconds: Word;
|
|
end;
|
|
|
|
function GetVersion : longint;
|
|
stdcall; external 'kernel32' name 'GetVersion';
|
|
procedure GetLocalTime(var t : TSystemTime);
|
|
stdcall; external 'kernel32' name 'GetLocalTime';
|
|
function SetLocalTime(const t : TSystemTime) : longbool;
|
|
stdcall; external 'kernel32' name 'SetLocalTime';
|
|
|
|
function dosversion : word;
|
|
begin
|
|
dosversion:=GetVersion and $ffff;
|
|
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
|
|
{ we need the time set privilege }
|
|
{ so this function crash currently }
|
|
{!!!!!}
|
|
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
|
|
{ we need the time set privilege }
|
|
{ so this function crash currently }
|
|
{!!!!!}
|
|
GetLocalTime(t);
|
|
t.wHour:=hour;
|
|
t.wMinute:=minute;
|
|
t.wSecond:=second;
|
|
t.wMilliSeconds:=sec100*10;
|
|
SetLocalTime(t);
|
|
end;
|
|
|
|
|
|
{******************************************************************************
|
|
--- Exec ---
|
|
******************************************************************************}
|
|
|
|
type
|
|
PProcessInformation = ^TProcessInformation;
|
|
TProcessInformation = record
|
|
hProcess: THandle;
|
|
hThread: THandle;
|
|
dwProcessId: DWORD;
|
|
dwThreadId: DWORD;
|
|
end;
|
|
|
|
function CreateProcess(lpApplicationName: PChar; lpCommandLine: PChar;
|
|
lpProcessAttributes, lpThreadAttributes: Pointer;
|
|
bInheritHandles: Longbool; dwCreationFlags: DWORD; lpEnvironment: Pointer;
|
|
lpCurrentDirectory: PChar; const lpStartupInfo: TStartupInfo;
|
|
var lpProcessInformation: TProcessInformation): longbool;
|
|
stdcall; external 'kernel32' name 'CreateProcessA';
|
|
function getExitCodeProcess(h:THandle;var code:longint):longbool;
|
|
stdcall; external 'kernel32' name 'GetExitCodeProcess';
|
|
function WaitForSingleObject(hHandle: THandle; dwMilliseconds: DWORD): DWORD;
|
|
stdcall; external 'kernel32' name 'WaitForSingleObject';
|
|
function CloseHandle(h : THandle) : longint;
|
|
stdcall; external 'kernel32' name 'CloseHandle';
|
|
|
|
procedure exec(const path : pathstr;const comline : comstr);
|
|
var
|
|
SI: TStartupInfo;
|
|
PI: TProcessInformation;
|
|
l : Longint;
|
|
CommandLine : array[0..511] of char;
|
|
AppParam : array[0..255] of char;
|
|
pathlocal : string;
|
|
begin
|
|
DosError:=0;
|
|
FillChar(SI, SizeOf(SI), 0);
|
|
SI.cb:=SizeOf(SI);
|
|
SI.wShowWindow:=1;
|
|
{ always surround the name of the application by quotes
|
|
so that long filenames will always be accepted. But don't
|
|
do it if there are already double quotes, since Win32 does not
|
|
like double quotes which are duplicated!
|
|
}
|
|
if pos('"',path) = 0 then
|
|
pathlocal:='"'+path+'"'
|
|
else
|
|
pathlocal := path;
|
|
Move(Pathlocal[1],CommandLine,length(Pathlocal));
|
|
|
|
AppParam[0]:=' ';
|
|
AppParam[1]:=' ';
|
|
Move(ComLine[1],AppParam[2],length(Comline));
|
|
AppParam[Length(ComLine)+2]:=#0;
|
|
{ concatenate both pathnames }
|
|
Move(Appparam[0],CommandLine[length(Pathlocal)],strlen(Appparam)+1);
|
|
if not CreateProcess(nil, PChar(@CommandLine),
|
|
Nil, Nil, ExecInheritsHandles,$20, Nil, Nil, SI, PI) then
|
|
begin
|
|
DosError:=Last2DosError(GetLastError);
|
|
exit;
|
|
end;
|
|
if WaitForSingleObject(PI.hProcess,dword($ffffffff))<>$ffffffff then
|
|
GetExitCodeProcess(PI.hProcess,l)
|
|
else
|
|
l:=-1;
|
|
CloseHandle(PI.hProcess);
|
|
CloseHandle(PI.hThread);
|
|
LastDosExitCode:=l;
|
|
end;
|
|
|
|
|
|
{******************************************************************************
|
|
--- Disk ---
|
|
******************************************************************************}
|
|
|
|
function GetDiskFreeSpace(drive:pchar;var sector_cluster,bytes_sector,
|
|
freeclusters,totalclusters:DWORD):longbool;
|
|
stdcall; external 'kernel32' name 'GetDiskFreeSpaceA';
|
|
type
|
|
TGetDiskFreeSpaceEx = function(drive:pchar;var availableforcaller,
|
|
total,free):longbool;stdcall;
|
|
|
|
var
|
|
GetDiskFreeSpaceEx : TGetDiskFreeSpaceEx;
|
|
|
|
function diskfree(drive : byte) : int64;
|
|
var
|
|
disk : array[1..4] of char;
|
|
secs,bytes,
|
|
free,total : DWORD;
|
|
qwtotal,qwfree,qwcaller : int64;
|
|
|
|
|
|
begin
|
|
if drive=0 then
|
|
begin
|
|
disk[1]:='\';
|
|
disk[2]:=#0;
|
|
end
|
|
else
|
|
begin
|
|
disk[1]:=chr(drive+64);
|
|
disk[2]:=':';
|
|
disk[3]:='\';
|
|
disk[4]:=#0;
|
|
end;
|
|
if assigned(GetDiskFreeSpaceEx) then
|
|
begin
|
|
if GetDiskFreeSpaceEx(@disk,qwcaller,qwtotal,qwfree) then
|
|
diskfree:=qwfree
|
|
else
|
|
diskfree:=-1;
|
|
end
|
|
else
|
|
begin
|
|
if GetDiskFreeSpace(@disk,secs,bytes,free,total) then
|
|
diskfree:=int64(free)*secs*bytes
|
|
else
|
|
diskfree:=-1;
|
|
end;
|
|
end;
|
|
|
|
|
|
function disksize(drive : byte) : int64;
|
|
var
|
|
disk : array[1..4] of char;
|
|
secs,bytes,
|
|
free,total : DWORD;
|
|
qwtotal,qwfree,qwcaller : int64;
|
|
|
|
begin
|
|
if drive=0 then
|
|
begin
|
|
disk[1]:='\';
|
|
disk[2]:=#0;
|
|
end
|
|
else
|
|
begin
|
|
disk[1]:=chr(drive+64);
|
|
disk[2]:=':';
|
|
disk[3]:='\';
|
|
disk[4]:=#0;
|
|
end;
|
|
if assigned(GetDiskFreeSpaceEx) then
|
|
begin
|
|
if GetDiskFreeSpaceEx(@disk,qwcaller,qwtotal,qwfree) then
|
|
disksize:=qwtotal
|
|
else
|
|
disksize:=-1;
|
|
end
|
|
else
|
|
begin
|
|
if GetDiskFreeSpace(@disk,secs,bytes,free,total) then
|
|
disksize:=int64(total)*secs*bytes
|
|
else
|
|
disksize:=-1;
|
|
end;
|
|
end;
|
|
|
|
|
|
{******************************************************************************
|
|
--- Findfirst FindNext ---
|
|
******************************************************************************}
|
|
|
|
{ Needed kernel calls }
|
|
|
|
function FindFirstFile (lpFileName: PChar; var lpFindFileData: TWinFindData): THandle;
|
|
stdcall; external 'kernel32' name 'FindFirstFileA';
|
|
function FindNextFile (hFindFile: THandle; var lpFindFileData: TWinFindData): LongBool;
|
|
stdcall; external 'kernel32' name 'FindNextFileA';
|
|
function FindCloseFile (hFindFile: THandle): LongBool;
|
|
stdcall; external 'kernel32' name 'FindClose';
|
|
|
|
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);
|
|
begin
|
|
{ Find file with correct attribute }
|
|
While (F.WinFindData.dwFileAttributes and DWORD(F.ExcludeAttr))<>0 do
|
|
begin
|
|
if not FindNextFile (F.FindHandle,F.WinFindData) then
|
|
begin
|
|
DosError:=Last2DosError(GetLastError);
|
|
if DosError=2 then
|
|
DosError:=18;
|
|
exit;
|
|
end;
|
|
end;
|
|
|
|
{ Convert some attributes back }
|
|
f.size:=F.WinFindData.NFileSizeLow;
|
|
f.attr:=WinToDosAttr(F.WinFindData.dwFileAttributes);
|
|
WinToDosTime(F.WinFindData.ftLastWriteTime,f.Time);
|
|
f.Name:=StrPas(@F.WinFindData.cFileName);
|
|
end;
|
|
|
|
|
|
procedure findfirst(const path : pathstr;attr : word;var f : searchRec);
|
|
begin
|
|
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 Win32 Call }
|
|
F.WinFindData.dwFileAttributes:=DosToWinAttr(f.attr);
|
|
F.FindHandle:=FindFirstFile (pchar(@f.Name),F.WinFindData);
|
|
|
|
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.WinFindData) 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
|
|
FindCloseFile(F.FindHandle);
|
|
end;
|
|
|
|
|
|
{******************************************************************************
|
|
--- File ---
|
|
******************************************************************************}
|
|
|
|
function GetWinFileTime(h : longint;creation,lastaccess,lastwrite : PWinFileTime) : longbool;
|
|
stdcall; external 'kernel32' name 'GetFileTime';
|
|
function SetWinFileTime(h : longint;creation,lastaccess,lastwrite : PWinFileTime) : longbool;
|
|
stdcall; external 'kernel32' name 'SetFileTime';
|
|
function SetFileAttributes(lpFileName : pchar;dwFileAttributes : longint) : longbool;
|
|
stdcall; external 'kernel32' name 'SetFileAttributesA';
|
|
function GetFileAttributes(lpFileName : pchar) : longint;
|
|
stdcall; external 'kernel32' name 'GetFileAttributesA';
|
|
|
|
|
|
{ <immobilizer> }
|
|
|
|
function GetFullPathName(lpFileName: PChar; nBufferLength: Longint; lpBuffer: PChar; var lpFilePart : PChar):DWORD;
|
|
stdcall; external 'kernel32' name 'GetFullPathNameA';
|
|
function GetShortPathName(lpszLongPath:pchar; lpszShortPath:pchar; cchBuffer:DWORD):DWORD;
|
|
stdcall; external 'kernel32' name 'GetShortPathNameA';
|
|
|
|
|
|
Function FSearch(path: pathstr; dirlist: string): pathstr;
|
|
var
|
|
i,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 }
|
|
for i:=1 to length(dirlist) do
|
|
if dirlist[i]='/' then dirlist[i]:='\';
|
|
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;
|
|
|
|
|
|
procedure getftime(var f;var time : longint);
|
|
var
|
|
ft : TWinFileTime;
|
|
begin
|
|
doserror:=0;
|
|
if GetWinFileTime(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 : TWinFileTime;
|
|
begin
|
|
doserror:=0;
|
|
if DosToWinTime(time,ft) and
|
|
SetWinFileTime(filerec(f).Handle,nil,nil,@ft) then
|
|
exit
|
|
else
|
|
DosError:=Last2DosError(GetLastError);
|
|
end;
|
|
|
|
|
|
procedure getfattr(var f;var attr : word);
|
|
var
|
|
l : longint;
|
|
begin
|
|
doserror:=0;
|
|
l:=GetFileAttributes(filerec(f).name);
|
|
if l=longint($ffffffff) then
|
|
begin
|
|
doserror:=getlasterror;
|
|
attr:=0;
|
|
end
|
|
else
|
|
attr:=l and $ffff;
|
|
end;
|
|
|
|
|
|
procedure setfattr(var f;attr : word);
|
|
begin
|
|
{ Fail for setting VolumeId }
|
|
if (attr and VolumeID)<>0 then
|
|
doserror:=5
|
|
else
|
|
if SetFileAttributes(filerec(f).name,attr) then
|
|
doserror:=0
|
|
else
|
|
doserror:=getlasterror;
|
|
end;
|
|
|
|
{ change to short filename if successful win32 call PM }
|
|
function GetShortName(var p : String) : boolean;
|
|
var
|
|
buffer : array[0..255] of char;
|
|
ret : longint;
|
|
begin
|
|
{we can't mess with p, because we have to return it if call is
|
|
unsuccesfully.}
|
|
|
|
if Length(p)>0 then {copy p to array of char}
|
|
move(p[1],buffer[0],length(p));
|
|
buffer[length(p)]:=chr(0);
|
|
|
|
{Should return value load loaddoserror?}
|
|
|
|
ret:=GetShortPathName(@buffer,@buffer,255);
|
|
if (Ret > 0) and (Ret <= 255) then
|
|
begin
|
|
Move (Buffer, P [1], Ret);
|
|
byte (P [0]) := Ret;
|
|
GetShortName := true;
|
|
end
|
|
else
|
|
GetShortName := false;
|
|
end;
|
|
|
|
{ change to long filename if successful DOS call PM }
|
|
function GetLongName(var p : String) : boolean;
|
|
|
|
var
|
|
SR: SearchRec;
|
|
FullFN, FinalFN, TestFN: string;
|
|
Found: boolean;
|
|
SPos: byte;
|
|
begin
|
|
if Length (P) = 0 then
|
|
GetLongName := false
|
|
else
|
|
begin
|
|
FullFN := FExpand (P); (* Needed to be done at the beginning to get proper case for all parts *)
|
|
SPos := 1;
|
|
if (Length (FullFN) > 2) then
|
|
if (FullFN [2] = DriveSeparator) then
|
|
SPos := 4
|
|
else
|
|
if (FullFN [1] = DirectorySeparator) and (FullFN [2] = DirectorySeparator) then
|
|
begin
|
|
SPos := 3;
|
|
while (Length (FullFN) > SPos) and (FullFN [SPos] <> DirectorySeparator) do
|
|
Inc (SPos);
|
|
if SPos >= Length (FullFN) then
|
|
SPos := 1
|
|
else
|
|
begin
|
|
Inc (SPos);
|
|
while (Length (FullFN) >= SPos) and (FullFN [SPos] <> DirectorySeparator) do
|
|
Inc (SPos);
|
|
if SPos <= Length (FullFN) then
|
|
Inc (SPos);
|
|
end;
|
|
end;
|
|
FinalFN := Copy (FullFN, 1, Pred (SPos));
|
|
Delete (FullFN, 1, Pred (SPos));
|
|
Found := true;
|
|
while (FullFN <> '') and Found do
|
|
begin
|
|
SPos := Pos (DirectorySeparator, FullFN);
|
|
TestFN := Copy (FullFN, 1, Pred (SPos));
|
|
Delete (FullFN, 1, Pred (SPos));
|
|
FindFirst (FinalFN + TestFN, AnyFile, SR);
|
|
if DosError <> 0 then
|
|
Found := false
|
|
else
|
|
begin
|
|
FinalFN := FinalFN + SR.Name;
|
|
if (FullFN <> '') and (FullFN [1] = DirectorySeparator) then
|
|
begin
|
|
FinalFN := FinalFN + DirectorySeparator;
|
|
Delete (FullFN, 1, 1);
|
|
end;
|
|
end;
|
|
FindClose (SR);
|
|
end;
|
|
if Found then
|
|
begin
|
|
GetLongName := true;
|
|
P := FinalFN;
|
|
end
|
|
else
|
|
GetLongName := false
|
|
end;
|
|
end;
|
|
|
|
{******************************************************************************
|
|
--- Environment ---
|
|
******************************************************************************}
|
|
|
|
{
|
|
The environment is a block of zero terminated strings
|
|
terminated by a #0
|
|
}
|
|
|
|
function GetEnvironmentStrings : pchar;
|
|
stdcall; external 'kernel32' name 'GetEnvironmentStringsA';
|
|
function FreeEnvironmentStrings(p : pchar) : longbool;
|
|
stdcall; external 'kernel32' name 'FreeEnvironmentStringsA';
|
|
|
|
function envcount : longint;
|
|
var
|
|
hp,p : pchar;
|
|
count : longint;
|
|
begin
|
|
p:=GetEnvironmentStrings;
|
|
hp:=p;
|
|
count:=0;
|
|
while hp^<>#0 do
|
|
begin
|
|
{ next string entry}
|
|
hp:=hp+strlen(hp)+1;
|
|
inc(count);
|
|
end;
|
|
FreeEnvironmentStrings(p);
|
|
envcount:=count;
|
|
end;
|
|
|
|
|
|
Function EnvStr (Index: longint): string;
|
|
var
|
|
hp,p : pchar;
|
|
count,i : longint;
|
|
begin
|
|
{ envcount takes some time in win32 }
|
|
count:=envcount;
|
|
|
|
{ range checking }
|
|
if (index<=0) or (index>count) then
|
|
begin
|
|
envstr:='';
|
|
exit;
|
|
end;
|
|
p:=GetEnvironmentStrings;
|
|
hp:=p;
|
|
|
|
{ retrive the string with the given index }
|
|
for i:=2 to index do
|
|
hp:=hp+strlen(hp)+1;
|
|
|
|
envstr:=strpas(hp);
|
|
FreeEnvironmentStrings(p);
|
|
end;
|
|
|
|
|
|
Function GetEnv(envvar: string): string;
|
|
var
|
|
s : string;
|
|
i : longint;
|
|
hp,p : pchar;
|
|
begin
|
|
getenv:='';
|
|
p:=GetEnvironmentStrings;
|
|
hp:=p;
|
|
while hp^<>#0 do
|
|
begin
|
|
s:=strpas(hp);
|
|
i:=pos('=',s);
|
|
if upcase(copy(s,1,i-1))=upcase(envvar) then
|
|
begin
|
|
{ getenv:=copy(s,i+1,length(s)-i);
|
|
this limits the size to 255-(i+1) }
|
|
getenv:=strpas(hp+i);
|
|
break;
|
|
end;
|
|
{ next string entry}
|
|
hp:=hp+strlen(hp)+1;
|
|
end;
|
|
FreeEnvironmentStrings(p);
|
|
end;
|
|
|
|
|
|
function FreeLibrary(hLibModule : THandle) : longbool;
|
|
stdcall; external 'kernel32' name 'FreeLibrary';
|
|
function GetVersionEx(var VersionInformation:OSVERSIONINFO) : longbool;
|
|
stdcall; external 'kernel32' name 'GetVersionExA';
|
|
function LoadLibrary(lpLibFileName : pchar):THandle;
|
|
stdcall; external 'kernel32' name 'LoadLibraryA';
|
|
function GetProcAddress(hModule : THandle;lpProcName : pchar) : pointer;
|
|
stdcall; external 'kernel32' name 'GetProcAddress';
|
|
|
|
var
|
|
oldexitproc : pointer;
|
|
|
|
procedure dosexitproc;
|
|
|
|
begin
|
|
exitproc:=oldexitproc;
|
|
if kernel32dll<>0 then
|
|
FreeLibrary(kernel32dll);
|
|
end;
|
|
|
|
begin
|
|
oldexitproc:=exitproc;
|
|
exitproc:=@dosexitproc;
|
|
versioninfo.dwOSVersionInfoSize:=sizeof(versioninfo);
|
|
GetVersionEx(versioninfo);
|
|
kernel32dll:=0;
|
|
GetDiskFreeSpaceEx:=nil;
|
|
{$ifndef win64}
|
|
if ((versioninfo.dwPlatformId=VER_PLATFORM_WIN32_WINDOWS) and
|
|
(versioninfo.dwBuildNUmber>=1000)) or
|
|
(versioninfo.dwPlatformId=VER_PLATFORM_WIN32_NT) then
|
|
{$endif win64}
|
|
begin
|
|
kernel32dll:=LoadLibrary('kernel32');
|
|
if kernel32dll<>0 then
|
|
GetDiskFreeSpaceEx:=TGetDiskFreeSpaceEx(GetProcAddress(kernel32dll,'GetDiskFreeSpaceExA'));
|
|
end;
|
|
end.
|