//===- tools/dsymutil/DebugMap.cpp - Generic debug map representation -----===// // // The LLVM Linker // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "DebugMap.h" #include "BinaryHolder.h" #include "llvm/ADT/STLExtras.h" #include "llvm/ADT/iterator_range.h" #include "llvm/Support/DataTypes.h" #include "llvm/Support/Format.h" #include "llvm/Support/raw_ostream.h" #include namespace llvm { namespace dsymutil { using namespace llvm::object; DebugMapObject::DebugMapObject(StringRef ObjectFilename, sys::TimeValue Timestamp) : Filename(ObjectFilename), Timestamp(Timestamp) {} 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). typedef std::pair Entry; std::vector Entries; Entries.reserve(Symbols.getNumItems()); for (const auto &Sym : make_range(Symbols.begin(), Symbols.end())) Entries.push_back(std::make_pair(Sym.getKey(), Sym.getValue())); std::sort( Entries.begin(), Entries.end(), [](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::TimeValue Timestamp) { Objects.emplace_back(new DebugMapObject(ObjectFilePath, Timestamp)); 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; }; } 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); } } 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; sys::TimeValue::SecondsType 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 *, llvm::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 = Obj.getTimestamp().toEpochTime(); Entries.reserve(Obj.Symbols.size()); for (auto &Entry : Obj.Symbols) Entries.push_back(std::make_pair(Entry.getKey(), Entry.getValue())); } dsymutil::DebugMapObject MappingTraits::YamlDMO::denormalize(IO &IO) { BinaryHolder BinHolder(/* Verbose =*/false); const auto &Ctxt = *reinterpret_cast(IO.getContext()); SmallString<80> Path(Ctxt.PrependPath); StringMap SymbolAddresses; 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'; } 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 // requiring binaries checked-in. If we generate the object files // during the test, we can't hardcode the symbols addresses, so // look them up here and rewrite them. for (const auto &Sym : ErrOrObjectFile->symbols()) { uint64_t Address = Sym.getValue(); ErrorOr Name = Sym.getName(); if (!Name || (Sym.getFlags() & (SymbolRef::SF_Absolute | SymbolRef::SF_Common))) continue; SymbolAddresses[*Name] = Address; } } sys::TimeValue TV; TV.fromEpochTime(Timestamp); dsymutil::DebugMapObject Res(Path, TV); 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; } } }