[Symbolize] Improve the ownership of parsed objects.

This code changes the way Symbolize handles parsed binaries: now
parsed OwningBinary<Binary> is not broken into (binary, memory buffer)
pair, and is just stored as-is in a cache. ObjectFile components
of Mach-O universal binaries are also stored explicitly in a
separate cache.

Additionally, this change:
* simplifies the code that parses/caches binaries: it's now done
  in a single place, not three different functions.
* makes flush() method behave as expected, and actually clear
  the cached parsed binaries and objects.
* fixes a dangling pointer issue described in
  http://reviews.llvm.org/D15638

llvm-svn: 256041
This commit is contained in:
Alexey Samsonov 2015-12-18 22:02:14 +00:00
parent 718a9b2844
commit 1eaae4c3b1
2 changed files with 67 additions and 71 deletions

View File

@ -13,13 +13,9 @@
#ifndef LLVM_DEBUGINFO_SYMBOLIZE_SYMBOLIZE_H #ifndef LLVM_DEBUGINFO_SYMBOLIZE_SYMBOLIZE_H
#define LLVM_DEBUGINFO_SYMBOLIZE_SYMBOLIZE_H #define LLVM_DEBUGINFO_SYMBOLIZE_SYMBOLIZE_H
#include "llvm/ADT/SmallVector.h"
#include "llvm/DebugInfo/DIContext.h"
#include "llvm/DebugInfo/Symbolize/SymbolizableModule.h" #include "llvm/DebugInfo/Symbolize/SymbolizableModule.h"
#include "llvm/Object/MachOUniversal.h"
#include "llvm/Object/ObjectFile.h" #include "llvm/Object/ObjectFile.h"
#include "llvm/Support/ErrorOr.h" #include "llvm/Support/ErrorOr.h"
#include "llvm/Support/MemoryBuffer.h"
#include <map> #include <map>
#include <memory> #include <memory>
#include <string> #include <string>
@ -63,6 +59,8 @@ public:
const SymbolizableModule *ModInfo); const SymbolizableModule *ModInfo);
private: private:
// Bundles together object file with code/data and object file with
// corresponding debug info. These objects can be the same.
typedef std::pair<ObjectFile*, ObjectFile*> ObjectPair; typedef std::pair<ObjectFile*, ObjectFile*> ObjectPair;
ErrorOr<SymbolizableModule *> ErrorOr<SymbolizableModule *>
@ -75,30 +73,29 @@ private:
const std::string &ArchName); const std::string &ArchName);
/// \brief Returns pair of pointers to object and debug object. /// \brief Returns pair of pointers to object and debug object.
ErrorOr<ObjectPair> getOrCreateObjects(const std::string &Path, ErrorOr<ObjectPair> getOrCreateObjectPair(const std::string &Path,
const std::string &ArchName); const std::string &ArchName);
/// \brief Returns a parsed object file for a given architecture in a
/// universal binary (or the binary itself if it is an object file).
ErrorOr<ObjectFile *> getObjectFileFromBinary(Binary *Bin,
const std::string &ArchName);
// Owns all the parsed binaries and object files. /// \brief Return a pointer to object file at specified path, for a specified
SmallVector<std::unique_ptr<Binary>, 4> ParsedBinariesAndObjects; /// architecture (e.g. if path refers to a Mach-O universal binary, only one
SmallVector<std::unique_ptr<MemoryBuffer>, 4> MemoryBuffers; /// object file from it will be returned).
void addOwningBinary(OwningBinary<Binary> OwningBin) { ErrorOr<ObjectFile *> getOrCreateObject(const std::string &Path,
std::unique_ptr<Binary> Bin; const std::string &ArchName);
std::unique_ptr<MemoryBuffer> MemBuf;
std::tie(Bin, MemBuf) = OwningBin.takeBinary();
ParsedBinariesAndObjects.push_back(std::move(Bin));
MemoryBuffers.push_back(std::move(MemBuf));
}
std::map<std::string, ErrorOr<std::unique_ptr<SymbolizableModule>>> Modules; std::map<std::string, ErrorOr<std::unique_ptr<SymbolizableModule>>> Modules;
std::map<std::pair<MachOUniversalBinary *, std::string>,
ErrorOr<ObjectFile *>> ObjectFileForArch; /// \brief Contains cached results of getOrCreateObjectPair().
std::map<std::pair<std::string, std::string>, ErrorOr<ObjectPair>> std::map<std::pair<std::string, std::string>, ErrorOr<ObjectPair>>
ObjectPairForPathArch; ObjectPairForPathArch;
/// \brief Contains parsed binary for each path, or parsing error.
std::map<std::string, ErrorOr<OwningBinary<Binary>>> BinaryForPath;
/// \brief Parsed object file for path/architecture pair, where "path" refers
/// to Mach-O universal binary.
std::map<std::pair<std::string, std::string>, ErrorOr<std::unique_ptr<ObjectFile>>>
ObjectForUBPathAndArch;
Options Opts; Options Opts;
}; };

View File

@ -22,6 +22,7 @@
#include "llvm/DebugInfo/PDB/PDBContext.h" #include "llvm/DebugInfo/PDB/PDBContext.h"
#include "llvm/Object/ELFObjectFile.h" #include "llvm/Object/ELFObjectFile.h"
#include "llvm/Object/MachO.h" #include "llvm/Object/MachO.h"
#include "llvm/Object/MachOUniversal.h"
#include "llvm/Support/COFF.h" #include "llvm/Support/COFF.h"
#include "llvm/Support/Casting.h" #include "llvm/Support/Casting.h"
#include "llvm/Support/Compression.h" #include "llvm/Support/Compression.h"
@ -109,9 +110,10 @@ ErrorOr<DIGlobal> LLVMSymbolizer::symbolizeData(const std::string &ModuleName,
} }
void LLVMSymbolizer::flush() { void LLVMSymbolizer::flush() {
Modules.clear(); ObjectForUBPathAndArch.clear();
BinaryForPath.clear();
ObjectPairForPathArch.clear(); ObjectPairForPathArch.clear();
ObjectFileForArch.clear(); Modules.clear();
} }
// For Path="/path/to/foo" and Basename="foo" assume that debug info is in // For Path="/path/to/foo" and Basename="foo" assume that debug info is in
@ -223,23 +225,16 @@ ObjectFile *LLVMSymbolizer::lookUpDsymFile(const std::string &ExePath,
for (const auto &Path : Opts.DsymHints) { for (const auto &Path : Opts.DsymHints) {
DsymPaths.push_back(getDarwinDWARFResourceForPath(Path, Filename)); DsymPaths.push_back(getDarwinDWARFResourceForPath(Path, Filename));
} }
for (const auto &path : DsymPaths) { for (const auto &Path : DsymPaths) {
ErrorOr<OwningBinary<Binary>> BinaryOrErr = createBinary(path); auto DbgObjOrErr = getOrCreateObject(Path, ArchName);
if (!BinaryOrErr)
continue;
OwningBinary<Binary> &B = BinaryOrErr.get();
auto DbgObjOrErr = getObjectFileFromBinary(B.getBinary(), ArchName);
if (!DbgObjOrErr) if (!DbgObjOrErr)
continue; continue;
ObjectFile *DbgObj = DbgObjOrErr.get(); ObjectFile *DbgObj = DbgObjOrErr.get();
const MachOObjectFile *MachDbgObj = const MachOObjectFile *MachDbgObj = dyn_cast<const MachOObjectFile>(DbgObj);
dyn_cast<const MachOObjectFile>(DbgObj);
if (!MachDbgObj) if (!MachDbgObj)
continue; continue;
if (darwinDsymMatchesBinary(MachDbgObj, MachExeObj)) { if (darwinDsymMatchesBinary(MachDbgObj, MachExeObj))
addOwningBinary(std::move(B));
return DbgObj; return DbgObj;
}
} }
return nullptr; return nullptr;
} }
@ -254,40 +249,25 @@ ObjectFile *LLVMSymbolizer::lookUpDebuglinkObject(const std::string &Path,
return nullptr; return nullptr;
if (!findDebugBinary(Path, DebuglinkName, CRCHash, DebugBinaryPath)) if (!findDebugBinary(Path, DebuglinkName, CRCHash, DebugBinaryPath))
return nullptr; return nullptr;
ErrorOr<OwningBinary<Binary>> DebugBinaryOrErr = auto DbgObjOrErr = getOrCreateObject(DebugBinaryPath, ArchName);
createBinary(DebugBinaryPath);
if (!DebugBinaryOrErr)
return nullptr;
OwningBinary<Binary> &DB = DebugBinaryOrErr.get();
auto DbgObjOrErr = getObjectFileFromBinary(DB.getBinary(), ArchName);
if (!DbgObjOrErr) if (!DbgObjOrErr)
return nullptr; return nullptr;
addOwningBinary(std::move(DB));
return DbgObjOrErr.get(); return DbgObjOrErr.get();
} }
ErrorOr<LLVMSymbolizer::ObjectPair> ErrorOr<LLVMSymbolizer::ObjectPair>
LLVMSymbolizer::getOrCreateObjects(const std::string &Path, LLVMSymbolizer::getOrCreateObjectPair(const std::string &Path,
const std::string &ArchName) { const std::string &ArchName) {
const auto &I = ObjectPairForPathArch.find(std::make_pair(Path, ArchName)); const auto &I = ObjectPairForPathArch.find(std::make_pair(Path, ArchName));
if (I != ObjectPairForPathArch.end()) if (I != ObjectPairForPathArch.end())
return I->second; return I->second;
ErrorOr<OwningBinary<Binary>> BinaryOrErr = createBinary(Path); auto ObjOrErr = getOrCreateObject(Path, ArchName);
if (auto EC = BinaryOrErr.getError()) {
ObjectPairForPathArch.insert(
std::make_pair(std::make_pair(Path, ArchName), EC));
return EC;
}
OwningBinary<Binary> &B = BinaryOrErr.get();
auto ObjOrErr = getObjectFileFromBinary(B.getBinary(), ArchName);
if (auto EC = ObjOrErr.getError()) { if (auto EC = ObjOrErr.getError()) {
ObjectPairForPathArch.insert( ObjectPairForPathArch.insert(
std::make_pair(std::make_pair(Path, ArchName), EC)); std::make_pair(std::make_pair(Path, ArchName), EC));
return EC; return EC;
} }
addOwningBinary(std::move(B));
ObjectFile *Obj = ObjOrErr.get(); ObjectFile *Obj = ObjOrErr.get();
assert(Obj != nullptr); assert(Obj != nullptr);
@ -306,24 +286,43 @@ LLVMSymbolizer::getOrCreateObjects(const std::string &Path,
} }
ErrorOr<ObjectFile *> ErrorOr<ObjectFile *>
LLVMSymbolizer::getObjectFileFromBinary(Binary *Bin, LLVMSymbolizer::getOrCreateObject(const std::string &Path,
const std::string &ArchName) { const std::string &ArchName) {
assert(Bin != nullptr); const auto &I = BinaryForPath.find(Path);
if (MachOUniversalBinary *UB = dyn_cast<MachOUniversalBinary>(Bin)) { Binary *Bin = nullptr;
const auto &I = ObjectFileForArch.find( if (I == BinaryForPath.end()) {
std::make_pair(UB, ArchName)); ErrorOr<OwningBinary<Binary>> BinOrErr = createBinary(Path);
if (I != ObjectFileForArch.end()) if (auto EC = BinOrErr.getError()) {
return I->second; BinaryForPath.insert(std::make_pair(Path, EC));
ErrorOr<std::unique_ptr<ObjectFile>> ParsedObj =
UB->getObjectForArch(ArchName);
if (auto EC = ParsedObj.getError()) {
ObjectFileForArch.insert(
std::make_pair(std::make_pair(UB, ArchName), EC));
return EC; return EC;
} }
ObjectFile *Res = ParsedObj.get().get(); Bin = BinOrErr->getBinary();
ParsedBinariesAndObjects.push_back(std::move(ParsedObj.get())); BinaryForPath.insert(std::make_pair(Path, std::move(BinOrErr.get())));
ObjectFileForArch.insert(std::make_pair(std::make_pair(UB, ArchName), Res)); } else if (auto EC = I->second.getError()) {
return EC;
} else {
Bin = I->second->getBinary();
}
assert(Bin != nullptr);
if (MachOUniversalBinary *UB = dyn_cast<MachOUniversalBinary>(Bin)) {
const auto &I = ObjectForUBPathAndArch.find(std::make_pair(Path, ArchName));
if (I != ObjectForUBPathAndArch.end()) {
if (auto EC = I->second.getError())
return EC;
return I->second->get();
}
ErrorOr<std::unique_ptr<ObjectFile>> ObjOrErr =
UB->getObjectForArch(ArchName);
if (auto EC = ObjOrErr.getError()) {
ObjectForUBPathAndArch.insert(
std::make_pair(std::make_pair(Path, ArchName), EC));
return EC;
}
ObjectFile *Res = ObjOrErr->get();
ObjectForUBPathAndArch.insert(std::make_pair(std::make_pair(Path, ArchName),
std::move(ObjOrErr.get())));
return Res; return Res;
} }
if (Bin->isObject()) { if (Bin->isObject()) {
@ -352,7 +351,7 @@ LLVMSymbolizer::getOrCreateModuleInfo(const std::string &ModuleName) {
ArchName = ArchStr; ArchName = ArchStr;
} }
} }
auto ObjectsOrErr = getOrCreateObjects(BinaryName, ArchName); auto ObjectsOrErr = getOrCreateObjectPair(BinaryName, ArchName);
if (auto EC = ObjectsOrErr.getError()) { if (auto EC = ObjectsOrErr.getError()) {
// Failed to find valid object file. // Failed to find valid object file.
Modules.insert(std::make_pair(ModuleName, EC)); Modules.insert(std::make_pair(ModuleName, EC));