| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560 | {    Copyright (c) 1998-2002 by Florian Klaempfl and Peter Vreman    This module provides some basic file/dir handling utils and classes    This program is free software; you can redistribute it and/or modify    it under the terms of the GNU General Public License as published by    the Free Software Foundation; either version 2 of the License, or    (at your option) any later version.    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.  See the    GNU General Public License for more details.    You should have received a copy of the GNU General Public License    along with this program; if not, write to the Free Software    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. ****************************************************************************}unit cfileutl;{$i fpcdefs.inc}{$define usedircache}interface    uses{$ifdef hasunix}      Baseunix,unix,{$endif hasunix}{$ifdef win32}      Windows,{$endif win32}{$if defined(go32v2) or defined(watcom)}      Dos,{$endif}{$ifdef macos}      macutils,{$endif macos}{$IFNDEF USE_FAKE_SYSUTILS}      SysUtils,{$ELSE}      fksysutl,{$ENDIF}      GlobType,      CUtils,CClasses,      Systems;    type      TCachedDirectory = class(TFPHashObject)      private        FDirectoryEntries : TFPHashList;        FCached : Boolean;        procedure FreeDirectoryEntries;        function GetItemAttr(const AName: TCmdStr): longint;        function TryUseCache: boolean;        procedure ForceUseCache;        procedure Reload;      public        constructor Create(AList:TFPHashObjectList;const AName:TCmdStr);        destructor  destroy;override;        function FileExists(const AName:TCmdStr):boolean;        function FileExistsCaseAware(const path, fn: TCmdStr; out FoundName: TCmdStr):boolean;        function DirectoryExists(const AName:TCmdStr):boolean;        property DirectoryEntries:TFPHashList read FDirectoryEntries;      end;      TCachedSearchRec = record        Name       : TCmdStr;        Attr       : longint;        Pattern    : TCmdStr;        CachedDir  : TCachedDirectory;        EntryIndex : longint;      end;      PCachedDirectoryEntry =  ^TCachedDirectoryEntry;      TCachedDirectoryEntry = record        RealName: TCmdStr;        Attr    : longint;      end;      TDirectoryCache = class      private        FDirectories : TFPHashObjectList;        function GetDirectory(const ADir:TCmdStr):TCachedDirectory;      public        constructor Create;        destructor  destroy;override;        function FileExists(const AName:TCmdStr):boolean;        function FileExistsCaseAware(const path, fn: TCmdStr; out FoundName: TCmdStr):boolean;        function DirectoryExists(const AName:TCmdStr):boolean;        function FindFirst(const APattern:TCmdStr;var Res:TCachedSearchRec):boolean;        function FindNext(var Res:TCachedSearchRec):boolean;        function FindClose(var Res:TCachedSearchRec):boolean;      end;      TSearchPathList = class(TCmdStrList)        procedure AddPath(s:TCmdStr;addfirst:boolean);overload;        procedure AddLibraryPath(const sysroot: TCmdStr; s:TCmdStr;addfirst:boolean);overload;        procedure AddList(list:TSearchPathList;addfirst:boolean);        function  FindFile(const f : TCmdStr;allowcache:boolean;var foundfile:TCmdStr):boolean;      end;    function  bstoslash(const s : TCmdStr) : TCmdStr;    {Gives the absolute path to the current directory}    function  GetCurrentDir:TCmdStr;    {Gives the relative path to the current directory,     with a trailing dir separator. E. g. on unix ./ }    function CurDirRelPath(systeminfo: tsysteminfo): TCmdStr;    function  path_absolute(const s : TCmdStr) : boolean;    Function  PathExists (const F : TCmdStr;allowcache:boolean) : Boolean;    Function  FileExists (const F : TCmdStr;allowcache:boolean) : Boolean;    function  FileExistsNonCase(const path,fn:TCmdStr;allowcache:boolean;var foundfile:TCmdStr):boolean;    Function  RemoveDir(d:TCmdStr):boolean;    Function  FixPath(const s:TCmdStr;allowdot:boolean):TCmdStr;    function  FixFileName(const s:TCmdStr):TCmdStr;    function  TargetFixPath(s:TCmdStr;allowdot:boolean):TCmdStr;    function  TargetFixFileName(const s:TCmdStr):TCmdStr;    procedure SplitBinCmd(const s:TCmdStr;var bstr: TCmdStr;var cstr:TCmdStr);    function  FindFile(const f : TCmdStr; const path : TCmdStr;allowcache:boolean;var foundfile:TCmdStr):boolean;{    function  FindFilePchar(const f : TCmdStr;path : pchar;allowcache:boolean;var foundfile:TCmdStr):boolean;}    function  FindFileInExeLocations(const bin:TCmdStr;allowcache:boolean;var foundfile:TCmdStr):boolean;    function  FindExe(const bin:TCmdStr;allowcache:boolean;var foundfile:TCmdStr):boolean;    function  GetShortName(const n:TCmdStr):TCmdStr;    function maybequoted(const s:string):string;    function maybequoted(const s:ansistring):ansistring;    function maybequoted_for_script(const s:ansistring; quote_script: tscripttype):ansistring;    procedure InitFileUtils;    procedure DoneFileUtils;    function UnixRequoteWithDoubleQuotes(const QuotedStr: TCmdStr): TCmdStr;    function RequotedExecuteProcess(const Path: AnsiString; const ComLine: AnsiString; Flags: TExecuteFlags = []): Longint;    function RequotedExecuteProcess(const Path: AnsiString; const ComLine: array of AnsiString; Flags: TExecuteFlags = []): Longint;    function Shell(const command:ansistring): longint;  { hide Sysutils.ExecuteProcess in units using this one after SysUtils}  const    ExecuteProcess = 'Do not use' deprecated 'Use cfileutil.RequotedExecuteProcess instead, ExecuteProcess cannot deal with single quotes as used by Unix command lines';{ * Since native Amiga commands can't handle Unix-style relative paths used by the compiler,    and some GNU tools, Unix2AmigaPath is needed to handle such situations (KB) * }{$IFDEF HASAMIGA}{ * PATHCONV is implemented in the Amiga/MorphOS system unit * }{$NOTE TODO Amiga: implement PathConv() in System unit, which works with AnsiString}function Unix2AmigaPath(path: ShortString): ShortString; external name 'PATHCONV';{$ELSE}function Unix2AmigaPath(path: String): String;{$ENDIF}{$if FPC_FULLVERSION < 20701}type  TRawByteSearchRec = TSearchRec;{$endif}implementation    uses      Comphook,      Globals;{$undef AllFilesMaskIsInRTL}{$if (FPC_VERSION > 2)}  {$define AllFilesMaskIsInRTL}{$endif FPC_VERSION}{$if (FPC_VERSION = 2) and (FPC_RELEASE > 2)}  {$define AllFilesMaskIsInRTL}{$endif}{$if (FPC_VERSION = 2) and (FPC_RELEASE = 2) and (FPC_PATCH > 0)}  {$define AllFilesMaskIsInRTL}{$endif}{$ifndef AllFilesMaskIsInRTL}  {$if defined(go32v2) or defined(watcom)}  const    AllFilesMask = '*.*';  {$else}  const    AllFilesMask = '*';  {$endif not (go32v2 or watcom)}{$endif not AllFilesMaskIsInRTL}    var      DirCache : TDirectoryCache;{$IFNDEF HASAMIGA}{ Stub function for Unix2Amiga Path conversion functionality, only available in  Amiga/MorphOS RTL. I'm open for better solutions. (KB) }function Unix2AmigaPath(path: String): String;begin  Unix2AmigaPath:=path;end;{$ENDIF}{****************************************************************************                           TCachedDirectory****************************************************************************}    constructor TCachedDirectory.create(AList:TFPHashObjectList;const AName:TCmdStr);      begin        inherited create(AList,AName);        FDirectoryEntries:=TFPHashList.Create;        FCached:=False;      end;    destructor TCachedDirectory.destroy;      begin        FreeDirectoryEntries;        FDirectoryEntries.Free;        inherited destroy;      end;    function TCachedDirectory.TryUseCache:boolean;      begin        Result:=True;        if FCached then          exit;        if not current_settings.disabledircache then          ForceUseCache        else          Result:=False;      end;    procedure TCachedDirectory.ForceUseCache;      begin        if not FCached then          begin            FCached:=True;            Reload;          end;      end;    procedure TCachedDirectory.FreeDirectoryEntries;      var        i: Integer;      begin        if not(tf_files_case_aware in source_info.flags) then          exit;        for i := 0 to DirectoryEntries.Count-1 do          dispose(PCachedDirectoryEntry(DirectoryEntries[i]));      end;    function TCachedDirectory.GetItemAttr(const AName: TCmdStr): longint;      var        entry: PCachedDirectoryEntry;      begin        if not(tf_files_case_sensitive in source_info.flags) then          if (tf_files_case_aware in source_info.flags) then            begin              entry:=PCachedDirectoryEntry(DirectoryEntries.Find(Lower(AName)));              if assigned(entry) then                Result:=entry^.Attr              else                Result:=0;            end          else            Result:=PtrUInt(DirectoryEntries.Find(Lower(AName)))        else          Result:=PtrUInt(DirectoryEntries.Find(AName));      end;    procedure TCachedDirectory.Reload;      var        dir   : TRawByteSearchRec;        entry : PCachedDirectoryEntry;      begin        FreeDirectoryEntries;        DirectoryEntries.Clear;        if findfirst(IncludeTrailingPathDelimiter(Name)+AllFilesMask,faAnyFile or faDirectory,dir) = 0 then          begin            repeat              if ((dir.attr and faDirectory)<>faDirectory) or                 ((dir.Name<>'.') and                  (dir.Name<>'..')) then                begin                  { Force Archive bit so the attribute always has a value. This is needed                    to be able to see the difference in the directoryentries lookup if a file                    exists or not }                  Dir.Attr:=Dir.Attr or faArchive;                  if not(tf_files_case_sensitive in source_info.flags) then                    if (tf_files_case_aware in source_info.flags) then                      begin                        new(entry);                        entry^.RealName:=Dir.Name;                        entry^.Attr:=Dir.Attr;                        DirectoryEntries.Add(Lower(Dir.Name),entry)                      end                    else                      DirectoryEntries.Add(Lower(Dir.Name),Pointer(Ptrint(Dir.Attr)))                  else                    DirectoryEntries.Add(Dir.Name,Pointer(Ptrint(Dir.Attr)));                end;            until findnext(dir) <> 0;            findclose(dir);          end;      end;    function TCachedDirectory.FileExists(const AName:TCmdStr):boolean;      var        Attr : Longint;      begin        if not TryUseCache then          begin            { prepend directory name again }            result:=cfileutl.FileExists(Name+AName,false);            exit;          end;        Attr:=GetItemAttr(AName);        if Attr<>0 then          Result:=((Attr and faDirectory)=0)        else          Result:=false;      end;    function TCachedDirectory.FileExistsCaseAware(const path, fn: TCmdStr; out FoundName: TCmdStr):boolean;      var        entry : PCachedDirectoryEntry;      begin        if (tf_files_case_aware in source_info.flags) then          begin            if not TryUseCache then              begin                Result:=FileExistsNonCase(path,fn,false,FoundName);                exit;              end;            entry:=PCachedDirectoryEntry(DirectoryEntries.Find(Lower(ExtractFileName(fn))));            if assigned(entry) and               (entry^.Attr<>0) and               ((entry^.Attr and faDirectory) = 0) then              begin                 FoundName:=ExtractFilePath(path+fn)+entry^.RealName;                 Result:=true              end            else              Result:=false;          end        else          { should not be called in this case, use plain FileExists }          Result:=False;      end;    function TCachedDirectory.DirectoryExists(const AName:TCmdStr):boolean;      var        Attr : Longint;      begin        if not TryUseCache then          begin            Result:=PathExists(Name+AName,false);            exit;          end;        Attr:=GetItemAttr(AName);        if Attr<>0 then          Result:=((Attr and faDirectory)=faDirectory)        else          Result:=false;      end;{****************************************************************************                           TDirectoryCache****************************************************************************}    constructor TDirectoryCache.create;      begin        inherited create;        FDirectories:=TFPHashObjectList.Create(true);      end;    destructor TDirectoryCache.destroy;      begin        FDirectories.Free;        inherited destroy;      end;    function TDirectoryCache.GetDirectory(const ADir:TCmdStr):TCachedDirectory;      var        CachedDir : TCachedDirectory;        DirName   : TCmdStr;      begin        if ADir='' then          DirName:='.'+source_info.DirSep        else          DirName:=ADir;        CachedDir:=TCachedDirectory(FDirectories.Find(DirName));        if not assigned(CachedDir) then          CachedDir:=TCachedDirectory.Create(FDirectories,DirName);        Result:=CachedDir;      end;    function TDirectoryCache.FileExists(const AName:TCmdStr):boolean;      var        CachedDir : TCachedDirectory;      begin        Result:=false;        CachedDir:=GetDirectory(ExtractFilePath(AName));        if assigned(CachedDir) then          Result:=CachedDir.FileExists(ExtractFileName(AName));      end;    function TDirectoryCache.FileExistsCaseAware(const path, fn: TCmdStr; out FoundName: TCmdStr):boolean;      var        CachedDir : TCachedDirectory;      begin        Result:=false;        CachedDir:=GetDirectory(ExtractFilePath(path+fn));        if assigned(CachedDir) then          Result:=CachedDir.FileExistsCaseAware(path,fn,FoundName);      end;    function TDirectoryCache.DirectoryExists(const AName:TCmdStr):boolean;      var        CachedDir : TCachedDirectory;      begin        Result:=false;        CachedDir:=GetDirectory(ExtractFilePath(AName));        if assigned(CachedDir) then          Result:=CachedDir.DirectoryExists(ExtractFileName(AName));      end;    function TDirectoryCache.FindFirst(const APattern:TCmdStr;var Res:TCachedSearchRec):boolean;      begin        Res.Pattern:=ExtractFileName(APattern);        Res.CachedDir:=GetDirectory(ExtractFilePath(APattern));        Res.CachedDir.ForceUseCache;        Res.EntryIndex:=0;        if assigned(Res.CachedDir) then          Result:=FindNext(Res)        else          Result:=false;      end;    function TDirectoryCache.FindNext(var Res:TCachedSearchRec):boolean;      var        entry: PCachedDirectoryEntry;      begin        if Res.EntryIndex<Res.CachedDir.DirectoryEntries.Count then          begin            if (tf_files_case_aware in source_info.flags) then              begin                entry:=Res.CachedDir.DirectoryEntries[Res.EntryIndex];                Res.Name:=entry^.RealName;                Res.Attr:=entry^.Attr;              end            else              begin                Res.Name:=Res.CachedDir.DirectoryEntries.NameOfIndex(Res.EntryIndex);                Res.Attr:=PtrUInt(Res.CachedDir.DirectoryEntries[Res.EntryIndex]);              end;            inc(Res.EntryIndex);            Result:=true;          end        else          Result:=false;      end;    function TDirectoryCache.FindClose(var Res:TCachedSearchRec):boolean;      begin        { nothing todo }        result:=true;      end;{****************************************************************************                                   Utils****************************************************************************}    function bstoslash(const s : TCmdStr) : TCmdStr;    {      return TCmdStr s with all \ changed into /    }      var         i : sizeint;      begin        bstoslash:=s;        for i:=1 to length(s) do         if s[i]='\' then          bstoslash[i]:='/';      end;   {Gives the absolute path to the current directory}     var       CachedCurrentDir : TCmdStr;   function GetCurrentDir:TCmdStr;     begin       if CachedCurrentDir='' then         begin           GetDir(0,CachedCurrentDir);           CachedCurrentDir:=FixPath(CachedCurrentDir,false);         end;       result:=CachedCurrentDir;     end;   {Gives the relative path to the current directory,    with a trailing dir separator. E. g. on unix ./ }   function CurDirRelPath(systeminfo: tsysteminfo): TCmdStr;   begin     if systeminfo.system <> system_powerpc_macosclassic then       CurDirRelPath:= '.'+systeminfo.DirSep     else       CurDirRelPath:= ':'   end;   function path_absolute(const s : TCmdStr) : boolean;   {     is path s an absolute path?   }     begin        result:=false;{$if defined(unix)}        if (length(s)>0) and (s[1] in AllowDirectorySeparators) then          result:=true;{$elseif defined(hasamiga)}        (* An Amiga path is absolute, if it has a volume/device name in it (contains ":"),           otherwise it's always a relative path, no matter if it starts with a directory           separator or not. (KB) *)        if (length(s)>0) and (Pos(':',s) <> 0) then          result:=true;{$elseif defined(macos)}        if IsMacFullPath(s) then          result:=true;{$elseif defined(netware)}        if (Pos (DriveSeparator, S) <> 0) or                ((Length (S) > 0) and (S [1] in AllowDirectorySeparators)) then          result:=true;{$elseif defined(win32) or defined(win64) or defined(go32v2) or defined(os2) or defined(watcom)}        if ((length(s)>0) and (s[1] in AllowDirectorySeparators)) or(* The following check for non-empty AllowDriveSeparators assumes that all   other platforms supporting drives and not handled as exceptions above   should work with DOS-like paths, i.e. use absolute paths with one letter   for drive followed by path separator *)           ((length(s)>2) and (s[2] in AllowDriveSeparators) and (s[3] in AllowDirectorySeparators)) then          result:=true;{$else}        if ((length(s)>0) and (s[1] in AllowDirectorySeparators)) or(* The following check for non-empty AllowDriveSeparators assumes that all   other platforms supporting drives and not handled as exceptions above   should work with DOS-like paths, i.e. use absolute paths with one letter   for drive followed by path separator *)           ((AllowDriveSeparators <> []) and (length(s)>2) and (s[2] in AllowDriveSeparators) and (s[3] in AllowDirectorySeparators)) then          result:=true;{$endif unix}     end;    Function FileExists ( Const F : TCmdStr;allowcache:boolean) : Boolean;      begin{$ifdef usedircache}        if allowcache then          Result:=DirCache.FileExists(F)        else{$endif usedircache}          Result:=SysUtils.FileExists(F);        if do_checkverbosity(V_Tried) then         begin           if Result then             do_comment(V_Tried,'Searching file '+F+'... found')           else             do_comment(V_Tried,'Searching file '+F+'... not found');         end;      end;    function FileExistsNonCase(const path,fn:TCmdStr;allowcache:boolean;var foundfile:TCmdStr):boolean;      var        fn2 : TCmdStr;      begin        result:=false;        if tf_files_case_sensitive in source_info.flags then          begin            {              Search order for case sensitive systems:               1. NormalCase               2. lowercase               3. UPPERCASE            }            FoundFile:=path+fn;            If (ftNone in AllowedFilenameTransFormations) and FileExists(FoundFile,allowcache) then             begin               result:=true;               exit;             end;            if (ftLowerCase in AllowedFilenameTransFormations) then              begin                fn2:=Lower(fn);                if (fn2<>fn) then                  begin                    FoundFile:=path+fn2;                    If FileExists(FoundFile,allowcache) then                     begin                       result:=true;                       exit;                     end;                  end;              end;            if (ftUpperCase in AllowedFilenameTransFormations)  then              begin                fn2:=Upper(fn);                if (fn2<>fn) then                  begin                  FoundFile:=path+fn2;                  If FileExists(FoundFile,allowcache) then                    begin                      result:=true;                      exit;                    end;                  end;              end;          end        else          if tf_files_case_aware in source_info.flags then            begin              {                Search order for case aware systems:                 1. NormalCase              }{$ifdef usedircache}              if allowcache then                begin                  result:=DirCache.FileExistsCaseAware(path,fn,fn2);                  if result then                    begin                      FoundFile:=fn2;                      exit;                    end;                end              else{$endif usedircache}                begin                  FoundFile:=path+fn;                  If FileExists(FoundFile,allowcache) then                    begin                      { don't know the real name in this case }                      result:=true;                      exit;                   end;                end;           end        else          begin            { None case sensitive only lowercase }            FoundFile:=path+Lower(fn);            If FileExists(FoundFile,allowcache) then             begin               result:=true;               exit;             end;          end;        { Set foundfile to something useful }        FoundFile:=fn;      end;    Function PathExists (const F : TCmdStr;allowcache:boolean) : Boolean;      Var        i: longint;        hs : TCmdStr;      begin        if F = '' then          begin            result := true;            exit;          end;        hs := ExpandFileName(F);        I := Pos (DriveSeparator, hs);        if (hs [Length (hs)] = DirectorySeparator) and           (((I = 0) and (Length (hs) > 1)) or (I <> Length (hs) - 1)) then          Delete (hs, Length (hs), 1);{$ifdef usedircache}        if allowcache then          Result:=DirCache.DirectoryExists(hs)        else{$endif usedircache}          Result:=SysUtils.DirectoryExists(hs);      end;    Function RemoveDir(d:TCmdStr):boolean;      begin        if d[length(d)]=source_info.DirSep then         Delete(d,length(d),1);        {$push}{$I-}         rmdir(d);        {$pop}        RemoveDir:=(ioresult=0);      end;    Function _FixPath(const s:TCmdStr;allowdot:boolean;const info:tsysteminfo;const drivesep:string):TCmdStr;      var        i, L : sizeint;      begin        Result := s;        L := Length(Result);        if L=0 then          exit;        { Fix separator }        for i:=1 to L do          if (Result[i] in ['/','\']) and (Result[i]<>info.DirSep) then            Result[i]:=info.DirSep;        { Remove . or ./ }        if (not allowdot) and (Result[1]='.') and ((L=1) or (L=2) and (Result[2]=info.DirSep)) then          exit('');        { Fix ending / }        if (Result[L]<>info.DirSep) and (Result[L]<>drivesep) then          Result:=Result+info.DirSep;        { return }        if not ((tf_files_case_aware in info.flags) or           (tf_files_case_sensitive in info.flags)) then          Result := lower(Result);      end;    Function FixPath(const s:TCmdStr;allowdot:boolean):TCmdStr;      begin        Result:=_FixPath(s,allowdot,source_info,DriveSeparator);      end;  {Actually the version in macutils.pp could be used,   but that would not work for crosscompiling, so this is a slightly modified   version of it.}  function TranslatePathToMac (const path: TCmdStr; mpw: Boolean): TCmdStr;    function GetVolumeIdentifier: TCmdStr;    begin      GetVolumeIdentifier := '{Boot}'      (*      if mpw then        GetVolumeIdentifier := '{Boot}'      else        GetVolumeIdentifier := macosBootVolumeName;      *)    end;    var      slashPos, oldpos, newpos, oldlen, maxpos: Longint;  begin    oldpos := 1;    slashPos := Pos('/', path);    TranslatePathToMac:='';    if (slashPos <> 0) then   {its a unix path}      begin        if slashPos = 1 then          begin      {its a full path}            oldpos := 2;            TranslatePathToMac := GetVolumeIdentifier;          end        else     {its a partial path}          TranslatePathToMac := ':';      end    else      begin        slashPos := Pos('\', path);        if (slashPos <> 0) then   {its a dos path}          begin            if slashPos = 1 then              begin      {its a full path, without drive letter}                oldpos := 2;                TranslatePathToMac := GetVolumeIdentifier;              end            else if (Length(path) >= 2) and (path[2] = ':') then {its a full path, with drive letter}              begin                oldpos := 4;                TranslatePathToMac := GetVolumeIdentifier;              end            else     {its a partial path}              TranslatePathToMac := ':';          end;      end;    if (slashPos <> 0) then   {its a unix or dos path}      begin        {Translate "/../" to "::" , "/./" to ":" and "/" to ":" }        newpos := Length(TranslatePathToMac);        oldlen := Length(path);        SetLength(TranslatePathToMac, newpos + oldlen);  {It will be no longer than what is already}                                                                        {prepended plus length of path.}        maxpos := Length(TranslatePathToMac);          {Get real maxpos, can be short if String is ShortString}        {There is never a slash in the beginning, because either it was an absolute path, and then the}        {drive and slash was removed, or it was a relative path without a preceding slash.}        while oldpos <= oldlen do          begin            {Check if special dirs, ./ or ../ }            if path[oldPos] = '.' then              if (oldpos + 1 <= oldlen) and (path[oldPos + 1] = '.') then                begin                  if (oldpos + 2 > oldlen) or (path[oldPos + 2] in ['/', '\']) then                    begin                      {It is "../" or ".."  translates to ":" }                      if newPos = maxPos then                        begin {Shouldn't actually happen, but..}                          Exit('');                        end;                      newPos := newPos + 1;                      TranslatePathToMac[newPos] := ':';                      oldPos := oldPos + 3;                      continue;  {Start over again}                    end;                end              else if (oldpos + 1 > oldlen) or (path[oldPos + 1] in ['/', '\']) then                begin                  {It is "./" or "."  ignor it }                  oldPos := oldPos + 2;                  continue;  {Start over again}                end;            {Collect file or dir name}            while (oldpos <= oldlen) and not (path[oldPos] in ['/', '\']) do              begin                if newPos = maxPos then                  begin {Shouldn't actually happen, but..}                    Exit('');                  end;                newPos := newPos + 1;                TranslatePathToMac[newPos] := path[oldPos];                oldPos := oldPos + 1;              end;            {When we come here there is either a slash or we are at the end.}            if (oldpos <= oldlen) then              begin                if newPos = maxPos then                  begin {Shouldn't actually happen, but..}                    Exit('');                  end;                newPos := newPos + 1;                TranslatePathToMac[newPos] := ':';                oldPos := oldPos + 1;              end;          end;        SetLength(TranslatePathToMac, newpos);      end    else if (path = '.') then      TranslatePathToMac := ':'    else if (path = '..') then      TranslatePathToMac := '::'    else      TranslatePathToMac := path;  {its a mac path}  end;   function _FixFileName(const s:TCmdStr; const info:tsysteminfo):TCmdStr;     var       i      : sizeint;     begin       if info.system = system_powerpc_macosclassic then         Result:=TranslatePathToMac(s, true)       else        begin          if (tf_files_case_aware in info.flags) or             (tf_files_case_sensitive in info.flags) then            Result:=s          else            Result:=Lower(s);          for i:=1 to length(s) do           case s[i] of             '/','\' :               if s[i]<>info.dirsep then                 Result[i]:=info.dirsep;           end;        end;     end;   function FixFileName(const s:TCmdStr):TCmdStr;     begin       result:=_FixFileName(s,source_info);     end;   Function TargetFixPath(s:TCmdStr;allowdot:boolean):TCmdStr;     begin       result:=_FixPath(s,allowdot,target_info,':');     end;   function TargetFixFileName(const s:TCmdStr):TCmdStr;     begin       result:=_FixFileName(s,target_info);     end;   procedure SplitBinCmd(const s:TCmdStr;var bstr:TCmdStr;var cstr:TCmdStr);     var       i : longint;     begin       i:=pos(' ',s);       if i>0 then        begin          bstr:=Copy(s,1,i-1);          cstr:=Copy(s,i+1,length(s)-i);        end       else        begin          bstr:=s;          cstr:='';        end;     end;    procedure TSearchPathList.AddPath(s:TCmdStr;addfirst:boolean);      begin        AddLibraryPath('',s,AddFirst);      end;   procedure TSearchPathList.AddLibraryPath(const sysroot: TCmdStr; s:TCmdStr;addfirst:boolean);     var       staridx,       i,j      : longint;       prefix,       suffix,       CurrentDir,       currPath : TCmdStr;       subdirfound : boolean;{$ifdef usedircache}       dir      : TCachedSearchRec;{$else usedircache}       dir      : TSearchRec;{$endif usedircache}       hp       : TCmdStrListItem;       procedure WarnNonExistingPath(const path : TCmdStr);       begin         if do_checkverbosity(V_Tried) then           do_comment(V_Tried,'Path "'+path+'" not found');       end;       procedure AddCurrPath;       begin         if addfirst then          begin            Remove(currPath);            Insert(currPath);          end         else          begin            { Check if already in path, then we don't add it }            hp:=Find(currPath);            if not assigned(hp) then             Concat(currPath);          end;       end;     begin       if s='' then        exit;     { Support default macro's }       DefaultReplacements(s);{$warnings off}       if PathSeparator <> ';' then        for i:=1 to length(s) do         if s[i]=PathSeparator then          s[i]:=';';{$warnings on}     { get current dir }       CurrentDir:=GetCurrentDir;       repeat         { get currpath }         if addfirst then          begin            j:=length(s);            while (j>0) and (s[j]<>';') do             dec(j);            currPath:= TrimSpace(Copy(s,j+1,length(s)-j));            if j=0 then             s:=''            else             System.Delete(s,j,length(s)-j+1);          end         else          begin            j:=Pos(';',s);            if j=0 then             j:=length(s)+1;            currPath:= TrimSpace(Copy(s,1,j-1));            System.Delete(s,1,j);          end;         { fix pathname }         DePascalQuote(currPath);         { GNU LD convention: if library search path starts with '=', it's relative to the           sysroot; otherwise, interpret it as a regular path }         if (length(currPath) >0) and (currPath[1]='=') then           currPath:=sysroot+FixPath(copy(currPath,2,length(currPath)-1),false);         if currPath='' then           currPath:= CurDirRelPath(source_info)         else          begin            currPath:=FixPath(ExpandFileName(currpath),false);            if (CurrentDir<>'') and (Copy(currPath,1,length(CurrentDir))=CurrentDir) then             begin{$ifdef hasamiga}               currPath:= CurrentDir+Copy(currPath,length(CurrentDir)+1,length(currPath));{$else}               currPath:= CurDirRelPath(source_info)+Copy(currPath,length(CurrentDir)+1,length(currPath));{$endif}             end;          end;         { wildcard adding ? }         staridx:=pos('*',currpath);         if staridx>0 then          begin            prefix:=ExtractFilePath(Copy(currpath,1,staridx));            suffix:=Copy(currpath,staridx+1,length(currpath));            subdirfound:=false;{$ifdef usedircache}            if DirCache.FindFirst(Prefix+AllFilesMask,dir) then              begin                repeat                  if (dir.attr and faDirectory)<>0 then                    begin                      subdirfound:=true;                      currpath:=prefix+dir.name+suffix;                      if (suffix='') or PathExists(currpath,true) then                        begin                          hp:=Find(currPath);                          if not assigned(hp) then                            AddCurrPath;                        end;                    end;                until not DirCache.FindNext(dir);              end;            DirCache.FindClose(dir);{$else usedircache}            if findfirst(prefix+AllFilesMask,faDirectory,dir) = 0 then              begin                repeat                  if (dir.name<>'.') and                      (dir.name<>'..') and                      ((dir.attr and faDirectory)<>0) then                    begin                      subdirfound:=true;                      currpath:=prefix+dir.name+suffix;                      if (suffix='') or PathExists(currpath,false) then                        begin                          hp:=Find(currPath);                          if not assigned(hp) then                            AddCurrPath;                        end;                    end;                until findnext(dir) <> 0;                FindClose(dir);              end;{$endif usedircache}            if not subdirfound then              WarnNonExistingPath(currpath);          end         else          begin            if PathExists(currpath,true) then             AddCurrPath            else             WarnNonExistingPath(currpath);          end;       until (s='');     end;   procedure TSearchPathList.AddList(list:TSearchPathList;addfirst:boolean);     var       s : TCmdStr;       hl : TSearchPathList;       hp,hp2 : TCmdStrListItem;     begin       if list.empty then        exit;       { create temp and reverse the list }       if addfirst then        begin          hl:=TSearchPathList.Create;          hp:=TCmdStrListItem(list.first);          while assigned(hp) do           begin             hl.insert(hp.Str);             hp:=TCmdStrListItem(hp.next);           end;          while not hl.empty do           begin             s:=hl.GetFirst;             Remove(s);             Insert(s);           end;          hl.Free;        end       else        begin          hp:=TCmdStrListItem(list.first);          while assigned(hp) do           begin             hp2:=Find(hp.Str);             { Check if already in path, then we don't add it }             if not assigned(hp2) then              Concat(hp.Str);             hp:=TCmdStrListItem(hp.next);           end;        end;     end;   function TSearchPathList.FindFile(const f :TCmdStr;allowcache:boolean;var foundfile:TCmdStr):boolean;     Var       p : TCmdStrListItem;     begin       FindFile:=false;       p:=TCmdStrListItem(first);       while assigned(p) do        begin          result:=FileExistsNonCase(p.Str,f,allowcache,FoundFile);          if result then            exit;          p:=TCmdStrListItem(p.next);        end;       { Return original filename if not found }       FoundFile:=f;     end;   function FindFile(const f : TCmdStr; const path : TCmdStr;allowcache:boolean;var foundfile:TCmdStr):boolean;     Var       StartPos, EndPos, L: LongInt;     begin       Result:=False;       if (path_absolute(f)) then         begin           Result:=FileExistsNonCase('',f, allowcache, foundfile);           if Result then             Exit;         end;       StartPos := 1;       L := Length(Path);       repeat         EndPos := StartPos;         while (EndPos <= L) and ((Path[EndPos] <> PathSeparator) and (Path[EndPos] <> ';')) do           Inc(EndPos);         Result := FileExistsNonCase(FixPath(Copy(Path, StartPos, EndPos-StartPos), False), f, allowcache, FoundFile);         if Result then           Exit;         StartPos := EndPos + 1;       until StartPos > L;       FoundFile:=f;     end;{   function FindFilePchar(const f : TCmdStr;path : pchar;allowcache:boolean;var foundfile:TCmdStr):boolean;      Var        singlepathstring : TCmdStr;        startpc,pc : pchar;     begin       FindFilePchar:=false;       if Assigned (Path) then        begin          pc:=path;          repeat             startpc:=pc;             while (pc^<>PathSeparator) and (pc^<>';') and (pc^<>#0) do              inc(pc);             SetLength(singlepathstring, pc-startpc);             move(startpc^,singlepathstring[1],pc-startpc);             singlepathstring:=FixPath(ExpandFileName(singlepathstring),false);             result:=FileExistsNonCase(singlepathstring,f,allowcache,FoundFile);             if result then               exit;             if (pc^=#0) then               break;             inc(pc);          until false;        end;       foundfile:=f;     end;}  function  FindFileInExeLocations(const bin:TCmdStr;allowcache:boolean;var foundfile:TCmdStr):boolean;    var      Path : TCmdStr;      found : boolean;    begin       found:=FindFile(FixFileName(bin),exepath,allowcache,foundfile);      if not found then       begin{$ifdef macos}         Path:=GetEnvironmentVariable('Commands');{$else}         Path:=GetEnvironmentVariable('PATH');{$endif}         found:=FindFile(FixFileName(bin),Path,allowcache,foundfile);       end;      FindFileInExeLocations:=found;    end;   function  FindExe(const bin:TCmdStr;allowcache:boolean;var foundfile:TCmdStr):boolean;     var       b : TCmdStr;     begin       { change extension only on platforms that use an exe extension, otherwise on OpenBSD         'ld.bfd' gets converted to 'ld' }       if source_info.exeext<>'' then         b:=ChangeFileExt(bin,source_info.exeext)       else         b:=bin;       FindExe:=FindFileInExeLocations(b,allowcache,foundfile);     end;    function GetShortName(const n:TCmdStr):TCmdStr;{$ifdef win32}      var        hs,hs2 : TCmdStr;        i : longint;{$endif}{$if defined(go32v2) or defined(watcom)}      var        hs : shortstring;{$endif}      begin        GetShortName:=n;{$ifdef win32}        hs:=n+#0;        hs2:='';        { may become longer in case of e.g. ".a" -> "a~1" or so }        setlength(hs2,length(hs)*2);        i:=Windows.GetShortPathName(@hs[1],@hs2[1],length(hs)*2);        if (i>0) and (i<=length(hs)*2) then          begin            setlength(hs2,strlen(@hs2[1]));            GetShortName:=hs2;          end;{$endif}{$if defined(go32v2) or defined(watcom)}        hs:=n;        if Dos.GetShortName(hs) then         GetShortName:=hs;{$endif}      end;    function maybequoted(const s:string):string;    const      FORBIDDEN_CHARS_DOS = ['!', '@', '#', '$', '%', '^', '&', '*', '(', ')',                         '{', '}', '''', '`', '~'];      FORBIDDEN_CHARS_OTHER = ['!', '@', '#', '$', '%', '^', '&', '*', '(', ')',                         '{', '}', '''', ':', '\', '`', '~'];    var      forbidden_chars: set of char;      i  : integer;      quote_script: tscripttype;      quote_char: ansichar;      quoted : boolean;    begin      if not(cs_link_on_target in current_settings.globalswitches) then        quote_script:=source_info.script      else        quote_script:=target_info.script;      if quote_script=script_dos then        forbidden_chars:=FORBIDDEN_CHARS_DOS      else        begin          forbidden_chars:=FORBIDDEN_CHARS_OTHER;          if quote_script=script_unix then            include(forbidden_chars,'"');        end;      if quote_script=script_unix then        quote_char:=''''      else        quote_char:='"';      quoted:=false;      result:=quote_char;      for i:=1 to length(s) do       begin         if s[i]=quote_char then           begin             quoted:=true;             result:=result+'\'+quote_char;           end         else case s[i] of           '\':             begin               if quote_script=script_unix then                 begin                   result:=result+'\\';                   quoted:=true                 end               else                 result:=result+'\';             end;           ' ',           #128..#255 :             begin               quoted:=true;               result:=result+s[i];             end;           else begin             if s[i] in forbidden_chars then               quoted:=True;             result:=result+s[i];           end;         end;       end;      if quoted then        result:=result+quote_char      else        result:=s;    end;    function maybequoted_for_script(const s:ansistring; quote_script: tscripttype):ansistring;      const        FORBIDDEN_CHARS_DOS = ['!', '@', '#', '$', '%', '^', '&', '*', '(', ')',                           '{', '}', '''', '`', '~'];        FORBIDDEN_CHARS_OTHER = ['!', '@', '#', '$', '%', '^', '&', '*', '(', ')',                           '{', '}', '''', ':', '\', '`', '~'];      var        forbidden_chars: set of char;        i  : integer;        quote_char: ansichar;        quoted : boolean;      begin        if quote_script=script_dos then          forbidden_chars:=FORBIDDEN_CHARS_DOS        else          begin            forbidden_chars:=FORBIDDEN_CHARS_OTHER;            if quote_script=script_unix then              include(forbidden_chars,'"');          end;        if quote_script=script_unix then          quote_char:=''''        else          quote_char:='"';        quoted:=false;        result:=quote_char;        for i:=1 to length(s) do         begin           if s[i]=quote_char then             begin               quoted:=true;               result:=result+'\'+quote_char;             end           else case s[i] of             '\':               begin                 if quote_script=script_unix then                   begin                     result:=result+'\\';                     quoted:=true                   end                 else                   result:=result+'\';               end;             ' ',             #128..#255 :               begin                 quoted:=true;                 result:=result+s[i];               end;             else begin               if s[i] in forbidden_chars then                 quoted:=True;               result:=result+s[i];             end;           end;         end;        if quoted then          result:=result+quote_char        else          result:=s;      end;    function maybequoted(const s:ansistring):ansistring;      var        quote_script: tscripttype;      begin        if not(cs_link_on_target in current_settings.globalswitches) then          quote_script:=source_info.script        else          quote_script:=target_info.script;        result:=maybequoted_for_script(s,quote_script);      end;    { requotes a string that was quoted for Unix for passing to ExecuteProcess,      because it only supports Windows-style quoting; this routine assumes that      everything that has to be quoted for Windows, was also quoted (but      differently for Unix) -- which is the case }    function UnixRequoteWithDoubleQuotes(const QuotedStr: TCmdStr): TCmdStr;      var        i: longint;        temp: TCmdStr;        inquotes: boolean;      begin        if QuotedStr='' then          begin            result:='';            exit;          end;        inquotes:=false;        result:='';        i:=1;        temp:='';        while i<=length(QuotedStr) do          begin            case QuotedStr[i] of              '''':                begin                  if not(inquotes) then                    begin                      inquotes:=true;                      temp:=''                    end                  else                    begin                      { requote for Windows }                      result:=result+maybequoted_for_script(temp,script_dos);                      inquotes:=false;                    end;                end;              '\':                begin                  if inquotes then                    temp:=temp+QuotedStr[i+1]                  else                    result:=result+QuotedStr[i+1];                  inc(i);                end;              else                begin                  if inquotes then                    temp:=temp+QuotedStr[i]                  else                    result:=result+QuotedStr[i];                end;            end;            inc(i);          end;      end;    function RequotedExecuteProcess(const Path: AnsiString; const ComLine: AnsiString; Flags: TExecuteFlags): Longint;      var        quote_script: tscripttype;      begin        if do_checkverbosity(V_Executable) then          do_comment(V_Executable,'Executing "'+Path+'" with command line "'+            ComLine+'"');        if (cs_link_on_target in current_settings.globalswitches) then          quote_script:=target_info.script        else          quote_script:=source_info.script;        if quote_script=script_unix then          result:=sysutils.ExecuteProcess(Path,UnixRequoteWithDoubleQuotes(ComLine),Flags)        else          result:=sysutils.ExecuteProcess(Path,ComLine,Flags)      end;    function RequotedExecuteProcess(const Path: AnsiString; const ComLine: array of AnsiString; Flags: TExecuteFlags): Longint;      var        i : longint;        st : string;      begin        if do_checkverbosity(V_Executable) then          begin            if high(ComLine)=0 then              st:=''            else              st:=ComLine[1];            for i:=2 to high(ComLine) do              st:=st+' '+ComLine[i];            do_comment(V_Executable,'Executing "'+Path+'" with command line "'+              st+'"');          end;        result:=sysutils.ExecuteProcess(Path,ComLine,Flags);      end;    function Shell(const command:ansistring): longint;      { This is already defined in the linux.ppu for linux, need for the *        expansion under linux }{$ifdef hasunix}      begin        do_comment(V_Executable,'Executing "'+Command+'" with fpSystem call');        result := Unix.fpsystem(command);      end;{$else hasunix}  {$ifdef hasamiga}      begin        do_comment(V_Executable,'Executing "'+Command+'" using RequotedExecuteProcess');        result := RequotedExecuteProcess('',command);      end;  {$else hasamiga}      var        comspec : string;      begin        comspec:=GetEnvironmentVariable('COMSPEC');        do_comment(V_Executable,'Executing "'+Command+'" using comspec "'            +ComSpec+'"');        result := RequotedExecuteProcess(comspec,' /C '+command);      end;   {$endif hasamiga}{$endif hasunix}{****************************************************************************                           Init / Done****************************************************************************}    procedure InitFileUtils;      begin        DirCache:=TDirectoryCache.Create;      end;    procedure DoneFileUtils;      begin        DirCache.Free;      end;end.
 |