diff options
| author | Dimitry Andric <dim@FreeBSD.org> | 2022-07-24 15:03:44 +0000 | 
|---|---|---|
| committer | Dimitry Andric <dim@FreeBSD.org> | 2022-07-24 15:03:44 +0000 | 
| commit | 4b4fe385e49bd883fd183b5f21c1ea486c722e61 (patch) | |
| tree | c3d8fdb355c9c73e57723718c22103aaf7d15aa6 /llvm/tools/llvm-dwarfutil/DebugInfoLinker.cpp | |
| parent | 1f917f69ff07f09b6dbb670971f57f8efe718b84 (diff) | |
Diffstat (limited to 'llvm/tools/llvm-dwarfutil/DebugInfoLinker.cpp')
| -rw-r--r-- | llvm/tools/llvm-dwarfutil/DebugInfoLinker.cpp | 277 | 
1 files changed, 277 insertions, 0 deletions
| diff --git a/llvm/tools/llvm-dwarfutil/DebugInfoLinker.cpp b/llvm/tools/llvm-dwarfutil/DebugInfoLinker.cpp new file mode 100644 index 000000000000..458a58c12ca7 --- /dev/null +++ b/llvm/tools/llvm-dwarfutil/DebugInfoLinker.cpp @@ -0,0 +1,277 @@ +//=== DebugInfoLinker.cpp -------------------------------------------------===// +// +// 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 +// +//===----------------------------------------------------------------------===// + +#include "DebugInfoLinker.h" +#include "Error.h" +#include "llvm/DWARFLinker/DWARFLinker.h" +#include "llvm/DWARFLinker/DWARFStreamer.h" +#include "llvm/DebugInfo/DWARF/DWARFContext.h" +#include "llvm/DebugInfo/DWARF/DWARFExpression.h" +#include "llvm/Object/ObjectFile.h" +#include <memory> +#include <vector> + +namespace llvm { +namespace dwarfutil { + +// ObjFileAddressMap allows to check whether specified DIE referencing +// dead addresses. It uses tombstone values to determine dead addresses. +// The concrete values of tombstone constants were discussed in +// https://reviews.llvm.org/D81784 and https://reviews.llvm.org/D84825. +// So we use following values as indicators of dead addresses: +// +// bfd: (LowPC == 0) or (LowPC == 1 and HighPC == 1 and  DWARF v4 (or less)) +//      or ([LowPC, HighPC] is not inside address ranges of .text sections). +// +// maxpc: (LowPC == -1) or (LowPC == -2 and  DWARF v4 (or less)) +//        That value is assumed to be compatible with +//        http://www.dwarfstd.org/ShowIssue.php?issue=200609.1 +// +// exec: [LowPC, HighPC] is not inside address ranges of .text sections +// +// universal: maxpc and bfd +class ObjFileAddressMap : public AddressesMap { +public: +  ObjFileAddressMap(DWARFContext &Context, const Options &Options, +                    object::ObjectFile &ObjFile) +      : Opts(Options) { +    // Remember addresses of existing text sections. +    for (const object::SectionRef &Sect : ObjFile.sections()) { +      if (!Sect.isText()) +        continue; +      const uint64_t Size = Sect.getSize(); +      if (Size == 0) +        continue; +      const uint64_t StartAddr = Sect.getAddress(); +      TextAddressRanges.insert({StartAddr, StartAddr + Size}); +    } + +    // Check CU address ranges for tombstone value. +    for (std::unique_ptr<DWARFUnit> &CU : Context.compile_units()) { +      Expected<llvm::DWARFAddressRangesVector> ARanges = +          CU->getUnitDIE().getAddressRanges(); +      if (ARanges) { +        for (auto &Range : *ARanges) { +          if (!isDeadAddressRange(Range.LowPC, Range.HighPC, CU->getVersion(), +                                  Options.Tombstone, CU->getAddressByteSize())) +            DWARFAddressRanges.insert({Range.LowPC, Range.HighPC}, 0); +        } +      } +    } +  } + +  // should be renamed into has valid address ranges +  bool hasValidRelocs() override { return !DWARFAddressRanges.empty(); } + +  bool isLiveSubprogram(const DWARFDie &DIE, +                        CompileUnit::DIEInfo &Info) override { +    assert((DIE.getTag() == dwarf::DW_TAG_subprogram || +            DIE.getTag() == dwarf::DW_TAG_label) && +           "Wrong type of input die"); + +    if (Optional<uint64_t> LowPC = +            dwarf::toAddress(DIE.find(dwarf::DW_AT_low_pc))) { +      if (!isDeadAddress(*LowPC, DIE.getDwarfUnit()->getVersion(), +                         Opts.Tombstone, +                         DIE.getDwarfUnit()->getAddressByteSize())) { +        Info.AddrAdjust = 0; +        Info.InDebugMap = true; +        return true; +      } +    } + +    return false; +  } + +  bool isLiveVariable(const DWARFDie &DIE, +                      CompileUnit::DIEInfo &Info) override { +    assert((DIE.getTag() == dwarf::DW_TAG_variable || +            DIE.getTag() == dwarf::DW_TAG_constant) && +           "Wrong type of input die"); + +    if (Expected<DWARFLocationExpressionsVector> Loc = +            DIE.getLocations(dwarf::DW_AT_location)) { +      DWARFUnit *U = DIE.getDwarfUnit(); +      for (const auto &Entry : *Loc) { +        DataExtractor Data(toStringRef(Entry.Expr), +                           U->getContext().isLittleEndian(), 0); +        DWARFExpression Expression(Data, U->getAddressByteSize(), +                                   U->getFormParams().Format); +        bool HasLiveAddresses = +            any_of(Expression, [&](const DWARFExpression::Operation &Op) { +              // TODO: add handling of dwarf::DW_OP_addrx +              return !Op.isError() && +                     (Op.getCode() == dwarf::DW_OP_addr && +                      !isDeadAddress(Op.getRawOperand(0), U->getVersion(), +                                     Opts.Tombstone, +                                     DIE.getDwarfUnit()->getAddressByteSize())); +            }); + +        if (HasLiveAddresses) { +          Info.AddrAdjust = 0; +          Info.InDebugMap = true; +          return true; +        } +      } +    } else { +      // FIXME: missing DW_AT_location is OK here, but other errors should be +      // reported to the user. +      consumeError(Loc.takeError()); +    } + +    return false; +  } + +  bool applyValidRelocs(MutableArrayRef<char>, uint64_t, bool) override { +    // no need to apply relocations to the linked binary. +    return false; +  } + +  RangesTy &getValidAddressRanges() override { return DWARFAddressRanges; }; + +  void clear() override { DWARFAddressRanges.clear(); } + +  llvm::Expected<uint64_t> relocateIndexedAddr(uint64_t, uint64_t) override { +    // should not be called. +    return object::createError("no relocations in linked binary"); +  } + +protected: +  // returns true if specified address range is inside address ranges +  // of executable sections. +  bool isInsideExecutableSectionsAddressRange(uint64_t LowPC, +                                              Optional<uint64_t> HighPC) { +    Optional<AddressRange> Range = +        TextAddressRanges.getRangeThatContains(LowPC); + +    if (HighPC) +      return Range.has_value() && Range->end() >= *HighPC; + +    return Range.has_value(); +  } + +  uint64_t isBFDDeadAddressRange(uint64_t LowPC, Optional<uint64_t> HighPC, +                                 uint16_t Version) { +    if (LowPC == 0) +      return true; + +    if ((Version <= 4) && HighPC && (LowPC == 1 && *HighPC == 1)) +      return true; + +    return !isInsideExecutableSectionsAddressRange(LowPC, HighPC); +  } + +  uint64_t isMAXPCDeadAddressRange(uint64_t LowPC, Optional<uint64_t> HighPC, +                                   uint16_t Version, uint8_t AddressByteSize) { +    if (Version <= 4 && HighPC) { +      if (LowPC == (dwarf::computeTombstoneAddress(AddressByteSize) - 1)) +        return true; +    } else if (LowPC == dwarf::computeTombstoneAddress(AddressByteSize)) +      return true; + +    if (!isInsideExecutableSectionsAddressRange(LowPC, HighPC)) +      warning("Address referencing invalid text section is not marked with " +              "tombstone value"); + +    return false; +  } + +  bool isDeadAddressRange(uint64_t LowPC, Optional<uint64_t> HighPC, +                          uint16_t Version, TombstoneKind Tombstone, +                          uint8_t AddressByteSize) { +    switch (Tombstone) { +    case TombstoneKind::BFD: +      return isBFDDeadAddressRange(LowPC, HighPC, Version); +    case TombstoneKind::MaxPC: +      return isMAXPCDeadAddressRange(LowPC, HighPC, Version, AddressByteSize); +    case TombstoneKind::Universal: +      return isBFDDeadAddressRange(LowPC, HighPC, Version) || +             isMAXPCDeadAddressRange(LowPC, HighPC, Version, AddressByteSize); +    case TombstoneKind::Exec: +      return !isInsideExecutableSectionsAddressRange(LowPC, HighPC); +    } + +    llvm_unreachable("Unknown tombstone value"); +  } + +  bool isDeadAddress(uint64_t LowPC, uint16_t Version, TombstoneKind Tombstone, +                     uint8_t AddressByteSize) { +    return isDeadAddressRange(LowPC, None, Version, Tombstone, AddressByteSize); +  } + +private: +  RangesTy DWARFAddressRanges; +  AddressRanges TextAddressRanges; +  const Options &Opts; +}; + +bool linkDebugInfo(object::ObjectFile &File, const Options &Options, +                   raw_pwrite_stream &OutStream) { + +  auto ReportWarn = [&](const Twine &Message, StringRef Context, +                        const DWARFDie *Die) { +    warning(Message, Context); + +    if (!Options.Verbose || !Die) +      return; + +    DIDumpOptions DumpOpts; +    DumpOpts.ChildRecurseDepth = 0; +    DumpOpts.Verbose = Options.Verbose; + +    WithColor::note() << "    in DIE:\n"; +    Die->dump(errs(), /*Indent=*/6, DumpOpts); +  }; +  auto ReportErr = [&](const Twine &Message, StringRef Context, +                       const DWARFDie *) { +    WithColor::error(errs(), Context) << Message << '\n'; +  }; + +  // Create output streamer. +  DwarfStreamer OutStreamer(OutputFileType::Object, OutStream, nullptr, +                            ReportWarn, ReportWarn); +  if (!OutStreamer.init(File.makeTriple(), "")) +    return false; + +  // Create DWARF linker. +  DWARFLinker DebugInfoLinker(&OutStreamer, DwarfLinkerClient::LLD); + +  DebugInfoLinker.setEstimatedObjfilesAmount(1); +  DebugInfoLinker.setAccelTableKind(DwarfLinkerAccelTableKind::None); +  DebugInfoLinker.setErrorHandler(ReportErr); +  DebugInfoLinker.setWarningHandler(ReportWarn); +  DebugInfoLinker.setNumThreads(Options.NumThreads); +  DebugInfoLinker.setNoODR(!Options.DoODRDeduplication); +  DebugInfoLinker.setVerbosity(Options.Verbose); +  DebugInfoLinker.setUpdate(!Options.DoGarbageCollection); + +  std::vector<std::unique_ptr<DWARFFile>> ObjectsForLinking(1); +  std::vector<std::unique_ptr<AddressesMap>> AddresssMapForLinking(1); +  std::vector<std::string> EmptyWarnings; + +  std::unique_ptr<DWARFContext> Context = DWARFContext::create(File); + +  // Add object files to the DWARFLinker. +  AddresssMapForLinking[0] = +      std::make_unique<ObjFileAddressMap>(*Context, Options, File); + +  ObjectsForLinking[0] = std::make_unique<DWARFFile>( +      File.getFileName(), &*Context, AddresssMapForLinking[0].get(), +      EmptyWarnings); + +  for (size_t I = 0; I < ObjectsForLinking.size(); I++) +    DebugInfoLinker.addObjectFile(*ObjectsForLinking[I]); + +  // Link debug info. +  DebugInfoLinker.link(); +  OutStreamer.finish(); +  return true; +} + +} // end of namespace dwarfutil +} // end of namespace llvm | 
