From 0ad18f888ed464b28e947dbc92ea4d6764caf128 Mon Sep 17 00:00:00 2001 From: Eugene Zelenko Date: Wed, 1 Nov 2017 21:16:06 +0000 Subject: [PATCH] [dsymutil, llvm-objcopy] Fix some Clang-tidy modernize and Include What You Use warnings; other minor fixes (NFC). llvm-svn: 317123 --- llvm/tools/dsymutil/DebugMap.cpp | 43 +++- llvm/tools/dsymutil/DebugMap.h | 54 +++-- llvm/tools/dsymutil/DwarfLinker.cpp | 216 +++++++++++++----- .../tools/dsymutil/NonRelocatableStringpool.h | 23 +- llvm/tools/dsymutil/dsymutil.cpp | 23 +- llvm/tools/dsymutil/dsymutil.h | 38 ++- llvm/tools/llvm-objcopy/Object.cpp | 79 ++++--- llvm/tools/llvm-objcopy/Object.h | 202 ++++++++-------- llvm/tools/llvm-objcopy/llvm-objcopy.cpp | 51 +++-- llvm/tools/llvm-objcopy/llvm-objcopy.h | 13 +- 10 files changed, 490 insertions(+), 252 deletions(-) diff --git a/llvm/tools/dsymutil/DebugMap.cpp b/llvm/tools/dsymutil/DebugMap.cpp index 7f2057674618..b543c212fe80 100644 --- a/llvm/tools/dsymutil/DebugMap.cpp +++ b/llvm/tools/dsymutil/DebugMap.cpp @@ -6,16 +6,34 @@ // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// + #include "DebugMap.h" #include "BinaryHolder.h" -#include "llvm/ADT/STLExtras.h" +#include "llvm/ADT/Optional.h" +#include "llvm/ADT/SmallString.h" +#include "llvm/ADT/StringMap.h" +#include "llvm/ADT/StringRef.h" +#include "llvm/ADT/Triple.h" #include "llvm/ADT/iterator_range.h" -#include "llvm/Support/DataTypes.h" +#include "llvm/BinaryFormat/MachO.h" +#include "llvm/Object/ObjectFile.h" +#include "llvm/Support/Chrono.h" +#include "llvm/Support/Error.h" #include "llvm/Support/Format.h" +#include "llvm/Support/MemoryBuffer.h" +#include "llvm/Support/Path.h" +#include "llvm/Support/YAMLTraits.h" #include "llvm/Support/raw_ostream.h" #include +#include +#include +#include +#include +#include +#include namespace llvm { + namespace dsymutil { using namespace llvm::object; @@ -39,7 +57,7 @@ void DebugMapObject::print(raw_ostream &OS) const { OS << getObjectFilename() << ":\n"; // Sort the symbols in alphabetical order, like llvm-nm (and to get // deterministic output for testing). - typedef std::pair Entry; + using Entry = std::pair; std::vector Entries; Entries.reserve(Symbols.getNumItems()); for (const auto &Sym : make_range(Symbols.begin(), Symbols.end())) @@ -97,11 +115,13 @@ void DebugMap::dump() const { print(errs()); } #endif namespace { + struct YAMLContext { StringRef PrependPath; Triple BinaryTriple; }; -} + +} // end anonymous namespace ErrorOr>> DebugMap::parseYAMLDebugMap(StringRef InputFile, StringRef PrependPath, @@ -124,7 +144,8 @@ DebugMap::parseYAMLDebugMap(StringRef InputFile, StringRef PrependPath, Result.push_back(std::move(Res)); return std::move(Result); } -} + +} // end namespace dsymutil namespace yaml { @@ -155,8 +176,7 @@ void MappingTraits::mapping( io.mapRequired("symbols", Norm->Entries); } -void ScalarTraits::output(const Triple &val, void *, - llvm::raw_ostream &out) { +void ScalarTraits::output(const Triple &val, void *, raw_ostream &out) { out << val.str(); } @@ -221,8 +241,7 @@ MappingTraits::YamlDMO::denormalize(IO &IO) { sys::path::append(Path, Filename); auto ErrOrObjectFiles = BinHolder.GetObjectFiles(Path); if (auto EC = ErrOrObjectFiles.getError()) { - llvm::errs() << "warning: Unable to open " << Path << " " << EC.message() - << '\n'; + errs() << "warning: Unable to open " << Path << " " << EC.message() << '\n'; } else if (auto ErrOrObjectFile = BinHolder.Get(Ctxt.BinaryTriple)) { // Rewrite the object file symbol addresses in the debug map. The // YAML input is mainly used to test llvm-dsymutil without @@ -256,5 +275,7 @@ MappingTraits::YamlDMO::denormalize(IO &IO) { } return Res; } -} -} + +} // end namespace yaml + +} // end namespace llvm diff --git a/llvm/tools/dsymutil/DebugMap.h b/llvm/tools/dsymutil/DebugMap.h index 0b564149488a..2bdd333c8380 100644 --- a/llvm/tools/dsymutil/DebugMap.h +++ b/llvm/tools/dsymutil/DebugMap.h @@ -1,4 +1,4 @@ -//=== tools/dsymutil/DebugMap.h - Generic debug map representation -*- C++ -*-// +//=- tools/dsymutil/DebugMap.h - Generic debug map representation -*- C++ -*-=// // // The LLVM Linker // @@ -6,7 +6,7 @@ // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// -/// +// /// \file /// /// This file contains the class declaration of the DebugMap @@ -16,27 +16,35 @@ /// The DebugMap is an input to the DwarfLinker class that will /// extract the Dwarf debug information from the referenced object /// files and link their usefull debug info together. -/// +// //===----------------------------------------------------------------------===// + #ifndef LLVM_TOOLS_DSYMUTIL_DEBUGMAP_H #define LLVM_TOOLS_DSYMUTIL_DEBUGMAP_H #include "llvm/ADT/DenseMap.h" +#include "llvm/ADT/Optional.h" #include "llvm/ADT/StringMap.h" +#include "llvm/ADT/StringRef.h" #include "llvm/ADT/Triple.h" #include "llvm/ADT/iterator_range.h" -#include "llvm/Object/ObjectFile.h" #include "llvm/Support/Chrono.h" #include "llvm/Support/ErrorOr.h" -#include "llvm/Support/Format.h" -#include "llvm/Support/Path.h" #include "llvm/Support/YAMLTraits.h" +#include +#include +#include +#include +#include +#include #include namespace llvm { + class raw_ostream; namespace dsymutil { + class DebugMapObject; /// \brief The DebugMap object stores the list of object files to @@ -67,20 +75,24 @@ class DebugMapObject; class DebugMap { Triple BinaryTriple; std::string BinaryPath; - typedef std::vector> ObjectContainer; + + using ObjectContainer = std::vector>; + ObjectContainer Objects; /// For YAML IO support. ///@{ friend yaml::MappingTraits>; friend yaml::MappingTraits; + DebugMap() = default; ///@} + public: DebugMap(const Triple &BinaryTriple, StringRef BinaryPath) : BinaryTriple(BinaryTriple), BinaryPath(BinaryPath) {} - typedef ObjectContainer::const_iterator const_iterator; + using const_iterator = ObjectContainer::const_iterator; iterator_range objects() const { return make_range(begin(), end()); @@ -122,23 +134,25 @@ public: Optional ObjectAddress; yaml::Hex64 BinaryAddress; yaml::Hex32 Size; + SymbolMapping(Optional ObjectAddr, uint64_t BinaryAddress, uint32_t Size) : BinaryAddress(BinaryAddress), Size(Size) { if (ObjectAddr) ObjectAddress = *ObjectAddr; } + /// For YAML IO support SymbolMapping() = default; }; - typedef std::pair YAMLSymbolMapping; - typedef StringMapEntry DebugMapEntry; + using YAMLSymbolMapping = std::pair; + using DebugMapEntry = StringMapEntry; /// \brief Adds a symbol mapping to this DebugMapObject. /// \returns false if the symbol was already registered. The request /// is discarded in this case. - bool addSymbol(llvm::StringRef SymName, Optional ObjectAddress, + bool addSymbol(StringRef SymName, Optional ObjectAddress, uint64_t LinkedAddress, uint32_t Size); /// \brief Lookup a symbol mapping. @@ -149,7 +163,7 @@ public: /// \returns null if the address isn't found. const DebugMapEntry *lookupObjectAddress(uint64_t Address) const; - llvm::StringRef getObjectFilename() const { return Filename; } + StringRef getObjectFilename() const { return Filename; } sys::TimePoint getTimestamp() const { return Timestamp; @@ -165,8 +179,10 @@ public: #ifndef NDEBUG void dump() const; #endif + private: friend class DebugMap; + /// DebugMapObjects can only be constructed by the owning DebugMap. DebugMapObject(StringRef ObjectFilename, sys::TimePoint Timestamp, uint8_t Type); @@ -181,6 +197,7 @@ private: ///@{ friend yaml::MappingTraits; friend yaml::SequenceTraits>>; + DebugMapObject() = default; public: @@ -188,8 +205,10 @@ public: DebugMapObject &operator=(DebugMapObject &&) = default; ///@} }; -} -} + +} // end namespace dsymutil + +} // end namespace llvm LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::dsymutil::DebugMapObject::YAMLSymbolMapping) @@ -211,7 +230,7 @@ template <> struct MappingTraits { }; template <> struct ScalarTraits { - static void output(const Triple &val, void *, llvm::raw_ostream &out); + static void output(const Triple &val, void *, raw_ostream &out); static StringRef input(StringRef scalar, void *, Triple &value); static bool mustQuote(StringRef) { return true; } }; @@ -232,7 +251,8 @@ template <> struct MappingTraits { template <> struct MappingTraits> { static void mapping(IO &io, std::unique_ptr &DM); }; -} -} + +} // end namespace yaml +} // end namespace llvm #endif // LLVM_TOOLS_DSYMUTIL_DEBUGMAP_H diff --git a/llvm/tools/dsymutil/DwarfLinker.cpp b/llvm/tools/dsymutil/DwarfLinker.cpp index a726ba4e7b61..9fb968cb5d25 100644 --- a/llvm/tools/dsymutil/DwarfLinker.cpp +++ b/llvm/tools/dsymutil/DwarfLinker.cpp @@ -6,21 +6,43 @@ // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// + #include "BinaryHolder.h" #include "DebugMap.h" #include "MachOUtils.h" #include "NonRelocatableStringpool.h" #include "dsymutil.h" +#include "llvm/ADT/ArrayRef.h" +#include "llvm/ADT/DenseMap.h" +#include "llvm/ADT/DenseMapInfo.h" +#include "llvm/ADT/DenseSet.h" +#include "llvm/ADT/FoldingSet.h" +#include "llvm/ADT/Hashing.h" #include "llvm/ADT/IntervalMap.h" +#include "llvm/ADT/None.h" +#include "llvm/ADT/Optional.h" +#include "llvm/ADT/PointerIntPair.h" #include "llvm/ADT/STLExtras.h" +#include "llvm/ADT/SmallString.h" #include "llvm/ADT/StringMap.h" +#include "llvm/ADT/StringRef.h" +#include "llvm/ADT/Triple.h" +#include "llvm/ADT/Twine.h" #include "llvm/BinaryFormat/Dwarf.h" +#include "llvm/BinaryFormat/MachO.h" #include "llvm/CodeGen/AsmPrinter.h" #include "llvm/CodeGen/DIE.h" #include "llvm/Config/config.h" +#include "llvm/DebugInfo/DIContext.h" +#include "llvm/DebugInfo/DWARF/DWARFAbbreviationDeclaration.h" #include "llvm/DebugInfo/DWARF/DWARFContext.h" -#include "llvm/DebugInfo/DWARF/DWARFDebugInfoEntry.h" +#include "llvm/DebugInfo/DWARF/DWARFDataExtractor.h" +#include "llvm/DebugInfo/DWARF/DWARFDebugLine.h" +#include "llvm/DebugInfo/DWARF/DWARFDebugRangeList.h" +#include "llvm/DebugInfo/DWARF/DWARFDie.h" #include "llvm/DebugInfo/DWARF/DWARFFormValue.h" +#include "llvm/DebugInfo/DWARF/DWARFSection.h" +#include "llvm/DebugInfo/DWARF/DWARFUnit.h" #include "llvm/MC/MCAsmBackend.h" #include "llvm/MC/MCAsmInfo.h" #include "llvm/MC/MCCodeEmitter.h" @@ -29,18 +51,47 @@ #include "llvm/MC/MCInstrInfo.h" #include "llvm/MC/MCObjectFileInfo.h" #include "llvm/MC/MCRegisterInfo.h" +#include "llvm/MC/MCSection.h" #include "llvm/MC/MCStreamer.h" #include "llvm/MC/MCSubtargetInfo.h" +#include "llvm/MC/MCTargetOptions.h" #include "llvm/MC/MCTargetOptionsCommandFlags.h" #include "llvm/Object/MachO.h" +#include "llvm/Object/ObjectFile.h" +#include "llvm/Object/SymbolicFile.h" +#include "llvm/Support/Allocator.h" +#include "llvm/Support/Casting.h" +#include "llvm/Support/Compiler.h" +#include "llvm/Support/DataExtractor.h" +#include "llvm/Support/Error.h" +#include "llvm/Support/ErrorHandling.h" +#include "llvm/Support/ErrorOr.h" +#include "llvm/Support/FileSystem.h" +#include "llvm/Support/Format.h" #include "llvm/Support/LEB128.h" +#include "llvm/Support/MathExtras.h" +#include "llvm/Support/MemoryBuffer.h" +#include "llvm/Support/Path.h" #include "llvm/Support/TargetRegistry.h" #include "llvm/Support/ToolOutputFile.h" +#include "llvm/Support/raw_ostream.h" #include "llvm/Target/TargetMachine.h" #include "llvm/Target/TargetOptions.h" +#include +#include +#include +#include +#include +#include +#include +#include +#include #include #include +#include #include +#include +#include namespace llvm { namespace dsymutil { @@ -52,7 +103,7 @@ using HalfOpenIntervalMap = IntervalMap::LeafSize, IntervalMapHalfOpenInfo>; -typedef HalfOpenIntervalMap FunctionIntervals; +using FunctionIntervals = HalfOpenIntervalMap; // FIXME: Delete this structure. struct PatchLocation { @@ -95,6 +146,8 @@ struct DeclMapInfo; /// specific DeclContext using a separate DenseMap keyed on the hash /// of the fully qualified name of the context. class DeclContext { + friend DeclMapInfo; + unsigned QualifiedNameHash = 0; uint32_t Line = 0; uint32_t ByteSize = 0; @@ -107,10 +160,8 @@ class DeclContext { uint32_t LastSeenCompileUnitID = 0; uint32_t CanonicalDIEOffset = 0; - friend DeclMapInfo; - public: - typedef DenseSet Map; + using Map = DenseSet; DeclContext() : DefinedInClangModule(0), Parent(*this) {} @@ -186,20 +237,35 @@ class CompileUnit { public: /// Information gathered about a DIE in the object file. struct DIEInfo { - int64_t AddrAdjust; ///< Address offset to apply to the described entity. - DeclContext *Ctxt; ///< ODR Declaration context. - DIE *Clone; ///< Cloned version of that DIE. - uint32_t ParentIdx; ///< The index of this DIE's parent. - bool Keep : 1; ///< Is the DIE part of the linked output? - bool InDebugMap : 1; ///< Was this DIE's entity found in the map? - bool Prune : 1; ///< Is this a pure forward declaration we can strip? - bool Incomplete : 1; ///< Does DIE transitively refer an incomplete decl? + /// Address offset to apply to the described entity. + int64_t AddrAdjust; + + /// ODR Declaration context. + DeclContext *Ctxt; + + /// Cloned version of that DIE. + DIE *Clone; + + /// The index of this DIE's parent. + uint32_t ParentIdx; + + /// Is the DIE part of the linked output? + bool Keep : 1; + + /// Was this DIE's entity found in the map? + bool InDebugMap : 1; + + /// Is this a pure forward declaration we can strip? + bool Prune : 1; + + /// Does DIE transitively refer an incomplete decl? + bool Incomplete : 1; }; CompileUnit(DWARFUnit &OrigUnit, unsigned ID, bool CanUseODR, StringRef ClangModuleName) - : OrigUnit(OrigUnit), ID(ID), LowPc(UINT64_MAX), HighPc(0), RangeAlloc(), - Ranges(RangeAlloc), ClangModuleName(ClangModuleName) { + : OrigUnit(OrigUnit), ID(ID), Ranges(RangeAlloc), + ClangModuleName(ClangModuleName) { Info.resize(OrigUnit.getNumDIEs()); auto CUDie = OrigUnit.getUnitDIE(false); @@ -245,7 +311,9 @@ public: Optional getUnitRangesAttribute() const { return UnitRangeAttribute; } + const FunctionIntervals &getFunctionRanges() const { return Ranges; } + const std::vector &getRangesAttributes() const { return RangeAttributes; } @@ -337,8 +405,8 @@ private: uint64_t StartOffset; uint64_t NextUnitOffset; - uint64_t LowPc; - uint64_t HighPc; + uint64_t LowPc = std::numeric_limits::max(); + uint64_t HighPc = 0; /// A list of attributes to fixup with the absolute offset of /// a DIE in the debug_info section. @@ -350,6 +418,7 @@ private: PatchLocation>> ForwardDIEReferences; FunctionIntervals::Allocator RangeAlloc; + /// The ranges in that interval map are the PC ranges for /// functions in this unit, associated with the PC offset to apply /// to the addresses to get the linked address. @@ -383,12 +452,16 @@ private: /// Is this unit subject to the ODR rule? bool HasODR; + /// Did a DIE actually contain a valid reloc? bool HasInterestingContent; + /// If this is a Clang module, this holds the module's name. std::string ClangModuleName; }; +} // end anonymous namespace + void CompileUnit::markEverythingAsKept() { for (auto &I : Info) // Mark everything that wasn't explicity marked for pruning. @@ -459,6 +532,8 @@ void CompileUnit::addTypeAccelerator(const DIE *Die, const char *Name, Pubtypes.emplace_back(Name, Die, Offset, false); } +namespace { + /// The Dwarf streaming logic /// /// All interactions with the MC layer that is used to build the debug @@ -573,6 +648,8 @@ public: uint32_t getFrameSectionSize() const { return FrameSectionSize; } }; +} // end anonymous namespace + bool DwarfStreamer::init(Triple TheTriple, StringRef OutputFilename) { std::string ErrorStr; std::string TripleName; @@ -919,7 +996,8 @@ void DwarfStreamer::emitLineTableForUnit(MCDwarfLineTableParams Params, if (Rows.empty()) { // We only have the dummy entry, dsymutil emits an entry with a 0 // address in that case. - MCDwarfLineAddr::Encode(*MC, Params, INT64_MAX, 0, EncodingOS); + MCDwarfLineAddr::Encode(*MC, Params, std::numeric_limits::max(), 0, + EncodingOS); MS->EmitBytes(EncodingOS.str()); LineSectionSize += EncodingBuffer.size(); MS->EmitLabel(LineEndSym); @@ -1019,7 +1097,8 @@ void DwarfStreamer::emitLineTableForUnit(MCDwarfLineTableParams Params, MS->EmitULEB128IntValue(AddressDelta); LineSectionSize += 1 + getULEB128Size(AddressDelta); } - MCDwarfLineAddr::Encode(*MC, Params, INT64_MAX, 0, EncodingOS); + MCDwarfLineAddr::Encode(*MC, Params, std::numeric_limits::max(), + 0, EncodingOS); MS->EmitBytes(EncodingOS.str()); LineSectionSize += EncodingBuffer.size(); EncodingBuffer.resize(0); @@ -1030,7 +1109,8 @@ void DwarfStreamer::emitLineTableForUnit(MCDwarfLineTableParams Params, } if (RowsSinceLastSequence) { - MCDwarfLineAddr::Encode(*MC, Params, INT64_MAX, 0, EncodingOS); + MCDwarfLineAddr::Encode(*MC, Params, std::numeric_limits::max(), 0, + EncodingOS); MS->EmitBytes(EncodingOS.str()); LineSectionSize += EncodingBuffer.size(); EncodingBuffer.resize(0); @@ -1112,6 +1192,8 @@ void DwarfStreamer::emitFDE(uint32_t CIEOffset, uint32_t AddrSize, FrameSectionSize += FDEBytes.size() + 8 + AddrSize; } +namespace { + /// The core of the Dwarf linking logic. /// /// The link of the dwarf information from the object files will be @@ -1179,13 +1261,13 @@ private: /// 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; + unsigned NextValidReloc = 0; public: - RelocationManager(DwarfLinker &Linker) - : Linker(Linker), NextValidReloc(0) {} + RelocationManager(DwarfLinker &Linker) : Linker(Linker) {} bool hasValidRelocs() const { return !ValidRelocs.empty(); } + /// Reset the NextValidReloc counter. void resetValidRelocs() { NextValidReloc = 0; } @@ -1284,8 +1366,10 @@ private: class DIECloner { DwarfLinker &Linker; RelocationManager &RelocMgr; + /// Allocator used for all the DIEValue objects. BumpPtrAllocator &DIEAlloc; + std::vector> &CompileUnits; LinkOptions Options; @@ -1318,25 +1402,35 @@ private: void cloneAllCompileUnits(DWARFContext &DwarfContext); private: - typedef DWARFAbbreviationDeclaration::AttributeSpec AttributeSpec; + using AttributeSpec = DWARFAbbreviationDeclaration::AttributeSpec; /// Information gathered and exchanged between the various /// clone*Attributes helpers about the attributes of a particular DIE. struct AttributesInfo { - const char *Name, *MangledName; ///< Names. - uint32_t NameOffset, MangledNameOffset; ///< Offsets in the string pool. + /// Names. + const char *Name = nullptr; + const char *MangledName = nullptr; - uint64_t OrigLowPc; ///< Value of AT_low_pc in the input DIE - uint64_t OrigHighPc; ///< Value of AT_high_pc in the input DIE - int64_t PCOffset; ///< Offset to apply to PC addresses inside a function. + /// Offsets in the string pool. + uint32_t NameOffset = 0; + uint32_t MangledNameOffset = 0; - bool HasLowPc; ///< Does the DIE have a low_pc attribute? - bool IsDeclaration; ///< Is this DIE only a declaration? + /// Value of AT_low_pc in the input DIE + uint64_t OrigLowPc = std::numeric_limits::max(); - AttributesInfo() - : Name(nullptr), MangledName(nullptr), NameOffset(0), - MangledNameOffset(0), OrigLowPc(UINT64_MAX), OrigHighPc(0), - PCOffset(0), HasLowPc(false), IsDeclaration(false) {} + /// Value of AT_high_pc in the input DIE + uint64_t OrigHighPc = 0; + + /// Offset to apply to PC addresses inside a function. + int64_t PCOffset = 0; + + /// Does the DIE have a low_pc attribute? + bool HasLowPc = false; + + /// Is this DIE only a declaration? + bool IsDeclaration = false; + + AttributesInfo() = default; }; /// Helper for cloneDIE. @@ -1397,13 +1491,6 @@ private: /// Assign an abbreviation number to \p Abbrev void AssignAbbrev(DIEAbbrev &Abbrev); - /// FoldingSet that uniques the abbreviations. - FoldingSet AbbreviationsSet; - /// Storage for the unique Abbreviations. - /// This is passed to AsmPrinter::emitDwarfAbbrevs(), thus it cannot - /// be changed to a vecot of unique_ptrs. - std::vector> Abbreviations; - /// Compute and emit debug_ranges section for \p Unit, and /// patch the attributes referencing it. void patchRangesForUnit(const CompileUnit &Unit, DWARFContext &Dwarf) const; @@ -1424,10 +1511,20 @@ private: void patchFrameInfoForObject(const DebugMapObject &, DWARFContext &, unsigned AddressSize); + /// FoldingSet that uniques the abbreviations. + FoldingSet AbbreviationsSet; + + /// Storage for the unique Abbreviations. + /// This is passed to AsmPrinter::emitDwarfAbbrevs(), thus it cannot + /// be changed to a vecot of unique_ptrs. + std::vector> Abbreviations; + /// DIELoc objects that need to be destructed (but not freed!). std::vector DIELocs; + /// DIEBlock objects that need to be destructed (but not freed!). std::vector DIEBlocks; + /// Allocator used for all the DIEValue objects. BumpPtrAllocator DIEAlloc; /// @} @@ -1451,13 +1548,15 @@ private: BinaryHolder BinHolder; std::unique_ptr Streamer; uint64_t OutputDebugInfoSize; - unsigned UnitID; ///< A unique ID that identifies each compile unit. + + /// A unique ID that identifies each compile unit. + unsigned UnitID; + unsigned MaxDwarfVersion = 0; /// The units of the current debug map object. std::vector> Units; - /// The debug map object currently under consideration. DebugMapObject *CurrentDebugObject; @@ -1489,6 +1588,8 @@ private: bool ArchiveHintDisplayed = false; }; +} // end anonymous namespace + /// Similar to DWARFUnitSection::getUnitForOffset(), but returning our /// CompileUnit object instead. static CompileUnit *getUnitForOffset( @@ -1633,7 +1734,7 @@ PointerIntPair DeclContextTree::getChildDeclContext( return PointerIntPair(nullptr); unsigned Line = 0; - unsigned ByteSize = UINT32_MAX; + unsigned ByteSize = std::numeric_limits::max(); if (!InClangModule) { // Gather some discriminating data about the DeclContext we will be @@ -1643,7 +1744,8 @@ PointerIntPair DeclContextTree::getChildDeclContext( // namespaces, use these additional data points to make the process // safer. This is disabled for clang modules, because forward // declarations of module-defined types do not have a file and line. - ByteSize = dwarf::toUnsigned(DIE.find(dwarf::DW_AT_byte_size), UINT64_MAX); + ByteSize = dwarf::toUnsigned(DIE.find(dwarf::DW_AT_byte_size), + std::numeric_limits::max()); if (Tag != dwarf::DW_TAG_namespace || !Name) { if (unsigned FileNum = dwarf::toUnsigned(DIE.find(dwarf::DW_AT_decl_file), 0)) { if (const auto *LT = U.getOrigUnit().getContext().getLineTableForUnit( @@ -2063,8 +2165,9 @@ hasValidRelocation(uint32_t StartOffset, uint32_t EndOffset, const auto &ValidReloc = ValidRelocs[NextValidReloc++]; const auto &Mapping = ValidReloc.Mapping->getValue(); - uint64_t ObjectAddress = - Mapping.ObjectAddress ? uint64_t(*Mapping.ObjectAddress) : UINT64_MAX; + uint64_t ObjectAddress = Mapping.ObjectAddress + ? uint64_t(*Mapping.ObjectAddress) + : std::numeric_limits::max(); if (Linker.Options.Verbose) outs() << "Found valid debug map entry: " << ValidReloc.Mapping->getKey() << " " << format("\t%016" PRIx64 " => %016" PRIx64, ObjectAddress, @@ -2539,11 +2642,13 @@ unsigned DwarfLinker::DIECloner::cloneAddressAttribute( // relocated because it happens to match the low_pc of the // enclosing subprogram. To prevent issues with that, always use // the low_pc from the input DIE if relocations have been applied. - Addr = (Info.OrigLowPc != UINT64_MAX ? Info.OrigLowPc : Addr) + + Addr = (Info.OrigLowPc != std::numeric_limits::max() + ? Info.OrigLowPc + : Addr) + Info.PCOffset; else if (Die.getTag() == dwarf::DW_TAG_compile_unit) { Addr = Unit.getLowPc(); - if (Addr == UINT64_MAX) + if (Addr == std::numeric_limits::max()) return 0; } Info.HasLowPc = true; @@ -2820,8 +2925,8 @@ DIE *DwarfLinker::DIECloner::cloneDIE( // Also store the low_pc. It might get relocated in an // inline_subprogram that happens at the beginning of its // inlining function. - AttrInfo.OrigLowPc = - dwarf::toAddress(InputDIE.find(dwarf::DW_AT_low_pc), UINT64_MAX); + AttrInfo.OrigLowPc = dwarf::toAddress(InputDIE.find(dwarf::DW_AT_low_pc), + std::numeric_limits::max()); } // Reset the Offset to 0 as we will be working on the local copy of @@ -3479,7 +3584,6 @@ void DwarfLinker::DIECloner::cloneAllCompileUnits(DWARFContext &DwarfContext) { } bool DwarfLinker::link(const DebugMap &Map) { - if (!createStreamer(Map.getTriple(), OutputFilename)) return false; @@ -3594,7 +3698,6 @@ bool DwarfLinker::link(const DebugMap &Map) { return Options.NoOutput ? true : Streamer->finish(Map); } -} /// Get the offset of string \p S in the string table. This /// can insert a new element or return the offset of a preexisitng @@ -3646,5 +3749,6 @@ bool linkDwarf(StringRef OutputFilename, const DebugMap &DM, DwarfLinker Linker(OutputFilename, Options); return Linker.link(DM); } -} -} + +} // end namespace dsymutil +} // end namespace llvm diff --git a/llvm/tools/dsymutil/NonRelocatableStringpool.h b/llvm/tools/dsymutil/NonRelocatableStringpool.h index fffd02084e34..aa21d77ad438 100644 --- a/llvm/tools/dsymutil/NonRelocatableStringpool.h +++ b/llvm/tools/dsymutil/NonRelocatableStringpool.h @@ -1,4 +1,4 @@ -//===-- NonRelocatableStringpool.h - A simple stringpool -----------------===// +//===- NonRelocatableStringpool.h - A simple stringpool --------*- C++ -*-===// // // The LLVM Compiler Infrastructure // @@ -6,10 +6,15 @@ // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// + #ifndef LLVM_TOOLS_DSYMUTIL_NONRELOCATABLESTRINGPOOL_H #define LLVM_TOOLS_DSYMUTIL_NONRELOCATABLESTRINGPOOL_H #include "llvm/ADT/StringMap.h" +#include "llvm/ADT/StringRef.h" +#include "llvm/Support/Allocator.h" +#include +#include namespace llvm { namespace dsymutil { @@ -25,11 +30,10 @@ public: /// \brief Entries are stored into the StringMap and simply linked /// together through the second element of this pair in order to /// keep track of insertion order. - typedef StringMap, BumpPtrAllocator> - MapTy; + using MapTy = + StringMap, BumpPtrAllocator>; - NonRelocatableStringpool() - : CurrentEndOffset(0), Sentinel(0), Last(&Sentinel) { + NonRelocatableStringpool() : Sentinel(0), Last(&Sentinel) { // Legacy dsymutil puts an empty string at the start of the line // table. getStringOffset(""); @@ -61,10 +65,11 @@ public: private: MapTy Strings; - uint32_t CurrentEndOffset; + uint32_t CurrentEndOffset = 0; MapTy::MapEntryTy Sentinel, *Last; }; -} -} -#endif +} // end namespace dsymutil +} // end namespace llvm + +#endif // LLVM_TOOLS_DSYMUTIL_NONRELOCATABLESTRINGPOOL_H diff --git a/llvm/tools/dsymutil/dsymutil.cpp b/llvm/tools/dsymutil/dsymutil.cpp index ef21677e895c..b6d6c909abcf 100644 --- a/llvm/tools/dsymutil/dsymutil.cpp +++ b/llvm/tools/dsymutil/dsymutil.cpp @@ -1,4 +1,4 @@ -//===-- dsymutil.cpp - Debug info dumping utility for llvm ----------------===// +//===- dsymutil.cpp - Debug info dumping utility for llvm -----------------===// // // The LLVM Compiler Infrastructure // @@ -15,25 +15,31 @@ #include "dsymutil.h" #include "DebugMap.h" #include "MachOUtils.h" +#include "llvm/ADT/SmallString.h" +#include "llvm/ADT/SmallVector.h" +#include "llvm/ADT/StringRef.h" +#include "llvm/ADT/Triple.h" #include "llvm/Object/MachO.h" +#include "llvm/Support/CommandLine.h" #include "llvm/Support/FileSystem.h" -#include "llvm/Support/FileUtilities.h" #include "llvm/Support/ManagedStatic.h" -#include "llvm/Support/Options.h" +#include "llvm/Support/Path.h" #include "llvm/Support/PrettyStackTrace.h" #include "llvm/Support/Signals.h" #include "llvm/Support/TargetSelect.h" #include "llvm/Support/ThreadPool.h" #include "llvm/Support/raw_ostream.h" +#include "llvm/Support/thread.h" +#include #include +#include #include +#include +using namespace llvm::cl; using namespace llvm::dsymutil; -namespace { -using namespace llvm::cl; - -OptionCategory DsymCategory("Specific Options"); +static OptionCategory DsymCategory("Specific Options"); static opt Help("h", desc("Alias for -help"), Hidden); static opt Version("v", desc("Alias for -version"), Hidden); @@ -77,10 +83,12 @@ static opt NoOutput("no-output", desc("Do the link in memory, but do not emit the result file."), init(false), cat(DsymCategory)); + static opt NoTimestamp("no-swiftmodule-timestamp", desc("Don't check timestamp for swiftmodule files."), init(false), cat(DsymCategory)); + static list ArchFlags( "arch", desc("Link DWARF debug information only for specified CPU architecture\n" @@ -103,7 +111,6 @@ static opt DumpDebugMap( static opt InputIsYAMLDebugMap( "y", desc("Treat the input file is a YAML debug map rather than a binary."), init(false), cat(DsymCategory)); -} static bool createPlistFile(llvm::StringRef BundleRoot) { if (NoOutput) diff --git a/llvm/tools/dsymutil/dsymutil.h b/llvm/tools/dsymutil/dsymutil.h index 8f5874210054..58379bbba226 100644 --- a/llvm/tools/dsymutil/dsymutil.h +++ b/llvm/tools/dsymutil/dsymutil.h @@ -6,38 +6,52 @@ // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// -/// +// /// \file /// /// This file contains the class declaration for the code that parses STABS /// debug maps that are embedded in the binaries symbol tables. -/// +// //===----------------------------------------------------------------------===// + #ifndef LLVM_TOOLS_DSYMUTIL_DSYMUTIL_H #define LLVM_TOOLS_DSYMUTIL_DSYMUTIL_H #include "DebugMap.h" +#include "llvm/ADT/ArrayRef.h" #include "llvm/ADT/StringRef.h" +#include "llvm/Support/Compiler.h" #include "llvm/Support/ErrorOr.h" #include +#include +#include namespace llvm { namespace dsymutil { struct LinkOptions { - bool Verbose; ///< Verbosity - bool NoOutput; ///< Skip emitting output - bool NoODR; ///< Do not unique types according to ODR - bool NoTimestamp; ///< Do not check swiftmodule timestamp - std::string PrependPath; ///< -oso-prepend-path + /// Verbosity + bool Verbose = false; - LinkOptions() : Verbose(false), NoOutput(false), NoTimestamp(false) {} + /// Skip emitting output + bool NoOutput = false; + + /// Do not unique types according to ODR + bool NoODR; + + /// Do not check swiftmodule timestamp + bool NoTimestamp = false; + + /// -oso-prepend-path + std::string PrependPath; + + LinkOptions() = default; }; /// \brief Extract the DebugMaps from the given file. /// The file has to be a MachO object file. Multiple debug maps can be /// returned when the file is universal (aka fat) binary. -llvm::ErrorOr>> +ErrorOr>> parseDebugMap(StringRef InputFile, ArrayRef Archs, StringRef PrependPath, bool Verbose, bool InputIsYAML); @@ -57,6 +71,8 @@ LLVM_ATTRIBUTE_NORETURN void exitDsymutil(int ExitStatus); void warn(const Twine &Warning, const Twine &Context); bool error(const Twine &Error, const Twine &Context); -} -} + +} // end namespace dsymutil +} // end namespace llvm + #endif // LLVM_TOOLS_DSYMUTIL_DSYMUTIL_H diff --git a/llvm/tools/llvm-objcopy/Object.cpp b/llvm/tools/llvm-objcopy/Object.cpp index f9acf001ae93..22ae47f1cace 100644 --- a/llvm/tools/llvm-objcopy/Object.cpp +++ b/llvm/tools/llvm-objcopy/Object.cpp @@ -1,4 +1,4 @@ -//===- Object.cpp -----------------------------------------------*- C++ -*-===// +//===- Object.cpp ---------------------------------------------------------===// // // The LLVM Compiler Infrastructure // @@ -6,16 +6,32 @@ // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// + #include "Object.h" #include "llvm-objcopy.h" +#include "llvm/ADT/ArrayRef.h" +#include "llvm/ADT/STLExtras.h" +#include "llvm/ADT/StringRef.h" +#include "llvm/ADT/Twine.h" +#include "llvm/ADT/iterator_range.h" +#include "llvm/BinaryFormat/ELF.h" +#include "llvm/Object/ELFObjectFile.h" +#include "llvm/Support/ErrorHandling.h" +#include "llvm/Support/FileOutputBuffer.h" +#include +#include +#include +#include +#include +#include using namespace llvm; using namespace object; using namespace ELF; template void Segment::writeHeader(FileOutputBuffer &Out) const { - typedef typename ELFT::Ehdr Elf_Ehdr; - typedef typename ELFT::Phdr Elf_Phdr; + using Elf_Ehdr = typename ELFT::Ehdr; + using Elf_Phdr = typename ELFT::Phdr; uint8_t *Buf = Out.getBufferStart(); Buf += sizeof(Elf_Ehdr) + Index * sizeof(Elf_Phdr); @@ -191,8 +207,7 @@ const Symbol *SymbolTableSection::getSymbolByIndex(uint32_t Index) const { } template -void SymbolTableSectionImpl::writeSection( - llvm::FileOutputBuffer &Out) const { +void SymbolTableSectionImpl::writeSection(FileOutputBuffer &Out) const { uint8_t *Buf = Out.getBufferStart(); Buf += Offset; typename ELFT::Sym *Sym = reinterpret_cast(Buf); @@ -212,9 +227,10 @@ template void RelocSectionWithSymtabBase::removeSectionReferences( const SectionBase *Sec) { if (Symbols == Sec) { - error("Symbol table " + Symbols->Name + " cannot be removed because it is " - "referenced by the relocation " - "section " + + error("Symbol table " + Symbols->Name + + " cannot be removed because it is " + "referenced by the relocation " + "section " + this->Name); } } @@ -229,9 +245,9 @@ void RelocSectionWithSymtabBase::initialize( " is not a symbol table")); if (Info != SHN_UNDEF) - setSection(SecTable.getSection(Info, - "Info field value " + Twine(Info) + - " in section " + Name + " is invalid")); + setSection(SecTable.getSection(Info, "Info field value " + Twine(Info) + + " in section " + Name + + " is invalid")); else setSection(nullptr); } @@ -263,7 +279,7 @@ void RelocationSection::writeRel(T *Buf) const { } template -void RelocationSection::writeSection(llvm::FileOutputBuffer &Out) const { +void RelocationSection::writeSection(FileOutputBuffer &Out) const { uint8_t *Buf = Out.getBufferStart() + Offset; if (Type == SHT_REL) writeRel(reinterpret_cast(Buf)); @@ -271,15 +287,16 @@ void RelocationSection::writeSection(llvm::FileOutputBuffer &Out) const { writeRel(reinterpret_cast(Buf)); } -void DynamicRelocationSection::writeSection(llvm::FileOutputBuffer &Out) const { +void DynamicRelocationSection::writeSection(FileOutputBuffer &Out) const { std::copy(std::begin(Contents), std::end(Contents), Out.getBufferStart() + Offset); } void SectionWithStrTab::removeSectionReferences(const SectionBase *Sec) { if (StrTab == Sec) { - error("String table " + StrTab->Name + " cannot be removed because it is " - "referenced by the section " + + error("String table " + StrTab->Name + + " cannot be removed because it is " + "referenced by the section " + this->Name); } } @@ -289,9 +306,9 @@ bool SectionWithStrTab::classof(const SectionBase *S) { } void SectionWithStrTab::initialize(SectionTableRef SecTable) { - auto StrTab = SecTable.getSection(Link, - "Link field value " + Twine(Link) + - " in section " + Name + " is invalid"); + auto StrTab = + SecTable.getSection(Link, "Link field value " + Twine(Link) + + " in section " + Name + " is invalid"); if (StrTab->Type != SHT_STRTAB) { error("Link field value " + Twine(Link) + " in section " + Name + " is not a string table"); @@ -377,10 +394,9 @@ void Object::readProgramHeaders(const ELFFile &ElfFile) { } template -void Object::initSymbolTable(const llvm::object::ELFFile &ElfFile, +void Object::initSymbolTable(const object::ELFFile &ElfFile, SymbolTableSection *SymTab, SectionTableRef SecTable) { - const Elf_Shdr &Shdr = *unwrapOrError(ElfFile.getSection(SymTab->Index)); StringRef StrTabData = unwrapOrError(ElfFile.getStringTableForSymtab(Shdr)); @@ -397,9 +413,9 @@ void Object::initSymbolTable(const llvm::object::ELFFile &ElfFile, } } else if (Sym.st_shndx != SHN_UNDEF) { DefSection = SecTable.getSection( - Sym.st_shndx, - "Symbol '" + Name + "' is defined in invalid section with index " + - Twine(Sym.st_shndx)); + Sym.st_shndx, "Symbol '" + Name + + "' is defined in invalid section with index " + + Twine(Sym.st_shndx)); } SymTab->addSymbol(Name, Sym.getBinding(), Sym.getType(), DefSection, @@ -437,14 +453,14 @@ SectionBase *SectionTableRef::getSection(uint16_t Index, Twine ErrMsg) { template T *SectionTableRef::getSectionOfType(uint16_t Index, Twine IndexErrMsg, Twine TypeErrMsg) { - if (T *Sec = llvm::dyn_cast(getSection(Index, IndexErrMsg))) + if (T *Sec = dyn_cast(getSection(Index, IndexErrMsg))) return Sec; error(TypeErrMsg); } template std::unique_ptr -Object::makeSection(const llvm::object::ELFFile &ElfFile, +Object::makeSection(const object::ELFFile &ElfFile, const Elf_Shdr &Shdr) { ArrayRef Data; switch (Shdr.sh_type) { @@ -621,7 +637,7 @@ void Object::writeSectionHeaders(FileOutputBuffer &Out) const { template void Object::writeSectionData(FileOutputBuffer &Out) const { for (auto &Section : Sections) - Section->writeSection(Out); + Section->writeSection(Out); } template @@ -797,15 +813,13 @@ void BinaryObject::write(FileOutputBuffer &Out) const { for (auto &Segment : this->Segments) { // GNU objcopy does not output segments that do not cover a section. Such // segments can sometimes be produced by LLD due to how LLD handles PT_PHDR. - if (Segment->Type == llvm::ELF::PT_LOAD && - Segment->firstSection() != nullptr) { + if (Segment->Type == PT_LOAD && Segment->firstSection() != nullptr) { Segment->writeSegment(Out); } } } template void BinaryObject::finalize() { - // Put all segments in offset order. auto CompareSegments = [](const SegPtr &A, const SegPtr &B) { return A->Offset < B->Offset; @@ -815,8 +829,7 @@ template void BinaryObject::finalize() { uint64_t Offset = 0; for (auto &Segment : this->Segments) { - if (Segment->Type == llvm::ELF::PT_LOAD && - Segment->firstSection() != nullptr) { + if (Segment->Type == PT_LOAD && Segment->firstSection() != nullptr) { Offset = alignTo(Offset, Segment->Align); Segment->Offset = Offset; Offset += Segment->FileSize; @@ -825,6 +838,8 @@ template void BinaryObject::finalize() { TotalSize = Offset; } +namespace llvm { + template class Object; template class Object; template class Object; @@ -839,3 +854,5 @@ template class BinaryObject; template class BinaryObject; template class BinaryObject; template class BinaryObject; + +} // end namespace llvm diff --git a/llvm/tools/llvm-objcopy/Object.h b/llvm/tools/llvm-objcopy/Object.h index f6088434805d..9c77f5900ce2 100644 --- a/llvm/tools/llvm-objcopy/Object.h +++ b/llvm/tools/llvm-objcopy/Object.h @@ -7,38 +7,46 @@ // //===----------------------------------------------------------------------===// -#ifndef LLVM_OBJCOPY_OBJECT_H -#define LLVM_OBJCOPY_OBJECT_H +#ifndef LLVM_TOOLS_OBJCOPY_OBJECT_H +#define LLVM_TOOLS_OBJCOPY_OBJECT_H +#include "llvm/ADT/ArrayRef.h" +#include "llvm/ADT/StringRef.h" +#include "llvm/ADT/Twine.h" +#include "llvm/BinaryFormat/ELF.h" #include "llvm/MC/StringTableBuilder.h" #include "llvm/Object/ELFObjectFile.h" -#include "llvm/Support/FileOutputBuffer.h" - +#include +#include +#include #include #include +#include -class Segment; +namespace llvm { + +class FileOutputBuffer; class SectionBase; +class Segment; class SectionTableRef { private: - llvm::ArrayRef> Sections; + ArrayRef> Sections; public: - SectionTableRef(llvm::ArrayRef> Secs) + SectionTableRef(ArrayRef> Secs) : Sections(Secs) {} SectionTableRef(const SectionTableRef &) = default; - SectionBase *getSection(uint16_t Index, llvm::Twine ErrMsg); + SectionBase *getSection(uint16_t Index, Twine ErrMsg); template - T *getSectionOfType(uint16_t Index, llvm::Twine IndexErrMsg, - llvm::Twine TypeErrMsg); + T *getSectionOfType(uint16_t Index, Twine IndexErrMsg, Twine TypeErrMsg); }; class SectionBase { public: - llvm::StringRef Name; + StringRef Name; Segment *ParentSegment = nullptr; uint64_t HeaderOffset; uint64_t OriginalOffset; @@ -49,18 +57,19 @@ public: uint32_t EntrySize = 0; uint64_t Flags = 0; uint64_t Info = 0; - uint64_t Link = llvm::ELF::SHN_UNDEF; + uint64_t Link = ELF::SHN_UNDEF; uint64_t NameIndex = 0; uint64_t Offset = 0; uint64_t Size = 0; - uint64_t Type = llvm::ELF::SHT_NULL; + uint64_t Type = ELF::SHT_NULL; + + virtual ~SectionBase() = default; - virtual ~SectionBase() {} virtual void initialize(SectionTableRef SecTable); virtual void finalize(); virtual void removeSectionReferences(const SectionBase *Sec); - template void writeHeader(llvm::FileOutputBuffer &Out) const; - virtual void writeSection(llvm::FileOutputBuffer &Out) const = 0; + template void writeHeader(FileOutputBuffer &Out) const; + virtual void writeSection(FileOutputBuffer &Out) const = 0; }; class Segment { @@ -77,7 +86,7 @@ private: }; std::set Sections; - llvm::ArrayRef Contents; + ArrayRef Contents; public: uint64_t Align; @@ -93,25 +102,28 @@ public: uint64_t OriginalOffset; Segment *ParentSegment = nullptr; - Segment(llvm::ArrayRef Data) : Contents(Data) {} + Segment(ArrayRef Data) : Contents(Data) {} + const SectionBase *firstSection() const { if (!Sections.empty()) return *Sections.begin(); return nullptr; } + void removeSection(const SectionBase *Sec) { Sections.erase(Sec); } void addSection(const SectionBase *Sec) { Sections.insert(Sec); } - template void writeHeader(llvm::FileOutputBuffer &Out) const; - void writeSegment(llvm::FileOutputBuffer &Out) const; + template void writeHeader(FileOutputBuffer &Out) const; + void writeSegment(FileOutputBuffer &Out) const; }; class Section : public SectionBase { private: - llvm::ArrayRef Contents; + ArrayRef Contents; public: - Section(llvm::ArrayRef Data) : Contents(Data) {} - void writeSection(llvm::FileOutputBuffer &Out) const override; + Section(ArrayRef Data) : Contents(Data) {} + + void writeSection(FileOutputBuffer &Out) const override; }; // There are two types of string tables that can exist, dynamic and not dynamic. @@ -124,21 +136,22 @@ public: // then agrees with the makeSection method used to construct most sections. class StringTableSection : public SectionBase { private: - llvm::StringTableBuilder StrTabBuilder; + StringTableBuilder StrTabBuilder; public: - StringTableSection() : StrTabBuilder(llvm::StringTableBuilder::ELF) { - Type = llvm::ELF::SHT_STRTAB; + StringTableSection() : StrTabBuilder(StringTableBuilder::ELF) { + Type = ELF::SHT_STRTAB; } - void addString(llvm::StringRef Name); - uint32_t findIndex(llvm::StringRef Name) const; + void addString(StringRef Name); + uint32_t findIndex(StringRef Name) const; void finalize() override; - void writeSection(llvm::FileOutputBuffer &Out) const override; + void writeSection(FileOutputBuffer &Out) const override; + static bool classof(const SectionBase *S) { - if (S->Flags & llvm::ELF::SHF_ALLOC) + if (S->Flags & ELF::SHF_ALLOC) return false; - return S->Type == llvm::ELF::SHT_STRTAB; + return S->Type == ELF::SHT_STRTAB; } }; @@ -148,12 +161,12 @@ public: // SYMBOL_SIMPLE_INDEX means that the st_shndx is just an index of a section. enum SymbolShndxType { SYMBOL_SIMPLE_INDEX = 0, - SYMBOL_ABS = llvm::ELF::SHN_ABS, - SYMBOL_COMMON = llvm::ELF::SHN_COMMON, - SYMBOL_HEXAGON_SCOMMON = llvm::ELF::SHN_HEXAGON_SCOMMON, - SYMBOL_HEXAGON_SCOMMON_2 = llvm::ELF::SHN_HEXAGON_SCOMMON_2, - SYMBOL_HEXAGON_SCOMMON_4 = llvm::ELF::SHN_HEXAGON_SCOMMON_4, - SYMBOL_HEXAGON_SCOMMON_8 = llvm::ELF::SHN_HEXAGON_SCOMMON_8, + SYMBOL_ABS = ELF::SHN_ABS, + SYMBOL_COMMON = ELF::SHN_COMMON, + SYMBOL_HEXAGON_SCOMMON = ELF::SHN_HEXAGON_SCOMMON, + SYMBOL_HEXAGON_SCOMMON_2 = ELF::SHN_HEXAGON_SCOMMON_2, + SYMBOL_HEXAGON_SCOMMON_4 = ELF::SHN_HEXAGON_SCOMMON_4, + SYMBOL_HEXAGON_SCOMMON_8 = ELF::SHN_HEXAGON_SCOMMON_8, }; struct Symbol { @@ -161,7 +174,7 @@ struct Symbol { SectionBase *DefinedIn = nullptr; SymbolShndxType ShndxType; uint32_t Index; - llvm::StringRef Name; + StringRef Name; uint32_t NameIndex; uint64_t Size; uint8_t Type; @@ -175,11 +188,11 @@ protected: std::vector> Symbols; StringTableSection *SymbolNames = nullptr; - typedef std::unique_ptr SymPtr; + using SymPtr = std::unique_ptr; public: void setStrTab(StringTableSection *StrTab) { SymbolNames = StrTab; } - void addSymbol(llvm::StringRef Name, uint8_t Bind, uint8_t Type, + void addSymbol(StringRef Name, uint8_t Bind, uint8_t Type, SectionBase *DefinedIn, uint64_t Value, uint16_t Shndx, uint64_t Sz); void addSymbolNames(); @@ -187,14 +200,15 @@ public: void removeSectionReferences(const SectionBase *Sec) override; void initialize(SectionTableRef SecTable) override; void finalize() override; + static bool classof(const SectionBase *S) { - return S->Type == llvm::ELF::SHT_SYMTAB; + return S->Type == ELF::SHT_SYMTAB; } }; // Only writeSection depends on the ELF type so we implement it in a subclass. template class SymbolTableSectionImpl : public SymbolTableSection { - void writeSection(llvm::FileOutputBuffer &Out) const override; + void writeSection(FileOutputBuffer &Out) const override; }; struct Relocation { @@ -222,7 +236,7 @@ public: void setSection(SectionBase *Sec) { SecToApplyRel = Sec; } static bool classof(const SectionBase *S) { - return S->Type == llvm::ELF::SHT_REL || S->Type == llvm::ELF::SHT_RELA; + return S->Type == ELF::SHT_REL || S->Type == ELF::SHT_RELA; } }; @@ -234,11 +248,10 @@ private: SymTabType *Symbols = nullptr; protected: - RelocSectionWithSymtabBase() {} + RelocSectionWithSymtabBase() = default; public: void setSymTab(SymTabType *StrTab) { Symbols = StrTab; } - void removeSectionReferences(const SectionBase *Sec) override; void initialize(SectionTableRef SecTable) override; void finalize() override; @@ -248,8 +261,8 @@ template class RelocationSection : public RelocSectionWithSymtabBase { private: - typedef typename ELFT::Rel Elf_Rel; - typedef typename ELFT::Rela Elf_Rela; + using Elf_Rel = typename ELFT::Rel; + using Elf_Rela = typename ELFT::Rela; std::vector Relocations; @@ -257,12 +270,12 @@ private: public: void addRelocation(Relocation Rel) { Relocations.push_back(Rel); } - void writeSection(llvm::FileOutputBuffer &Out) const override; + void writeSection(FileOutputBuffer &Out) const override; static bool classof(const SectionBase *S) { - if (S->Flags & llvm::ELF::SHF_ALLOC) + if (S->Flags & ELF::SHF_ALLOC) return false; - return S->Type == llvm::ELF::SHT_REL || S->Type == llvm::ELF::SHT_RELA; + return S->Type == ELF::SHT_REL || S->Type == ELF::SHT_RELA; } }; @@ -271,7 +284,8 @@ private: const SectionBase *StrTab = nullptr; public: - SectionWithStrTab(llvm::ArrayRef Data) : Section(Data) {} + SectionWithStrTab(ArrayRef Data) : Section(Data) {} + void setStrTab(const SectionBase *StringTable) { StrTab = StringTable; } void removeSectionReferences(const SectionBase *Sec) override; void initialize(SectionTableRef SecTable) override; @@ -281,51 +295,54 @@ public: class DynamicSymbolTableSection : public SectionWithStrTab { public: - DynamicSymbolTableSection(llvm::ArrayRef Data) - : SectionWithStrTab(Data) {} + DynamicSymbolTableSection(ArrayRef Data) : SectionWithStrTab(Data) {} + static bool classof(const SectionBase *S) { - return S->Type == llvm::ELF::SHT_DYNSYM; + return S->Type == ELF::SHT_DYNSYM; } }; class DynamicSection : public SectionWithStrTab { public: - DynamicSection(llvm::ArrayRef Data) : SectionWithStrTab(Data) {} + DynamicSection(ArrayRef Data) : SectionWithStrTab(Data) {} + static bool classof(const SectionBase *S) { - return S->Type == llvm::ELF::SHT_DYNAMIC; + return S->Type == ELF::SHT_DYNAMIC; } }; class DynamicRelocationSection : public RelocSectionWithSymtabBase { private: - llvm::ArrayRef Contents; + ArrayRef Contents; public: - DynamicRelocationSection(llvm::ArrayRef Data) : Contents(Data) {} - void writeSection(llvm::FileOutputBuffer &Out) const override; + DynamicRelocationSection(ArrayRef Data) : Contents(Data) {} + + void writeSection(FileOutputBuffer &Out) const override; + static bool classof(const SectionBase *S) { - if (!(S->Flags & llvm::ELF::SHF_ALLOC)) + if (!(S->Flags & ELF::SHF_ALLOC)) return false; - return S->Type == llvm::ELF::SHT_REL || S->Type == llvm::ELF::SHT_RELA; + return S->Type == ELF::SHT_REL || S->Type == ELF::SHT_RELA; } }; template class Object { private: - typedef std::unique_ptr SecPtr; - typedef std::unique_ptr SegPtr; + using SecPtr = std::unique_ptr; + using SegPtr = std::unique_ptr; - typedef typename ELFT::Shdr Elf_Shdr; - typedef typename ELFT::Ehdr Elf_Ehdr; - typedef typename ELFT::Phdr Elf_Phdr; + using Elf_Shdr = typename ELFT::Shdr; + using Elf_Ehdr = typename ELFT::Ehdr; + using Elf_Phdr = typename ELFT::Phdr; - void initSymbolTable(const llvm::object::ELFFile &ElfFile, + void initSymbolTable(const object::ELFFile &ElfFile, SymbolTableSection *SymTab, SectionTableRef SecTable); - SecPtr makeSection(const llvm::object::ELFFile &ElfFile, + SecPtr makeSection(const object::ELFFile &ElfFile, const Elf_Shdr &Shdr); - void readProgramHeaders(const llvm::object::ELFFile &ElfFile); - SectionTableRef readSectionHeaders(const llvm::object::ELFFile &ElfFile); + void readProgramHeaders(const object::ELFFile &ElfFile); + SectionTableRef readSectionHeaders(const object::ELFFile &ElfFile); protected: StringTableSection *SectionNames = nullptr; @@ -333,10 +350,10 @@ protected: std::vector Sections; std::vector Segments; - void writeHeader(llvm::FileOutputBuffer &Out) const; - void writeProgramHeaders(llvm::FileOutputBuffer &Out) const; - void writeSectionData(llvm::FileOutputBuffer &Out) const; - void writeSectionHeaders(llvm::FileOutputBuffer &Out) const; + void writeHeader(FileOutputBuffer &Out) const; + void writeProgramHeaders(FileOutputBuffer &Out) const; + void writeSectionData(FileOutputBuffer &Out) const; + void writeSectionHeaders(FileOutputBuffer &Out) const; public: uint8_t Ident[16]; @@ -348,45 +365,50 @@ public: uint32_t Flags; bool WriteSectionHeaders = true; - Object(const llvm::object::ELFObjectFile &Obj); + Object(const object::ELFObjectFile &Obj); + virtual ~Object() = default; + void removeSections(std::function ToRemove); virtual size_t totalSize() const = 0; virtual void finalize() = 0; - virtual void write(llvm::FileOutputBuffer &Out) const = 0; - virtual ~Object() = default; + virtual void write(FileOutputBuffer &Out) const = 0; }; template class ELFObject : public Object { private: - typedef std::unique_ptr SecPtr; - typedef std::unique_ptr SegPtr; + using SecPtr = std::unique_ptr; + using SegPtr = std::unique_ptr; - typedef typename ELFT::Shdr Elf_Shdr; - typedef typename ELFT::Ehdr Elf_Ehdr; - typedef typename ELFT::Phdr Elf_Phdr; + using Elf_Shdr = typename ELFT::Shdr; + using Elf_Ehdr = typename ELFT::Ehdr; + using Elf_Phdr = typename ELFT::Phdr; void sortSections(); void assignOffsets(); public: - ELFObject(const llvm::object::ELFObjectFile &Obj) : Object(Obj) {} + ELFObject(const object::ELFObjectFile &Obj) : Object(Obj) {} + void finalize() override; size_t totalSize() const override; - void write(llvm::FileOutputBuffer &Out) const override; + void write(FileOutputBuffer &Out) const override; }; template class BinaryObject : public Object { private: - typedef std::unique_ptr SecPtr; - typedef std::unique_ptr SegPtr; + using SecPtr = std::unique_ptr; + using SegPtr = std::unique_ptr; uint64_t TotalSize; public: - BinaryObject(const llvm::object::ELFObjectFile &Obj) - : Object(Obj) {} + BinaryObject(const object::ELFObjectFile &Obj) : Object(Obj) {} + void finalize() override; size_t totalSize() const override; - void write(llvm::FileOutputBuffer &Out) const override; + void write(FileOutputBuffer &Out) const override; }; -#endif + +} // end namespace llvm + +#endif // LLVM_TOOLS_OBJCOPY_OBJECT_H diff --git a/llvm/tools/llvm-objcopy/llvm-objcopy.cpp b/llvm/tools/llvm-objcopy/llvm-objcopy.cpp index 7f55a434b334..f3e9c7750a64 100644 --- a/llvm/tools/llvm-objcopy/llvm-objcopy.cpp +++ b/llvm/tools/llvm-objcopy/llvm-objcopy.cpp @@ -1,4 +1,4 @@ -//===- llvm-objcopy.cpp -----------------------------------------*- C++ -*-===// +//===- llvm-objcopy.cpp ---------------------------------------------------===// // // The LLVM Compiler Infrastructure // @@ -6,17 +6,37 @@ // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// + #include "llvm-objcopy.h" #include "Object.h" +#include "llvm/ADT/STLExtras.h" +#include "llvm/ADT/StringRef.h" +#include "llvm/ADT/Twine.h" +#include "llvm/BinaryFormat/ELF.h" +#include "llvm/Object/Binary.h" +#include "llvm/Object/ELFObjectFile.h" +#include "llvm/Object/ELFTypes.h" +#include "llvm/Object/Error.h" +#include "llvm/Support/Casting.h" #include "llvm/Support/CommandLine.h" +#include "llvm/Support/Compiler.h" +#include "llvm/Support/Error.h" +#include "llvm/Support/ErrorHandling.h" +#include "llvm/Support/ErrorOr.h" #include "llvm/Support/FileOutputBuffer.h" +#include "llvm/Support/ManagedStatic.h" #include "llvm/Support/PrettyStackTrace.h" #include "llvm/Support/Signals.h" -#include "llvm/Support/ToolOutputFile.h" - +#include "llvm/Support/raw_ostream.h" +#include +#include +#include +#include +#include #include #include #include +#include using namespace llvm; using namespace object; @@ -39,7 +59,7 @@ LLVM_ATTRIBUTE_NORETURN void reportError(StringRef File, std::error_code EC) { exit(1); } -LLVM_ATTRIBUTE_NORETURN void reportError(StringRef File, llvm::Error E) { +LLVM_ATTRIBUTE_NORETURN void reportError(StringRef File, Error E) { assert(E); std::string Buf; raw_string_ostream OS(Buf); @@ -48,22 +68,23 @@ LLVM_ATTRIBUTE_NORETURN void reportError(StringRef File, llvm::Error E) { errs() << ToolName << ": '" << File << "': " << Buf; exit(1); } -} -cl::opt InputFilename(cl::Positional, cl::desc("")); -cl::opt OutputFilename(cl::Positional, cl::desc(""), +} // end namespace llvm + +static cl::opt InputFilename(cl::Positional, cl::desc("")); +static cl::opt OutputFilename(cl::Positional, cl::desc(""), cl::init("-")); -cl::opt +static cl::opt OutputFormat("O", cl::desc("set output format to one of the following:" "\n\tbinary")); -cl::list ToRemove("remove-section", - cl::desc("Remove a specific section")); -cl::alias ToRemoveA("R", cl::desc("Alias for remove-section"), - cl::aliasopt(ToRemove)); -cl::opt StripSections("strip-sections", - cl::desc("Remove all section headers")); +static cl::list ToRemove("remove-section", + cl::desc("Remove a specific section")); +static cl::alias ToRemoveA("R", cl::desc("Alias for remove-section"), + cl::aliasopt(ToRemove)); +static cl::opt StripSections("strip-sections", + cl::desc("Remove all section headers")); -typedef std::function SectionPred; +using SectionPred = std::function; void CopyBinary(const ELFObjectFile &ObjFile) { std::unique_ptr Buffer; diff --git a/llvm/tools/llvm-objcopy/llvm-objcopy.h b/llvm/tools/llvm-objcopy/llvm-objcopy.h index de7bf367ac87..6732e410d8e0 100644 --- a/llvm/tools/llvm-objcopy/llvm-objcopy.h +++ b/llvm/tools/llvm-objcopy/llvm-objcopy.h @@ -6,11 +6,15 @@ // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// -#ifndef LLVM_OBJCOPY_H -#define LLVM_OBJCOPY_H + +#ifndef LLVM_TOOLS_OBJCOPY_OBJCOPY_H +#define LLVM_TOOLS_OBJCOPY_OBJCOPY_H #include "llvm/ADT/Twine.h" +#include "llvm/Support/Compiler.h" #include "llvm/Support/Error.h" +#include "llvm/Support/raw_ostream.h" +#include namespace llvm { @@ -27,6 +31,7 @@ template T unwrapOrError(Expected EO) { OS.flush(); error(Buf); } -} -#endif +} // end namespace llvm + +#endif // LLVM_TOOLS_OBJCOPY_OBJCOPY_H