mirror of
				https://github.com/c64scene-ar/llvm-6502.git
				synced 2025-10-30 16:17:05 +00:00 
			
		
		
		
	(instead of basenames) from DWARF. Use this behavior in llvm-dwarfdump tool. Reviewed by Benjamin Kramer. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@160496 91177308-0d34-0410-b5e6-96231b3b80d8
		
			
				
	
	
		
			129 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			129 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //===-- llvm-dwarfdump.cpp - Debug info dumping utility for llvm -----------===//
 | |
| //
 | |
| //                     The LLVM Compiler Infrastructure
 | |
| //
 | |
| // This file is distributed under the University of Illinois Open Source
 | |
| // License. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| //
 | |
| // This program is a utility that works like "dwarfdump".
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "llvm/ADT/OwningPtr.h"
 | |
| #include "llvm/ADT/Triple.h"
 | |
| #include "llvm/ADT/STLExtras.h"
 | |
| #include "llvm/Object/ObjectFile.h"
 | |
| #include "llvm/DebugInfo/DIContext.h"
 | |
| #include "llvm/Support/CommandLine.h"
 | |
| #include "llvm/Support/Debug.h"
 | |
| #include "llvm/Support/Format.h"
 | |
| #include "llvm/Support/ManagedStatic.h"
 | |
| #include "llvm/Support/MemoryBuffer.h"
 | |
| #include "llvm/Support/MemoryObject.h"
 | |
| #include "llvm/Support/PrettyStackTrace.h"
 | |
| #include "llvm/Support/Signals.h"
 | |
| #include "llvm/Support/raw_ostream.h"
 | |
| #include "llvm/Support/system_error.h"
 | |
| #include <algorithm>
 | |
| #include <cstring>
 | |
| using namespace llvm;
 | |
| using namespace object;
 | |
| 
 | |
| static cl::list<std::string>
 | |
| InputFilenames(cl::Positional, cl::desc("<input object files>"),
 | |
|                cl::ZeroOrMore);
 | |
| 
 | |
| static cl::opt<unsigned long long>
 | |
| Address("address", cl::init(-1ULL),
 | |
|         cl::desc("Print line information for a given address"));
 | |
| 
 | |
| static cl::opt<bool>
 | |
| PrintFunctions("functions", cl::init(false),
 | |
|                cl::desc("Print function names as well as line information "
 | |
|                         "for a given address"));
 | |
| 
 | |
| static void DumpInput(const StringRef &Filename) {
 | |
|   OwningPtr<MemoryBuffer> Buff;
 | |
| 
 | |
|   if (error_code ec = MemoryBuffer::getFileOrSTDIN(Filename, Buff)) {
 | |
|     errs() << Filename << ": " << ec.message() << "\n";
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   OwningPtr<ObjectFile> Obj(ObjectFile::createObjectFile(Buff.take()));
 | |
| 
 | |
|   StringRef DebugInfoSection;
 | |
|   StringRef DebugAbbrevSection;
 | |
|   StringRef DebugLineSection;
 | |
|   StringRef DebugArangesSection;
 | |
|   StringRef DebugStringSection;
 | |
| 
 | |
|   error_code ec;
 | |
|   for (section_iterator i = Obj->begin_sections(),
 | |
|                         e = Obj->end_sections();
 | |
|                         i != e; i.increment(ec)) {
 | |
|     StringRef name;
 | |
|     i->getName(name);
 | |
|     StringRef data;
 | |
|     i->getContents(data);
 | |
| 
 | |
|     if (name.startswith("__DWARF,"))
 | |
|       name = name.substr(8); // Skip "__DWARF," prefix.
 | |
|     name = name.substr(name.find_first_not_of("._")); // Skip . and _ prefixes.
 | |
|     if (name == "debug_info")
 | |
|       DebugInfoSection = data;
 | |
|     else if (name == "debug_abbrev")
 | |
|       DebugAbbrevSection = data;
 | |
|     else if (name == "debug_line")
 | |
|       DebugLineSection = data;
 | |
|     else if (name == "debug_aranges")
 | |
|       DebugArangesSection = data;
 | |
|     else if (name == "debug_str")
 | |
|       DebugStringSection = data;
 | |
|   }
 | |
| 
 | |
|   OwningPtr<DIContext> dictx(DIContext::getDWARFContext(/*FIXME*/true,
 | |
|                                                         DebugInfoSection,
 | |
|                                                         DebugAbbrevSection,
 | |
|                                                         DebugArangesSection,
 | |
|                                                         DebugLineSection,
 | |
|                                                         DebugStringSection));
 | |
|   if (Address == -1ULL) {
 | |
|     outs() << Filename
 | |
|            << ":\tfile format " << Obj->getFileFormatName() << "\n\n";
 | |
|     // Dump the complete DWARF structure.
 | |
|     dictx->dump(outs());
 | |
|   } else {
 | |
|     // Print line info for the specified address.
 | |
|     int spec_flags = DILineInfoSpecifier::FileLineInfo |
 | |
|                      DILineInfoSpecifier::AbsoluteFilePath;
 | |
|     if (PrintFunctions)
 | |
|       spec_flags |= DILineInfoSpecifier::FunctionName;
 | |
|     DILineInfo dli = dictx->getLineInfoForAddress(Address, spec_flags);
 | |
|     if (PrintFunctions)
 | |
|       outs() << (dli.getFunctionName() ? dli.getFunctionName() : "<unknown>")
 | |
|              << "\n";
 | |
|     outs() << (dli.getFileName() ? dli.getFileName() : "<unknown>") << ':'
 | |
|            << dli.getLine() << ':' << dli.getColumn() << '\n';
 | |
|   }
 | |
| }
 | |
| 
 | |
| int main(int argc, char **argv) {
 | |
|   // Print a stack trace if we signal out.
 | |
|   sys::PrintStackTraceOnErrorSignal();
 | |
|   PrettyStackTraceProgram X(argc, argv);
 | |
|   llvm_shutdown_obj Y;  // Call llvm_shutdown() on exit.
 | |
| 
 | |
|   cl::ParseCommandLineOptions(argc, argv, "llvm dwarf dumper\n");
 | |
| 
 | |
|   // Defaults to a.out if no filenames specified.
 | |
|   if (InputFilenames.size() == 0)
 | |
|     InputFilenames.push_back("a.out");
 | |
| 
 | |
|   std::for_each(InputFilenames.begin(), InputFilenames.end(), DumpInput);
 | |
| 
 | |
|   return 0;
 | |
| }
 |