mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-10-24 07:01:44 +02:00
1046 lines
26 KiB
ObjectPascal
1046 lines
26 KiB
ObjectPascal
{
|
|
$Id$
|
|
This file is part of the Free Pascal run time library.
|
|
Copyright (c) 1999-2000 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
|
|
|
|
{ Include Win32 Consts,Types }
|
|
{$I win32.inc}
|
|
|
|
Const
|
|
Max_Path = 260;
|
|
|
|
{Bitmasks for CPU Flags}
|
|
fcarry = $0001;
|
|
fparity = $0004;
|
|
fauxiliary = $0010;
|
|
fzero = $0040;
|
|
fsign = $0080;
|
|
foverflow = $0800;
|
|
|
|
{Bitmasks for file attribute}
|
|
readonly = $01;
|
|
hidden = $02;
|
|
sysfile = $04;
|
|
volumeid = $08;
|
|
directory = $10;
|
|
archive = $20;
|
|
anyfile = $3F;
|
|
|
|
{File Status}
|
|
fmclosed = $D7B0;
|
|
fminput = $D7B1;
|
|
fmoutput = $D7B2;
|
|
fminout = $D7B3;
|
|
|
|
|
|
Type
|
|
{ Needed for Win95 LFN Support }
|
|
ComStr = String[255];
|
|
PathStr = String[255];
|
|
DirStr = String[255];
|
|
NameStr = String[255];
|
|
ExtStr = String[255];
|
|
|
|
{
|
|
filerec.inc contains the definition of the filerec.
|
|
textrec.inc contains the definition of the textrec.
|
|
It is in a separate file to make it available in other units without
|
|
having to use the DOS unit for it.
|
|
}
|
|
{$i filerec.inc}
|
|
{$i textrec.inc}
|
|
|
|
DateTime = packed record
|
|
Year,
|
|
Month,
|
|
Day,
|
|
Hour,
|
|
Min,
|
|
Sec : word;
|
|
End;
|
|
|
|
PWin32FindData = ^TWin32FindData;
|
|
TWin32FindData = record
|
|
dwFileAttributes: Cardinal;
|
|
ftCreationTime: TFileTime;
|
|
ftLastAccessTime: TFileTime;
|
|
ftLastWriteTime: TFileTime;
|
|
nFileSizeHigh: Cardinal;
|
|
nFileSizeLow: Cardinal;
|
|
dwReserved0: Cardinal;
|
|
dwReserved1: Cardinal;
|
|
cFileName: array[0..MAX_PATH - 1] of Char;
|
|
cAlternateFileName: array[0..13] of Char;
|
|
// The structure should be 320 bytes long...
|
|
pad : system.integer;
|
|
end;
|
|
|
|
Searchrec = Packed Record
|
|
FindHandle : THandle;
|
|
W32FindData : TWin32FindData;
|
|
ExcludeAttr : longint;
|
|
time : longint;
|
|
size : longint;
|
|
attr : longint;
|
|
name : string;
|
|
end;
|
|
|
|
|
|
registers = packed record
|
|
case i : integer of
|
|
0 : (ax,f1,bx,f2,cx,f3,dx,f4,bp,f5,si,f51,di,f6,ds,f7,es,f8,flags,fs,gs : word);
|
|
1 : (al,ah,f9,f10,bl,bh,f11,f12,cl,ch,f13,f14,dl,dh : byte);
|
|
2 : (eax, ebx, ecx, edx, ebp, esi, edi : longint);
|
|
end;
|
|
|
|
Var
|
|
DosError : integer;
|
|
|
|
{Interrupt}
|
|
Procedure Intr(intno: byte; var regs: registers);
|
|
Procedure MSDos(var regs: registers);
|
|
|
|
{Info/Date/Time}
|
|
Function DosVersion: Word;
|
|
Procedure GetDate(var year, month, mday, wday: word);
|
|
Procedure GetTime(var hour, minute, second, sec100: word);
|
|
procedure SetDate(year,month,day: word);
|
|
Procedure SetTime(hour,minute,second,sec100: word);
|
|
Procedure UnpackTime(p: longint; var t: datetime);
|
|
Procedure PackTime(var t: datetime; var p: longint);
|
|
|
|
{Exec}
|
|
Procedure Exec(const path: pathstr; const comline: comstr);
|
|
Function DosExitCode: word;
|
|
|
|
{Disk}
|
|
Function DiskFree(drive: byte) : int64;
|
|
Function DiskSize(drive: byte) : int64;
|
|
Procedure FindFirst(const path: pathstr; attr: word; var f: searchRec);
|
|
Procedure FindNext(var f: searchRec);
|
|
Procedure FindClose(Var f: SearchRec);
|
|
|
|
{File}
|
|
Procedure GetFAttr(var f; var attr: word);
|
|
Procedure GetFTime(var f; var time: longint);
|
|
Function FSearch(path: pathstr; dirlist: string): pathstr;
|
|
Function FExpand(const path: pathstr): pathstr;
|
|
Procedure FSplit(path: pathstr; var dir: dirstr; var name: namestr; var ext: extstr);
|
|
function GetShortName(var p : String) : boolean;
|
|
function GetLongName(var p : String) : boolean;
|
|
|
|
{Environment}
|
|
Function EnvCount: longint;
|
|
Function EnvStr(index: integer): string;
|
|
Function GetEnv(envvar: string): string;
|
|
|
|
{Misc}
|
|
Procedure SetFAttr(var f; attr: word);
|
|
Procedure SetFTime(var f; time: longint);
|
|
Procedure GetCBreak(var breakvalue: boolean);
|
|
Procedure SetCBreak(breakvalue: boolean);
|
|
Procedure GetVerify(var verify: boolean);
|
|
Procedure SetVerify(verify: boolean);
|
|
|
|
{Do Nothing Functions}
|
|
Procedure SwapVectors;
|
|
Procedure GetIntVec(intno: byte; var vector: pointer);
|
|
Procedure SetIntVec(intno: byte; vector: pointer);
|
|
Procedure Keep(exitcode: word);
|
|
|
|
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 : BOOL = false;
|
|
|
|
implementation
|
|
uses strings;
|
|
type
|
|
OSVERSIONINFO = record
|
|
dwOSVersionInfoSize : DWORD;
|
|
dwMajorVersion : DWORD;
|
|
dwMinorVersion : DWORD;
|
|
dwBuildNumber : DWORD;
|
|
dwPlatformId : DWORD;
|
|
szCSDVersion : array[0..127] of char;
|
|
end;
|
|
|
|
LPOSVERSIONINFO = ^OSVERSIONINFO;
|
|
|
|
var
|
|
versioninfo : OSVERSIONINFO;
|
|
kernel32dll : THandle;
|
|
|
|
{******************************************************************************
|
|
--- Conversion ---
|
|
******************************************************************************}
|
|
|
|
function GetLastError : DWORD;
|
|
external 'kernel32' name 'GetLastError';
|
|
function FileTimeToDosDateTime(const ft :TFileTime;var data,time : word) : longbool;
|
|
external 'kernel32' name 'FileTimeToDosDateTime';
|
|
function DosDateTimeToFileTime(date,time : word;var ft :TFileTime) : longbool;
|
|
external 'kernel32' name 'DosDateTimeToFileTime';
|
|
function FileTimeToLocalFileTime(const ft : TFileTime;var lft : TFileTime) : longbool;
|
|
external 'kernel32' name 'FileTimeToLocalFileTime';
|
|
function LocalFileTimeToFileTime(const lft : TFileTime;var ft : TFileTime) : longbool;
|
|
external 'kernel32' name 'LocalFileTimeToFileTime';
|
|
|
|
type
|
|
Longrec=packed record
|
|
lo,hi : word;
|
|
end;
|
|
|
|
function Last2DosError(d:dword):integer;
|
|
begin
|
|
Last2DosError:=d;
|
|
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 : TFileTime):longbool;
|
|
var
|
|
lft : TFileTime;
|
|
begin
|
|
DosToWinTime:=DosDateTimeToFileTime(longrec(dtime).hi,longrec(dtime).lo,lft) and
|
|
LocalFileTimeToFileTime(lft,Wtime);
|
|
end;
|
|
|
|
|
|
Function WinToDosTime (Const Wtime : TFileTime;var DTime:longint):longbool;
|
|
var
|
|
lft : TFileTime;
|
|
begin
|
|
WinToDosTime:=FileTimeToLocalFileTime(WTime,lft) and
|
|
FileTimeToDosDateTime(lft,longrec(dtime).hi,longrec(dtime).lo);
|
|
end;
|
|
|
|
|
|
{******************************************************************************
|
|
--- Dos Interrupt ---
|
|
******************************************************************************}
|
|
|
|
procedure intr(intno : byte;var regs : registers);
|
|
begin
|
|
{ !!!!!!!! }
|
|
end;
|
|
|
|
procedure msdos(var regs : registers);
|
|
begin
|
|
{ !!!!!!!! }
|
|
end;
|
|
|
|
|
|
{******************************************************************************
|
|
--- Info / Date / Time ---
|
|
******************************************************************************}
|
|
|
|
function GetVersion : longint;
|
|
external 'kernel32' name 'GetVersion';
|
|
procedure GetLocalTime(var t : TSystemTime);
|
|
external 'kernel32' name 'GetLocalTime';
|
|
function SetLocalTime(const t : TSystemTime) : longbool;
|
|
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;
|
|
|
|
|
|
Procedure packtime(var t : datetime;var p : longint);
|
|
Begin
|
|
p:=(t.sec shr 1)+(t.min shl 5)+(t.hour shl 11)+(t.day shl 16)+(t.month shl 21)+((t.year-1980) shl 25);
|
|
End;
|
|
|
|
|
|
Procedure unpacktime(p : longint;var t : datetime);
|
|
Begin
|
|
with t do
|
|
begin
|
|
sec:=(p and 31) shl 1;
|
|
min:=(p shr 5) and 63;
|
|
hour:=(p shr 11) and 31;
|
|
day:=(p shr 16) and 31;
|
|
month:=(p shr 21) and 15;
|
|
year:=(p shr 25)+1980;
|
|
end;
|
|
End;
|
|
|
|
|
|
{******************************************************************************
|
|
--- Exec ---
|
|
******************************************************************************}
|
|
|
|
function CreateProcess(lpApplicationName: PChar; lpCommandLine: PChar;
|
|
lpProcessAttributes, lpThreadAttributes: PSecurityAttributes;
|
|
bInheritHandles: BOOL; dwCreationFlags: DWORD; lpEnvironment: Pointer;
|
|
lpCurrentDirectory: PChar; const lpStartupInfo: TStartupInfo;
|
|
var lpProcessInformation: TProcessInformation): longbool;
|
|
external 'kernel32' name 'CreateProcessA';
|
|
function getExitCodeProcess(h:THandle;var code:longint):longbool;
|
|
external 'kernel32' name 'GetExitCodeProcess';
|
|
function WaitForSingleObject(hHandle: THandle; dwMilliseconds: DWORD): DWORD;
|
|
external 'kernel32' name 'WaitForSingleObject';
|
|
function CloseHandle(h : THandle) : longint;
|
|
external 'kernel32' name 'CloseHandle';
|
|
|
|
var
|
|
lastdosexitcode : longint;
|
|
|
|
procedure exec(const path : pathstr;const comline : comstr);
|
|
var
|
|
SI: TStartupInfo;
|
|
PI: TProcessInformation;
|
|
Proc : THandle;
|
|
l : Longint;
|
|
AppPath,
|
|
AppParam : array[0..255] of char;
|
|
begin
|
|
FillChar(SI, SizeOf(SI), 0);
|
|
SI.cb:=SizeOf(SI);
|
|
SI.wShowWindow:=1;
|
|
Move(Path[1],AppPath,length(Path));
|
|
AppPath[Length(Path)]:=#0;
|
|
AppParam[0]:='-';
|
|
AppParam[1]:=' ';
|
|
Move(ComLine[1],AppParam[2],length(Comline));
|
|
AppParam[Length(ComLine)+2]:=#0;
|
|
if not CreateProcess(PChar(@AppPath), PChar(@AppParam),
|
|
Nil, Nil, ExecInheritsHandles,$20, Nil, Nil, SI, PI) then
|
|
begin
|
|
DosError:=Last2DosError(GetLastError);
|
|
exit;
|
|
end;
|
|
Proc:=PI.hProcess;
|
|
CloseHandle(PI.hThread);
|
|
if WaitForSingleObject(Proc, Infinite) <> $ffffffff then
|
|
GetExitCodeProcess(Proc,l)
|
|
else
|
|
l:=-1;
|
|
CloseHandle(Proc);
|
|
LastDosExitCode:=l;
|
|
end;
|
|
|
|
|
|
function dosexitcode : word;
|
|
begin
|
|
dosexitcode:=lastdosexitcode and $ffff;
|
|
end;
|
|
|
|
|
|
procedure getcbreak(var breakvalue : boolean);
|
|
begin
|
|
{ !! No Win32 Function !! }
|
|
breakvalue := true;
|
|
end;
|
|
|
|
|
|
procedure setcbreak(breakvalue : boolean);
|
|
begin
|
|
{ !! No Win32 Function !! }
|
|
end;
|
|
|
|
|
|
procedure getverify(var verify : boolean);
|
|
begin
|
|
{ !! No Win32 Function !! }
|
|
verify := true;
|
|
end;
|
|
|
|
|
|
procedure setverify(verify : boolean);
|
|
begin
|
|
{ !! No Win32 Function !! }
|
|
end;
|
|
|
|
|
|
{******************************************************************************
|
|
--- Disk ---
|
|
******************************************************************************}
|
|
|
|
function GetDiskFreeSpace(drive:pchar;var sector_cluster,bytes_sector,
|
|
freeclusters,totalclusters:longint):longbool;
|
|
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 : longint;
|
|
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 : longint;
|
|
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: TWIN32FindData): THandle;
|
|
external 'kernel32' name 'FindFirstFileA';
|
|
function FindNextFile (hFindFile: THandle; var lpFindFileData: TWIN32FindData): LongBool;
|
|
external 'kernel32' name 'FindNextFileA';
|
|
function FindCloseFile (hFindFile: THandle): LongBool;
|
|
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.W32FindData.dwFileAttributes and cardinal(F.ExcludeAttr))<>0 do
|
|
begin
|
|
if not FindNextFile (F.FindHandle,F.W32FindData) then
|
|
begin
|
|
DosError:=Last2DosError(GetLastError);
|
|
exit;
|
|
end;
|
|
end;
|
|
{ Convert some attributes back }
|
|
f.size:=F.W32FindData.NFileSizeLow;
|
|
f.attr:=WinToDosAttr(F.W32FindData.dwFileAttributes);
|
|
WinToDosTime(F.W32FindData.ftLastWriteTime,f.Time);
|
|
f.Name:=StrPas(@F.W32FindData.cFileName);
|
|
end;
|
|
|
|
|
|
procedure findfirst(const path : pathstr;attr : word;var f : searchRec);
|
|
begin
|
|
{ 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.FindHandle:=FindFirstFile (pchar(@f.Name),F.W32FindData);
|
|
If longint(F.FindHandle)=Invalid_Handle_value then
|
|
begin
|
|
DosError:=Last2DosError(GetLastError);
|
|
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);
|
|
exit;
|
|
end;
|
|
{ Find file with correct attribute }
|
|
FindMatch(f);
|
|
end;
|
|
|
|
|
|
procedure swapvectors;
|
|
begin
|
|
end;
|
|
|
|
|
|
Procedure FindClose(Var f: SearchRec);
|
|
begin
|
|
If longint(F.FindHandle)<>Invalid_Handle_value then
|
|
FindCloseFile(F.FindHandle);
|
|
end;
|
|
|
|
|
|
{******************************************************************************
|
|
--- File ---
|
|
******************************************************************************}
|
|
|
|
function GetFileTime(h : longint;creation,lastaccess,lastwrite : PFileTime) : longbool;
|
|
external 'kernel32' name 'GetFileTime';
|
|
function SetFileTime(h : longint;creation,lastaccess,lastwrite : PFileTime) : longbool;
|
|
external 'kernel32' name 'SetFileTime';
|
|
function SetFileAttributes(lpFileName : pchar;dwFileAttributes : longint) : longbool;
|
|
external 'kernel32' name 'SetFileAttributesA';
|
|
function GetFileAttributes(lpFileName : pchar) : longint;
|
|
external 'kernel32' name 'GetFileAttributesA';
|
|
|
|
procedure fsplit(path : pathstr;var dir : dirstr;var name : namestr;var ext : extstr);
|
|
var
|
|
dotpos,p1,i : longint;
|
|
begin
|
|
{ allow slash as backslash }
|
|
for i:=1 to length(path) do
|
|
if path[i]='/' then path[i]:='\';
|
|
{ get drive name }
|
|
p1:=pos(':',path);
|
|
if p1>0 then
|
|
begin
|
|
dir:=path[1]+':';
|
|
delete(path,1,p1);
|
|
end
|
|
else
|
|
dir:='';
|
|
{ split the path and the name, there are no more path informtions }
|
|
{ if path contains no backslashes }
|
|
while true do
|
|
begin
|
|
p1:=pos('\',path);
|
|
if p1=0 then
|
|
break;
|
|
dir:=dir+copy(path,1,p1);
|
|
delete(path,1,p1);
|
|
end;
|
|
{ try to find out a extension }
|
|
Ext:='';
|
|
i:=Length(Path);
|
|
DotPos:=256;
|
|
While (i>0) Do
|
|
Begin
|
|
If (Path[i]='.') Then
|
|
begin
|
|
DotPos:=i;
|
|
break;
|
|
end;
|
|
Dec(i);
|
|
end;
|
|
Ext:=Copy(Path,DotPos,255);
|
|
Name:=Copy(Path,1,DotPos - 1);
|
|
end;
|
|
|
|
{ <immobilizer> }
|
|
|
|
function GetFullPathName(lpFileName: PChar; nBufferLength: Longint; lpBuffer: PChar; var lpFilePart : PChar):DWORD;
|
|
external 'kernel32' name 'GetFullPathNameA';
|
|
|
|
function GetShortPathName(lpszLongPath:pchar; lpszShortPath:pchar; cchBuffer:DWORD):DWORD;
|
|
external 'kernel32' name 'GetShortPathNameA';
|
|
|
|
|
|
(*
|
|
function FExpand (const Path: PathStr): PathStr;
|
|
- declared in fexpand.inc
|
|
*)
|
|
|
|
{$DEFINE FPC_FEXPAND_UNC} (* UNC paths are supported *)
|
|
{$DEFINE FPC_FEXPAND_DRIVES} (* Full paths begin with drive specification *)
|
|
|
|
{$I fexpand.inc}
|
|
|
|
{$UNDEF FPC_FEXPAND_DRIVES}
|
|
{$UNDEF FPC_FEXPAND_UNC}
|
|
|
|
|
|
function SearchPath(lpPath : PChar; lpFileName : PChar; lpExtension : PChar; nBufferLength : Longint; lpBuffer : PChar;
|
|
var lpFilePart : PChar) : Longint; external 'kernel32' name 'SearchPathA';
|
|
|
|
Function FSearch(path: pathstr; dirlist: string): pathstr;
|
|
var temp : PChar;
|
|
value : Array [0..255] of char;
|
|
i : Longint;
|
|
dir,dir2 : dirstr;
|
|
lastchar : char;
|
|
name : namestr;
|
|
ext : extstr;
|
|
s : SearchRec;
|
|
found : boolean;
|
|
begin
|
|
{ check if the file specified exists }
|
|
findfirst(path,anyfile,s);
|
|
found:=(doserror=0);
|
|
findclose(s);
|
|
if found then
|
|
begin
|
|
fsearch:=path;
|
|
exit;
|
|
end;
|
|
{ search the path }
|
|
fsearch:='';
|
|
for i:=1 to length(path) do
|
|
if path[i]='/' then
|
|
path[i]:='\';
|
|
fsplit(path,dir,name,ext);
|
|
for i:=1 to length(dirlist) do
|
|
if dirlist[i]='/' then
|
|
dirlist[i]:='\';
|
|
{ allow slash as backslash }
|
|
StringToPchar(name);
|
|
StringToPchar(ext);
|
|
|
|
StringToPchar(dir);
|
|
if SearchPath(@dir, @name, @ext, 255, @value, temp)>0 then
|
|
begin
|
|
fsearch := strpas(value);
|
|
exit;
|
|
end;
|
|
PCharToString(dir);
|
|
|
|
repeat
|
|
i:=pos(';',dirlist);
|
|
while i=1 do
|
|
begin
|
|
delete(dirlist,1,1);
|
|
i:=pos(';',dirlist);
|
|
end;
|
|
if i=0 then
|
|
begin
|
|
dir2:=dirlist;
|
|
dirlist:='';
|
|
end
|
|
else
|
|
begin
|
|
dir2:=Copy(dirlist,1,i-1);
|
|
dirlist:=Copy(dirlist,i+1,255);
|
|
end;
|
|
{ don't add anything if dir2 is empty string }
|
|
if dir2<>'' then
|
|
lastchar:=dir2[length(dir2)]
|
|
else
|
|
lastchar:='\';
|
|
if (lastchar<>'\') and (lastchar<>':') then
|
|
dir2:=dir2+'\'+dir
|
|
else
|
|
dir2:=dir2+dir;
|
|
StringToPchar(dir2);
|
|
if SearchPath(@dir2, @name, @ext, 255, @value, temp)>0 then
|
|
begin
|
|
fsearch := strpas(value);
|
|
exit;
|
|
end;
|
|
until dirlist='';
|
|
|
|
end;
|
|
|
|
{ </immobilizer> }
|
|
|
|
procedure getftime(var f;var time : longint);
|
|
var
|
|
ft : TFileTime;
|
|
begin
|
|
if GetFileTime(filerec(f).Handle,nil,nil,@ft) and
|
|
WinToDosTime(ft,time) then
|
|
exit
|
|
else
|
|
time:=0;
|
|
end;
|
|
|
|
|
|
procedure setftime(var f;time : longint);
|
|
var
|
|
ft : TFileTime;
|
|
begin
|
|
if DosToWinTime(time,ft) then
|
|
SetFileTime(filerec(f).Handle,nil,nil,@ft);
|
|
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
|
|
doserror:=0;
|
|
if not(SetFileAttributes(filerec(f).name,attr)) then
|
|
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 then
|
|
p:=strpas(buffer);
|
|
GetShortName:=ret<>0;
|
|
end;
|
|
|
|
{ change to long filename if successful DOS call PM }
|
|
function GetLongName(var p : String) : boolean;
|
|
|
|
var
|
|
lfn,sfn : array[0..255] of char;
|
|
filename : pchar;
|
|
ret : longint;
|
|
begin
|
|
{contrary to shortname, SDK does not mention input buffer can be equal
|
|
to output.}
|
|
|
|
if Length(p)>0 then {copy p to array of char}
|
|
move(p[1],sfn[0],length(p));
|
|
sfn[length(p)]:=chr(0);
|
|
fillchar(lfn,sizeof(lfn),#0);
|
|
filename:=nil;
|
|
|
|
{Should return value load loaddoserror?}
|
|
|
|
ret:=GetFullPathName(@sfn,255,@lfn,filename);
|
|
if ret=0 then
|
|
p:=strpas(lfn); {lfn here returns full path, filename only fn}
|
|
GetLongName:=ret<>0;
|
|
end;
|
|
|
|
{******************************************************************************
|
|
--- Environment ---
|
|
******************************************************************************}
|
|
|
|
{
|
|
The environment is a block of zero terminated strings
|
|
terminated by a #0
|
|
}
|
|
|
|
function GetEnvironmentStrings : pchar;
|
|
external 'kernel32' name 'GetEnvironmentStringsA';
|
|
function FreeEnvironmentStrings(p : pchar) : longbool;
|
|
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: integer): 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);
|
|
break;
|
|
end;
|
|
{ next string entry}
|
|
hp:=hp+strlen(hp)+1;
|
|
end;
|
|
FreeEnvironmentStrings(p);
|
|
end;
|
|
|
|
|
|
{******************************************************************************
|
|
--- Not Supported ---
|
|
******************************************************************************}
|
|
|
|
Procedure keep(exitcode : word);
|
|
Begin
|
|
End;
|
|
|
|
Procedure getintvec(intno : byte;var vector : pointer);
|
|
Begin
|
|
End;
|
|
|
|
Procedure setintvec(intno : byte;vector : pointer);
|
|
Begin
|
|
End;
|
|
|
|
function FreeLibrary(hLibModule : THANDLE) : longbool;
|
|
external 'kernel32' name 'FreeLibrary';
|
|
function GetVersionEx(var VersionInformation:OSVERSIONINFO) : longbool;
|
|
external 'kernel32' name 'GetVersionExA';
|
|
function LoadLibrary(lpLibFileName : pchar):THandle;
|
|
external 'kernel32' name 'LoadLibraryA';
|
|
function GetProcAddress(hModule : THandle;lpProcName : pchar) : pointer;
|
|
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;
|
|
if ((versioninfo.dwPlatformId=VER_PLATFORM_WIN32_WINDOWS) and
|
|
(versioninfo.dwBuildNUmber>=1000)) or
|
|
(versioninfo.dwPlatformId=VER_PLATFORM_WIN32_NT) then
|
|
begin
|
|
kernel32dll:=LoadLibrary('kernel32');
|
|
if kernel32dll<>0 then
|
|
GetDiskFreeSpaceEx:=TGetDiskFreeSpaceEx(GetProcAddress(kernel32dll,'GetDiskFreeSpaceExA'));
|
|
end;
|
|
end.
|
|
{
|
|
$Log$
|
|
Revision 1.11 2001-11-23 01:35:09 carl
|
|
* Range check error fix
|
|
|
|
Revision 1.10 2001/11/23 00:36:26 carl
|
|
* updated behavior of some routines to conform to docs
|
|
|
|
Revision 1.9 2001/06/13 22:21:53 hajny
|
|
+ platform specific information
|
|
|
|
Revision 1.8 2001/03/16 20:09:58 hajny
|
|
* universal FExpand
|
|
|
|
Revision 1.7 2000/12/18 17:28:58 jonas
|
|
* fixed range check errors
|
|
|
|
Revision 1.6 2000/09/06 20:47:34 peter
|
|
* removed previous fsplit() patch as it's not the correct behaviour for
|
|
LFNs. The code showing the bug could easily be adapted (merged)
|
|
|
|
Revision 1.5 2000/09/04 20:17:54 peter
|
|
* fixed previous commit (merged)
|
|
|
|
Revision 1.4 2000/09/04 19:38:13 peter
|
|
* fsplit with .. fix from Thomas (merged)
|
|
|
|
Revision 1.3 2000/08/24 19:02:36 peter
|
|
* fsearch checks if file exists first (merged)
|
|
|
|
Revision 1.2 2000/07/13 11:33:57 michael
|
|
+ removed logs
|
|
|
|
}
|