dsymutil: Factor out the relocation handling into a RelocationManager (NFC)

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@247490 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Adrian Prantl 2015-09-11 23:45:30 +00:00
parent 830d0f8976
commit 0a0a960468

View File

@ -1103,52 +1103,74 @@ private:
/// \brief Called at the end of a debug object link. /// \brief Called at the end of a debug object link.
void endDebugObject(); void endDebugObject();
/// \defgroup FindValidRelocations Translate debug map into a list /// Keeps track of relocations.
/// of relevant relocations class RelocationManager {
/// struct ValidReloc {
/// @{ uint32_t Offset;
struct ValidReloc { uint32_t Size;
uint32_t Offset; uint64_t Addend;
uint32_t Size; const DebugMapObject::DebugMapEntry *Mapping;
uint64_t Addend;
const DebugMapObject::DebugMapEntry *Mapping;
ValidReloc(uint32_t Offset, uint32_t Size, uint64_t Addend, ValidReloc(uint32_t Offset, uint32_t Size, uint64_t Addend,
const DebugMapObject::DebugMapEntry *Mapping) const DebugMapObject::DebugMapEntry *Mapping)
: Offset(Offset), Size(Size), Addend(Addend), Mapping(Mapping) {} : Offset(Offset), Size(Size), Addend(Addend), Mapping(Mapping) {}
bool operator<(const ValidReloc &RHS) const { return Offset < RHS.Offset; } bool operator<(const ValidReloc &RHS) const {
return Offset < RHS.Offset;
}
};
DwarfLinker &Linker;
/// \brief The valid relocations for the current DebugMapObject.
/// This vector is sorted by relocation offset.
std::vector<ValidReloc> ValidRelocs;
/// \brief Index into ValidRelocs of the next relocation to
/// consider. As we walk the DIEs in acsending file offset and as
/// ValidRelocs is sorted by file offset, keeping this index
/// uptodate is all we have to do to have a cheap lookup during the
/// root DIE selection and during DIE cloning.
unsigned NextValidReloc;
public:
RelocationManager(DwarfLinker &Linker)
: Linker(Linker), NextValidReloc(0) {}
bool hasValidRelocs() const { return !ValidRelocs.empty(); }
/// \brief Reset the NextValidReloc counter.
void resetValidRelocs() { NextValidReloc = 0; }
/// \defgroup FindValidRelocations Translate debug map into a list
/// of relevant relocations
///
/// @{
bool findValidRelocsInDebugInfo(const object::ObjectFile &Obj,
const DebugMapObject &DMO);
bool findValidRelocs(const object::SectionRef &Section,
const object::ObjectFile &Obj,
const DebugMapObject &DMO);
void findValidRelocsMachO(const object::SectionRef &Section,
const object::MachOObjectFile &Obj,
const DebugMapObject &DMO);
/// @}
bool hasValidRelocation(uint32_t StartOffset, uint32_t EndOffset,
CompileUnit::DIEInfo &Info);
bool applyValidRelocs(MutableArrayRef<char> Data, uint32_t BaseOffset,
bool isLittleEndian);
}; };
/// \brief The valid relocations for the current DebugMapObject.
/// This vector is sorted by relocation offset.
std::vector<ValidReloc> ValidRelocs;
/// \brief Index into ValidRelocs of the next relocation to
/// consider. As we walk the DIEs in acsending file offset and as
/// ValidRelocs is sorted by file offset, keeping this index
/// uptodate is all we have to do to have a cheap lookup during the
/// root DIE selection and during DIE cloning.
unsigned NextValidReloc;
bool findValidRelocsInDebugInfo(const object::ObjectFile &Obj,
const DebugMapObject &DMO);
bool findValidRelocs(const object::SectionRef &Section,
const object::ObjectFile &Obj,
const DebugMapObject &DMO);
void findValidRelocsMachO(const object::SectionRef &Section,
const object::MachOObjectFile &Obj,
const DebugMapObject &DMO);
/// @}
/// \defgroup FindRootDIEs Find DIEs corresponding to debug map entries. /// \defgroup FindRootDIEs Find DIEs corresponding to debug map entries.
/// ///
/// @{ /// @{
/// \brief Recursively walk the \p DIE tree and look for DIEs to /// \brief Recursively walk the \p DIE tree and look for DIEs to
/// keep. Store that information in \p CU's DIEInfo. /// keep. Store that information in \p CU's DIEInfo.
void lookForDIEsToKeep(const DWARFDebugInfoEntryMinimal &DIE, void lookForDIEsToKeep(RelocationManager &RelocMgr,
const DWARFDebugInfoEntryMinimal &DIE,
const DebugMapObject &DMO, CompileUnit &CU, const DebugMapObject &DMO, CompileUnit &CU,
unsigned Flags); unsigned Flags);
@ -1164,20 +1186,24 @@ private:
/// \brief Mark the passed DIE as well as all the ones it depends on /// \brief Mark the passed DIE as well as all the ones it depends on
/// as kept. /// as kept.
void keepDIEAndDenpendencies(const DWARFDebugInfoEntryMinimal &DIE, void keepDIEAndDenpendencies(RelocationManager &RelocMgr,
const DWARFDebugInfoEntryMinimal &DIE,
CompileUnit::DIEInfo &MyInfo, CompileUnit::DIEInfo &MyInfo,
const DebugMapObject &DMO, CompileUnit &CU, const DebugMapObject &DMO, CompileUnit &CU,
bool UseODR); bool UseODR);
unsigned shouldKeepDIE(const DWARFDebugInfoEntryMinimal &DIE, unsigned shouldKeepDIE(RelocationManager &RelocMgr,
const DWARFDebugInfoEntryMinimal &DIE,
CompileUnit &Unit, CompileUnit::DIEInfo &MyInfo, CompileUnit &Unit, CompileUnit::DIEInfo &MyInfo,
unsigned Flags); unsigned Flags);
unsigned shouldKeepVariableDIE(const DWARFDebugInfoEntryMinimal &DIE, unsigned shouldKeepVariableDIE(RelocationManager &RelocMgr,
const DWARFDebugInfoEntryMinimal &DIE,
CompileUnit &Unit, CompileUnit &Unit,
CompileUnit::DIEInfo &MyInfo, unsigned Flags); CompileUnit::DIEInfo &MyInfo, unsigned Flags);
unsigned shouldKeepSubprogramDIE(const DWARFDebugInfoEntryMinimal &DIE, unsigned shouldKeepSubprogramDIE(RelocationManager &RelocMgr,
const DWARFDebugInfoEntryMinimal &DIE,
CompileUnit &Unit, CompileUnit &Unit,
CompileUnit::DIEInfo &MyInfo, CompileUnit::DIEInfo &MyInfo,
unsigned Flags); unsigned Flags);
@ -1200,9 +1226,19 @@ private:
/// applied to the entry point of the function to get the linked address. /// applied to the entry point of the function to get the linked address.
/// ///
/// \returns the root of the cloned tree. /// \returns the root of the cloned tree.
DIE *cloneDIE(const DWARFDebugInfoEntryMinimal &InputDIE, CompileUnit &U, DIE *cloneDIE(RelocationManager &RelocMgr,
const DWARFDebugInfoEntryMinimal &InputDIE, CompileUnit &U,
int64_t PCOffset, uint32_t OutOffset, unsigned Flags); int64_t PCOffset, uint32_t OutOffset, unsigned Flags);
/// Construct the output DIE tree by cloning the DIEs we chose to
/// keep above. If there are no valid relocs, then there's nothing
/// to clone/emit.
void cloneCompileUnit(RelocationManager &RelocMgr,
MutableArrayRef<CompileUnit> CompileUnit,
DWARFContextInMemory &DwarfContext);
typedef DWARFAbbreviationDeclaration::AttributeSpec AttributeSpec; typedef DWARFAbbreviationDeclaration::AttributeSpec AttributeSpec;
/// \brief Information gathered and exchanged between the various /// \brief Information gathered and exchanged between the various
@ -1257,10 +1293,6 @@ private:
const DWARFFormValue &Val, unsigned AttrSize, const DWARFFormValue &Val, unsigned AttrSize,
AttributesInfo &Info); AttributesInfo &Info);
/// \brief Helper for cloneDIE.
bool applyValidRelocs(MutableArrayRef<char> Data, uint32_t BaseOffset,
bool isLittleEndian);
/// \brief Assign an abbreviation number to \p Abbrev /// \brief Assign an abbreviation number to \p Abbrev
void AssignAbbrev(DIEAbbrev &Abbrev); void AssignAbbrev(DIEAbbrev &Abbrev);
@ -1334,6 +1366,7 @@ private:
LinkOptions Options; LinkOptions Options;
BinaryHolder BinHolder; BinaryHolder BinHolder;
std::unique_ptr<DwarfStreamer> Streamer; std::unique_ptr<DwarfStreamer> Streamer;
uint64_t OutputDebugInfoSize;
/// The units of the current debug map object. /// The units of the current debug map object.
std::vector<CompileUnit> Units; std::vector<CompileUnit> Units;
@ -1701,7 +1734,6 @@ static unsigned getRefAddrSize(const DWARFUnit &U) {
void DwarfLinker::startDebugObject(DWARFContext &Dwarf, DebugMapObject &Obj) { void DwarfLinker::startDebugObject(DWARFContext &Dwarf, DebugMapObject &Obj) {
Units.reserve(Dwarf.getNumCompileUnits()); Units.reserve(Dwarf.getNumCompileUnits());
NextValidReloc = 0;
// Iterate over the debug map entries and put all the ones that are // Iterate over the debug map entries and put all the ones that are
// functions (because they have a size) into the Ranges map. This // functions (because they have a size) into the Ranges map. This
// map is very similar to the FunctionRanges that are stored in each // map is very similar to the FunctionRanges that are stored in each
@ -1727,7 +1759,6 @@ void DwarfLinker::startDebugObject(DWARFContext &Dwarf, DebugMapObject &Obj) {
void DwarfLinker::endDebugObject() { void DwarfLinker::endDebugObject() {
Units.clear(); Units.clear();
ValidRelocs.clear();
Ranges.clear(); Ranges.clear();
for (auto I = DIEBlocks.begin(), E = DIEBlocks.end(); I != E; ++I) for (auto I = DIEBlocks.begin(), E = DIEBlocks.end(); I != E; ++I)
@ -1743,9 +1774,10 @@ void DwarfLinker::endDebugObject() {
/// \brief Iterate over the relocations of the given \p Section and /// \brief Iterate over the relocations of the given \p Section and
/// store the ones that correspond to debug map entries into the /// store the ones that correspond to debug map entries into the
/// ValidRelocs array. /// ValidRelocs array.
void DwarfLinker::findValidRelocsMachO(const object::SectionRef &Section, void DwarfLinker::RelocationManager::
const object::MachOObjectFile &Obj, findValidRelocsMachO(const object::SectionRef &Section,
const DebugMapObject &DMO) { const object::MachOObjectFile &Obj,
const DebugMapObject &DMO) {
StringRef Contents; StringRef Contents;
Section.getContents(Contents); Section.getContents(Contents);
DataExtractor Data(Contents, Obj.isLittleEndian(), 0); DataExtractor Data(Contents, Obj.isLittleEndian(), 0);
@ -1756,7 +1788,7 @@ void DwarfLinker::findValidRelocsMachO(const object::SectionRef &Section,
unsigned RelocSize = 1 << Obj.getAnyRelocationLength(MachOReloc); unsigned RelocSize = 1 << Obj.getAnyRelocationLength(MachOReloc);
uint64_t Offset64 = Reloc.getOffset(); uint64_t Offset64 = Reloc.getOffset();
if ((RelocSize != 4 && RelocSize != 8)) { if ((RelocSize != 4 && RelocSize != 8)) {
reportWarning(" unsupported relocation in debug_info section."); Linker.reportWarning(" unsupported relocation in debug_info section.");
continue; continue;
} }
uint32_t Offset = Offset64; uint32_t Offset = Offset64;
@ -1767,7 +1799,7 @@ void DwarfLinker::findValidRelocsMachO(const object::SectionRef &Section,
if (Sym != Obj.symbol_end()) { if (Sym != Obj.symbol_end()) {
ErrorOr<StringRef> SymbolName = Sym->getName(); ErrorOr<StringRef> SymbolName = Sym->getName();
if (!SymbolName) { if (!SymbolName) {
reportWarning("error getting relocation symbol name."); Linker.reportWarning("error getting relocation symbol name.");
continue; continue;
} }
if (const auto *Mapping = DMO.lookupSymbol(*SymbolName)) if (const auto *Mapping = DMO.lookupSymbol(*SymbolName))
@ -1783,14 +1815,15 @@ void DwarfLinker::findValidRelocsMachO(const object::SectionRef &Section,
/// \brief Dispatch the valid relocation finding logic to the /// \brief Dispatch the valid relocation finding logic to the
/// appropriate handler depending on the object file format. /// appropriate handler depending on the object file format.
bool DwarfLinker::findValidRelocs(const object::SectionRef &Section, bool DwarfLinker::RelocationManager::findValidRelocs(
const object::ObjectFile &Obj, const object::SectionRef &Section, const object::ObjectFile &Obj,
const DebugMapObject &DMO) { const DebugMapObject &DMO) {
// Dispatch to the right handler depending on the file type. // Dispatch to the right handler depending on the file type.
if (auto *MachOObj = dyn_cast<object::MachOObjectFile>(&Obj)) if (auto *MachOObj = dyn_cast<object::MachOObjectFile>(&Obj))
findValidRelocsMachO(Section, *MachOObj, DMO); findValidRelocsMachO(Section, *MachOObj, DMO);
else else
reportWarning(Twine("unsupported object file type: ") + Obj.getFileName()); Linker.reportWarning(Twine("unsupported object file type: ") +
Obj.getFileName());
if (ValidRelocs.empty()) if (ValidRelocs.empty())
return false; return false;
@ -1808,8 +1841,9 @@ bool DwarfLinker::findValidRelocs(const object::SectionRef &Section,
/// link by indicating which DIEs refer to symbols present in the /// link by indicating which DIEs refer to symbols present in the
/// linked binary. /// linked binary.
/// \returns wether there are any valid relocations in the debug info. /// \returns wether there are any valid relocations in the debug info.
bool DwarfLinker::findValidRelocsInDebugInfo(const object::ObjectFile &Obj, bool DwarfLinker::RelocationManager::
const DebugMapObject &DMO) { findValidRelocsInDebugInfo(const object::ObjectFile &Obj,
const DebugMapObject &DMO) {
// Find the debug_info section. // Find the debug_info section.
for (const object::SectionRef &Section : Obj.sections()) { for (const object::SectionRef &Section : Obj.sections()) {
StringRef SectionName; StringRef SectionName;
@ -1828,8 +1862,9 @@ bool DwarfLinker::findValidRelocsInDebugInfo(const object::ObjectFile &Obj,
/// This function must be called with offsets in strictly ascending /// This function must be called with offsets in strictly ascending
/// order because it never looks back at relocations it already 'went past'. /// order because it never looks back at relocations it already 'went past'.
/// \returns true and sets Info.InDebugMap if it is the case. /// \returns true and sets Info.InDebugMap if it is the case.
bool DwarfLinker::hasValidRelocation(uint32_t StartOffset, uint32_t EndOffset, bool DwarfLinker::RelocationManager::
CompileUnit::DIEInfo &Info) { hasValidRelocation(uint32_t StartOffset, uint32_t EndOffset,
CompileUnit::DIEInfo &Info) {
assert(NextValidReloc == 0 || assert(NextValidReloc == 0 ||
StartOffset > ValidRelocs[NextValidReloc - 1].Offset); StartOffset > ValidRelocs[NextValidReloc - 1].Offset);
if (NextValidReloc >= ValidRelocs.size()) if (NextValidReloc >= ValidRelocs.size())
@ -1849,7 +1884,7 @@ bool DwarfLinker::hasValidRelocation(uint32_t StartOffset, uint32_t EndOffset,
const auto &ValidReloc = ValidRelocs[NextValidReloc++]; const auto &ValidReloc = ValidRelocs[NextValidReloc++];
const auto &Mapping = ValidReloc.Mapping->getValue(); const auto &Mapping = ValidReloc.Mapping->getValue();
if (Options.Verbose) if (Linker.Options.Verbose)
outs() << "Found valid debug map entry: " << ValidReloc.Mapping->getKey() outs() << "Found valid debug map entry: " << ValidReloc.Mapping->getKey()
<< " " << format("\t%016" PRIx64 " => %016" PRIx64, << " " << format("\t%016" PRIx64 " => %016" PRIx64,
uint64_t(Mapping.ObjectAddress), uint64_t(Mapping.ObjectAddress),
@ -1882,9 +1917,11 @@ getAttributeOffsets(const DWARFAbbreviationDeclaration *Abbrev, unsigned Idx,
/// \brief Check if a variable describing DIE should be kept. /// \brief Check if a variable describing DIE should be kept.
/// \returns updated TraversalFlags. /// \returns updated TraversalFlags.
unsigned DwarfLinker::shouldKeepVariableDIE( unsigned DwarfLinker::shouldKeepVariableDIE(RelocationManager &RelocMgr,
const DWARFDebugInfoEntryMinimal &DIE, CompileUnit &Unit, const DWARFDebugInfoEntryMinimal &DIE,
CompileUnit::DIEInfo &MyInfo, unsigned Flags) { CompileUnit &Unit,
CompileUnit::DIEInfo &MyInfo,
unsigned Flags) {
const auto *Abbrev = DIE.getAbbreviationDeclarationPtr(); const auto *Abbrev = DIE.getAbbreviationDeclarationPtr();
// Global variables with constant value can always be kept. // Global variables with constant value can always be kept.
@ -1909,7 +1946,7 @@ unsigned DwarfLinker::shouldKeepVariableDIE(
// always check in the variable has a valid relocation, so that the // always check in the variable has a valid relocation, so that the
// DIEInfo is filled. However, we don't want a static variable in a // DIEInfo is filled. However, we don't want a static variable in a
// function to force us to keep the enclosing function. // function to force us to keep the enclosing function.
if (!hasValidRelocation(LocationOffset, LocationEndOffset, MyInfo) || if (!RelocMgr.hasValidRelocation(LocationOffset, LocationEndOffset, MyInfo) ||
(Flags & TF_InFunctionScope)) (Flags & TF_InFunctionScope))
return Flags; return Flags;
@ -1922,6 +1959,7 @@ unsigned DwarfLinker::shouldKeepVariableDIE(
/// \brief Check if a function describing DIE should be kept. /// \brief Check if a function describing DIE should be kept.
/// \returns updated TraversalFlags. /// \returns updated TraversalFlags.
unsigned DwarfLinker::shouldKeepSubprogramDIE( unsigned DwarfLinker::shouldKeepSubprogramDIE(
RelocationManager &RelocMgr,
const DWARFDebugInfoEntryMinimal &DIE, CompileUnit &Unit, const DWARFDebugInfoEntryMinimal &DIE, CompileUnit &Unit,
CompileUnit::DIEInfo &MyInfo, unsigned Flags) { CompileUnit::DIEInfo &MyInfo, unsigned Flags) {
const auto *Abbrev = DIE.getAbbreviationDeclarationPtr(); const auto *Abbrev = DIE.getAbbreviationDeclarationPtr();
@ -1942,7 +1980,7 @@ unsigned DwarfLinker::shouldKeepSubprogramDIE(
DIE.getAttributeValueAsAddress(&OrigUnit, dwarf::DW_AT_low_pc, -1ULL); DIE.getAttributeValueAsAddress(&OrigUnit, dwarf::DW_AT_low_pc, -1ULL);
assert(LowPc != -1ULL && "low_pc attribute is not an address."); assert(LowPc != -1ULL && "low_pc attribute is not an address.");
if (LowPc == -1ULL || if (LowPc == -1ULL ||
!hasValidRelocation(LowPcOffset, LowPcEndOffset, MyInfo)) !RelocMgr.hasValidRelocation(LowPcOffset, LowPcEndOffset, MyInfo))
return Flags; return Flags;
if (Options.Verbose) if (Options.Verbose)
@ -1973,16 +2011,17 @@ unsigned DwarfLinker::shouldKeepSubprogramDIE(
/// \brief Check if a DIE should be kept. /// \brief Check if a DIE should be kept.
/// \returns updated TraversalFlags. /// \returns updated TraversalFlags.
unsigned DwarfLinker::shouldKeepDIE(const DWARFDebugInfoEntryMinimal &DIE, unsigned DwarfLinker::shouldKeepDIE(RelocationManager &RelocMgr,
const DWARFDebugInfoEntryMinimal &DIE,
CompileUnit &Unit, CompileUnit &Unit,
CompileUnit::DIEInfo &MyInfo, CompileUnit::DIEInfo &MyInfo,
unsigned Flags) { unsigned Flags) {
switch (DIE.getTag()) { switch (DIE.getTag()) {
case dwarf::DW_TAG_constant: case dwarf::DW_TAG_constant:
case dwarf::DW_TAG_variable: case dwarf::DW_TAG_variable:
return shouldKeepVariableDIE(DIE, Unit, MyInfo, Flags); return shouldKeepVariableDIE(RelocMgr, DIE, Unit, MyInfo, Flags);
case dwarf::DW_TAG_subprogram: case dwarf::DW_TAG_subprogram:
return shouldKeepSubprogramDIE(DIE, Unit, MyInfo, Flags); return shouldKeepSubprogramDIE(RelocMgr, DIE, Unit, MyInfo, Flags);
case dwarf::DW_TAG_module: case dwarf::DW_TAG_module:
case dwarf::DW_TAG_imported_module: case dwarf::DW_TAG_imported_module:
case dwarf::DW_TAG_imported_declaration: case dwarf::DW_TAG_imported_declaration:
@ -2002,7 +2041,8 @@ unsigned DwarfLinker::shouldKeepDIE(const DWARFDebugInfoEntryMinimal &DIE,
/// back to lookForDIEsToKeep while adding TF_DependencyWalk to the /// back to lookForDIEsToKeep while adding TF_DependencyWalk to the
/// TraversalFlags to inform it that it's not doing the primary DIE /// TraversalFlags to inform it that it's not doing the primary DIE
/// tree walk. /// tree walk.
void DwarfLinker::keepDIEAndDenpendencies(const DWARFDebugInfoEntryMinimal &Die, void DwarfLinker::keepDIEAndDenpendencies(RelocationManager &RelocMgr,
const DWARFDebugInfoEntryMinimal &Die,
CompileUnit::DIEInfo &MyInfo, CompileUnit::DIEInfo &MyInfo,
const DebugMapObject &DMO, const DebugMapObject &DMO,
CompileUnit &CU, bool UseODR) { CompileUnit &CU, bool UseODR) {
@ -2013,7 +2053,7 @@ void DwarfLinker::keepDIEAndDenpendencies(const DWARFDebugInfoEntryMinimal &Die,
unsigned AncestorIdx = MyInfo.ParentIdx; unsigned AncestorIdx = MyInfo.ParentIdx;
while (!CU.getInfo(AncestorIdx).Keep) { while (!CU.getInfo(AncestorIdx).Keep) {
unsigned ODRFlag = UseODR ? TF_ODR : 0; unsigned ODRFlag = UseODR ? TF_ODR : 0;
lookForDIEsToKeep(*Unit.getDIEAtIndex(AncestorIdx), DMO, CU, lookForDIEsToKeep(RelocMgr, *Unit.getDIEAtIndex(AncestorIdx), DMO, CU,
TF_ParentWalk | TF_Keep | TF_DependencyWalk | ODRFlag); TF_ParentWalk | TF_Keep | TF_DependencyWalk | ODRFlag);
AncestorIdx = CU.getInfo(AncestorIdx).ParentIdx; AncestorIdx = CU.getInfo(AncestorIdx).ParentIdx;
} }
@ -2053,7 +2093,7 @@ void DwarfLinker::keepDIEAndDenpendencies(const DWARFDebugInfoEntryMinimal &Die,
continue; continue;
unsigned ODRFlag = UseODR ? TF_ODR : 0; unsigned ODRFlag = UseODR ? TF_ODR : 0;
lookForDIEsToKeep(*RefDIE, DMO, *ReferencedCU, lookForDIEsToKeep(RelocMgr, *RefDIE, DMO, *ReferencedCU,
TF_Keep | TF_DependencyWalk | ODRFlag); TF_Keep | TF_DependencyWalk | ODRFlag);
} }
} }
@ -2071,7 +2111,8 @@ void DwarfLinker::keepDIEAndDenpendencies(const DWARFDebugInfoEntryMinimal &Die,
/// also called, but during these dependency walks the file order is /// also called, but during these dependency walks the file order is
/// not respected. The TF_DependencyWalk flag tells us which kind of /// not respected. The TF_DependencyWalk flag tells us which kind of
/// traversal we are currently doing. /// traversal we are currently doing.
void DwarfLinker::lookForDIEsToKeep(const DWARFDebugInfoEntryMinimal &Die, void DwarfLinker::lookForDIEsToKeep(RelocationManager &RelocMgr,
const DWARFDebugInfoEntryMinimal &Die,
const DebugMapObject &DMO, CompileUnit &CU, const DebugMapObject &DMO, CompileUnit &CU,
unsigned Flags) { unsigned Flags) {
unsigned Idx = CU.getOrigUnit().getDIEIndex(&Die); unsigned Idx = CU.getOrigUnit().getDIEIndex(&Die);
@ -2087,12 +2128,12 @@ void DwarfLinker::lookForDIEsToKeep(const DWARFDebugInfoEntryMinimal &Die,
// We must not call shouldKeepDIE while called from keepDIEAndDenpendencies, // We must not call shouldKeepDIE while called from keepDIEAndDenpendencies,
// because it would screw up the relocation finding logic. // because it would screw up the relocation finding logic.
if (!(Flags & TF_DependencyWalk)) if (!(Flags & TF_DependencyWalk))
Flags = shouldKeepDIE(Die, CU, MyInfo, Flags); Flags = shouldKeepDIE(RelocMgr, Die, CU, MyInfo, Flags);
// If it is a newly kept DIE mark it as well as all its dependencies as kept. // If it is a newly kept DIE mark it as well as all its dependencies as kept.
if (!AlreadyKept && (Flags & TF_Keep)) { if (!AlreadyKept && (Flags & TF_Keep)) {
bool UseOdr = (Flags & TF_DependencyWalk) ? (Flags & TF_ODR) : CU.hasODR(); bool UseOdr = (Flags & TF_DependencyWalk) ? (Flags & TF_ODR) : CU.hasODR();
keepDIEAndDenpendencies(Die, MyInfo, DMO, CU, UseOdr); keepDIEAndDenpendencies(RelocMgr, Die, MyInfo, DMO, CU, UseOdr);
} }
// The TF_ParentWalk flag tells us that we are currently walking up // The TF_ParentWalk flag tells us that we are currently walking up
// the parent chain of a required DIE, and we don't want to mark all // the parent chain of a required DIE, and we don't want to mark all
@ -2108,7 +2149,7 @@ void DwarfLinker::lookForDIEsToKeep(const DWARFDebugInfoEntryMinimal &Die,
for (auto *Child = Die.getFirstChild(); Child && !Child->isNULL(); for (auto *Child = Die.getFirstChild(); Child && !Child->isNULL();
Child = Child->getSibling()) Child = Child->getSibling())
lookForDIEsToKeep(*Child, DMO, CU, Flags); lookForDIEsToKeep(RelocMgr, *Child, DMO, CU, Flags);
} }
/// \brief Assign an abbreviation numer to \p Abbrev. /// \brief Assign an abbreviation numer to \p Abbrev.
@ -2412,8 +2453,9 @@ unsigned DwarfLinker::cloneAttribute(DIE &Die,
/// monotonic \p BaseOffset values. /// monotonic \p BaseOffset values.
/// ///
/// \returns wether any reloc has been applied. /// \returns wether any reloc has been applied.
bool DwarfLinker::applyValidRelocs(MutableArrayRef<char> Data, bool DwarfLinker::RelocationManager::
uint32_t BaseOffset, bool isLittleEndian) { applyValidRelocs(MutableArrayRef<char> Data, uint32_t BaseOffset,
bool isLittleEndian) {
assert((NextValidReloc == 0 || assert((NextValidReloc == 0 ||
BaseOffset > ValidRelocs[NextValidReloc - 1].Offset) && BaseOffset > ValidRelocs[NextValidReloc - 1].Offset) &&
"BaseOffset should only be increasing."); "BaseOffset should only be increasing.");
@ -2512,7 +2554,8 @@ shouldSkipAttribute(DWARFAbbreviationDeclaration::AttributeSpec AttrSpec,
/// lie in the linked compile unit. /// lie in the linked compile unit.
/// ///
/// \returns the cloned DIE object or null if nothing was selected. /// \returns the cloned DIE object or null if nothing was selected.
DIE *DwarfLinker::cloneDIE(const DWARFDebugInfoEntryMinimal &InputDIE, DIE *DwarfLinker::cloneDIE(RelocationManager &RelocMgr,
const DWARFDebugInfoEntryMinimal &InputDIE,
CompileUnit &Unit, int64_t PCOffset, CompileUnit &Unit, int64_t PCOffset,
uint32_t OutOffset, unsigned Flags) { uint32_t OutOffset, unsigned Flags) {
DWARFUnit &U = Unit.getOrigUnit(); DWARFUnit &U = Unit.getOrigUnit();
@ -2551,7 +2594,7 @@ DIE *DwarfLinker::cloneDIE(const DWARFDebugInfoEntryMinimal &InputDIE,
SmallString<40> DIECopy(Data.getData().substr(Offset, NextOffset - Offset)); SmallString<40> DIECopy(Data.getData().substr(Offset, NextOffset - Offset));
Data = DataExtractor(DIECopy, Data.isLittleEndian(), Data.getAddressSize()); Data = DataExtractor(DIECopy, Data.isLittleEndian(), Data.getAddressSize());
// Modify the copy with relocated addresses. // Modify the copy with relocated addresses.
if (applyValidRelocs(DIECopy, Offset, Data.isLittleEndian())) { if (RelocMgr.applyValidRelocs(DIECopy, Offset, Data.isLittleEndian())) {
// If we applied relocations, we store the value of high_pc that was // If we applied relocations, we store the value of high_pc that was
// potentially stored in the input DIE. If high_pc is an address // potentially stored in the input DIE. If high_pc is an address
// (Dwarf version == 2), then it might have been relocated to a // (Dwarf version == 2), then it might have been relocated to a
@ -2652,7 +2695,8 @@ DIE *DwarfLinker::cloneDIE(const DWARFDebugInfoEntryMinimal &InputDIE,
// Recursively clone children. // Recursively clone children.
for (auto *Child = InputDIE.getFirstChild(); Child && !Child->isNULL(); for (auto *Child = InputDIE.getFirstChild(); Child && !Child->isNULL();
Child = Child->getSibling()) { Child = Child->getSibling()) {
if (DIE *Clone = cloneDIE(*Child, Unit, PCOffset, OutOffset, Flags)) { if (DIE *Clone =
cloneDIE(RelocMgr, *Child, Unit, PCOffset, OutOffset, Flags)) {
Die->addChild(Clone); Die->addChild(Clone);
OutOffset = Clone->getOffset() + Clone->getSize(); OutOffset = Clone->getOffset() + Clone->getSize();
} }
@ -3020,13 +3064,53 @@ DwarfLinker::loadObject(BinaryHolder &BinaryHolder, DebugMapObject &Obj,
return ErrOrObj; return ErrOrObj;
} }
void DwarfLinker::cloneCompileUnit(RelocationManager &RelocMgr,
MutableArrayRef<CompileUnit> CompileUnits,
DWARFContextInMemory &DwarfContext) {
if (!Streamer)
return;
for (auto &CurrentUnit : CompileUnits) {
const auto *InputDIE = CurrentUnit.getOrigUnit().getUnitDIE();
CurrentUnit.setStartOffset(OutputDebugInfoSize);
DIE *OutputDIE = cloneDIE(RelocMgr, *InputDIE, CurrentUnit,
0 /* PC offset */, 11 /* Unit Header size */, 0);
CurrentUnit.setOutputUnitDIE(OutputDIE);
OutputDebugInfoSize = CurrentUnit.computeNextUnitOffset();
if (Options.NoOutput)
continue;
// FIXME: for compatibility with the classic dsymutil, we emit
// an empty line table for the unit, even if the unit doesn't
// actually exist in the DIE tree.
patchLineTableForUnit(CurrentUnit, DwarfContext);
if (!OutputDIE)
continue;
patchRangesForUnit(CurrentUnit, DwarfContext);
Streamer->emitLocationsForUnit(CurrentUnit, DwarfContext);
emitAcceleratorEntriesForUnit(CurrentUnit);
}
if (Options.NoOutput)
return;
// Emit all the compile unit's debug information.
for (auto &CurrentUnit : CompileUnits) {
generateUnitRanges(CurrentUnit);
CurrentUnit.fixupForwardReferences();
Streamer->emitCompileUnitHeader(CurrentUnit);
if (!CurrentUnit.getOutputUnitDIE())
continue;
Streamer->emitDIE(*CurrentUnit.getOutputUnitDIE());
}
}
bool DwarfLinker::link(const DebugMap &Map) { bool DwarfLinker::link(const DebugMap &Map) {
if (!createStreamer(Map.getTriple(), OutputFilename)) if (!createStreamer(Map.getTriple(), OutputFilename))
return false; return false;
// Size of the DIEs (and headers) generated for the linked output. // Size of the DIEs (and headers) generated for the linked output.
uint64_t OutputDebugInfoSize = 0; OutputDebugInfoSize = 0;
// A unique ID that identifies each compile unit. // A unique ID that identifies each compile unit.
unsigned UnitID = 0; unsigned UnitID = 0;
for (const auto &Obj : Map.objects()) { for (const auto &Obj : Map.objects()) {
@ -3039,7 +3123,8 @@ bool DwarfLinker::link(const DebugMap &Map) {
continue; continue;
// Look for relocations that correspond to debug map entries. // Look for relocations that correspond to debug map entries.
if (!findValidRelocsInDebugInfo(*ErrOrObj, *Obj)) { RelocationManager RelocMgr(*this);
if (!RelocMgr.findValidRelocsInDebugInfo(*ErrOrObj, *Obj)) {
if (Options.Verbose) if (Options.Verbose)
outs() << "No valid relocations found. Skipping.\n"; outs() << "No valid relocations found. Skipping.\n";
continue; continue;
@ -3068,50 +3153,16 @@ bool DwarfLinker::link(const DebugMap &Map) {
// references require the ParentIdx to be setup for every CU in // references require the ParentIdx to be setup for every CU in
// the object file before calling this. // the object file before calling this.
for (auto &CurrentUnit : Units) for (auto &CurrentUnit : Units)
lookForDIEsToKeep(*CurrentUnit.getOrigUnit().getUnitDIE(), *Obj, lookForDIEsToKeep(RelocMgr, *CurrentUnit.getOrigUnit().getUnitDIE(), *Obj,
CurrentUnit, 0); CurrentUnit, 0);
// The calls to applyValidRelocs inside cloneDIE will walk the // The calls to applyValidRelocs inside cloneDIE will walk the
// reloc array again (in the same way findValidRelocsInDebugInfo() // reloc array again (in the same way findValidRelocsInDebugInfo()
// did). We need to reset the NextValidReloc index to the beginning. // did). We need to reset the NextValidReloc index to the beginning.
NextValidReloc = 0; RelocMgr.resetValidRelocs();
if (RelocMgr.hasValidRelocs())
// Construct the output DIE tree by cloning the DIEs we chose to cloneCompileUnit(RelocMgr, Units, DwarfContext);
// keep above. If there are no valid relocs, then there's nothing if (!Options.NoOutput && !Units.empty())
// to clone/emit.
if (!ValidRelocs.empty())
for (auto &CurrentUnit : Units) {
const auto *InputDIE = CurrentUnit.getOrigUnit().getUnitDIE();
CurrentUnit.setStartOffset(OutputDebugInfoSize);
DIE *OutputDIE = cloneDIE(*InputDIE, CurrentUnit, 0 /* PCOffset */,
11 /* Unit Header size */, /* Flags =*/0);
CurrentUnit.setOutputUnitDIE(OutputDIE);
OutputDebugInfoSize = CurrentUnit.computeNextUnitOffset();
if (Options.NoOutput)
continue;
// FIXME: for compatibility with the classic dsymutil, we emit
// an empty line table for the unit, even if the unit doesn't
// actually exist in the DIE tree.
patchLineTableForUnit(CurrentUnit, DwarfContext);
if (!OutputDIE)
continue;
patchRangesForUnit(CurrentUnit, DwarfContext);
Streamer->emitLocationsForUnit(CurrentUnit, DwarfContext);
emitAcceleratorEntriesForUnit(CurrentUnit);
}
// Emit all the compile unit's debug information.
if (!ValidRelocs.empty() && !Options.NoOutput)
for (auto &CurrentUnit : Units) {
generateUnitRanges(CurrentUnit);
CurrentUnit.fixupForwardReferences();
Streamer->emitCompileUnitHeader(CurrentUnit);
if (!CurrentUnit.getOutputUnitDIE())
continue;
Streamer->emitDIE(*CurrentUnit.getOutputUnitDIE());
}
if (!ValidRelocs.empty() && !Options.NoOutput && !Units.empty())
patchFrameInfoForObject(*Obj, DwarfContext, patchFrameInfoForObject(*Obj, DwarfContext,
Units[0].getOrigUnit().getAddressByteSize()); Units[0].getOrigUnit().getAddressByteSize());