//===- tools/dsymutil/DebugMap.cpp - Generic debug map representation -----===// // // 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 "DebugMap.h" #include "BinaryHolder.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/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/WithColor.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; DebugMapObject::DebugMapObject(StringRef ObjectFilename, sys::TimePoint Timestamp, uint8_t Type) : Filename(std::string(ObjectFilename)), Timestamp(Timestamp), Type(Type) {} bool DebugMapObject::addSymbol(StringRef Name, Optional ObjectAddress, uint64_t LinkedAddress, uint32_t Size) { auto InsertResult = Symbols.insert( std::make_pair(Name, SymbolMapping(ObjectAddress, LinkedAddress, Size))); if (ObjectAddress && InsertResult.second) AddressToMapping[*ObjectAddress] = &*InsertResult.first; return InsertResult.second; } 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). using Entry = std::pair; std::vector Entries; Entries.reserve(Symbols.getNumItems()); for (const auto &Sym : Symbols) Entries.push_back(std::make_pair(Sym.getKey(), Sym.getValue())); llvm::sort(Entries, [](const Entry &LHS, const Entry &RHS) { return LHS.first < RHS.first; }); for (const auto &Sym : Entries) { if (Sym.second.ObjectAddress) OS << format("\t%016" PRIx64, uint64_t(*Sym.second.ObjectAddress)); else OS << "\t????????????????"; OS << format(" => %016" PRIx64 "+0x%x\t%s\n", uint64_t(Sym.second.BinaryAddress), uint32_t(Sym.second.Size), Sym.first.data()); } OS << '\n'; } #ifndef NDEBUG void DebugMapObject::dump() const { print(errs()); } #endif DebugMapObject & DebugMap::addDebugMapObject(StringRef ObjectFilePath, sys::TimePoint Timestamp, uint8_t Type) { Objects.emplace_back(new DebugMapObject(ObjectFilePath, Timestamp, Type)); return *Objects.back(); } const DebugMapObject::DebugMapEntry * DebugMapObject::lookupSymbol(StringRef SymbolName) const { StringMap::const_iterator Sym = Symbols.find(SymbolName); if (Sym == Symbols.end()) return nullptr; return &*Sym; } const DebugMapObject::DebugMapEntry * DebugMapObject::lookupObjectAddress(uint64_t Address) const { auto Mapping = AddressToMapping.find(Address); if (Mapping == AddressToMapping.end()) return nullptr; return Mapping->getSecond(); } void DebugMap::print(raw_ostream &OS) const { yaml::Output yout(OS, /* Ctxt = */ nullptr, /* WrapColumn = */ 0); yout << const_cast(*this); } #ifndef NDEBUG void DebugMap::dump() const { print(errs()); } #endif namespace { struct YAMLContext { StringRef PrependPath; Triple BinaryTriple; }; } // end anonymous namespace ErrorOr>> DebugMap::parseYAMLDebugMap(StringRef InputFile, StringRef PrependPath, bool Verbose) { auto ErrOrFile = MemoryBuffer::getFileOrSTDIN(InputFile); if (auto Err = ErrOrFile.getError()) return Err; YAMLContext Ctxt; Ctxt.PrependPath = PrependPath; std::unique_ptr Res; yaml::Input yin((*ErrOrFile)->getBuffer(), &Ctxt); yin >> Res; if (auto EC = yin.error()) return EC; std::vector> Result; Result.push_back(std::move(Res)); return std::move(Result); } } // end namespace dsymutil namespace yaml { // Normalize/Denormalize between YAML and a DebugMapObject. struct MappingTraits::YamlDMO { YamlDMO(IO &io) { Timestamp = 0; } YamlDMO(IO &io, dsymutil::DebugMapObject &Obj); dsymutil::DebugMapObject denormalize(IO &IO); std::string Filename; int64_t Timestamp; std::vector Entries; }; void MappingTraits>:: mapping(IO &io, std::pair &s) { io.mapRequired("sym", s.first); io.mapOptional("objAddr", s.second.ObjectAddress); io.mapRequired("binAddr", s.second.BinaryAddress); io.mapOptional("size", s.second.Size); } void MappingTraits::mapping( IO &io, dsymutil::DebugMapObject &DMO) { MappingNormalization Norm(io, DMO); io.mapRequired("filename", Norm->Filename); io.mapOptional("timestamp", Norm->Timestamp); io.mapRequired("symbols", Norm->Entries); } void ScalarTraits::output(const Triple &val, void *, raw_ostream &out) { out << val.str(); } StringRef ScalarTraits::input(StringRef scalar, void *, Triple &value) { value = Triple(scalar); return StringRef(); } size_t SequenceTraits>>::size( IO &io, std::vector> &seq) { return seq.size(); } dsymutil::DebugMapObject & SequenceTraits>>::element( IO &, std::vector> &seq, size_t index) { if (index >= seq.size()) { seq.resize(index + 1); seq[index].reset(new dsymutil::DebugMapObject); } return *seq[index]; } void MappingTraits::mapping(IO &io, dsymutil::DebugMap &DM) { io.mapRequired("triple", DM.BinaryTriple); io.mapOptional("binary-path", DM.BinaryPath); if (void *Ctxt = io.getContext()) reinterpret_cast(Ctxt)->BinaryTriple = DM.BinaryTriple; io.mapOptional("objects", DM.Objects); } void MappingTraits>::mapping( IO &io, std::unique_ptr &DM) { if (!DM) DM.reset(new DebugMap()); io.mapRequired("triple", DM->BinaryTriple); io.mapOptional("binary-path", DM->BinaryPath); if (void *Ctxt = io.getContext()) reinterpret_cast(Ctxt)->BinaryTriple = DM->BinaryTriple; io.mapOptional("objects", DM->Objects); } MappingTraits::YamlDMO::YamlDMO( IO &io, dsymutil::DebugMapObject &Obj) { Filename = Obj.Filename; Timestamp = sys::toTimeT(Obj.getTimestamp()); Entries.reserve(Obj.Symbols.size()); for (auto &Entry : Obj.Symbols) Entries.push_back( std::make_pair(std::string(Entry.getKey()), Entry.getValue())); } dsymutil::DebugMapObject MappingTraits::YamlDMO::denormalize(IO &IO) { BinaryHolder BinHolder(vfs::getRealFileSystem(), /* Verbose =*/false); const auto &Ctxt = *reinterpret_cast(IO.getContext()); SmallString<80> Path(Ctxt.PrependPath); StringMap SymbolAddresses; sys::path::append(Path, Filename); auto ObjectEntry = BinHolder.getObjectEntry(Path); if (!ObjectEntry) { auto Err = ObjectEntry.takeError(); WithColor::warning() << "Unable to open " << Path << " " << toString(std::move(Err)) << '\n'; } else { auto Object = ObjectEntry->getObject(Ctxt.BinaryTriple); if (!Object) { auto Err = Object.takeError(); WithColor::warning() << "Unable to open " << Path << " " << toString(std::move(Err)) << '\n'; } else { for (const auto &Sym : Object->symbols()) { Expected AddressOrErr = Sym.getValue(); if (!AddressOrErr) { // TODO: Actually report errors helpfully. consumeError(AddressOrErr.takeError()); continue; } Expected Name = Sym.getName(); Expected FlagsOrErr = Sym.getFlags(); if (!Name || !FlagsOrErr || (*FlagsOrErr & (SymbolRef::SF_Absolute | SymbolRef::SF_Common))) { // TODO: Actually report errors helpfully. if (!FlagsOrErr) consumeError(FlagsOrErr.takeError()); if (!Name) consumeError(Name.takeError()); continue; } SymbolAddresses[*Name] = *AddressOrErr; } } } dsymutil::DebugMapObject Res(Path, sys::toTimePoint(Timestamp), MachO::N_OSO); for (auto &Entry : Entries) { auto &Mapping = Entry.second; Optional ObjAddress; if (Mapping.ObjectAddress) ObjAddress = *Mapping.ObjectAddress; auto AddressIt = SymbolAddresses.find(Entry.first); if (AddressIt != SymbolAddresses.end()) ObjAddress = AddressIt->getValue(); Res.addSymbol(Entry.first, ObjAddress, Mapping.BinaryAddress, Mapping.Size); } return Res; } } // end namespace yaml } // end namespace llvm