mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-05-29 00:42:38 +02:00
435 lines
11 KiB
PHP
435 lines
11 KiB
PHP
{
|
|
*********************************************************************
|
|
Copyright (C) 1997, 1998 Gertjan Schouten
|
|
|
|
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.
|
|
|
|
**********************************************************************
|
|
|
|
System Utilities For Free Pascal
|
|
}
|
|
|
|
function ChangeFileExt(const FileName, Extension: string): string;
|
|
var
|
|
i : longint;
|
|
EndSep : Set of Char;
|
|
begin
|
|
i := Length(FileName);
|
|
EndSep:=AllowDirectorySeparators+AllowDriveSeparators+[ExtensionSeparator];
|
|
while (I > 0) and not(FileName[I] in EndSep) do
|
|
Dec(I);
|
|
if (I = 0) or (FileName[I] <> ExtensionSeparator) then
|
|
I := Length(FileName)+1;
|
|
Result := Copy(FileName, 1, I - 1) + Extension;
|
|
end;
|
|
|
|
function ExtractFilePath(const FileName: string): string;
|
|
var
|
|
i : longint;
|
|
EndSep : Set of Char;
|
|
begin
|
|
i := Length(FileName);
|
|
EndSep:=AllowDirectorySeparators+AllowDriveSeparators;
|
|
while (i > 0) and not (FileName[i] in EndSep) do
|
|
Dec(i);
|
|
If I>0 then
|
|
Result := Copy(FileName, 1, i)
|
|
else
|
|
Result:='';
|
|
end;
|
|
|
|
function ExtractFileDir(const FileName: string): string;
|
|
var
|
|
i : longint;
|
|
EndSep : Set of Char;
|
|
begin
|
|
I := Length(FileName);
|
|
EndSep:=AllowDirectorySeparators+AllowDriveSeparators;
|
|
while (I > 0) and not (FileName[I] in EndSep) do
|
|
Dec(I);
|
|
if (I > 1) and (FileName[I] in AllowDirectorySeparators) and
|
|
not (FileName[I - 1] in EndSep) then
|
|
Dec(I);
|
|
Result := Copy(FileName, 1, I);
|
|
end;
|
|
|
|
function ExtractFileDrive(const FileName: string): string;
|
|
var
|
|
i,l: longint;
|
|
begin
|
|
Result := '';
|
|
l:=Length(FileName);
|
|
if (l<2) then
|
|
exit;
|
|
{$IF DEFINED(AMIGA) OR DEFINED(MORPHOS)}
|
|
i:=Pos(DriveSeparator,FileName);
|
|
if (i > 0) then Result:=Copy(FileName,1,i);
|
|
{$ELSE}
|
|
If (FileName[2] in AllowDriveSeparators) then
|
|
result:=Copy(FileName,1,2)
|
|
else if (FileName[1] in AllowDirectorySeparators) and
|
|
(FileName[2] in AllowDirectorySeparators) then
|
|
begin
|
|
i := 2;
|
|
|
|
{ skip share }
|
|
While (i<l) and Not (Filename[i+1] in AllowDirectorySeparators) do
|
|
inc(i);
|
|
inc(i);
|
|
|
|
While (i<l) and Not (Filename[i+1] in AllowDirectorySeparators) do
|
|
inc(i);
|
|
Result:=Copy(FileName,1,i);
|
|
end;
|
|
{$ENDIF}
|
|
end;
|
|
|
|
function ExtractFileName(const FileName: string): string;
|
|
var
|
|
i : longint;
|
|
EndSep : Set of Char;
|
|
begin
|
|
I := Length(FileName);
|
|
EndSep:=AllowDirectorySeparators+AllowDriveSeparators;
|
|
while (I > 0) and not (FileName[I] in EndSep) do
|
|
Dec(I);
|
|
Result := Copy(FileName, I + 1, MaxInt);
|
|
end;
|
|
|
|
function ExtractFileExt(const FileName: string): string;
|
|
var
|
|
i : longint;
|
|
EndSep : Set of Char;
|
|
begin
|
|
I := Length(FileName);
|
|
EndSep:=AllowDirectorySeparators+AllowDriveSeparators+[ExtensionSeparator];
|
|
while (I > 0) and not (FileName[I] in EndSep) do
|
|
Dec(I);
|
|
if (I > 0) and (FileName[I] = ExtensionSeparator) then
|
|
Result := Copy(FileName, I, MaxInt)
|
|
else
|
|
Result := '';
|
|
end;
|
|
|
|
function ExtractShortPathName(Const FileName : String) : String;
|
|
|
|
begin
|
|
{$ifdef MSWINDOWS}
|
|
SetLength(Result,Max_Path);
|
|
SetLength(Result,GetShortPathName(PChar(FileName), Pchar(Result),Length(Result)));
|
|
{$else}
|
|
Result:=FileName;
|
|
{$endif}
|
|
end;
|
|
|
|
type
|
|
PathStr=string;
|
|
|
|
{$DEFINE FPC_FEXPAND_SYSUTILS}
|
|
|
|
{$I fexpand.inc}
|
|
|
|
|
|
function ExpandFileName (Const FileName : string): String;
|
|
|
|
Var S : String;
|
|
|
|
Begin
|
|
S:=FileName;
|
|
DoDirSeparators(S);
|
|
Result:=Fexpand(S);
|
|
end;
|
|
|
|
|
|
{$ifndef HASEXPANDUNCFILENAME}
|
|
function ExpandUNCFileName (Const FileName : string): String;
|
|
begin
|
|
Result:=ExpandFileName (FileName);
|
|
//!! Here should follow code to replace the drive: part with UNC...
|
|
end;
|
|
{$endif HASEXPANDUNCFILENAME}
|
|
|
|
|
|
function ExpandFileNameCase (const FileName: string; out MatchFound: TFilenameCaseMatch): string;
|
|
var
|
|
SR: TSearchRec;
|
|
ItemsFound: byte;
|
|
FoundPath: string;
|
|
RestPos: SizeUInt;
|
|
Root: string;
|
|
|
|
procedure TryCase (const Base, Rest: string);
|
|
var
|
|
SR: TSearchRec;
|
|
RC: longint;
|
|
NextDirPos: SizeUInt;
|
|
NextPart: string;
|
|
NextRest: string;
|
|
SearchBase: string;
|
|
begin
|
|
NextDirPos := 1;
|
|
while (NextDirPos <= Length (Rest)) and
|
|
not (Rest [NextDirPos] in (AllowDirectorySeparators)) do
|
|
Inc (NextDirPos);
|
|
NextPart := Copy (Rest, 1, Pred (NextDirPos));
|
|
{$IFDEF FPC_FEXPAND_DIRSEP_IS_UPDIR}
|
|
if (Length (Rest) >= NextDirPos) and
|
|
(Rest [NextDirPos] in AllowDirectorySeparators) then
|
|
{$ELSE FPC_FEXPAND_DIRSEP_IS_UPDIR}
|
|
while (Length (Rest) >= NextDirPos) and
|
|
(Rest [NextDirPos] in AllowDirectorySeparators) do
|
|
{$ENDIF FPC_FEXPAND_DIRSEP_IS_UPDIR}
|
|
Inc (NextDirPos);
|
|
NextRest := Copy (Rest, NextDirPos, Length (Rest) - Pred (NextDirPos));
|
|
if (Base = '') or (Base [Length (Base)] in AllowDirectorySeparators) then
|
|
SearchBase := Base
|
|
else
|
|
SearchBase := Base + DirectorySeparator;
|
|
RC := FindFirst (SearchBase + AllFilesMask, faAnyFile, SR);
|
|
while (RC = 0) and (ItemsFound < 2) do
|
|
begin
|
|
if UpCase (NextPart) = UpCase (SR.Name) then
|
|
begin
|
|
if Length (NextPart) = Length (Rest) then
|
|
begin
|
|
Inc (ItemsFound);
|
|
if ItemsFound = 1 then
|
|
FoundPath := SearchBase + SR.Name;
|
|
end
|
|
else if SR.Attr and faDirectory = faDirectory then
|
|
TryCase (SearchBase + SR.Name + DirectorySeparator, NextRest);
|
|
end;
|
|
if ItemsFound < 2 then
|
|
RC := FindNext (SR);
|
|
end;
|
|
FindClose (SR);
|
|
end;
|
|
|
|
begin
|
|
Result := ExpandFileName (FileName);
|
|
if FileName = '' then
|
|
MatchFound := mkExactMatch
|
|
else
|
|
if (FindFirst (FileName, faAnyFile, SR) = 0) or
|
|
(* Special check for a root directory or a directory with a trailing slash *)
|
|
(* which are not found using FindFirst. *)
|
|
DirectoryExists (FileName) then
|
|
begin
|
|
MatchFound := mkExactMatch;
|
|
Result := ExtractFilePath (Result) + SR.Name;
|
|
FindClose (SR);
|
|
end
|
|
else
|
|
begin
|
|
(* Better close the search handle here before starting the recursive search *)
|
|
FindClose (SR);
|
|
MatchFound := mkNone;
|
|
if FileNameCaseSensitive then
|
|
begin
|
|
ItemsFound := 0;
|
|
FoundPath := '';
|
|
RestPos := Length (ExtractFileDrive (FileName)) + 1;
|
|
if (Length (FileName) > RestPos) then
|
|
begin
|
|
{$IFDEF FPC_FEXPAND_DIRSEP_IS_UPDIR}
|
|
if (Length (FileName) >= RestPos) and
|
|
(FileName [RestPos] in AllowDirectorySeparators) then
|
|
{$ELSE FPC_FEXPAND_DIRSEP_IS_UPDIR}
|
|
while (Length (FileName) >= RestPos) and
|
|
(FileName [RestPos] in AllowDirectorySeparators) do
|
|
{$ENDIF FPC_FEXPAND_DIRSEP_IS_UPDIR}
|
|
Inc (RestPos);
|
|
Root := Copy (FileName, 1, Pred (RestPos));
|
|
TryCase (Root, Copy (FileName, RestPos, Length (FileName) - Length (Root)));
|
|
if ItemsFound > 0 then
|
|
begin
|
|
Result := ExpandFileName (FoundPath);
|
|
if ItemsFound = 1 then
|
|
MatchFound := mkSingleMatch
|
|
else
|
|
MatchFound := mkAmbiguous;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
|
|
Const
|
|
MaxDirs = 129;
|
|
|
|
function ExtractRelativepath (Const BaseName,DestName : String): String;
|
|
|
|
Var Source, Dest : String;
|
|
Sc,Dc,I,J : Longint;
|
|
SD,DD : Array[1..MaxDirs] of PChar;
|
|
|
|
Const OneLevelBack = '..'+DirectorySeparator;
|
|
|
|
begin
|
|
If Uppercase(ExtractFileDrive(BaseName))<>Uppercase(ExtractFileDrive(DestName)) Then
|
|
begin
|
|
Result:=DestName;
|
|
exit;
|
|
end;
|
|
Source:=ExcludeTrailingPathDelimiter(ExtractFilePath(BaseName));
|
|
Dest:=ExcludeTrailingPathDelimiter(ExtractFilePath(DestName));
|
|
SC:=GetDirs (Source,SD);
|
|
DC:=GetDirs (Dest,DD);
|
|
I:=1;
|
|
While (I<=DC) and (I<=SC) do
|
|
begin
|
|
If StrIcomp(DD[i],SD[i])=0 then
|
|
Inc(i)
|
|
else
|
|
Break;
|
|
end;
|
|
Result:='';
|
|
For J:=I to SC do Result:=Result+OneLevelBack;
|
|
For J:=I to DC do Result:=Result+DD[J]+DirectorySeparator;
|
|
Result:=Result+ExtractFileName(DestNAme);
|
|
end;
|
|
|
|
Procedure DoDirSeparators (Var FileName : String);
|
|
|
|
VAr I : longint;
|
|
|
|
begin
|
|
For I:=1 to Length(FileName) do
|
|
If FileName[I] in AllowDirectorySeparators then
|
|
FileName[i]:=DirectorySeparator;
|
|
end;
|
|
|
|
|
|
Function SetDirSeparators (Const FileName : string) : String;
|
|
|
|
begin
|
|
Result:=FileName;
|
|
DoDirSeparators (Result);
|
|
end;
|
|
|
|
{
|
|
DirName is split in a #0 separated list of directory names,
|
|
Dirs is an array of pchars, pointing to these directory names.
|
|
The function returns the number of directories found, or -1
|
|
if none were found.
|
|
}
|
|
|
|
Function GetDirs (Var DirName : String; Var Dirs : Array of pchar) : Longint;
|
|
|
|
Var I : Longint;
|
|
|
|
begin
|
|
I:=1;
|
|
Result:=-1;
|
|
While I<=Length(DirName) do
|
|
begin
|
|
If (DirName[i] in AllowDirectorySeparators) and
|
|
{ avoid error in case last char=pathdelim }
|
|
(length(dirname)>i) then
|
|
begin
|
|
DirName[i]:=#0;
|
|
Inc(Result);
|
|
Dirs[Result]:=@DirName[I+1];
|
|
end;
|
|
Inc(I);
|
|
end;
|
|
If Result>-1 then inc(Result);
|
|
end;
|
|
|
|
function IncludeTrailingPathDelimiter(Const Path : String) : String;
|
|
|
|
Var
|
|
l : Integer;
|
|
|
|
begin
|
|
Result:=Path;
|
|
l:=Length(Result);
|
|
If (L=0) or not(Result[l] in AllowDirectorySeparators) then
|
|
Result:=Result+DirectorySeparator;
|
|
end;
|
|
|
|
function IncludeTrailingBackslash(Const Path : String) : String;
|
|
|
|
begin
|
|
Result:=IncludeTrailingPathDelimiter(Path);
|
|
end;
|
|
|
|
function ExcludeTrailingBackslash(Const Path: string): string;
|
|
|
|
begin
|
|
Result:=ExcludeTrailingPathDelimiter(Path);
|
|
end;
|
|
|
|
function ExcludeTrailingPathDelimiter(Const Path: string): string;
|
|
|
|
Var
|
|
L : Integer;
|
|
|
|
begin
|
|
L:=Length(Path);
|
|
If (L>0) and (Path[L] in AllowDirectorySeparators) then
|
|
Dec(L);
|
|
Result:=Copy(Path,1,L);
|
|
end;
|
|
|
|
function IncludeLeadingPathDelimiter(Const Path : String) : String;
|
|
|
|
Var
|
|
l : Integer;
|
|
|
|
begin
|
|
Result:=Path;
|
|
l:=Length(Result);
|
|
If (L=0) or not(Result[1] in AllowDirectorySeparators) then
|
|
Result:=DirectorySeparator+Result;
|
|
end;
|
|
|
|
function ExcludeLeadingPathDelimiter(Const Path: string): string;
|
|
|
|
Var
|
|
L : Integer;
|
|
|
|
begin
|
|
Result:=Path;
|
|
L:=Length(Result);
|
|
If (L>0) and (Result[1] in AllowDirectorySeparators) then
|
|
Delete(Result,1,1);
|
|
end;
|
|
|
|
function IsPathDelimiter(Const Path: string; Index: Integer): Boolean;
|
|
|
|
begin
|
|
Result:=(Index>0) and (Index<=Length(Path)) and (Path[Index] in AllowDirectorySeparators);
|
|
end;
|
|
|
|
function ConcatPaths(const Paths: array of String): String;
|
|
var
|
|
I: Integer;
|
|
begin
|
|
if Length(Paths) > 0 then
|
|
begin
|
|
Result := Paths[0];
|
|
for I := 1 to Length(Paths) - 1 do
|
|
Result := IncludeTrailingPathDelimiter(Result) + ExcludeLeadingPathDelimiter(Paths[I]);
|
|
end else
|
|
Result := '';
|
|
end;
|
|
|
|
Function GetFileHandle(var f : File):THandle;
|
|
|
|
begin
|
|
result:=filerec(f).handle;
|
|
end;
|
|
|
|
Function GetFileHandle(var f : Text):THandle;
|
|
begin
|
|
result:=textrec(f).handle;
|
|
end;
|
|
|