| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105 | {    Copyright (c) 1998-2002 by Peter Vreman    This unit handles the linker and binder calls for programs and    libraries    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 link;{$i fpcdefs.inc}interface    uses	  sysutils,      cclasses,      systems,      fmodule,      globtype,      ogbase;    Type      TLinkerInfo=record        ExeCmd,        DllCmd        : array[1..3] of string;        ResName       : string[100];        ScriptName    : string[100];        ExtraOptions  : TCmdStr;        DynamicLinker : string[100];      end;      TLinker = class(TAbstractLinker)      public         HasResources,         HasExports      : boolean;         SysInitUnit     : string[20];         ObjectFiles,         SharedLibFiles,         StaticLibFiles  : TCmdStrList;         Constructor Create;virtual;         Destructor Destroy;override;         procedure AddModuleFiles(hp:tmodule);         Procedure AddObject(const S,unitpath : TCmdStr;isunit:boolean);         Procedure AddStaticLibrary(const S : TCmdStr);         Procedure AddSharedLibrary(S : TCmdStr);         Procedure AddStaticCLibrary(const S : TCmdStr);         Procedure AddSharedCLibrary(S : TCmdStr);         procedure AddImportSymbol(const libname,symname:TCmdStr;OrdNr: longint;isvar:boolean);virtual;         Procedure InitSysInitUnitName;virtual;         Function  MakeExecutable:boolean;virtual;         Function  MakeSharedLibrary:boolean;virtual;         Function  MakeStaticLibrary:boolean;virtual;         procedure ExpandAndApplyOrder(var Src:TCmdStrList);         procedure LoadPredefinedLibraryOrder;virtual;         function  ReOrderEntries : boolean;       end;      TExternalLinker = class(TLinker)      public         Info : TLinkerInfo;         Constructor Create;override;         Destructor Destroy;override;         Function  FindUtil(const s:TCmdStr):TCmdStr;         Function  DoExec(const command:TCmdStr; para:TCmdStr;showinfo,useshell:boolean):boolean;         procedure SetDefaultInfo;virtual;         Function  MakeStaticLibrary:boolean;override;       end;      TInternalLinker = class(TLinker)      private         FCExeOutput : TExeOutputClass;         FCObjInput  : TObjInputClass;         { Libraries }         FStaticLibraryList : TFPHashObjectList;         FImportLibraryList : TFPHashObjectList;         procedure Load_ReadObject(const para:TCmdStr);         procedure Load_ReadStaticLibrary(const para:TCmdStr);         procedure ParseScript_Load;         procedure ParseScript_Order;         procedure ParseScript_CalcPos;         procedure PrintLinkerScript;         function  RunLinkScript(const outputname:TCmdStr):boolean;      protected         property CObjInput:TObjInputClass read FCObjInput write FCObjInput;         property CExeOutput:TExeOutputClass read FCExeOutput write FCExeOutput;         property StaticLibraryList:TFPHashObjectList read FStaticLibraryList;         property ImportLibraryList:TFPHashObjectList read FImportLibraryList;         procedure DefaultLinkScript;virtual;abstract;         linkscript : TCmdStrList;      public         IsSharedLibrary : boolean;         Constructor Create;override;         Destructor Destroy;override;         Function  MakeExecutable:boolean;override;         Function  MakeSharedLibrary:boolean;override;         procedure AddImportSymbol(const libname,symname:TCmdStr;OrdNr: longint;isvar:boolean);override;       end;    var      Linker  : TLinker;    function FindObjectFile(s : TCmdStr;const unitpath:TCmdStr;isunit:boolean) : TCmdStr;    function FindLibraryFile(s:TCmdStr;const prefix,ext:TCmdStr;var foundfile : TCmdStr) : boolean;    function FindDLL(const s:TCmdStr;var founddll:TCmdStr):boolean;    procedure InitLinker;    procedure DoneLinker;Implementation    uses      cutils,cfileutils,      script,globals,verbose,comphook,ppu,      aasmbase,aasmtai,aasmdata,aasmcpu,      owbase,owar,ogmap;    type     TLinkerClass = class of Tlinker;{*****************************************************************************                                   Helpers*****************************************************************************}    { searches an object file }    function FindObjectFile(s:TCmdStr;const unitpath:TCmdStr;isunit:boolean) : TCmdStr;      var        found : boolean;        foundfile : TCmdStr;      begin        findobjectfile:='';        if s='' then         exit;        {When linking on target, the units has not been assembled yet,         so there is no object files to look for at         the host. Look for the corresponding assembler file instead,         because it will be assembled to object file on the target.}        if isunit and (cs_link_on_target in current_settings.globalswitches) then          s:=ChangeFileExt(s,target_info.asmext);        { when it does not belong to the unit then check if          the specified file exists without searching any paths }        if not isunit then         begin           if FileExists(FixFileName(s),false) then            begin              foundfile:=ScriptFixFileName(s);              found:=true;            end;         end;        if pos('.',s)=0 then         s:=s+target_info.objext;        { find object file           1. output unit path           2. output exe path           3. specified unit path (if specified)           4. cwd           5. unit search path           6. local object path           7. global object path           8. exepath (not when linking on target)          for all finds don't use the directory caching }        found:=false;        if isunit and (OutputUnitDir<>'') then          found:=FindFile(s,OutPutUnitDir,false,foundfile)        else          if OutputExeDir<>'' then            found:=FindFile(s,OutPutExeDir,false,foundfile);        if (not found) and (unitpath<>'') then         found:=FindFile(s,unitpath,false,foundfile);        if (not found) then         found:=FindFile(s, CurDirRelPath(source_info),false,foundfile);        if (not found) then         found:=UnitSearchPath.FindFile(s,false,foundfile);        if (not found) then         found:=current_module.localobjectsearchpath.FindFile(s,false,foundfile);        if (not found) then         found:=objectsearchpath.FindFile(s,false,foundfile);        if not(cs_link_on_target in current_settings.globalswitches) and (not found) then         found:=FindFile(s,exepath,false,foundfile);        if not(cs_link_nolink in current_settings.globalswitches) and (not found) then         Message1(exec_w_objfile_not_found,s);        {Restore file extension}        if isunit and (cs_link_on_target in current_settings.globalswitches) then          foundfile:= ChangeFileExt(foundfile,target_info.objext);        findobjectfile:=ScriptFixFileName(foundfile);      end;    { searches a (windows) DLL file }    function FindDLL(const s:TCmdStr;var founddll:TCmdStr):boolean;      var        sysdir : TCmdStr;        Found : boolean;      begin        Found:=false;        { Look for DLL in:          1. Current dir          2. Library Path          3. windir,windir/system,windir/system32 }        Found:=FindFile(s,'.'+source_info.DirSep,false,founddll);        if (not found) then         Found:=librarysearchpath.FindFile(s,false,founddll);        if (not found) then         begin           sysdir:=FixPath(GetEnvironmentVariable('windir'),false);           Found:=FindFile(s,sysdir+';'+sysdir+'system'+source_info.DirSep+';'+sysdir+'system32'+source_info.DirSep,false,founddll);         end;        if (not found) then         begin           message1(exec_w_libfile_not_found,s);           FoundDll:=s;         end;        FindDll:=Found;      end;    { searches an library file }    function FindLibraryFile(s:TCmdStr;const prefix,ext:TCmdStr;var foundfile : TCmdStr) : boolean;      var        found : boolean;        paths : TCmdStr;      begin        findlibraryfile:=false;        foundfile:=s;        if s='' then         exit;        { split path from filename }        paths:=ExtractFilePath(s);        s:=ExtractFileName(s);        { add prefix 'lib' }        if (prefix<>'') and (Copy(s,1,length(prefix))<>prefix) then         s:=prefix+s;        { add extension }        if (ext<>'') and (Copy(s,length(s)-length(ext)+1,length(ext))<>ext) then         s:=s+ext;        { readd the split path }        s:=paths+s;        if FileExists(s,false) then         begin           foundfile:=ScriptFixFileName(s);           FindLibraryFile:=true;           exit;         end;        { find libary           1. cwd           2. local libary dir           3. global libary dir           4. exe path of the compiler (not when linking on target)          for all searches don't use the directory cache }        found:=FindFile(s, CurDirRelPath(source_info), false,foundfile);        if (not found) and (current_module.outputpath^<>'') then         found:=FindFile(s,current_module.outputpath^,false,foundfile);        if (not found) then         found:=current_module.locallibrarysearchpath.FindFile(s,false,foundfile);        if (not found) then         found:=librarysearchpath.FindFile(s,false,foundfile);        if not(cs_link_on_target in current_settings.globalswitches) and (not found) then         found:=FindFile(s,exepath,false,foundfile);        foundfile:=ScriptFixFileName(foundfile);        findlibraryfile:=found;      end;{*****************************************************************************                                   TLINKER*****************************************************************************}    Constructor TLinker.Create;      begin        Inherited Create;        ObjectFiles:=TCmdStrList.Create_no_double;        SharedLibFiles:=TCmdStrList.Create_no_double;        StaticLibFiles:=TCmdStrList.Create_no_double;      end;    Destructor TLinker.Destroy;      begin        ObjectFiles.Free;        SharedLibFiles.Free;        StaticLibFiles.Free;      end;    procedure TLinker.AddModuleFiles(hp:tmodule);      var        mask : longint;        i,j  : longint;        ImportLibrary : TImportLibrary;        ImportSymbol  : TImportSymbol;      begin        with hp do         begin           if (flags and uf_has_resourcefiles)<>0 then             HasResources:=true;           if (flags and uf_has_exports)<>0 then             HasExports:=true;         { link unit files }           if (flags and uf_no_link)=0 then            begin              { create mask which unit files need linking }              mask:=link_always;              { static linking ? }              if (cs_link_static in current_settings.globalswitches) then               begin                 if (flags and uf_static_linked)=0 then                  begin                    { if smart not avail then try static linking }                    if (flags and uf_smart_linked)<>0 then                     begin                       Message1(exec_t_unit_not_static_linkable_switch_to_smart,modulename^);                       mask:=mask or link_smart;                     end                    else                     Message1(exec_e_unit_not_smart_or_static_linkable,modulename^);                  end                 else                   mask:=mask or link_static;               end;              { smart linking ? }              if (cs_link_smart in current_settings.globalswitches) then               begin                 if (flags and uf_smart_linked)=0 then                  begin                    { if smart not avail then try static linking }                    if (flags and uf_static_linked)<>0 then                     begin                       Message1(exec_t_unit_not_smart_linkable_switch_to_static,modulename^);                       mask:=mask or link_static;                     end                    else                     Message1(exec_e_unit_not_smart_or_static_linkable,modulename^);                  end                 else                  mask:=mask or link_smart;               end;              { shared linking }              if (cs_link_shared in current_settings.globalswitches) then               begin                 if (flags and uf_shared_linked)=0 then                  begin                    { if shared not avail then try static linking }                    if (flags and uf_static_linked)<>0 then                     begin                       Message1(exec_t_unit_not_shared_linkable_switch_to_static,modulename^);                       mask:=mask or link_static;                     end                    else                     Message1(exec_e_unit_not_shared_or_static_linkable,modulename^);                  end                 else                  mask:=mask or link_shared;               end;              { unit files }              while not linkunitofiles.empty do                AddObject(linkunitofiles.getusemask(mask),path^,true);              while not linkunitstaticlibs.empty do                AddStaticLibrary(linkunitstaticlibs.getusemask(mask));              while not linkunitsharedlibs.empty do                AddSharedLibrary(linkunitsharedlibs.getusemask(mask));            end;           { Other needed .o and libs, specified using $L,$LINKLIB,external }           mask:=link_always;           while not linkotherofiles.empty do            AddObject(linkotherofiles.Getusemask(mask),path^,false);           while not linkotherstaticlibs.empty do            AddStaticCLibrary(linkotherstaticlibs.Getusemask(mask));           while not linkothersharedlibs.empty do            AddSharedCLibrary(linkothersharedlibs.Getusemask(mask));           { Known Library/DLL Imports }           for i:=0 to ImportLibraryList.Count-1 do             begin               ImportLibrary:=TImportLibrary(ImportLibraryList[i]);               for j:=0 to ImportLibrary.ImportSymbolList.Count-1 do                 begin                   ImportSymbol:=TImportSymbol(ImportLibrary.ImportSymbolList[j]);                   AddImportSymbol(ImportLibrary.Name,ImportSymbol.Name,ImportSymbol.OrdNr,ImportSymbol.IsVar);                 end;             end;         end;      end;    procedure TLinker.AddImportSymbol(const libname,symname:TCmdStr;OrdNr: longint;isvar:boolean);      begin      end;    Procedure TLinker.AddObject(const S,unitpath : TCmdStr;isunit:boolean);      begin        ObjectFiles.Concat(FindObjectFile(s,unitpath,isunit));      end;    Procedure TLinker.AddSharedLibrary(S:TCmdStr);      begin        if s='' then          exit;        { remove prefix 'lib' }        if Copy(s,1,length(target_info.sharedlibprefix))=target_info.sharedlibprefix then          Delete(s,1,length(target_info.sharedlibprefix));        { remove extension if any }        if Copy(s,length(s)-length(target_info.sharedlibext)+1,length(target_info.sharedlibext))=target_info.sharedlibext then          Delete(s,length(s)-length(target_info.sharedlibext)+1,length(target_info.sharedlibext)+1);        { ready to be added }        SharedLibFiles.Concat(S);      end;    Procedure TLinker.AddStaticLibrary(const S:TCmdStr);      var        ns : TCmdStr;        found : boolean;      begin        if s='' then          exit;        found:=FindLibraryFile(s,target_info.staticlibprefix,target_info.staticlibext,ns);        if not(cs_link_nolink in current_settings.globalswitches) and (not found) then          Message1(exec_w_libfile_not_found,s);        StaticLibFiles.Concat(ns);      end;    Procedure TLinker.AddSharedCLibrary(S:TCmdStr);      begin        if s='' then          exit;        { remove prefix 'lib' }        if Copy(s,1,length(target_info.sharedclibprefix))=target_info.sharedclibprefix then          Delete(s,1,length(target_info.sharedclibprefix));        { remove extension if any }        if Copy(s,length(s)-length(target_info.sharedclibext)+1,length(target_info.sharedclibext))=target_info.sharedclibext then          Delete(s,length(s)-length(target_info.sharedclibext)+1,length(target_info.sharedclibext)+1);        { ready to be added }        SharedLibFiles.Concat(S);      end;    Procedure TLinker.AddStaticCLibrary(const S:TCmdStr);      var        ns : TCmdStr;        found : boolean;      begin        if s='' then         exit;        found:=FindLibraryFile(s,target_info.staticclibprefix,target_info.staticclibext,ns);        if not(cs_link_nolink in current_settings.globalswitches) and (not found) then         Message1(exec_w_libfile_not_found,s);        StaticLibFiles.Concat(ns);      end;    procedure AddImportSymbol(const libname,symname:TCmdStr;OrdNr: longint;isvar:boolean);      begin      end;    procedure TLinker.InitSysInitUnitName;      begin      end;    function TLinker.MakeExecutable:boolean;      begin        MakeExecutable:=false;        Message(exec_e_exe_not_supported);      end;    Function TLinker.MakeSharedLibrary:boolean;      begin        MakeSharedLibrary:=false;        Message(exec_e_dll_not_supported);      end;    Function TLinker.MakeStaticLibrary:boolean;      begin        MakeStaticLibrary:=false;        Message(exec_e_dll_not_supported);      end;    Procedure TLinker.ExpandAndApplyOrder(var Src:TCmdStrList);      var        p : TLinkStrMap;        i : longint;      begin        // call Virtual TLinker method to initialize        LoadPredefinedLibraryOrder;        // something to do?        if (LinkLibraryAliases.count=0) and (LinkLibraryOrder.Count=0) Then          exit;        p:=TLinkStrMap.Create;        // expand libaliases, clears src        LinkLibraryAliases.expand(src,p);        // writeln(src.count,' ',p.count,' ',linklibraryorder.count,' ',linklibraryaliases.count);        // apply order        p.UpdateWeights(LinkLibraryOrder);        p.SortOnWeight;        // put back in src        for i:=0 to p.count-1 do          src.insert(p[i].Key);        p.free;      end;    procedure TLinker.LoadPredefinedLibraryOrder;      begin      end;    function  TLinker.ReOrderEntries : boolean;      begin        result:=(LinkLibraryOrder.count>0) or (LinkLibraryAliases.count>0);      end;{*****************************************************************************                              TEXTERNALLINKER*****************************************************************************}    Constructor TExternalLinker.Create;      begin        inherited Create;        { set generic defaults }        FillChar(Info,sizeof(Info),0);        if cs_link_on_target in current_settings.globalswitches then          begin            Info.ResName:=outputexedir+ChangeFileExt(inputfilename,'_link.res');            Info.ScriptName:=outputexedir+ChangeFileExt(inputfilename,'_script.res');          end        else          begin            Info.ResName:='link.res';            Info.ScriptName:='script.res';          end;        { set the linker specific defaults }        SetDefaultInfo;        { Allow Parameter overrides for linker info }        with Info do         begin           if ParaLinkOptions<>'' then            ExtraOptions:=ParaLinkOptions;           if ParaDynamicLinker<>'' then            DynamicLinker:=ParaDynamicLinker;         end;      end;    Destructor TExternalLinker.Destroy;      begin        inherited destroy;      end;    Procedure TExternalLinker.SetDefaultInfo;      begin      end;    Function TExternalLinker.FindUtil(const s:TCmdStr):TCmdStr;      var        Found    : boolean;        FoundBin : TCmdStr;        UtilExe  : TCmdStr;      begin        if cs_link_on_target in current_settings.globalswitches then          begin            { If linking on target, don't add any path PM }            FindUtil:=ChangeFileExt(s,target_info.exeext);            exit;          end;        UtilExe:=ChangeFileExt(s,source_info.exeext);        FoundBin:='';        Found:=false;        if utilsdirectory<>'' then         Found:=FindFile(utilexe,utilsdirectory,false,Foundbin);        if (not Found) then         Found:=FindExe(utilexe,false,Foundbin);        if (not Found) and not(cs_link_nolink in current_settings.globalswitches) then         begin           Message1(exec_e_util_not_found,utilexe);           current_settings.globalswitches:=current_settings.globalswitches+[cs_link_nolink];         end;        if (FoundBin<>'') then         Message1(exec_t_using_util,FoundBin);        FindUtil:=FoundBin;      end;    Function TExternalLinker.DoExec(const command:TCmdStr; para:TCmdStr;showinfo,useshell:boolean):boolean;      var        exitcode: longint;      begin        DoExec:=true;        if not(cs_link_nolink in current_settings.globalswitches) then         begin           FlushOutput;           if useshell then             exitcode:=shell(maybequoted(command)+' '+para)           else             try               exitcode:=ExecuteProcess(command,para);             except on E:EOSError do               begin                 Message(exec_e_cant_call_linker);                 current_settings.globalswitches:=current_settings.globalswitches+[cs_link_nolink];                 DoExec:=false;               end;             end;           if (exitcode<>0) then             begin               Message(exec_e_error_while_linking);               current_settings.globalswitches:=current_settings.globalswitches+[cs_link_nolink];               DoExec:=false;             end;         end;      { Update asmres when externmode is set }        if cs_link_nolink in current_settings.globalswitches then         begin           if showinfo then             begin               if DLLsource then                 AsmRes.AddLinkCommand(Command,Para,current_module.sharedlibfilename^)               else                 AsmRes.AddLinkCommand(Command,Para,current_module.exefilename^);             end           else            AsmRes.AddLinkCommand(Command,Para,'');         end;      end;    Function TExternalLinker.MakeStaticLibrary:boolean;        function GetNextFiles(const maxCmdLength : AInt; var item : TCmdStrListItem) : TCmdStr;          begin            result := '';            while (assigned(item) and ((length(result) + length(item.str) + 1) < maxCmdLength)) do begin              result := result + ' ' + item.str;              item := TCmdStrListItem(item.next);            end;          end;      var        binstr, scriptfile : TCmdStr;        cmdstr, nextcmd, smartpath : TCmdStr;        current : TCmdStrListItem;        script: Text;        scripted_ar : boolean;        success : boolean;      begin        MakeStaticLibrary:=false;      { remove the library, to be sure that it is rewritten }        DeleteFile(current_module.staticlibfilename^);      { Call AR }        smartpath:=FixPath(ChangeFileExt(current_module.asmfilename^,target_info.smartext),false);        SplitBinCmd(target_ar.arcmd,binstr,cmdstr);        binstr := FindUtil(utilsprefix + binstr);        scripted_ar:=target_ar.id=ar_gnu_ar_scripted;        if scripted_ar then          begin            scriptfile := FixFileName(smartpath+'arscript.txt');            Replace(cmdstr,'$SCRIPT',maybequoted(scriptfile));            Assign(script, scriptfile);            Rewrite(script);            try              writeln(script, 'CREATE ' + current_module.staticlibfilename^);              current := TCmdStrListItem(SmartLinkOFiles.First);              while current <> nil do                begin                  writeln(script, 'ADDMOD ' + current.str);                  current := TCmdStrListItem(current.next);                end;              writeln(script, 'SAVE');              writeln(script, 'END');            finally              Close(script);            end;            success:=DoExec(binstr,cmdstr,false,true);          end        else          begin            Replace(cmdstr,'$LIB',maybequoted(current_module.staticlibfilename^));            { create AR commands }            success := true;            current := TCmdStrListItem(SmartLinkOFiles.First);            repeat              nextcmd := cmdstr;              Replace(nextcmd,'$FILES',GetNextFiles(2047, current));              success:=DoExec(binstr,nextcmd,false,true);            until (not assigned(current)) or (not success);          end;        if (target_ar.arfinishcmd <> '') then          begin            SplitBinCmd(target_ar.arfinishcmd,binstr,cmdstr);            binstr := FindUtil(utilsprefix + binstr);            Replace(cmdstr,'$LIB',maybequoted(current_module.staticlibfilename^));            success:=DoExec(binstr,cmdstr,false,true);          end;        { Clean up }        if not(cs_asm_leave in current_settings.globalswitches) then         if not(cs_link_nolink in current_settings.globalswitches) then          begin            while not SmartLinkOFiles.Empty do              DeleteFile(SmartLinkOFiles.GetFirst);            if scripted_ar then              DeleteFile(scriptfile);            RemoveDir(smartpath);          end         else          begin            AsmRes.AddDeleteCommand(FixFileName(smartpath+current_module.asmprefix^+'*'+target_info.objext));            if scripted_ar then              AsmRes.AddDeleteCommand(scriptfile);            AsmRes.AddDeleteDirCommand(smartpath);          end;        MakeStaticLibrary:=success;      end;{*****************************************************************************                              TINTERNALLINKER*****************************************************************************}    Constructor TInternalLinker.Create;      begin        inherited Create;        linkscript:=TCmdStrList.Create;        FStaticLibraryList:=TFPHashObjectList.Create(true);        FImportLibraryList:=TFPHashObjectList.Create(true);        exemap:=nil;        exeoutput:=nil;        CObjInput:=TObjInput;      end;    Destructor TInternalLinker.Destroy;      begin        linkscript.free;        StaticLibraryList.Free;        ImportLibraryList.Free;        if assigned(exeoutput) then          begin            exeoutput.free;            exeoutput:=nil;          end;        if assigned(exemap) then          begin            exemap.free;            exemap:=nil;          end;        inherited destroy;      end;    procedure TInternalLinker.AddImportSymbol(const libname,symname:TCmdStr;OrdNr: longint;isvar:boolean);      var        ImportLibrary : TImportLibrary;        ImportSymbol  : TFPHashObject;      begin        ImportLibrary:=TImportLibrary(ImportLibraryList.Find(libname));        if not assigned(ImportLibrary) then          ImportLibrary:=TImportLibrary.Create(ImportLibraryList,libname);        ImportSymbol:=TFPHashObject(ImportLibrary.ImportSymbolList.Find(symname));        if not assigned(ImportSymbol) then          ImportSymbol:=TImportSymbol.Create(ImportLibrary.ImportSymbolList,symname,OrdNr,isvar);      end;    procedure TInternalLinker.Load_ReadObject(const para:TCmdStr);      var        objdata   : TObjData;        objinput  : TObjinput;        objreader : TObjectReader;        fn        : TCmdStr;      begin        fn:=FindObjectFile(para,'',false);        Comment(V_Tried,'Reading object '+fn);        objinput:=CObjInput.Create;        objdata:=objinput.newObjData(para);        objreader:=TObjectreader.create;        if objreader.openfile(fn) then          begin            if objinput.ReadObjData(objreader,objdata) then              exeoutput.addobjdata(objdata);          end;        { release input object }        objinput.free;        objreader.free;      end;    procedure TInternalLinker.Load_ReadStaticLibrary(const para:TCmdStr);      var        objreader : TObjectReader;      begin{$warning TODO Cleanup ignoring of   FPC generated libimp*.a files}        { Don't load import libraries }        if copy(ExtractFileName(para),1,6)='libimp' then          exit;        Comment(V_Tried,'Opening library '+para);        objreader:=TArObjectreader.create(para);        TStaticLibrary.Create(StaticLibraryList,para,objreader,CObjInput);      end;    procedure TInternalLinker.ParseScript_Load;      var        s,        para,        keyword : String;        hp : TCmdStrListItem;      begin        exeoutput.Load_Start;        hp:=TCmdStrListItem(linkscript.first);        while assigned(hp) do          begin            s:=hp.str;            if (s='') or (s[1]='#') then              continue;            keyword:=Upper(GetToken(s,' '));            para:=GetToken(s,' ');            if keyword='SYMBOL' then              ExeOutput.Load_Symbol(para)            else if keyword='ENTRYNAME' then              ExeOutput.Load_EntryName(para)            else if keyword='ISSHAREDLIBRARY' then              ExeOutput.Load_IsSharedLibrary            else if keyword='IMAGEBASE' then              ExeOutput.Load_ImageBase(para)            else if keyword='READOBJECT' then              Load_ReadObject(para)            else if keyword='READSTATICLIBRARY' then              Load_ReadStaticLibrary(para);            hp:=TCmdStrListItem(hp.next);          end;      end;    procedure TInternalLinker.ParseScript_Order;      var        s,        para,        keyword : String;        hp : TCmdStrListItem;      begin        exeoutput.Order_Start;        hp:=TCmdStrListItem(linkscript.first);        while assigned(hp) do          begin            s:=hp.str;            if (s='') or (s[1]='#') then              continue;            keyword:=Upper(GetToken(s,' '));            para:=GetToken(s,' ');            if keyword='EXESECTION' then              ExeOutput.Order_ExeSection(para)            else if keyword='ENDEXESECTION' then              ExeOutput.Order_EndExeSection            else if keyword='OBJSECTION' then              ExeOutput.Order_ObjSection(para)            else if keyword='ZEROS' then              ExeOutput.Order_Zeros(para)            else if keyword='SYMBOL' then              ExeOutput.Order_Symbol(para);            hp:=TCmdStrListItem(hp.next);          end;        exeoutput.Order_End;      end;    procedure TInternalLinker.ParseScript_CalcPos;      var        s,        para,        keyword : String;        hp : TCmdStrListItem;      begin        exeoutput.CalcPos_Start;        hp:=TCmdStrListItem(linkscript.first);        while assigned(hp) do          begin            s:=hp.str;            if (s='') or (s[1]='#') then              continue;            keyword:=Upper(GetToken(s,' '));            para:=GetToken(s,' ');            if keyword='EXESECTION' then              ExeOutput.CalcPos_ExeSection(para)            else if keyword='ENDEXESECTION' then              ExeOutput.CalcPos_EndExeSection            else if keyword='HEADER' then              ExeOutput.CalcPos_Header            else if keyword='SYMBOLS' then              ExeOutput.CalcPos_Symbols;            hp:=TCmdStrListItem(hp.next);          end;      end;    procedure TInternalLinker.PrintLinkerScript;      var        hp : TCmdStrListItem;      begin        if not assigned(exemap) then          exit;        exemap.Add('Used linker script');        exemap.Add('');        hp:=TCmdStrListItem(linkscript.first);        while assigned(hp) do          begin            exemap.Add(hp.str);            hp:=TCmdStrListItem(hp.next);          end;      end;    function TInternalLinker.RunLinkScript(const outputname:TCmdStr):boolean;      label        myexit;      var        bsssize : aint;        bsssec : TExeSection;      begin        result:=false;        Message1(exec_i_linking,outputname);        FlushOutput;{$warning TODO Load custom linker script}        DefaultLinkScript;        exeoutput:=CExeOutput.Create;        if (cs_link_map in current_settings.globalswitches) then          exemap:=texemap.create(current_module.mapfilename^);        PrintLinkerScript;        { Load .o files and resolve symbols }        ParseScript_Load;        exeoutput.ResolveSymbols(StaticLibraryList);        { Generate symbols and code to do the importing }        exeoutput.GenerateLibraryImports(ImportLibraryList);        { Fill external symbols data }        exeoutput.FixupSymbols;        if ErrorCount>0 then          goto myexit;        { Create .exe sections and add .o sections }        ParseScript_Order;        exeoutput.RemoveUnreferencedSections;        exeoutput.MergeStabs;        exeoutput.RemoveEmptySections;        if ErrorCount>0 then          goto myexit;        { Calc positions in mem and file }        ParseScript_CalcPos;        exeoutput.FixupRelocations;        exeoutput.PrintMemoryMap;        if ErrorCount>0 then          goto myexit;        exeoutput.WriteExeFile(outputname);        if cs_link_separate_dbg_file in current_settings.globalswitches then          exeoutput.WriteDbgFile(ChangeFileExt(outputname,'.dbg'));{$warning TODO fixed section names}        status.codesize:=exeoutput.findexesection('.text').size;        status.datasize:=exeoutput.findexesection('.data').size;        bsssec:=exeoutput.findexesection('.bss');        if assigned(bsssec) then          bsssize:=bsssec.size        else          bsssize:=0;        { Executable info }        Message1(execinfo_x_codesize,tostr(status.codesize));        Message1(execinfo_x_initdatasize,tostr(status.datasize));        Message1(execinfo_x_uninitdatasize,tostr(bsssize));        Message1(execinfo_x_stackreserve,tostr(stacksize));      myexit:        { close map }        if assigned(exemap) then          begin            exemap.free;            exemap:=nil;          end;        { close exe }        exeoutput.free;        exeoutput:=nil;        result:=true;      end;    function TInternalLinker.MakeExecutable:boolean;      begin        IsSharedLibrary:=false;        result:=RunLinkScript(current_module.exefilename^);      end;    function TInternalLinker.MakeSharedLibrary:boolean;      begin        IsSharedLibrary:=true;        result:=RunLinkScript(current_module.sharedlibfilename^);      end;{*****************************************************************************                                 Init/Done*****************************************************************************}    procedure InitLinker;      var        lk : TlinkerClass;      begin        if (cs_link_extern in current_settings.globalswitches) and           assigned(target_info.linkextern) then          begin            lk:=TlinkerClass(target_info.linkextern);            linker:=lk.Create;          end        else          if assigned(target_info.link) then            begin              lk:=TLinkerClass(target_info.link);              linker:=lk.Create;            end        else          linker:=Tlinker.Create;      end;    procedure DoneLinker;      begin        if assigned(linker) then         Linker.Free;      end;{*****************************************************************************                                   Initialize*****************************************************************************}    const      ar_gnu_ar_info : tarinfo =          (            id          : ar_gnu_ar;            arcmd       : 'ar qS $LIB $FILES';            arfinishcmd : 'ar s $LIB'          );      ar_gnu_ar_scripted_info : tarinfo =          (            id    : ar_gnu_ar_scripted;            arcmd : 'ar -M < $SCRIPT';            arfinishcmd : ''          );      ar_gnu_gar_info : tarinfo =          ( id          : ar_gnu_gar;            arcmd       : 'gar qS $LIB $FILES';            arfinishcmd : 'gar s $LIB'          );initialization  RegisterAr(ar_gnu_ar_info);  RegisterAr(ar_gnu_ar_scripted_info);  RegisterAr(ar_gnu_gar_info);end.
 |