12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124 |
- {
- Copyright (c) 1998-2002 by Peter Vreman (original Linux)
- (c) 2000 by Marco van de Voort (FreeBSD mods)
- This unit implements support import,export,link routines
- for the (i386)FreeBSD 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_bsd;
- {$i fpcdefs.inc}
- interface
- implementation
- uses
- sysutils,
- cutils,cfileutl,cclasses,
- verbose,systems,globtype,globals,
- symconst,cscript,
- fmodule,aasmbase,aasmtai,aasmdata,aasmcpu,cpubase,symsym,symdef,
- import,export,link,comprsrc,rescmn,i_bsd,expunix,
- cgutils,cgbase,cgobj,cpuinfo,ogbase;
- type
- timportlibdarwin=class(timportlib)
- procedure generatelib;override;
- end;
- timportlibbsd=class(timportlib)
- procedure generatelib;override;
- end;
- texportlibbsd=class(texportlibunix)
- end;
- texportlibdarwin=class(texportlibbsd)
- procedure setinitname(list: TAsmList; const s: string); override;
- procedure setfininame(list: TAsmList; const s: string); override;
- end;
- tlinkerbsd=class(texternallinker)
- private
- LdSupportsNoResponseFile : boolean;
- LibrarySuffix : Char;
- Function WriteResponseFile(isdll:boolean) : Boolean;
- function GetDarwinCrt1ObjName(isdll: boolean): TCmdStr;
- Function GetDarwinPrtobjName(isdll: boolean): TCmdStr;
- public
- constructor Create;override;
- procedure SetDefaultInfo;override;
- function MakeExecutable:boolean;override;
- function MakeSharedLibrary:boolean;override;
- procedure LoadPredefinedLibraryOrder; override;
- procedure InitSysInitUnitName; override;
- end;
- {*****************************************************************************
- TIMPORTLIBDARWIN
- *****************************************************************************}
- procedure timportlibdarwin.generatelib;
- begin
- end;
- {*****************************************************************************
- TEXPORTLIBDARWIN
- *****************************************************************************}
- procedure texportlibdarwin.setinitname(list: TAsmList; const s: string);
- begin
- new_section(list,sec_init_func,'',sizeof(pint));
- list.concat(Tai_const.Createname(s,0));
- end;
- procedure texportlibdarwin.setfininame(list: TAsmList; const s: string);
- begin
- new_section(list,sec_term_func,'',sizeof(pint));
- list.concat(Tai_const.Createname(s,0));
- end;
- {*****************************************************************************
- TIMPORTLIBBSD
- *****************************************************************************}
- procedure timportlibbsd.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;
- {*****************************************************************************
- TLINKERBSD
- *****************************************************************************}
- Constructor TLinkerBSD.Create;
- begin
- Inherited Create;
- if not Dontlinkstdlibpath Then
- if target_info.system in systems_darwin then
- { Mac OS X doesn't have a /lib }
- LibrarySearchPath.AddPath(sysrootpath,'/usr/lib',true)
- else if target_info.system in systems_openbsd then
- LibrarySearchPath.AddPath(sysrootpath,'/usr/lib;${X11BASE}/lib;${LOCALBASE}/lib',true)
- else
- LibrarySearchPath.AddPath(sysrootpath,'/lib;/usr/lib;/usr/X11R6/lib',true);
- end;
- procedure TLinkerBSD.SetDefaultInfo;
- {
- This will also detect which libc version will be used
- }
- begin
- LibrarySuffix:=' ';
- LdSupportsNoResponseFile := (target_info.system in ([system_m68k_netbsd]+systems_darwin));
- with Info do
- begin
- if LdSupportsNoResponseFile then
- begin
- if not(target_info.system in systems_darwin) then
- begin
- ExeCmd[1]:='ld $TARGET $EMUL $OPT $DYNLINK $STATIC $GCSECTIONS $STRIP -L. -o $EXE $CATRES $FILELIST';
- DllCmd[1]:='ld $TARGET $EMUL $OPT -shared -L. -o $EXE $CATRES $FILELIST'
- end
- else
- begin
- {$ifndef cpu64bitaddr}
- { Set the size of the page at address zero to 64kb, so nothing
- is loaded below that address. This avoids problems with the
- strange Windows-compatible resource handling that assumes
- that addresses below 64kb do not exist.
- On 64bit systems, page zero is 4GB by default, so no problems
- there.
- }
- { In case of valgrind, don't do that, because it cannot deal with
- a custom pagezero size -- in general, this should not cause any
- problems because the resources are added at the end and most
- programs with problems that require Valgrind will have more
- than 60KB of data (first 4KB of address space is always invalid)
- }
- ExeCmd[1]:='ld $PRTOBJ $TARGET $EMUL $OPT $DYNLINK $STATIC $GCSECTIONS $STRIP -multiply_defined suppress -L. -o $EXE $CATRES $FILELIST';
- if not(cs_gdb_valgrind in current_settings.globalswitches) then
- ExeCmd[1]:=ExeCmd[1]+' -pagezero_size 0x10000';
- {$else ndef cpu64bitaddr}
- ExeCmd[1]:='ld $PRTOBJ $TARGET $EMUL $OPT $DYNLINK $STATIC $GCSECTIONS $STRIP -multiply_defined suppress -L. -o $EXE $CATRES $FILELIST';
- {$endif ndef cpu64bitaddr}
- if (apptype<>app_bundle) then
- DllCmd[1]:='ld $PRTOBJ $TARGET $EMUL $OPT $GCSECTIONS -dynamic -dylib -multiply_defined suppress -L. -o $EXE $CATRES $FILELIST'
- else
- DllCmd[1]:='ld $PRTOBJ $TARGET $EMUL $OPT $GCSECTIONS -dynamic -bundle -multiply_defined suppress -L. -o $EXE $CATRES $FILELIST'
- end
- end
- else
- begin
- ExeCmd[1]:='ld $TARGET $EMUL $OPT $DYNLINK $STATIC $GCSECTIONS $STRIP -L. -o $EXE $RES';
- DllCmd[1]:='ld $TARGET $EMUL $OPT $INIT $FINI $SONAME -shared -L. -o $EXE $RES';
- end;
- if not(target_info.system in systems_darwin) then
- DllCmd[2]:='strip --strip-unneeded $EXE'
- else
- DllCmd[2]:='strip -x $EXE';
- { OpenBSD seems to use a wrong dynamic linker by default }
- if target_info.system in systems_openbsd then
- DynamicLinker:='/usr/libexec/ld.so'
- else if target_info.system in systems_netbsd then
- DynamicLinker:='/usr/libexec/ld.elf_so'
- else if target_info.system=system_x86_64_dragonfly then
- DynamicLinker:='/libexec/ld-elf.so.2'
- else
- DynamicLinker:='';
- end;
- end;
- procedure TLinkerBSD.LoadPredefinedLibraryOrder;
- // put your linkorder/linkalias overrides here.
- // Note: assumes only called when reordering/aliasing is used.
- Begin
- if not(target_info.system in systems_darwin) then
- begin
- if (target_info.system =system_i386_freebsd) and
- not (cs_link_no_default_lib_order in current_settings.globalswitches) Then
- Begin
- LinkLibraryOrder.add('gcc','',15);
- LinkLibraryOrder.add('c','',50); // c and c_p mutual. excl?
- LinkLibraryOrder.add('c_p','',55);
- LinkLibraryOrder.add('pthread','',75); // pthread and c_r should be mutually exclusive
- LinkLibraryOrder.add('c_r','',76);
- LinkLibraryOrder.add('kvm','',80); // must be before ncurses
- if (cs_link_pthread in current_settings.globalswitches) Then // convert libpthread to libc_r.
- LinkLibraryAliases.add('pthread','c_r');
- end;
- end
- else
- begin
- LinkLibraryOrder.add('gcc','',15);
- LinkLibraryOrder.add('c','',50);
- end;
- End;
- procedure TLinkerBSD.InitSysInitUnitName;
- begin
- if target_info.system in systems_darwin then
- SysInitUnit:='sysinit'
- else
- inherited InitSysInitUnitName;
- end;
- function TLinkerBSD.GetDarwinCrt1ObjName(isdll: boolean): TCmdStr;
- begin
- if not isdll then
- begin
- if not(cs_profile in current_settings.moduleswitches) then
- begin
- case target_info.system of
- system_powerpc_darwin,
- system_powerpc64_darwin,
- system_i386_darwin,
- system_x86_64_darwin:
- begin
- { 10.8 and later: no crt1.* }
- if CompareVersionStrings(MacOSXVersionMin,'10.8')>=0 then
- exit('');
- { x86: crt1.10.6.o -> crt1.10.5.o -> crt1.o }
- { others: crt1.10.5 -> crt1.o }
- if (target_info.system in [system_i386_darwin,system_x86_64_darwin]) and
- (CompareVersionStrings(MacOSXVersionMin,'10.6')>=0) then
- exit('crt1.10.6.o');
- if CompareVersionStrings(MacOSXVersionMin,'10.5')>=0 then
- exit('crt1.10.5.o');
- end;
- system_arm_darwin:
- begin
- { iOS:
- iOS 6 and later: nothing
- iOS 3.1 - 5.x: crt1.3.1.o
- pre-iOS 3.1: crt1.o
- }
- if (CompareVersionStrings(iPhoneOSVersionMin,'6.0')>=0) then
- exit('');
- if (CompareVersionStrings(iPhoneOSVersionMin,'3.1')>=0) then
- exit('crt1.3.1.o');
- end;
- system_i386_iphonesim,
- system_x86_64_iphonesim:
- begin
- { "recent versions" must not use anything (https://github.com/llvm-mirror/clang/commit/e6d04f3d152a22077022cf9287d4c538a0918ab0 )
- What those recent versions could be, is anyone's guess. It
- still seems to work with 8.1 and no longer with 8.3, so use
- 8.1 as a cut-off point }
- if (CompareVersionStrings(iPhoneOSVersionMin,'8.1')>0) then
- exit('');
- end;
- system_aarch64_darwin:
- { never anything }
- exit('');
- end;
- { nothing special -> default }
- result:='crt1.o';
- end
- else
- begin
- result:='gcrt1.o';
- { 10.8 and later: tell the linker to use 'start' instead of "_main"
- as entry point }
- if CompareVersionStrings(MacOSXVersionMin,'10.8')>=0 then
- Info.ExeCmd[1]:=Info.ExeCmd[1]+' -no_new_main';
- end;
- end
- else
- begin
- if (apptype=app_bundle) then
- begin
- case target_info.system of
- system_powerpc_darwin,
- system_powerpc64_darwin,
- system_i386_darwin,
- system_x86_64_darwin:
- begin
- { < 10.6: bundle1.o
- >= 10.6: nothing }
- if CompareVersionStrings(MacOSXVersionMin,'10.6')>=0 then
- exit('');
- end;
- system_arm_darwin,
- system_aarch64_darwin:
- begin
- { iOS: < 3.1: bundle1.o
- >= 3.1: nothing }
- if (CompareVersionStrings(iPhoneOSVersionMin,'3.1')>=0) then
- exit('');
- end;
- system_i386_iphonesim,
- system_x86_64_iphonesim:
- begin
- { see rule for crt1.o }
- if (CompareVersionStrings(iPhoneOSVersionMin,'8.1')>0) then
- exit('');
- end;
- end;
- result:='bundle1.o';
- end
- else
- begin
- case target_info.system of
- system_powerpc_darwin,
- system_powerpc64_darwin,
- system_i386_darwin,
- system_x86_64_darwin:
- begin
- { >= 10.6: nothing
- = 10.5: dylib1.10.5.o
- < 10.5: dylib1.o
- }
- if CompareVersionStrings(MacOSXVersionMin,'10.6')>=0 then
- exit('');
- if CompareVersionStrings(MacOSXVersionMin,'10.5')>=0 then
- exit('dylib1.10.5.o');
- end;
- system_arm_darwin,
- system_aarch64_darwin:
- begin
- { iOS: < 3.1: dylib1.o
- >= 3.1: nothing }
- if (CompareVersionStrings(iPhoneOSVersionMin,'3.1')>=0) then
- exit('');
- end;
- system_i386_iphonesim,
- system_x86_64_iphonesim:
- begin
- { see rule for crt1.o }
- if (CompareVersionStrings(iPhoneOSVersionMin,'8.1')>0) then
- exit('');
- end;
- end;
- result:='dylib1.o';
- end;
- end;
- end;
- Function TLinkerBSD.GetDarwinPrtobjName(isdll: boolean): TCmdStr;
- var
- startupfile: TCmdStr;
- begin
- result:='';
- startupfile:=GetDarwinCrt1ObjName(isdll);
- if startupfile<>'' then
- begin
- if not librarysearchpath.FindFile(startupfile,false,result) then
- result:='/usr/lib/'+startupfile;
- end;
- result:=maybequoted(result);
- end;
- Function TLinkerBSD.WriteResponseFile(isdll:boolean) : Boolean;
- Var
- linkres : TLinkRes;
- FilesList : TLinkRes;
- i : longint;
- cprtobj,
- gprtobj,
- prtobj : string[80];
- HPath : TCmdStrListItem;
- s,s1,s2 : TCmdStr;
- linkdynamic,
- linklibc : boolean;
- Fl1,Fl2 : Boolean;
- IsDarwin : Boolean;
- ReOrder : Boolean;
- begin
- WriteResponseFile:=False;
- ReOrder:=False;
- linkdynamic:=False;
- IsDarwin:=target_info.system in systems_darwin;
- { set special options for some targets }
- if not IsDarwin Then
- begin
- if isdll and
- (target_info.system in systems_bsd) then
- begin
- prtobj:='dllprt0';
- cprtobj:='dllprt0';
- gprtobj:='dllprt0';
- end
- else
- begin
- prtobj:='prt0';
- cprtobj:='cprt0';
- gprtobj:='gprt0';
- end;
- linkdynamic:=not(SharedLibFiles.empty);
- linklibc:=(SharedLibFiles.Find('c')<>nil);
- // this one is a bit complex.
- // Only reorder for now if -XL or -XO params are given
- // or when -Xf.
- reorder:= linklibc and
- (
- ReorderEntries
- or
- (cs_link_pthread in current_settings.globalswitches));
- if cs_profile in current_settings.moduleswitches then
- begin
- prtobj:=gprtobj;
- AddSharedLibrary('c');
- LibrarySuffix:='p';
- linklibc:=true;
- end
- else
- begin
- if linklibc then
- prtobj:=cprtobj;
- end;
- // after this point addition of shared libs not allowed.
- end
- else
- begin
- { for darwin: always link dynamically against libc }
- linklibc := true;
- {$ifdef MACOSX104ORHIGHER}
- { not sure what this is for, but gcc always links against it }
- if not(cs_profile in current_settings.moduleswitches) then
- AddSharedLibrary('SystemStubs')
- else
- AddSharedLibrary('SystemStubs_profile');
- {$endif MACOSX104ORHIGHER}
- reorder:=reorderentries;
- prtobj:='';
- end;
- if reorder Then
- ExpandAndApplyOrder(SharedLibFiles);
- { Open link.res file }
- LinkRes:=TLinkRes.Create(outputexedir+Info.ResName,not LdSupportsNoResponseFile);
- if (target_info.system in systems_darwin) and
- (sysrootpath<>'') then
- begin
- LinkRes.Add('-syslibroot');
- LinkRes.Add(sysrootpath);
- end;
- if (target_info.system in systems_darwin) then
- begin
- LinkRes.Add('-arch');
- case target_info.system of
- system_powerpc_darwin:
- LinkRes.Add('ppc');
- system_i386_darwin,
- system_i386_iphonesim:
- LinkRes.Add('i386');
- system_powerpc64_darwin:
- LinkRes.Add('ppc64');
- system_x86_64_darwin,
- system_x86_64_iphonesim:
- LinkRes.Add('x86_64');
- system_arm_darwin:
- { current versions of the linker require the sub-architecture type
- to be specified }
- LinkRes.Add(lower(cputypestr[current_settings.cputype]));
- system_aarch64_darwin:
- LinkRes.Add('arm64');
- else
- internalerror(2014121801);
- end;
- if MacOSXVersionMin<>'' then
- begin
- LinkRes.Add('-macosx_version_min');
- LinkRes.Add(MacOSXVersionMin);
- end
- else if iPhoneOSVersionMin<>'' then
- begin
- LinkRes.Add('-iphoneos_version_min');
- LinkRes.Add(iPhoneOSVersionMin);
- end;
- end;
- { Write path to search libraries }
- HPath:=TCmdStrListItem(current_module.locallibrarysearchpath.First);
- while assigned(HPath) do
- begin
- if LdSupportsNoResponseFile then
- LinkRes.Add('-L'+HPath.Str)
- else
- LinkRes.Add('SEARCH_DIR("'+HPath.Str+'")');
- HPath:=TCmdStrListItem(HPath.Next);
- end;
- HPath:=TCmdStrListItem(LibrarySearchPath.First);
- while assigned(HPath) do
- begin
- if LdSupportsNoResponseFile then
- LinkRes.Add('-L'+HPath.Str)
- else
- LinkRes.Add('SEARCH_DIR("'+HPath.Str+'")');
- HPath:=TCmdStrListItem(HPath.Next);
- end;
- if (target_info.system in systems_darwin) then
- begin
- HPath:=TCmdStrListItem(current_module.localframeworksearchpath.First);
- while assigned(HPath) do
- begin
- LinkRes.Add('-F'+HPath.Str);
- HPath:=TCmdStrListItem(HPath.Next);
- end;
- HPath:=TCmdStrListItem(FrameworkSearchPath.First);
- while assigned(HPath) do
- begin
- LinkRes.Add('-F'+HPath.Str);
- HPath:=TCmdStrListItem(HPath.Next);
- end;
- end;
- { force local symbol resolution (i.e., inside the shared }
- { library itself) for all non-exorted symbols, otherwise }
- { several RTL symbols of FPC-compiled shared libraries }
- { will be bound to those of a single shared library or }
- { to the main program }
- if (isdll) and (target_info.system in systems_bsd) then
- begin
- LinkRes.add('VERSION');
- LinkRes.add('{');
- LinkRes.add(' {');
- if not texportlibunix(exportlib).exportedsymnames.empty then
- begin
- LinkRes.add(' global:');
- repeat
- LinkRes.add(' '+texportlibunix(exportlib).exportedsymnames.getfirst+';');
- until texportlibunix(exportlib).exportedsymnames.empty;
- end;
- LinkRes.add(' local:');
- LinkRes.add(' *;');
- LinkRes.add(' };');
- LinkRes.add('}');
- end;
- if not LdSupportsNoResponseFile then
- LinkRes.Add('INPUT(');
- { add objectfiles, start with prt0 always }
- if prtobj<>'' then
- LinkRes.AddFileName(FindObjectFile(prtobj,'',false));
- { try to add crti and crtbegin if linking to C }
- if linklibc and
- not IsDarwin Then
- begin
- if librarysearchpath.FindFile('crti.o',false,s) then
- LinkRes.AddFileName(s);
- if cs_create_pic in current_settings.moduleswitches then
- begin
- if librarysearchpath.FindFile('crtbeginS.o',false,s) then
- LinkRes.AddFileName(s);
- end
- else
- if (cs_link_staticflag in current_settings.globalswitches) and
- librarysearchpath.FindFile('crtbeginT.o',false,s) then
- LinkRes.AddFileName(s)
- else if librarysearchpath.FindFile('crtbegin.o',false,s) then
- LinkRes.AddFileName(s);
- end;
- { main objectfiles }
- { Generate linkfiles.res file if needed }
- { Only needed on Windows, due to the limitation of 8196 characters for command line }
- if (LdSupportsNoResponseFile) and
- (source_info.system in systems_all_windows) then
- begin
- FilesList:=TLinkRes.Create(outputexedir+'linkfiles.res',false);
- while not ObjectFiles.Empty do
- begin
- s:=ObjectFiles.GetFirst;
- if s<>'' then
- begin
- repeat
- i:=Pos(source_info.dirsep,s);
- if i>0 then
- s[i]:=target_info.dirsep;
- until i=0;
- FilesList.Add(s);
- end;
- end;
- FilesList.writetodisk;
- FilesList.Free;
- end
- else
- begin
- while not ObjectFiles.Empty do
- begin
- s:=ObjectFiles.GetFirst;
- if s<>'' then
- if LdSupportsNoResponseFile then
- LinkRes.AddFileName(s)
- else
- LinkRes.AddFileName(maybequoted(s));
- end;
- end;
- if not LdSupportsNoResponseFile then
- LinkRes.Add(')');
- { Write staticlibraries }
- if not StaticLibFiles.Empty then
- begin
- if not LdSupportsNoResponseFile then
- LinkRes.Add('GROUP(');
- While not StaticLibFiles.Empty do
- begin
- S:=StaticLibFiles.GetFirst;
- if LdSupportsNoResponseFile then
- LinkRes.AddFileName(s)
- else
- LinkRes.AddFileName(maybequoted(s))
- end;
- if not LdSupportsNoResponseFile then
- LinkRes.Add(')');
- end;
- { Write sharedlibraries like -l<lib>, also add the needed dynamic linker
- here to be sure that it gets linked this is needed for glibc2 systems (PFV) }
- if not SharedLibFiles.Empty then
- begin
- if not LdSupportsNoResponseFile then
- LinkRes.Add('INPUT(');
- While not SharedLibFiles.Empty do
- begin
- S:=SharedLibFiles.GetFirst;
- if (s<>'c') or reorder then
- begin
- i:=Pos(target_info.sharedlibext,S);
- if i>0 then
- Delete(S,i,255);
- LinkRes.Add('-l'+s);
- end
- else
- begin
- linklibc:=true;
- linkdynamic:=false; { libc will include the ld-* for us }
- end;
- end;
- { be sure that libc is the last lib }
- if linklibc and not reorder then
- Begin
- If LibrarySuffix=' ' Then
- LinkRes.Add('-lc')
- else
- LinkRes.Add('-lc_'+LibrarySuffix);
- If LibrarySuffix='r' Then
- LinkRes.Add('-lc');
- end;
- { when we have -static for the linker the we also need libgcc }
- if (cs_link_staticflag in current_settings.globalswitches) then
- LinkRes.Add('-lgcc');
- if linkdynamic and (Info.DynamicLinker<>'') and
- not(target_info.system in systems_openbsd) then
- LinkRes.AddFileName(Info.DynamicLinker);
- if not LdSupportsNoResponseFile then
- LinkRes.Add(')');
- end;
- { frameworks for Darwin }
- if IsDarwin then
- while not FrameworkFiles.empty do
- begin
- LinkRes.Add('-framework');
- LinkRes.Add(FrameworkFiles.GetFirst);
- end;
- { objects which must be at the end }
- if linklibc and
- not IsDarwin Then
- begin
- if cs_create_pic in current_settings.moduleswitches then
- Fl1:=librarysearchpath.FindFile('crtendS.o',false,s1)
- else
- Fl1:=librarysearchpath.FindFile('crtend.o',false,s1);
- Fl2:=librarysearchpath.FindFile('crtn.o',false,s2);
- if Fl1 or Fl2 then
- begin
- LinkRes.Add('INPUT(');
- If Fl1 Then
- LinkRes.AddFileName(s1);
- If Fl2 Then
- LinkRes.AddFileName(s2);
- LinkRes.Add(')');
- end;
- end;
- { Write and Close response }
- linkres.writetodisk;
- linkres.Free;
- WriteResponseFile:=True;
- end;
- function TLinkerBSD.MakeExecutable:boolean;
- var
- binstr,
- cmdstr,
- targetstr,
- emulstr,
- extdbgbinstr,
- extdbgcmdstr: TCmdStr;
- linkscript: TAsmScript;
- DynLinkStr : string[60];
- GCSectionsStr,
- StaticStr,
- StripStr : string[63];
- success,
- useshell : boolean;
- begin
- if not(cs_link_nolink in current_settings.globalswitches) then
- Message1(exec_i_linking,current_module.exefilename);
- { Create some replacements }
- StaticStr:='';
- StripStr:='';
- DynLinkStr:='';
- GCSectionsStr:='';
- linkscript:=nil;
- { i386_freebsd needs -b elf32-i386-freebsd and -m elf_i386_fbsd
- to avoid creation of a i386:x86_64 arch binary }
- if target_info.system=system_i386_freebsd then
- begin
- targetstr:='-b elf32-i386-freebsd';
- emulstr:='-m elf_i386_fbsd';
- end
- else
- begin
- targetstr:='';
- emulstr:='';
- end;
- if (cs_link_staticflag in current_settings.globalswitches) then
- begin
- if (target_info.system=system_m68k_netbsd) and
- ((cs_link_on_target in current_settings.globalswitches) or
- (target_info.system=source_info.system)) then
- StaticStr:='-Bstatic'
- else
- StaticStr:='-static';
- end;
- if (cs_link_strip in current_settings.globalswitches) then
- if (target_info.system in systems_darwin) then
- StripStr:='-x'
- else
- StripStr:='-s';
- if (cs_link_smart in current_settings.globalswitches) and
- (tf_smartlink_sections in target_info.flags) then
- if not(target_info.system in systems_darwin) then
- GCSectionsStr:='--gc-sections'
- else
- GCSectionsStr:='-dead_strip -no_dead_strip_inits_and_terms';
- if(not(target_info.system in systems_darwin) and
- (cs_profile in current_settings.moduleswitches)) or
- ((Info.DynamicLinker<>'') and (not SharedLibFiles.Empty)) then
- DynLinkStr:='-dynamic-linker='+Info.DynamicLinker;
- if CShared Then
- begin
- if not(target_info.system in systems_darwin) then
- DynLinKStr:=DynLinkStr+' --shared'
- else
- DynLinKStr:=DynLinkStr+' -dynamic'; // one dash!
- end;
- { Use -nopie on OpenBSD if PIC support is turned off }
- if (target_info.system in systems_openbsd) and
- not(cs_create_pic in current_settings.moduleswitches) then
- Info.ExtraOptions:=Info.ExtraOptions+' -nopie';
- { -N seems to be needed on NetBSD/earm }
- if (target_info.system in [system_arm_netbsd]) then
- Info.ExtraOptions:=Info.ExtraOptions+' -N';
- { Write used files and libraries }
- WriteResponseFile(false);
- { Call linker }
- SplitBinCmd(Info.ExeCmd[1],binstr,cmdstr);
- Replace(cmdstr,'$EXE',maybequoted(current_module.exefilename));
- Replace(cmdstr,'$OPT',Info.ExtraOptions);
- Replace(cmdstr,'$TARGET',targetstr);
- Replace(cmdstr,'$EMUL',EmulStr);
- Replace(cmdstr,'$CATRES',CatFileContent(outputexedir+Info.ResName));
- Replace(cmdstr,'$RES',maybequoted(outputexedir+Info.ResName));
- if (LdSupportsNoResponseFile) and (source_info.system in systems_all_windows) then
- Replace(cmdstr,'$FILELIST','-filelist '+maybequoted(outputexedir+'linkfiles.res'))
- else
- Replace(cmdstr,'$FILELIST','');
- Replace(cmdstr,'$STATIC',StaticStr);
- Replace(cmdstr,'$STRIP',StripStr);
- Replace(cmdstr,'$GCSECTIONS',GCSectionsStr);
- Replace(cmdstr,'$DYNLINK',DynLinkStr);
- if (target_info.system in systems_darwin) then
- Replace(cmdstr,'$PRTOBJ',GetDarwinPrtobjName(false));
- BinStr:=FindUtil(utilsprefix+BinStr);
- { create dsym file? }
- extdbgbinstr:='';
- extdbgcmdstr:='';
- if (target_info.system in systems_darwin) and
- (target_dbg.id in [dbg_dwarf2,dbg_dwarf3]) and
- (cs_link_separate_dbg_file in current_settings.globalswitches) then
- begin
- extdbgbinstr:=FindUtil(utilsprefix+'dsymutil');
- extdbgcmdstr:=maybequoted(current_module.exefilename);
- end;
- if (LdSupportsNoResponseFile) and
- not(cs_link_nolink in current_settings.globalswitches) then
- begin
- { we have to use a script to use the IFS hack }
- linkscript:=GenerateScript(outputexedir+'ppaslink');
- linkscript.AddLinkCommand(BinStr,CmdStr,'');
- if (extdbgcmdstr<>'') 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;
- useshell:=not (tf_no_backquote_support in source_info.flags);
- success:=DoExec(BinStr,CmdStr,true,LdSupportsNoResponseFile or useshell);
- if (success and
- (extdbgbinstr<>'') and
- (cs_link_nolink in current_settings.globalswitches)) then
- success:=DoExec(extdbgbinstr,extdbgcmdstr,false,LdSupportsNoResponseFile);
- { Remove ReponseFile }
- if (success) and not(cs_link_nolink in current_settings.globalswitches) then
- begin
- DeleteFile(outputexedir+Info.ResName);
- if LdSupportsNoResponseFile Then
- begin
- DeleteFile(linkscript.fn);
- linkscript.free
- end;
- end;
- { Remove linkfiles.res }
- if (success) and (LdSupportsNoResponseFile) and (source_info.system in systems_all_windows) then
- DeleteFile(outputexedir+'linkfiles.res');
- MakeExecutable:=success; { otherwise a recursive call to link method }
- end;
- Function TLinkerBSD.MakeSharedLibrary:boolean;
- var
- InitStr,
- FiniStr,
- SoNameStr : string[80];
- linkscript: TAsmScript;
- binstr,
- cmdstr,
- targetstr,
- emulstr,
- extdbgbinstr,
- extdbgcmdstr : TCmdStr;
- GCSectionsStr : string[63];
- exportedsyms: text;
- success : boolean;
- begin
- MakeSharedLibrary:=false;
- GCSectionsStr:='';
- linkscript:=nil;
- if not(cs_link_nolink in current_settings.globalswitches) then
- Message1(exec_i_linking,current_module.sharedlibfilename);
- { Write used files and libraries }
- WriteResponseFile(true);
- if (cs_link_smart in current_settings.globalswitches) and
- (tf_smartlink_sections in target_info.flags) then
- if not(target_info.system in systems_darwin) then
- { disabled because not tested
- GCSectionsStr:='--gc-sections' }
- else
- GCSectionsStr:='-dead_strip -no_dead_strip_inits_and_terms';
- { i386_freebsd needs -b elf32-i386-freebsd and -m elf_i386_fbsd
- to avoid creation of a i386:x86_64 arch binary }
- if target_info.system=system_i386_freebsd then
- begin
- targetstr:='-b elf32-i386-freebsd';
- emulstr:='-m elf_i386_fbsd';
- end
- else
- begin
- targetstr:='';
- emulstr:='';
- end;
- InitStr:='-init FPC_LIB_START';
- FiniStr:='-fini FPC_LIB_EXIT';
- SoNameStr:='-soname '+ExtractFileName(current_module.sharedlibfilename);
- { Call linker }
- SplitBinCmd(Info.DllCmd[1],binstr,cmdstr);
- {$ifndef darwin}
- Replace(cmdstr,'$EXE',maybequoted(current_module.sharedlibfilename));
- {$else darwin}
- Replace(cmdstr,'$EXE',maybequoted(ExpandFileName(current_module.sharedlibfilename)));
- {$endif darwin}
- Replace(cmdstr,'$OPT',Info.ExtraOptions);
- Replace(cmdstr,'$TARGET',targetstr);
- Replace(cmdstr,'$EMUL',EmulStr);
- Replace(cmdstr,'$CATRES',CatFileContent(outputexedir+Info.ResName));
- if (LdSupportsNoResponseFile) and (source_info.system in systems_all_windows) then
- Replace(cmdstr,'$FILELIST','-filelist '+maybequoted(outputexedir+'linkfiles.res'))
- else
- Replace(cmdstr,'$FILELIST','');
- Replace(cmdstr,'$RES',maybequoted(outputexedir+Info.ResName));
- Replace(cmdstr,'$INIT',InitStr);
- Replace(cmdstr,'$FINI',FiniStr);
- Replace(cmdstr,'$GCSECTIONS',GCSectionsStr);
- Replace(cmdstr,'$SONAME',SoNameStr);
- if (target_info.system in systems_darwin) then
- Replace(cmdstr,'$PRTOBJ',GetDarwinPrtobjName(true));
- BinStr:=FindUtil(utilsprefix+BinStr);
- { create dsym file? }
- extdbgbinstr:='';
- extdbgcmdstr:='';
- if (target_info.system in systems_darwin) and
- (target_dbg.id in [dbg_dwarf2,dbg_dwarf3]) and
- (cs_link_separate_dbg_file in current_settings.globalswitches) then
- begin
- extdbgbinstr:=FindUtil(utilsprefix+'dsymutil');
- extdbgcmdstr:=maybequoted(current_module.sharedlibfilename);
- end;
- if (target_info.system in systems_darwin) then
- begin
- { exported symbols for darwin }
- 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+' -exported_symbols_list '+maybequoted(outputexedir)+'linksyms.fpc';
- end;
- end;
- if (LdSupportsNoResponseFile) and
- not(cs_link_nolink in current_settings.globalswitches) 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,LdSupportsNoResponseFile);
- if (success and
- (extdbgbinstr<>'') and
- (cs_link_nolink in current_settings.globalswitches)) then
- success:=DoExec(extdbgbinstr,extdbgcmdstr,false,LdSupportsNoResponseFile);
- { Strip the library ? }
- if success and (cs_link_strip in current_settings.globalswitches) then
- begin
- SplitBinCmd(Info.DllCmd[2],binstr,cmdstr);
- Replace(cmdstr,'$EXE',maybequoted(current_module.sharedlibfilename));
- success:=DoExec(FindUtil(utilsprefix+binstr),cmdstr,false,false);
- end;
- { Remove ReponseFile }
- if (success) and not(cs_link_nolink in current_settings.globalswitches) then
- begin
- DeleteFile(outputexedir+Info.ResName);
- if LdSupportsNoResponseFile Then
- begin
- DeleteFile(linkscript.fn);
- linkscript.free
- end;
- if (target_info.system in systems_darwin) then
- DeleteFile(outputexedir+'linksyms.fpc');
- end;
- { Remove linkfiles.res }
- if (success) and (LdSupportsNoResponseFile) and (source_info.system in systems_all_windows) then
- DeleteFile(outputexedir+'linkfiles.res');
- MakeSharedLibrary:=success; { otherwise a recursive call to link method }
- end;
- {*****************************************************************************
- Initialize
- *****************************************************************************}
- initialization
- RegisterLinker(ld_bsd,TLinkerBSD);
- {$ifdef x86_64}
- RegisterImport(system_x86_64_dragonfly,timportlibbsd);
- RegisterExport(system_x86_64_dragonfly,texportlibbsd);
- RegisterTarget(system_x86_64_dragonfly_info);
- RegisterImport(system_x86_64_freebsd,timportlibbsd);
- RegisterExport(system_x86_64_freebsd,texportlibbsd);
- RegisterTarget(system_x86_64_freebsd_info);
- RegisterImport(system_x86_64_openbsd,timportlibbsd);
- RegisterExport(system_x86_64_openbsd,texportlibbsd);
- RegisterTarget(system_x86_64_openbsd_info);
- RegisterImport(system_x86_64_netbsd,timportlibbsd);
- RegisterExport(system_x86_64_netbsd,texportlibbsd);
- RegisterTarget(system_x86_64_netbsd_info);
- RegisterImport(system_x86_64_darwin,timportlibdarwin);
- RegisterExport(system_x86_64_darwin,texportlibdarwin);
- RegisterTarget(system_x86_64_darwin_info);
- RegisterImport(system_x86_64_iphonesim,timportlibdarwin);
- RegisterExport(system_x86_64_iphonesim,texportlibdarwin);
- RegisterTarget(system_x86_64_iphonesim_info);
- {$endif}
- {$ifdef i386}
- RegisterImport(system_i386_freebsd,timportlibbsd);
- RegisterExport(system_i386_freebsd,texportlibbsd);
- RegisterTarget(system_i386_freebsd_info);
- RegisterImport(system_i386_netbsd,timportlibbsd);
- RegisterExport(system_i386_netbsd,texportlibbsd);
- RegisterTarget(system_i386_netbsd_info);
- RegisterImport(system_i386_openbsd,timportlibbsd);
- RegisterExport(system_i386_openbsd,texportlibbsd);
- RegisterTarget(system_i386_openbsd_info);
- RegisterImport(system_i386_darwin,timportlibdarwin);
- RegisterExport(system_i386_darwin,texportlibdarwin);
- RegisterTarget(system_i386_darwin_info);
- RegisterImport(system_i386_iphonesim,timportlibdarwin);
- RegisterExport(system_i386_iphonesim,texportlibdarwin);
- RegisterTarget(system_i386_iphonesim_info);
- {$endif i386}
- {$ifdef m68k}
- RegisterImport(system_m68k_netbsd,timportlibbsd);
- RegisterExport(system_m68k_netbsd,texportlibbsd);
- RegisterTarget(system_m68k_netbsd_info);
- {$endif m68k}
- {$ifdef powerpc}
- RegisterImport(system_powerpc_darwin,timportlibdarwin);
- RegisterExport(system_powerpc_darwin,texportlibdarwin);
- RegisterTarget(system_powerpc_darwin_info);
- RegisterImport(system_powerpc_netbsd,timportlibbsd);
- RegisterExport(system_powerpc_netbsd,texportlibbsd);
- RegisterTarget(system_powerpc_netbsd_info);
- {$endif powerpc}
- {$ifdef powerpc64}
- RegisterImport(system_powerpc64_darwin,timportlibdarwin);
- RegisterExport(system_powerpc64_darwin,texportlibdarwin);
- RegisterTarget(system_powerpc64_darwin_info);
- {$endif powerpc64}
- {$ifdef arm}
- RegisterImport(system_arm_darwin,timportlibdarwin);
- RegisterExport(system_arm_darwin,texportlibdarwin);
- RegisterTarget(system_arm_darwin_info);
- RegisterImport(system_arm_netbsd,timportlibbsd);
- RegisterExport(system_arm_netbsd,texportlibbsd);
- RegisterTarget(system_arm_netbsd_info);
- {$endif arm}
- {$ifdef aarch64}
- RegisterImport(system_aarch64_darwin,timportlibdarwin);
- RegisterExport(system_aarch64_darwin,texportlibdarwin);
- RegisterTarget(system_aarch64_darwin_info);
- {$endif aarch64}
- RegisterRes(res_elf_info,TWinLikeResourceFile);
- RegisterRes(res_macho_info,TWinLikeResourceFile);
- end.
|