This patch introduces support for Integrated Distributed ThinLTO (DTLTO) in ELF LLD. DTLTO enables the distribution of ThinLTO backend compilations via external distribution systems, such as Incredibuild, during the traditional link step: https://llvm.org/docs/DTLTO.html. It is expected that users will invoke DTLTO through the compiler driver (e.g., Clang) rather than calling LLD directly. A Clang-side interface for DTLTO will be added in a follow-up patch. Note: Bitcode members of archives (thin or non-thin) are not currently supported. This will be addressed in a future change. As a consequence of this lack of support, this patch is not sufficient to allow for self-hosting an LLVM build with DTLTO. Theoretically, --start-lib/--end-lib could be used instead of archives in a self-host build. However, it's unclear how --start-lib/--end-lib can be easily used with the LLVM build system. Testing: - ELF LLD `lit` test coverage has been added, using a mock distributor to avoid requiring Clang. - Cross-project `lit` tests cover integration with Clang. For the design discussion of the DTLTO feature, see: #126654.
802 lines
25 KiB
C++
802 lines
25 KiB
C++
//===- Config.h -------------------------------------------------*- C++ -*-===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLD_ELF_CONFIG_H
|
|
#define LLD_ELF_CONFIG_H
|
|
|
|
#include "lld/Common/CommonLinkerContext.h"
|
|
#include "lld/Common/ErrorHandler.h"
|
|
#include "llvm/ADT/CachedHashString.h"
|
|
#include "llvm/ADT/DenseSet.h"
|
|
#include "llvm/ADT/MapVector.h"
|
|
#include "llvm/ADT/SetVector.h"
|
|
#include "llvm/ADT/SmallSet.h"
|
|
#include "llvm/ADT/StringRef.h"
|
|
#include "llvm/ADT/StringSet.h"
|
|
#include "llvm/BinaryFormat/ELF.h"
|
|
#include "llvm/Option/ArgList.h"
|
|
#include "llvm/Support/CachePruning.h"
|
|
#include "llvm/Support/CodeGen.h"
|
|
#include "llvm/Support/Compiler.h"
|
|
#include "llvm/Support/Compression.h"
|
|
#include "llvm/Support/Endian.h"
|
|
#include "llvm/Support/FileSystem.h"
|
|
#include "llvm/Support/GlobPattern.h"
|
|
#include "llvm/Support/TarWriter.h"
|
|
#include <atomic>
|
|
#include <memory>
|
|
#include <mutex>
|
|
#include <optional>
|
|
#include <vector>
|
|
|
|
namespace lld::elf {
|
|
|
|
class InputFile;
|
|
class BinaryFile;
|
|
class BitcodeFile;
|
|
class ELFFileBase;
|
|
class SharedFile;
|
|
class InputSectionBase;
|
|
class EhInputSection;
|
|
class Defined;
|
|
class Undefined;
|
|
class Symbol;
|
|
class SymbolTable;
|
|
class BitcodeCompiler;
|
|
class OutputSection;
|
|
class LinkerScript;
|
|
class TargetInfo;
|
|
struct Ctx;
|
|
struct Partition;
|
|
struct PhdrEntry;
|
|
|
|
class BssSection;
|
|
class GdbIndexSection;
|
|
class GotPltSection;
|
|
class GotSection;
|
|
class IBTPltSection;
|
|
class IgotPltSection;
|
|
class InputSection;
|
|
class IpltSection;
|
|
class MipsGotSection;
|
|
class MipsRldMapSection;
|
|
class PPC32Got2Section;
|
|
class PPC64LongBranchTargetSection;
|
|
class PltSection;
|
|
class RelocationBaseSection;
|
|
class RelroPaddingSection;
|
|
class StringTableSection;
|
|
class SymbolTableBaseSection;
|
|
class SymtabShndxSection;
|
|
class SyntheticSection;
|
|
|
|
enum ELFKind : uint8_t {
|
|
ELFNoneKind,
|
|
ELF32LEKind,
|
|
ELF32BEKind,
|
|
ELF64LEKind,
|
|
ELF64BEKind
|
|
};
|
|
|
|
// For -Bno-symbolic, -Bsymbolic-non-weak-functions, -Bsymbolic-functions,
|
|
// -Bsymbolic-non-weak, -Bsymbolic.
|
|
enum class BsymbolicKind { None, NonWeakFunctions, Functions, NonWeak, All };
|
|
|
|
// For --build-id.
|
|
enum class BuildIdKind { None, Fast, Md5, Sha1, Hexstring, Uuid };
|
|
|
|
// For --call-graph-profile-sort={none,hfsort,cdsort}.
|
|
enum class CGProfileSortKind { None, Hfsort, Cdsort };
|
|
|
|
// For --discard-{all,locals,none}.
|
|
enum class DiscardPolicy { Default, All, Locals, None };
|
|
|
|
// For --icf={none,safe,all}.
|
|
enum class ICFLevel { None, Safe, All };
|
|
|
|
// For --strip-{all,debug}.
|
|
enum class StripPolicy { None, All, Debug };
|
|
|
|
// For --unresolved-symbols.
|
|
enum class UnresolvedPolicy { ReportError, Warn, Ignore };
|
|
|
|
// For --orphan-handling.
|
|
enum class OrphanHandlingPolicy { Place, Warn, Error };
|
|
|
|
// For --sort-section and linkerscript sorting rules.
|
|
enum class SortSectionPolicy {
|
|
Default,
|
|
None,
|
|
Alignment,
|
|
Name,
|
|
Priority,
|
|
Reverse,
|
|
};
|
|
|
|
// For --target2
|
|
enum class Target2Policy { Abs, Rel, GotRel };
|
|
|
|
// For tracking ARM Float Argument PCS
|
|
enum class ARMVFPArgKind { Default, Base, VFP, ToolChain };
|
|
|
|
// For -z noseparate-code, -z separate-code and -z separate-loadable-segments.
|
|
enum class SeparateSegmentKind { None, Code, Loadable };
|
|
|
|
// For -z *stack
|
|
enum class GnuStackKind { None, Exec, NoExec };
|
|
|
|
// For --lto=
|
|
enum LtoKind : uint8_t {UnifiedThin, UnifiedRegular, Default};
|
|
|
|
// For -z gcs=
|
|
enum class GcsPolicy { Implicit, Never, Always };
|
|
|
|
// For -z zicfilp=
|
|
enum class ZicfilpPolicy { Implicit, Never, Unlabeled, FuncSig };
|
|
|
|
// For -z zicfiss=
|
|
enum class ZicfissPolicy { Implicit, Never, Always };
|
|
|
|
// For some options that resemble -z bti-report={none,warning,error}
|
|
enum class ReportPolicy { None, Warning, Error };
|
|
|
|
// Describes the signing schema for a file using the PAuth ABI extension.
|
|
// Two files are considered compatible when both `platform` and `version` match.
|
|
// The pair (0, 0) is reserved to indicate incompatibility with the PAuth ABI.
|
|
struct AArch64PauthAbiCoreInfo {
|
|
uint64_t platform;
|
|
uint64_t version;
|
|
// Returns true if the core info is not the reserved (0, 0) value.
|
|
bool isValid() const { return platform || version; }
|
|
static constexpr size_t size() { return sizeof(platform) + sizeof(version); }
|
|
bool operator==(const AArch64PauthAbiCoreInfo &other) const {
|
|
return platform == other.platform && version == other.version;
|
|
}
|
|
bool operator!=(const AArch64PauthAbiCoreInfo &other) const {
|
|
return !(*this == other);
|
|
}
|
|
};
|
|
|
|
struct SymbolVersion {
|
|
llvm::StringRef name;
|
|
bool isExternCpp;
|
|
bool hasWildcard;
|
|
};
|
|
|
|
// This struct contains symbols version definition that
|
|
// can be found in version script if it is used for link.
|
|
struct VersionDefinition {
|
|
llvm::StringRef name;
|
|
uint16_t id;
|
|
SmallVector<SymbolVersion, 0> nonLocalPatterns;
|
|
SmallVector<SymbolVersion, 0> localPatterns;
|
|
};
|
|
|
|
class LinkerDriver {
|
|
public:
|
|
LinkerDriver(Ctx &ctx);
|
|
LinkerDriver(LinkerDriver &) = delete;
|
|
void linkerMain(ArrayRef<const char *> args);
|
|
void addFile(StringRef path, bool withLOption);
|
|
void addLibrary(StringRef name);
|
|
|
|
private:
|
|
Ctx &ctx;
|
|
void createFiles(llvm::opt::InputArgList &args);
|
|
void inferMachineType();
|
|
template <class ELFT> void link(llvm::opt::InputArgList &args);
|
|
template <class ELFT> void compileBitcodeFiles(bool skipLinkedOutput);
|
|
bool tryAddFatLTOFile(MemoryBufferRef mb, StringRef archiveName,
|
|
uint64_t offsetInArchive, bool lazy);
|
|
// True if we are in --whole-archive and --no-whole-archive.
|
|
bool inWholeArchive = false;
|
|
|
|
// True if we are in --start-lib and --end-lib.
|
|
bool inLib = false;
|
|
|
|
std::unique_ptr<BitcodeCompiler> lto;
|
|
SmallVector<std::unique_ptr<InputFile>, 0> files, ltoObjectFiles;
|
|
|
|
public:
|
|
// See InputFile::groupId.
|
|
uint32_t nextGroupId;
|
|
bool isInGroup;
|
|
std::unique_ptr<InputFile> armCmseImpLib;
|
|
SmallVector<std::pair<StringRef, unsigned>, 0> archiveFiles;
|
|
};
|
|
|
|
// This struct contains the global configuration for the linker.
|
|
// Most fields are direct mapping from the command line options
|
|
// and such fields have the same name as the corresponding options.
|
|
// Most fields are initialized by the ctx.driver.
|
|
struct Config {
|
|
uint8_t osabi = 0;
|
|
uint32_t andFeatures = 0;
|
|
llvm::CachePruningPolicy thinLTOCachePolicy;
|
|
llvm::SetVector<llvm::CachedHashString> dependencyFiles; // for --dependency-file
|
|
llvm::StringMap<uint64_t> sectionStartMap;
|
|
llvm::StringRef bfdname;
|
|
llvm::StringRef chroot;
|
|
llvm::StringRef dependencyFile;
|
|
llvm::StringRef dwoDir;
|
|
llvm::StringRef dynamicLinker;
|
|
llvm::StringRef entry;
|
|
llvm::StringRef emulation;
|
|
llvm::StringRef fini;
|
|
llvm::StringRef init;
|
|
llvm::StringRef ltoAAPipeline;
|
|
llvm::StringRef ltoCSProfileFile;
|
|
llvm::StringRef ltoNewPmPasses;
|
|
llvm::StringRef ltoObjPath;
|
|
llvm::StringRef ltoSampleProfile;
|
|
llvm::StringRef mapFile;
|
|
llvm::StringRef outputFile;
|
|
llvm::StringRef optRemarksFilename;
|
|
std::optional<uint64_t> optRemarksHotnessThreshold = 0;
|
|
llvm::StringRef optRemarksPasses;
|
|
llvm::StringRef optRemarksFormat;
|
|
llvm::StringRef optStatsFilename;
|
|
llvm::StringRef progName;
|
|
llvm::StringRef printArchiveStats;
|
|
llvm::StringRef printSymbolOrder;
|
|
llvm::StringRef soName;
|
|
llvm::StringRef sysroot;
|
|
llvm::StringRef thinLTOCacheDir;
|
|
llvm::StringRef thinLTOIndexOnlyArg;
|
|
llvm::StringRef whyExtract;
|
|
llvm::SmallVector<llvm::GlobPattern, 0> whyLive;
|
|
llvm::StringRef cmseInputLib;
|
|
llvm::StringRef cmseOutputLib;
|
|
ReportPolicy zBtiReport = ReportPolicy::None;
|
|
ReportPolicy zCetReport = ReportPolicy::None;
|
|
ReportPolicy zPauthReport = ReportPolicy::None;
|
|
ReportPolicy zGcsReport = ReportPolicy::None;
|
|
ReportPolicy zGcsReportDynamic = ReportPolicy::None;
|
|
ReportPolicy zExecuteOnlyReport = ReportPolicy::None;
|
|
ReportPolicy zZicfilpUnlabeledReport = ReportPolicy::None;
|
|
ReportPolicy zZicfilpFuncSigReport = ReportPolicy::None;
|
|
ReportPolicy zZicfissReport = ReportPolicy::None;
|
|
bool ltoBBAddrMap;
|
|
llvm::StringRef ltoBasicBlockSections;
|
|
std::pair<llvm::StringRef, llvm::StringRef> thinLTOObjectSuffixReplace;
|
|
llvm::StringRef thinLTOPrefixReplaceOld;
|
|
llvm::StringRef thinLTOPrefixReplaceNew;
|
|
llvm::StringRef thinLTOPrefixReplaceNativeObject;
|
|
std::string rpath;
|
|
llvm::SmallVector<VersionDefinition, 0> versionDefinitions;
|
|
llvm::SmallVector<llvm::StringRef, 0> auxiliaryList;
|
|
llvm::SmallVector<llvm::StringRef, 0> filterList;
|
|
llvm::SmallVector<llvm::StringRef, 0> passPlugins;
|
|
llvm::SmallVector<llvm::StringRef, 0> searchPaths;
|
|
llvm::SmallVector<llvm::StringRef, 0> symbolOrderingFile;
|
|
llvm::SmallVector<llvm::StringRef, 0> thinLTOModulesToCompile;
|
|
llvm::StringRef dtltoDistributor;
|
|
llvm::SmallVector<llvm::StringRef, 0> dtltoDistributorArgs;
|
|
llvm::StringRef dtltoCompiler;
|
|
llvm::SmallVector<llvm::StringRef, 0> dtltoCompilerArgs;
|
|
llvm::SmallVector<llvm::StringRef, 0> undefined;
|
|
llvm::SmallVector<SymbolVersion, 0> dynamicList;
|
|
llvm::SmallVector<uint8_t, 0> buildIdVector;
|
|
llvm::SmallVector<llvm::StringRef, 0> mllvmOpts;
|
|
llvm::MapVector<std::pair<const InputSectionBase *, const InputSectionBase *>,
|
|
uint64_t>
|
|
callGraphProfile;
|
|
bool cmseImplib = false;
|
|
bool allowMultipleDefinition;
|
|
bool fatLTOObjects;
|
|
bool androidPackDynRelocs = false;
|
|
bool armHasArmISA = false;
|
|
bool armHasThumb2ISA = false;
|
|
bool armHasBlx = false;
|
|
bool armHasMovtMovw = false;
|
|
bool armJ1J2BranchEncoding = false;
|
|
bool armCMSESupport = false;
|
|
bool asNeeded = false;
|
|
bool armBe8 = false;
|
|
BsymbolicKind bsymbolic = BsymbolicKind::None;
|
|
CGProfileSortKind callGraphProfileSort;
|
|
llvm::StringRef irpgoProfilePath;
|
|
bool bpStartupFunctionSort = false;
|
|
bool bpCompressionSortStartupFunctions = false;
|
|
bool bpFunctionOrderForCompression = false;
|
|
bool bpDataOrderForCompression = false;
|
|
bool bpVerboseSectionOrderer = false;
|
|
bool branchToBranch = false;
|
|
bool checkSections;
|
|
bool checkDynamicRelocs;
|
|
std::optional<llvm::DebugCompressionType> compressDebugSections;
|
|
llvm::SmallVector<
|
|
std::tuple<llvm::GlobPattern, llvm::DebugCompressionType, unsigned>, 0>
|
|
compressSections;
|
|
bool cref;
|
|
llvm::SmallVector<std::pair<llvm::GlobPattern, uint64_t>, 0>
|
|
deadRelocInNonAlloc;
|
|
bool debugNames;
|
|
bool demangle = true;
|
|
bool dependentLibraries;
|
|
bool disableVerify;
|
|
bool ehFrameHdr;
|
|
bool emitLLVM;
|
|
bool emitRelocs;
|
|
bool enableNewDtags;
|
|
bool enableNonContiguousRegions;
|
|
bool executeOnly;
|
|
bool exportDynamic;
|
|
bool fixCortexA53Errata843419;
|
|
bool fixCortexA8;
|
|
bool formatBinary = false;
|
|
bool fortranCommon;
|
|
bool gcSections;
|
|
bool gdbIndex;
|
|
bool gnuHash = false;
|
|
bool gnuUnique;
|
|
bool ignoreDataAddressEquality;
|
|
bool ignoreFunctionAddressEquality;
|
|
bool ltoCSProfileGenerate;
|
|
bool ltoPGOWarnMismatch;
|
|
bool ltoDebugPassManager;
|
|
bool ltoEmitAsm;
|
|
bool ltoUniqueBasicBlockSectionNames;
|
|
bool ltoValidateAllVtablesHaveTypeInfos;
|
|
bool ltoWholeProgramVisibility;
|
|
bool mergeArmExidx;
|
|
bool mipsN32Abi = false;
|
|
bool mmapOutputFile;
|
|
bool nmagic;
|
|
bool noinhibitExec;
|
|
bool nostdlib;
|
|
bool oFormatBinary;
|
|
bool omagic;
|
|
bool optEB = false;
|
|
bool optEL = false;
|
|
bool optimizeBBJumps;
|
|
bool optRemarksWithHotness;
|
|
bool picThunk;
|
|
bool pie;
|
|
bool printGcSections;
|
|
bool printIcfSections;
|
|
bool printMemoryUsage;
|
|
std::optional<uint64_t> randomizeSectionPadding;
|
|
bool rejectMismatch;
|
|
bool relax;
|
|
bool relaxGP;
|
|
bool relocatable;
|
|
bool resolveGroups;
|
|
bool relrGlibc = false;
|
|
bool relrPackDynRelocs = false;
|
|
llvm::DenseSet<llvm::StringRef> saveTempsArgs;
|
|
llvm::SmallVector<std::pair<llvm::GlobPattern, uint32_t>, 0> shuffleSections;
|
|
bool singleRoRx;
|
|
bool singleXoRx;
|
|
bool shared;
|
|
bool symbolic;
|
|
bool isStatic = false;
|
|
bool sysvHash = false;
|
|
bool target1Rel;
|
|
bool trace;
|
|
bool thinLTOEmitImportsFiles;
|
|
bool thinLTOEmitIndexFiles;
|
|
bool thinLTOIndexOnly;
|
|
bool timeTraceEnabled;
|
|
bool tocOptimize;
|
|
bool pcRelOptimize;
|
|
bool undefinedVersion;
|
|
bool unique;
|
|
bool useAndroidRelrTags = false;
|
|
bool warnBackrefs;
|
|
llvm::SmallVector<llvm::GlobPattern, 0> warnBackrefsExclude;
|
|
bool warnCommon;
|
|
bool warnMissingEntry;
|
|
bool warnSymbolOrdering;
|
|
bool writeAddends;
|
|
bool zCombreloc;
|
|
bool zCopyreloc;
|
|
bool zDynamicUndefined;
|
|
bool zForceBti;
|
|
bool zForceIbt;
|
|
bool zGlobal;
|
|
bool zHazardplt;
|
|
bool zIfuncNoplt;
|
|
bool zInitfirst;
|
|
bool zInterpose;
|
|
bool zKeepTextSectionPrefix;
|
|
bool zLrodataAfterBss;
|
|
bool zNoBtCfi;
|
|
bool zNodefaultlib;
|
|
bool zNodelete;
|
|
bool zNodlopen;
|
|
bool zNow;
|
|
bool zOrigin;
|
|
bool zPacPlt;
|
|
bool zRelro;
|
|
bool zRodynamic;
|
|
bool zSectionHeader;
|
|
bool zShstk;
|
|
bool zStartStopGC;
|
|
uint8_t zStartStopVisibility;
|
|
bool zText;
|
|
bool zRetpolineplt;
|
|
bool zWxneeded;
|
|
ZicfilpPolicy zZicfilp;
|
|
ZicfissPolicy zZicfiss;
|
|
DiscardPolicy discard;
|
|
GnuStackKind zGnustack;
|
|
ICFLevel icf;
|
|
OrphanHandlingPolicy orphanHandling;
|
|
SortSectionPolicy sortSection;
|
|
StripPolicy strip;
|
|
UnresolvedPolicy unresolvedSymbols;
|
|
UnresolvedPolicy unresolvedSymbolsInShlib;
|
|
Target2Policy target2;
|
|
GcsPolicy zGcs;
|
|
bool power10Stubs;
|
|
ARMVFPArgKind armVFPArgs = ARMVFPArgKind::Default;
|
|
BuildIdKind buildId = BuildIdKind::None;
|
|
SeparateSegmentKind zSeparate;
|
|
ELFKind ekind = ELFNoneKind;
|
|
uint16_t emachine = llvm::ELF::EM_NONE;
|
|
std::optional<uint64_t> imageBase;
|
|
uint64_t commonPageSize;
|
|
uint64_t maxPageSize;
|
|
uint64_t mipsGotSize;
|
|
uint64_t zStackSize;
|
|
unsigned ltoPartitions;
|
|
unsigned ltoo;
|
|
llvm::CodeGenOptLevel ltoCgo;
|
|
unsigned optimize;
|
|
StringRef thinLTOJobs;
|
|
unsigned timeTraceGranularity;
|
|
int32_t splitStackAdjustSize;
|
|
SmallVector<uint8_t, 0> packageMetadata;
|
|
|
|
// The following config options do not directly correspond to any
|
|
// particular command line options.
|
|
|
|
// True if we need to pass through relocations in input files to the
|
|
// output file. Usually false because we consume relocations.
|
|
bool copyRelocs;
|
|
|
|
// True if the target is ELF64. False if ELF32.
|
|
bool is64;
|
|
|
|
// True if the target is little-endian. False if big-endian.
|
|
bool isLE;
|
|
|
|
// endianness::little if isLE is true. endianness::big otherwise.
|
|
llvm::endianness endianness;
|
|
|
|
// True if the target is the little-endian MIPS64.
|
|
//
|
|
// The reason why we have this variable only for the MIPS is because
|
|
// we use this often. Some ELF headers for MIPS64EL are in a
|
|
// mixed-endian (which is horrible and I'd say that's a serious spec
|
|
// bug), and we need to know whether we are reading MIPS ELF files or
|
|
// not in various places.
|
|
//
|
|
// (Note that MIPS64EL is not a typo for MIPS64LE. This is the official
|
|
// name whatever that means. A fun hypothesis is that "EL" is short for
|
|
// little-endian written in the little-endian order, but I don't know
|
|
// if that's true.)
|
|
bool isMips64EL;
|
|
|
|
// True if we need to set the DF_STATIC_TLS flag to an output file, which
|
|
// works as a hint to the dynamic loader that the shared object contains code
|
|
// compiled with the initial-exec TLS model.
|
|
bool hasTlsIe = false;
|
|
|
|
// Holds set of ELF header flags for the target.
|
|
uint32_t eflags = 0;
|
|
|
|
// The ELF spec defines two types of relocation table entries, RELA and
|
|
// REL. RELA is a triplet of (offset, info, addend) while REL is a
|
|
// tuple of (offset, info). Addends for REL are implicit and read from
|
|
// the location where the relocations are applied. So, REL is more
|
|
// compact than RELA but requires a bit of more work to process.
|
|
//
|
|
// (From the linker writer's view, this distinction is not necessary.
|
|
// If the ELF had chosen whichever and sticked with it, it would have
|
|
// been easier to write code to process relocations, but it's too late
|
|
// to change the spec.)
|
|
//
|
|
// Each ABI defines its relocation type. IsRela is true if target
|
|
// uses RELA. As far as we know, all 64-bit ABIs are using RELA. A
|
|
// few 32-bit ABIs are using RELA too.
|
|
bool isRela;
|
|
|
|
// True if we are creating position-independent code.
|
|
bool isPic;
|
|
|
|
// 4 for ELF32, 8 for ELF64.
|
|
int wordsize;
|
|
|
|
// Mode of MTE to write to the ELF note. Should be one of NT_MEMTAG_ASYNC (for
|
|
// async), NT_MEMTAG_SYNC (for sync), or NT_MEMTAG_LEVEL_NONE (for none). If
|
|
// async or sync is enabled, write the ELF note specifying the default MTE
|
|
// mode.
|
|
int androidMemtagMode;
|
|
// Signal to the dynamic loader to enable heap MTE.
|
|
bool androidMemtagHeap;
|
|
// Signal to the dynamic loader that this binary expects stack MTE. Generally,
|
|
// this means to map the primary and thread stacks as PROT_MTE. Note: This is
|
|
// not supported on Android 11 & 12.
|
|
bool androidMemtagStack;
|
|
|
|
// When using a unified pre-link LTO pipeline, specify the backend LTO mode.
|
|
LtoKind ltoKind = LtoKind::Default;
|
|
|
|
unsigned threadCount;
|
|
|
|
// If an input file equals a key, remap it to the value.
|
|
llvm::DenseMap<llvm::StringRef, llvm::StringRef> remapInputs;
|
|
// If an input file matches a wildcard pattern, remap it to the value.
|
|
llvm::SmallVector<std::pair<llvm::GlobPattern, llvm::StringRef>, 0>
|
|
remapInputsWildcards;
|
|
};
|
|
|
|
// Some index properties of a symbol are stored separately in this auxiliary
|
|
// struct to decrease sizeof(SymbolUnion) in the majority of cases.
|
|
struct SymbolAux {
|
|
uint32_t gotIdx = -1;
|
|
uint32_t pltIdx = -1;
|
|
uint32_t tlsDescIdx = -1;
|
|
uint32_t tlsGdIdx = -1;
|
|
};
|
|
|
|
struct DuplicateSymbol {
|
|
const Symbol *sym;
|
|
const InputFile *file;
|
|
InputSectionBase *section;
|
|
uint64_t value;
|
|
};
|
|
|
|
struct UndefinedDiag {
|
|
Undefined *sym;
|
|
struct Loc {
|
|
InputSectionBase *sec;
|
|
uint64_t offset;
|
|
};
|
|
SmallVector<Loc, 0> locs;
|
|
bool isWarning;
|
|
};
|
|
|
|
// Linker generated sections which can be used as inputs and are not specific to
|
|
// a partition.
|
|
struct InStruct {
|
|
std::unique_ptr<InputSection> attributes;
|
|
std::unique_ptr<SyntheticSection> riscvAttributes;
|
|
std::unique_ptr<BssSection> bss;
|
|
std::unique_ptr<BssSection> bssRelRo;
|
|
std::unique_ptr<SyntheticSection> gnuProperty;
|
|
std::unique_ptr<SyntheticSection> gnuStack;
|
|
std::unique_ptr<GotSection> got;
|
|
std::unique_ptr<GotPltSection> gotPlt;
|
|
std::unique_ptr<IgotPltSection> igotPlt;
|
|
std::unique_ptr<RelroPaddingSection> relroPadding;
|
|
std::unique_ptr<SyntheticSection> armCmseSGSection;
|
|
std::unique_ptr<PPC64LongBranchTargetSection> ppc64LongBranchTarget;
|
|
std::unique_ptr<SyntheticSection> mipsAbiFlags;
|
|
std::unique_ptr<MipsGotSection> mipsGot;
|
|
std::unique_ptr<SyntheticSection> mipsOptions;
|
|
std::unique_ptr<SyntheticSection> mipsReginfo;
|
|
std::unique_ptr<MipsRldMapSection> mipsRldMap;
|
|
std::unique_ptr<SyntheticSection> partEnd;
|
|
std::unique_ptr<SyntheticSection> partIndex;
|
|
std::unique_ptr<PltSection> plt;
|
|
std::unique_ptr<IpltSection> iplt;
|
|
std::unique_ptr<PPC32Got2Section> ppc32Got2;
|
|
std::unique_ptr<IBTPltSection> ibtPlt;
|
|
std::unique_ptr<RelocationBaseSection> relaPlt;
|
|
// Non-SHF_ALLOC sections
|
|
std::unique_ptr<SyntheticSection> debugNames;
|
|
std::unique_ptr<GdbIndexSection> gdbIndex;
|
|
std::unique_ptr<StringTableSection> shStrTab;
|
|
std::unique_ptr<StringTableSection> strTab;
|
|
std::unique_ptr<SymbolTableBaseSection> symTab;
|
|
std::unique_ptr<SymtabShndxSection> symTabShndx;
|
|
};
|
|
|
|
struct Ctx : CommonLinkerContext {
|
|
Config arg;
|
|
LinkerDriver driver;
|
|
LinkerScript *script;
|
|
std::unique_ptr<TargetInfo> target;
|
|
|
|
// These variables are initialized by Writer and should not be used before
|
|
// Writer is initialized.
|
|
uint8_t *bufferStart = nullptr;
|
|
Partition *mainPart = nullptr;
|
|
PhdrEntry *tlsPhdr = nullptr;
|
|
struct OutSections {
|
|
std::unique_ptr<OutputSection> elfHeader;
|
|
std::unique_ptr<OutputSection> programHeaders;
|
|
OutputSection *preinitArray = nullptr;
|
|
OutputSection *initArray = nullptr;
|
|
OutputSection *finiArray = nullptr;
|
|
};
|
|
OutSections out;
|
|
SmallVector<OutputSection *, 0> outputSections;
|
|
std::vector<Partition> partitions;
|
|
|
|
InStruct in;
|
|
|
|
// Some linker-generated symbols need to be created as
|
|
// Defined symbols.
|
|
struct ElfSym {
|
|
// __bss_start
|
|
Defined *bss;
|
|
|
|
// etext and _etext
|
|
Defined *etext1;
|
|
Defined *etext2;
|
|
|
|
// edata and _edata
|
|
Defined *edata1;
|
|
Defined *edata2;
|
|
|
|
// end and _end
|
|
Defined *end1;
|
|
Defined *end2;
|
|
|
|
// The _GLOBAL_OFFSET_TABLE_ symbol is defined by target convention to
|
|
// be at some offset from the base of the .got section, usually 0 or
|
|
// the end of the .got.
|
|
Defined *globalOffsetTable;
|
|
|
|
// _gp, _gp_disp and __gnu_local_gp symbols. Only for MIPS.
|
|
Defined *mipsGp;
|
|
Defined *mipsGpDisp;
|
|
Defined *mipsLocalGp;
|
|
|
|
// __global_pointer$ for RISC-V.
|
|
Defined *riscvGlobalPointer;
|
|
|
|
// __rel{,a}_iplt_{start,end} symbols.
|
|
Defined *relaIpltStart;
|
|
Defined *relaIpltEnd;
|
|
|
|
// _TLS_MODULE_BASE_ on targets that support TLSDESC.
|
|
Defined *tlsModuleBase;
|
|
};
|
|
ElfSym sym{};
|
|
std::unique_ptr<SymbolTable> symtab;
|
|
SmallVector<Symbol *, 0> synthesizedSymbols;
|
|
|
|
SmallVector<std::unique_ptr<MemoryBuffer>> memoryBuffers;
|
|
SmallVector<ELFFileBase *, 0> objectFiles;
|
|
SmallVector<SharedFile *, 0> sharedFiles;
|
|
SmallVector<BinaryFile *, 0> binaryFiles;
|
|
SmallVector<BitcodeFile *, 0> bitcodeFiles;
|
|
SmallVector<BitcodeFile *, 0> lazyBitcodeFiles;
|
|
SmallVector<InputSectionBase *, 0> inputSections;
|
|
SmallVector<EhInputSection *, 0> ehInputSections;
|
|
|
|
SmallVector<SymbolAux, 0> symAux;
|
|
// Duplicate symbol candidates.
|
|
SmallVector<DuplicateSymbol, 0> duplicates;
|
|
// Undefined diagnostics are collected in a vector and emitted once all of
|
|
// them are known, so that some postprocessing on the list of undefined
|
|
// symbols can happen before lld emits diagnostics.
|
|
std::mutex relocMutex;
|
|
SmallVector<UndefinedDiag, 0> undefErrs;
|
|
// Symbols in a non-prevailing COMDAT group which should be changed to an
|
|
// Undefined.
|
|
SmallVector<std::pair<Symbol *, unsigned>, 0> nonPrevailingSyms;
|
|
// A tuple of (reference, extractedFile, sym). Used by --why-extract=.
|
|
SmallVector<std::tuple<std::string, const InputFile *, const Symbol &>, 0>
|
|
whyExtractRecords;
|
|
// A mapping from a symbol to an InputFile referencing it backward. Used by
|
|
// --warn-backrefs.
|
|
llvm::DenseMap<const Symbol *,
|
|
std::pair<const InputFile *, const InputFile *>>
|
|
backwardReferences;
|
|
llvm::SmallSet<llvm::StringRef, 0> auxiliaryFiles;
|
|
// If --reproduce is specified, all input files are written to this tar
|
|
// archive.
|
|
std::unique_ptr<llvm::TarWriter> tar;
|
|
// InputFile for linker created symbols with no source location.
|
|
InputFile *internalFile = nullptr;
|
|
// True if symbols can be exported (isExported) or preemptible.
|
|
bool hasDynsym = false;
|
|
// True if SHT_LLVM_SYMPART is used.
|
|
std::atomic<bool> hasSympart{false};
|
|
// True if there are TLS IE relocations. Set DF_STATIC_TLS if -shared.
|
|
std::atomic<bool> hasTlsIe{false};
|
|
// True if we need to reserve two .got entries for local-dynamic TLS model.
|
|
std::atomic<bool> needsTlsLd{false};
|
|
// True if all native vtable symbols have corresponding type info symbols
|
|
// during LTO.
|
|
bool ltoAllVtablesHaveTypeInfos = false;
|
|
// Number of Vernaux entries (needed shared object names).
|
|
uint32_t vernauxNum = 0;
|
|
|
|
// Each symbol assignment and DEFINED(sym) reference is assigned an increasing
|
|
// order. Each DEFINED(sym) evaluation checks whether the reference happens
|
|
// before a possible `sym = expr;`.
|
|
unsigned scriptSymOrderCounter = 1;
|
|
llvm::DenseMap<const Symbol *, unsigned> scriptSymOrder;
|
|
|
|
// The set of TOC entries (.toc + addend) for which we should not apply
|
|
// toc-indirect to toc-relative relaxation. const Symbol * refers to the
|
|
// STT_SECTION symbol associated to the .toc input section.
|
|
llvm::DenseSet<std::pair<const Symbol *, uint64_t>> ppc64noTocRelax;
|
|
|
|
Ctx();
|
|
|
|
llvm::raw_fd_ostream openAuxiliaryFile(llvm::StringRef, std::error_code &);
|
|
|
|
std::optional<AArch64PauthAbiCoreInfo> aarch64PauthAbiCoreInfo;
|
|
};
|
|
|
|
// The first two elements of versionDefinitions represent VER_NDX_LOCAL and
|
|
// VER_NDX_GLOBAL. This helper returns other elements.
|
|
static inline ArrayRef<VersionDefinition> namedVersionDefs(Ctx &ctx) {
|
|
return llvm::ArrayRef(ctx.arg.versionDefinitions).slice(2);
|
|
}
|
|
|
|
struct ELFSyncStream : SyncStream {
|
|
Ctx &ctx;
|
|
ELFSyncStream(Ctx &ctx, DiagLevel level)
|
|
: SyncStream(ctx.e, level), ctx(ctx) {}
|
|
};
|
|
|
|
template <typename T>
|
|
std::enable_if_t<!std::is_pointer_v<std::remove_reference_t<T>>,
|
|
const ELFSyncStream &>
|
|
operator<<(const ELFSyncStream &s, T &&v) {
|
|
s.os << std::forward<T>(v);
|
|
return s;
|
|
}
|
|
|
|
inline const ELFSyncStream &operator<<(const ELFSyncStream &s, const char *v) {
|
|
s.os << v;
|
|
return s;
|
|
}
|
|
|
|
inline const ELFSyncStream &operator<<(const ELFSyncStream &s, Error v) {
|
|
s.os << llvm::toString(std::move(v));
|
|
return s;
|
|
}
|
|
|
|
// Report a log if --verbose is specified.
|
|
ELFSyncStream Log(Ctx &ctx);
|
|
|
|
// Print a message to stdout.
|
|
ELFSyncStream Msg(Ctx &ctx);
|
|
|
|
// Report a warning. Upgraded to an error if --fatal-warnings is specified.
|
|
ELFSyncStream Warn(Ctx &ctx);
|
|
|
|
// Report an error that will suppress the output file generation. Downgraded to
|
|
// a warning if --noinhibit-exec is specified.
|
|
ELFSyncStream Err(Ctx &ctx);
|
|
|
|
// Report an error regardless of --noinhibit-exec.
|
|
ELFSyncStream ErrAlways(Ctx &ctx);
|
|
|
|
// Report a fatal error that exits immediately. This should generally be avoided
|
|
// in favor of Err.
|
|
ELFSyncStream Fatal(Ctx &ctx);
|
|
|
|
uint64_t errCount(Ctx &ctx);
|
|
|
|
ELFSyncStream InternalErr(Ctx &ctx, const uint8_t *buf);
|
|
|
|
#define CHECK2(E, S) lld::check2((E), [&] { return toStr(ctx, S); })
|
|
|
|
inline DiagLevel toDiagLevel(ReportPolicy policy) {
|
|
if (policy == ReportPolicy::Error)
|
|
return DiagLevel::Err;
|
|
else if (policy == ReportPolicy::Warning)
|
|
return DiagLevel::Warn;
|
|
return DiagLevel::None;
|
|
}
|
|
|
|
} // namespace lld::elf
|
|
|
|
#endif
|