28#define DEBUG_TYPE "ReaderHandler"
33 if (
Error Err = createReaders())
35 if (
Error Err = printReaders())
37 if (
Error Err = compareReaders())
47 auto CreateOneReader = [&]() -> std::unique_ptr<LVReader> {
52 return std::make_unique<LVCodeViewReader>(
Filename, FileFormatName,
55 if (Obj.isELF() || Obj.isMachO() || Obj.isWasm())
56 return std::make_unique<LVDWARFReader>(
Filename, FileFormatName, Obj,
61 return std::make_unique<LVCodeViewReader>(
Filename, FileFormatName,
Pdb,
65 return std::make_unique<LVIRReader>(
Filename, FileFormatName, Ir, W);
69 const StringRef IRFileExt =
".ll";
77 std::unique_ptr<LVReader> ReaderObj = CreateOneReader();
80 "unable to create reader for: '%s'",
83 LVReader *Reader = ReaderObj.get();
84 Readers.emplace_back(std::move(ReaderObj));
91 for (
const Archive::Child &Child : Arch.
children(Err)) {
92 Expected<MemoryBufferRef> BuffOrErr = Child.getMemoryBufferRef();
96 Expected<StringRef> NameOrErr = Child.getName();
101 if (
Error Err = handleBuffer(Readers,
Name, BuffOrErr.
get()))
118 std::unique_ptr<IPDBSession> Session;
132 if (ConvertedPath ==
Path)
133 return std::string(ExePath);
148 return std::string(ObjPath);
155 MemoryBufferRef Buffer, StringRef ExePath) {
160 if (!ExePath.
empty())
167 std::vector<StringRef> ExecutableExtensions = {
"exe",
"dll"};
168 for (StringRef Extension : ExecutableExtensions) {
170 if (ExecutableImage.empty())
180 std::vector<StringRef> ObjectExtensions = {
"o",
"obj",
"lib"};
181 for (StringRef Extension : ObjectExtensions) {
183 if (ObjectImage.empty())
185 if (
Error Err = handleFile(Readers, ObjectImage)) {
201 "Binary object format in '%s' does not have debug info.",
205 return handleFile(Readers, PdbPath.
get(),
Filename);
212 return handleObject(Readers,
Filename, Buffer);
214 return handleObject(Readers,
Filename, *BinOrErr.
get());
220 std::string ConvertedPath =
222 ErrorOr<std::unique_ptr<MemoryBuffer>> BuffOrErr =
226 "File '%s' does not exist.",
227 ConvertedPath.c_str());
229 std::unique_ptr<MemoryBuffer> Buffer = std::move(BuffOrErr.
get());
230 return handleBuffer(Readers, ConvertedPath, *Buffer, ExePath);
234 MachOUniversalBinary &Mach) {
235 for (
const MachOUniversalBinary::ObjectForArch &ObjForArch : Mach.
objects()) {
236 std::string ObjName = (Twine(
Filename) + Twine(
"(") +
237 Twine(ObjForArch.getArchFlagName()) + Twine(
")"))
239 if (Expected<std::unique_ptr<MachOObjectFile>> MachOOrErr =
240 ObjForArch.getAsObjectFile()) {
241 MachOObjectFile &
Obj = **MachOOrErr;
244 createReader(
Filename, Readers, Input,
Obj.getFileFormatName()))
249 if (Expected<std::unique_ptr<Archive>> ArchiveOrErr =
250 ObjForArch.getAsArchive()) {
251 if (
Error Err = handleArchive(Readers, ObjName, *ArchiveOrErr.get()))
263 return createReader(
Filename, Readers, Input,
267 return handleMach(Readers,
Filename, *Fat);
270 return handleArchive(Readers,
Filename, *Arch);
273 return createReader(
Filename, Readers, Input,
"Bitcode IR");
276 "Binary object format in '%s' is not supported.",
281 StringRef Buffer, StringRef ExePath) {
282 std::unique_ptr<IPDBSession> Session;
287 std::unique_ptr<NativeSession> PdbSession;
288 PdbSession.reset(
static_cast<NativeSession *
>(Session.release()));
290 StringRef FileFormatName;
293 FileFormatName = Buffer.
substr(0, Pos - 1);
294 return createReader(
Filename, Readers, Input, FileFormatName, ExePath);
298 MemoryBufferRef Buffer) {
303Error LVReaderHandler::createReaders() {
305 for (std::string &Object : Objects) {
307 if (
Error Err = createReader(Object, Readers))
309 TheReaders.insert(TheReaders.end(),
310 std::make_move_iterator(Readers.begin()),
311 std::make_move_iterator(Readers.end()));
317Error LVReaderHandler::printReaders() {
319 if (
options().getPrintExecute())
320 for (
const std::unique_ptr<LVReader> &Reader : TheReaders)
327Error LVReaderHandler::compareReaders() {
329 size_t ReadersCount = TheReaders.size();
330 if (
options().getCompareExecute() && ReadersCount >= 2) {
332 size_t ViewPairs = ReadersCount / 2;
334 for (
size_t Pair = 0, Index = 0; Pair < ViewPairs; ++Pair) {
335 if (
Error Err =
Compare.execute(TheReaders[Index].get(),
336 TheReaders[Index + 1].get()))
static constexpr StringRef IRFileFormatName
static std::string searchForObj(const StringRef Path, const StringRef Extension)
static std::string searchForExe(const StringRef Path, const StringRef Extension)
static constexpr StringLiteral Filename
Represents either an error or a value T.
std::error_code getError() const
Lightweight error class with error context and mandatory checking.
static ErrorSuccess success()
Create a success value.
Tagged union holding either a T or a Error.
Error takeError()
Take ownership of the stored error.
reference get()
Returns a reference to the stored T value.
StringRef getBuffer() const
static ErrorOr< std::unique_ptr< MemoryBuffer > > getFileOrSTDIN(const Twine &Filename, bool IsText=false, bool RequiresNullTerminator=true, std::optional< Align > Alignment=std::nullopt)
Open the specified file as a MemoryBuffer, or open stdin if the Filename is "-".
SmallString - A SmallString is just a SmallVector with methods and accessors that make it work better...
Represent a constant reference to a string, i.e.
constexpr StringRef substr(size_t Start, size_t N=npos) const
Return a reference to the substring from [Start, Start + N).
constexpr bool empty() const
Check if the string is empty.
size_t find_first_of(char C, size_t From=0) const
Find the first character in the string that is C, or npos if not found.
LLVM_ABI void print(raw_ostream &OS) const
iterator_range< child_iterator > children(Error &Err, bool SkipInternal=true) const
iterator_range< object_iterator > objects() const
static Expected< std::string > searchForPdb(const PdbSearchOptions &Opts)
This class implements an extremely fast bulk output stream that can only output to a stream.
std::vector< std::unique_ptr< LVReader > > LVReaders
PointerUnion< object::ObjectFile *, pdb::PDBFile *, object::IRObjectFile *, MemoryBufferRef *, StringRef * > InputHandle
LLVM_ABI Expected< std::unique_ptr< Binary > > createBinary(MemoryBufferRef Source, LLVMContext *Context=nullptr, bool InitContent=true)
Create a Binary from Source, autodetecting the file type.
LLVM_ABI Error loadDataForEXE(PDB_ReaderType Type, StringRef Path, std::unique_ptr< IPDBSession > &Session)
LLVM_ABI Error loadDataForPDB(PDB_ReaderType Type, StringRef Path, std::unique_ptr< IPDBSession > &Session)
LLVM_ABI bool exists(const basic_file_status &status)
Does file exist?
LLVM_ABI void replace_extension(SmallVectorImpl< char > &path, const Twine &extension, Style style=Style::native)
Replace the file extension of path with extension.
LLVM_ABI std::string convert_to_slash(StringRef path, Style style=Style::native)
Replaces backslashes with slashes if Windows.
LLVM_ABI StringRef extension(StringRef path LLVM_LIFETIME_BOUND, Style style=Style::native)
Get extension.
This is an optimization pass for GlobalISel generic memory operations.
LLVM_ABI file_magic identify_magic(StringRef magic)
Identify the type of a binary file based on how magical it is.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
Error createStringError(std::error_code EC, char const *Fmt, const Ts &... Vals)
Create formatted StringError object.
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
bool isa(const From &Val)
isa<X> - Return true if the parameter to the template is an instance of one of the template type argu...
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
LLVM_ABI std::error_code errorToErrorCode(Error Err)
Helper for converting an ECError to a std::error_code.
void consumeError(Error Err)
Consume a Error without doing anything.
@ pdb
Windows PDB debug info file.
@ pecoff_executable
PECOFF executable file.