| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438 | {    Copyright (c) 2011 by Jonas Maebe    This unit implements support import,export,link routines    for the AIX target    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 t_aix;{$i fpcdefs.inc}interface  uses    aasmdata,    symsym,symdef,ppu,    import,export,expunix,link;  type    timportlibaix=class(timportlib)      procedure generatelib;override;    end;    texportlibaix=class(texportlibunix)      procedure setfininame(list: TAsmList; const s: string); override;    end;    TLinkerAIX=class(texternallinker)    private      prtobj  : string[80];      assumebinutils,use_gld : boolean;      Function  WriteResponseFile(isdll:boolean) : Boolean;    public      constructor Create;override;      procedure SetDefaultInfo;override;      function  MakeExecutable:boolean;override;      function  MakeSharedLibrary:boolean;override;    end;implementation  uses    SysUtils,    cutils,cfileutl,cclasses,    verbose,systems,globtype,globals,    symconst,cscript,    fmodule,    aasmbase,aasmtai,aasmcpu,cpubase,    cgbase,cgobj,cgutils,ogbase,ncgutil,    comprsrc,    rescmn, i_aix    ;{*****************************************************************************                               timportlibaix*****************************************************************************}    procedure timportlibaix.generatelib;      var        i : longint;        ImportLibrary : TImportLibrary;      begin        for i:=0 to current_module.ImportLibraryList.Count-1 do          begin            ImportLibrary:=TImportLibrary(current_module.ImportLibraryList[i]);            current_module.linkothersharedlibs.add(ImportLibrary.Name,link_always);          end;      end;{*****************************************************************************                               texportlibaix*****************************************************************************}    procedure texportlibaix.setfininame(list: TAsmList; const s: string);      begin        inherited setfininame(list,s);      end;{*****************************************************************************                                  TLinkerAIX*****************************************************************************}Constructor TLinkerAIX.Create;begin  Inherited Create;  if not Dontlinkstdlibpath then    if not(cs_profile in current_settings.moduleswitches) then      LibrarySearchPath.AddPath(sysrootpath,'/usr/lib;/usr/X11R6/lib;/opt/freeware/lib',true)    else      LibrarySearchPath.AddPath(sysrootpath,'/usr/lib/profiled;/usr/X11R6/lib;/opt/freeware/lib',true)end;procedure TLinkerAIX.SetDefaultInfo;begin  with Info do   begin     { the -bpT and -bpD options set the base addresses for text and data       sections. They are required because otherwise they are both 0 and hence       overlap, which confuses gdb. GCC uses those same options. -btextro makes       sure that the binary does not contain any relocations in the text       section (otherwise you get an error at load time instead of at link time       in case something is wrong) }     ExeCmd[1]:='$LDBIN -bpT:0x10000000 -bpD:0x20000000 -btextro $OPT $STRIP -L. -o $EXE $CATRES' {$ifdef powerpc64}+' -b64'{$endif};     DllCmd[1]:='$LDBIN -bpT:0x10000000 -bpD:0x20000000 -btextro $OPT $INITFINI $STRIP -G -L. -o $EXE $CATRES' {$ifdef powerpc64}+' -b64'{$endif};     if cs_debuginfo in current_settings.moduleswitches then       begin         { debugging helpers }         ExeCmd[1]:=ExeCmd[1]+' -lg -bexport:/usr/lib/libg.exp';         DllCmd[1]:=DllCmd[1]+' -lg -bexport:/usr/lib/libg.exp';       end;   end;{$if defined(powerpc)}   if not(cs_profile in current_settings.moduleswitches) then     prtobj:=sysrootpath+'/lib/crt0.o'   else     prtobj:=sysrootpath+'/lib/gcrt0.o';{$elseif defined(powerpc64)}  if not(cs_profile in current_settings.moduleswitches) then    prtobj:=sysrootpath+'/lib/crt0_64.o'  else    prtobj:=sysrootpath+'/lib/gcrt0_64.o';{$else}{$error unsupported AIX architecture}{$endif}  assumebinutils:=    not(cs_link_native in current_settings.globalswitches) or    (not(cs_link_on_target in current_settings.globalswitches) and     not(source_info.system in systems_aix)) ;  use_gld:=assumebinutils and (source_info.system in systems_aix)end;Function TLinkerAIX.WriteResponseFile(isdll:boolean) : Boolean;Var  linkres      : TLinkRes;  i            : longint;  HPath        : TCmdStrListItem;  s,s1         : TCmdStr;begin  result:=False;  { Open link.res file }  LinkRes:=TLinkRes.Create(outputexedir+Info.ResName,assumebinutils);  with linkres do    begin      { Write path to search libraries }      HPath:=TCmdStrListItem(current_module.locallibrarysearchpath.First);      while assigned(HPath) do       begin         if assumebinutils then           Add('SEARCH_DIR("'+HPath.Str+'")')         else           Add('-L'+HPath.Str);         HPath:=TCmdStrListItem(HPath.Next);       end;      HPath:=TCmdStrListItem(LibrarySearchPath.First);      while assigned(HPath) do       begin         if assumebinutils then           Add('SEARCH_DIR("'+HPath.Str+'")')         else           Add('-L'+HPath.Str);         HPath:=TCmdStrListItem(HPath.Next);       end;       if assumebinutils then        StartSection('INPUT(');      { add objectfiles, start with prt0 always }       if assumebinutils then         AddFileName(maybequoted(FindObjectFile(prtobj,'',false)))       else         AddFileName(FindObjectFile(prtobj,'',false));      { main objectfiles }      while not ObjectFiles.Empty do       begin         s:=ObjectFiles.GetFirst;         if s<>'' then          if assumebinutils then            AddFileName(maybequoted(s))          else            AddFileName(s)       end;      { Write staticlibraries }      if not StaticLibFiles.Empty then       begin         While not StaticLibFiles.Empty do          begin            S:=StaticLibFiles.GetFirst;            if assumebinutils then              AddFileName(maybequoted(s))            else              AddFileName(s);          end;       end;      { Write sharedlibraries like -l<lib> }      While not SharedLibFiles.Empty do        begin          S:=SharedLibFiles.GetFirst;          i:=Pos(target_info.sharedlibext,S);          if i>0 then            Delete(S,i,255);          Add('-l'+s);        end;       { when we have -static for the linker the we also need libgcc }       if (cs_link_staticflag in current_settings.globalswitches) then         begin           Add('-lgcc');           if librarysearchpath.FindFile('libgcc_eh.a',false,s1) then             Add('-lgcc_eh');         end;       if assumebinutils then         EndSection(')');      { Write and Close response }      writetodisk;      Free;    end;  WriteResponseFile:=True;end;function TLinkerAIX.MakeExecutable:boolean;var  linkscript: TAsmScript;  binstr,  cmdstr  : TCmdStr;  success : boolean;  StripStr   : string[40];begin  if not(cs_link_nolink in current_settings.globalswitches) then   Message1(exec_i_linking,current_module.exefilename);  linkscript:=nil;{ Create some replacements }  StripStr:='';  if (cs_link_strip in current_settings.globalswitches) and     not(cs_link_separate_dbg_file in current_settings.globalswitches) then   StripStr:='-s';  if (cs_link_map in current_settings.globalswitches) then   StripStr:='-bmap:'+maybequoted(ChangeFileExt(current_module.exefilename,'.map'));{ Write used files and libraries }  WriteResponseFile(false);{ Call linker }  SplitBinCmd(Info.ExeCmd[1],binstr,cmdstr);  if assumebinutils and (source_info.system in systems_aix) then    Replace(binstr,'$LDBIN','gld')  else    Replace(binstr,'$LDBIN','ld');  binstr:=FindUtil(utilsprefix+BinStr);  Replace(cmdstr,'$EXE',maybequoted(current_module.exefilename));  Replace(cmdstr,'$OPT',Info.ExtraOptions);  { the native AIX linker does not support linkres files, so we need    CatFileContent(). The binutils cross-linker does support such files, so    use them when cross-compiling to avoid overflowing the Windows maximum    command line length  }  if not(cs_link_on_target in current_settings.globalswitches) and     not(source_info.system in systems_aix) then    Replace(cmdstr,'$CATRES',outputexedir+Info.ResName)  else    Replace(cmdstr,'$CATRES',CatFileContent(outputexedir+Info.ResName));  Replace(cmdstr,'$STRIP',StripStr);  { create dynamic symbol table? }  if HasExports then    cmdstr:=cmdstr+' -E';  { custom sysroot? (for cross-compiling -> assume gnu ld) }  if sysrootpath<>'' then    cmdstr:=cmdstr+' --sysroot='+sysrootpath;  if not(cs_link_nolink in current_settings.globalswitches) and     not(tf_no_backquote_support in source_info.flags) then    begin      { we have to use a script to use the IFS hack }      linkscript:=GenerateScript(outputexedir+'ppaslink');      linkscript.AddLinkCommand(binstr,CmdStr,'');//      if (extdbgbinstr<>'') then//        linkscript.AddLinkCommand(extdbgbinstr,extdbgcmdstr,'');      linkscript.WriteToDisk;      BinStr:=linkscript.fn;      if not path_absolute(BinStr) then        if cs_link_on_target in current_settings.globalswitches then          BinStr:='.'+target_info.dirsep+BinStr        else          BinStr:='.'+source_info.dirsep+BinStr;      CmdStr:='';    end;  success:=DoExec(BinStr,cmdstr,true,true);  { Remove ReponseFile }  if (success) and not(cs_link_nolink in current_settings.globalswitches) then    begin      DeleteFile(outputexedir+Info.ResName);      if assigned(linkscript) then        DeleteFile(linkscript.fn);    end;  linkscript.free;  MakeExecutable:=success;   { otherwise a recursive call to link method }end;Function TLinkerAIX.MakeSharedLibrary:boolean;const{$ifdef cpu64bitaddr}  libobj = 'shr_64.o';{$else}  libobj = 'shr.o';{$endif}var  linkscript  : TAsmScript;  exportedsyms: text;  InitFiniStr : string[80];  StripStr,  binstr,  cmdstr,  libfn: TCmdStr;  success : boolean;begin  MakeSharedLibrary:=false;  if not(cs_link_nolink in current_settings.globalswitches) then   Message1(exec_i_linking,current_module.sharedlibfilename);{ Write used files and libraries }  WriteResponseFile(true); { Create some replacements }  InitFiniStr:='-binitfini:'+exportlib.initname+':'+exportlib.fininame;  Replace(InitFiniStr,'$','.');  if cs_link_strip in current_settings.globalswitches then    StripStr:='-s'  else    StripStr:='';{ Call linker }  SplitBinCmd(Info.DllCmd[1],binstr,cmdstr);  if assumebinutils and (source_info.system in systems_aix) then    Replace(binstr,'$LDBIN','gld')  else    Replace(binstr,'$LDBIN','ld');  binstr:=FindUtil(utilsprefix+BinStr);  { on AIX, shared libraries are special object files that are stored inside    an archive. In that archive, the 32 bit version of the library is called    shr.o and the 64 bit version shr_64.o }  Replace(cmdstr,'$EXE',maybequoted(libobj));  Replace(cmdstr,'$OPT',Info.ExtraOptions);  if not(cs_link_on_target in current_settings.globalswitches) and     not(source_info.system in systems_aix) then    Replace(cmdstr,'$CATRES',outputexedir+Info.ResName)  else    Replace(cmdstr,'$CATRES',CatFileContent(outputexedir+Info.ResName));  Replace(cmdstr,'$INITFINI',InitFiniStr);  Replace(cmdstr,'$STRIP',StripStr);  { exported symbols }  if not texportlibunix(exportlib).exportedsymnames.empty then    begin      assign(exportedsyms,outputexedir+'linksyms.fpc');      rewrite(exportedsyms);      repeat        writeln(exportedsyms,texportlibunix(exportlib).exportedsymnames.getfirst);      until texportlibunix(exportlib).exportedsymnames.empty;      close(exportedsyms);      cmdstr:=cmdstr+' -bE:'+maybequoted(outputexedir)+'linksyms.fpc';    end;  libfn:=maybequoted(current_module.sharedlibfilename);  { we have to use a script to use the IFS hack }  linkscript:=GenerateScript(outputexedir+'ppaslink');  linkscript.AddLinkCommand(binstr,CmdStr,'');  { delete the target static library containing the dynamic object file in    case it already existed }  if FileExists(libfn,true) then    linkscript.AddDeleteCommand(libfn);  { and create the new one }  linkscript.AddLinkCommand(FindUtil(utilsprefix+'ar'),' -X32_64 -q '+libfn+(' '+libobj),'');  linkscript.WriteToDisk;  BinStr:=linkscript.fn;  if not path_absolute(BinStr) then    if cs_link_on_target in current_settings.globalswitches then      BinStr:='.'+target_info.dirsep+BinStr    else      BinStr:='.'+source_info.dirsep+BinStr;  CmdStr:='';  success:=DoExec(BinStr,cmdstr,true,true);  { Remove ReponseFile }  if (success) and not(cs_link_nolink in current_settings.globalswitches) then    begin      DeleteFile(libobj);      DeleteFile(outputexedir+Info.ResName);      DeleteFile(outputexedir+'linksyms.fpc');      DeleteFile(linkscript.fn);    end;  linkscript.free;  MakeSharedLibrary:=success;   { otherwise a recursive call to link method }end;{*****************************************************************************                                  Initialize*****************************************************************************}initialization  RegisterLinker(ld_aix,TLinkerAIX);{$ifdef powerpc}  RegisterImport(system_powerpc_aix,timportlibaix);  RegisterExport(system_powerpc_aix,texportlibaix);  RegisterTarget(system_powerpc_aix_info);{$endif powerpc}{$ifdef powerpc64}  RegisterImport(system_powerpc64_aix,timportlibaix);  RegisterExport(system_powerpc64_aix,texportlibaix);  RegisterTarget(system_powerpc64_aix_info);{$endif powerpc64}  RegisterRes(res_xcoff_info,TWinLikeResourceFile);end.
 |