Add a WITH_CLANG_EXTRAS option for src.conf(5), disabled by default,
that builds the following additional llvm/clang tools: - bugpoint - llc - lli - llvm-ar - llvm-as - llvm-bcanalyzer - llvm-diff - llvm-dis - llvm-extract - llvm-ld - llvm-link - llvm-mc - llvm-nm - llvm-objdump - llvm-prof - llvm-ranlib - llvm-rtdyld - llvm-stub - macho-dump - opt These tools are mainly useful for people that want to manipulate llvm bitcode (.bc) and llvm assembly language (.ll) files, or want to tinker with llvm and clang themselves. MFC after: 2 weeks
This commit is contained in:
commit
eb8951e7f7
247
contrib/llvm/tools/bugpoint/BugDriver.cpp
Normal file
247
contrib/llvm/tools/bugpoint/BugDriver.cpp
Normal file
@ -0,0 +1,247 @@
|
||||
//===- BugDriver.cpp - Top-Level BugPoint class implementation ------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This class contains all of the shared state and information that is used by
|
||||
// the BugPoint tool to track down errors in optimizations. This class is the
|
||||
// main driver class that invokes all sub-functionality.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "BugDriver.h"
|
||||
#include "ToolRunner.h"
|
||||
#include "llvm/Linker.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/Pass.h"
|
||||
#include "llvm/Support/IRReader.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/FileUtilities.h"
|
||||
#include "llvm/Support/SourceMgr.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/Host.h"
|
||||
#include <memory>
|
||||
using namespace llvm;
|
||||
|
||||
namespace llvm {
|
||||
Triple TargetTriple;
|
||||
}
|
||||
|
||||
// Anonymous namespace to define command line options for debugging.
|
||||
//
|
||||
namespace {
|
||||
// Output - The user can specify a file containing the expected output of the
|
||||
// program. If this filename is set, it is used as the reference diff source,
|
||||
// otherwise the raw input run through an interpreter is used as the reference
|
||||
// source.
|
||||
//
|
||||
cl::opt<std::string>
|
||||
OutputFile("output", cl::desc("Specify a reference program output "
|
||||
"(for miscompilation detection)"));
|
||||
}
|
||||
|
||||
/// setNewProgram - If we reduce or update the program somehow, call this method
|
||||
/// to update bugdriver with it. This deletes the old module and sets the
|
||||
/// specified one as the current program.
|
||||
void BugDriver::setNewProgram(Module *M) {
|
||||
delete Program;
|
||||
Program = M;
|
||||
}
|
||||
|
||||
|
||||
/// getPassesString - Turn a list of passes into a string which indicates the
|
||||
/// command line options that must be passed to add the passes.
|
||||
///
|
||||
std::string llvm::getPassesString(const std::vector<std::string> &Passes) {
|
||||
std::string Result;
|
||||
for (unsigned i = 0, e = Passes.size(); i != e; ++i) {
|
||||
if (i) Result += " ";
|
||||
Result += "-";
|
||||
Result += Passes[i];
|
||||
}
|
||||
return Result;
|
||||
}
|
||||
|
||||
BugDriver::BugDriver(const char *toolname, bool find_bugs,
|
||||
unsigned timeout, unsigned memlimit, bool use_valgrind,
|
||||
LLVMContext& ctxt)
|
||||
: Context(ctxt), ToolName(toolname), ReferenceOutputFile(OutputFile),
|
||||
Program(0), Interpreter(0), SafeInterpreter(0), gcc(0),
|
||||
run_find_bugs(find_bugs), Timeout(timeout),
|
||||
MemoryLimit(memlimit), UseValgrind(use_valgrind) {}
|
||||
|
||||
BugDriver::~BugDriver() {
|
||||
delete Program;
|
||||
}
|
||||
|
||||
|
||||
/// ParseInputFile - Given a bitcode or assembly input filename, parse and
|
||||
/// return it, or return null if not possible.
|
||||
///
|
||||
Module *llvm::ParseInputFile(const std::string &Filename,
|
||||
LLVMContext& Ctxt) {
|
||||
SMDiagnostic Err;
|
||||
Module *Result = ParseIRFile(Filename, Err, Ctxt);
|
||||
if (!Result)
|
||||
Err.Print("bugpoint", errs());
|
||||
|
||||
// If we don't have an override triple, use the first one to configure
|
||||
// bugpoint, or use the host triple if none provided.
|
||||
if (Result) {
|
||||
if (TargetTriple.getTriple().empty()) {
|
||||
Triple TheTriple(Result->getTargetTriple());
|
||||
|
||||
if (TheTriple.getTriple().empty())
|
||||
TheTriple.setTriple(sys::getHostTriple());
|
||||
|
||||
TargetTriple.setTriple(TheTriple.getTriple());
|
||||
}
|
||||
|
||||
Result->setTargetTriple(TargetTriple.getTriple()); // override the triple
|
||||
}
|
||||
return Result;
|
||||
}
|
||||
|
||||
// This method takes the specified list of LLVM input files, attempts to load
|
||||
// them, either as assembly or bitcode, then link them together. It returns
|
||||
// true on failure (if, for example, an input bitcode file could not be
|
||||
// parsed), and false on success.
|
||||
//
|
||||
bool BugDriver::addSources(const std::vector<std::string> &Filenames) {
|
||||
assert(Program == 0 && "Cannot call addSources multiple times!");
|
||||
assert(!Filenames.empty() && "Must specify at least on input filename!");
|
||||
|
||||
// Load the first input file.
|
||||
Program = ParseInputFile(Filenames[0], Context);
|
||||
if (Program == 0) return true;
|
||||
|
||||
outs() << "Read input file : '" << Filenames[0] << "'\n";
|
||||
|
||||
for (unsigned i = 1, e = Filenames.size(); i != e; ++i) {
|
||||
std::auto_ptr<Module> M(ParseInputFile(Filenames[i], Context));
|
||||
if (M.get() == 0) return true;
|
||||
|
||||
outs() << "Linking in input file: '" << Filenames[i] << "'\n";
|
||||
std::string ErrorMessage;
|
||||
if (Linker::LinkModules(Program, M.get(), Linker::DestroySource,
|
||||
&ErrorMessage)) {
|
||||
errs() << ToolName << ": error linking in '" << Filenames[i] << "': "
|
||||
<< ErrorMessage << '\n';
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
outs() << "*** All input ok\n";
|
||||
|
||||
// All input files read successfully!
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
|
||||
/// run - The top level method that is invoked after all of the instance
|
||||
/// variables are set up from command line arguments.
|
||||
///
|
||||
bool BugDriver::run(std::string &ErrMsg) {
|
||||
if (run_find_bugs) {
|
||||
// Rearrange the passes and apply them to the program. Repeat this process
|
||||
// until the user kills the program or we find a bug.
|
||||
return runManyPasses(PassesToRun, ErrMsg);
|
||||
}
|
||||
|
||||
// If we're not running as a child, the first thing that we must do is
|
||||
// determine what the problem is. Does the optimization series crash the
|
||||
// compiler, or does it produce illegal code? We make the top-level
|
||||
// decision by trying to run all of the passes on the the input program,
|
||||
// which should generate a bitcode file. If it does generate a bitcode
|
||||
// file, then we know the compiler didn't crash, so try to diagnose a
|
||||
// miscompilation.
|
||||
if (!PassesToRun.empty()) {
|
||||
outs() << "Running selected passes on program to test for crash: ";
|
||||
if (runPasses(Program, PassesToRun))
|
||||
return debugOptimizerCrash();
|
||||
}
|
||||
|
||||
// Set up the execution environment, selecting a method to run LLVM bitcode.
|
||||
if (initializeExecutionEnvironment()) return true;
|
||||
|
||||
// Test to see if we have a code generator crash.
|
||||
outs() << "Running the code generator to test for a crash: ";
|
||||
std::string Error;
|
||||
compileProgram(Program, &Error);
|
||||
if (!Error.empty()) {
|
||||
outs() << Error;
|
||||
return debugCodeGeneratorCrash(ErrMsg);
|
||||
}
|
||||
outs() << '\n';
|
||||
|
||||
// Run the raw input to see where we are coming from. If a reference output
|
||||
// was specified, make sure that the raw output matches it. If not, it's a
|
||||
// problem in the front-end or the code generator.
|
||||
//
|
||||
bool CreatedOutput = false;
|
||||
if (ReferenceOutputFile.empty()) {
|
||||
outs() << "Generating reference output from raw program: ";
|
||||
if (!createReferenceFile(Program)) {
|
||||
return debugCodeGeneratorCrash(ErrMsg);
|
||||
}
|
||||
CreatedOutput = true;
|
||||
}
|
||||
|
||||
// Make sure the reference output file gets deleted on exit from this
|
||||
// function, if appropriate.
|
||||
sys::Path ROF(ReferenceOutputFile);
|
||||
FileRemover RemoverInstance(ROF.str(), CreatedOutput && !SaveTemps);
|
||||
|
||||
// Diff the output of the raw program against the reference output. If it
|
||||
// matches, then we assume there is a miscompilation bug and try to
|
||||
// diagnose it.
|
||||
outs() << "*** Checking the code generator...\n";
|
||||
bool Diff = diffProgram(Program, "", "", false, &Error);
|
||||
if (!Error.empty()) {
|
||||
errs() << Error;
|
||||
return debugCodeGeneratorCrash(ErrMsg);
|
||||
}
|
||||
if (!Diff) {
|
||||
outs() << "\n*** Output matches: Debugging miscompilation!\n";
|
||||
debugMiscompilation(&Error);
|
||||
if (!Error.empty()) {
|
||||
errs() << Error;
|
||||
return debugCodeGeneratorCrash(ErrMsg);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
outs() << "\n*** Input program does not match reference diff!\n";
|
||||
outs() << "Debugging code generator problem!\n";
|
||||
bool Failure = debugCodeGenerator(&Error);
|
||||
if (!Error.empty()) {
|
||||
errs() << Error;
|
||||
return debugCodeGeneratorCrash(ErrMsg);
|
||||
}
|
||||
return Failure;
|
||||
}
|
||||
|
||||
void llvm::PrintFunctionList(const std::vector<Function*> &Funcs) {
|
||||
unsigned NumPrint = Funcs.size();
|
||||
if (NumPrint > 10) NumPrint = 10;
|
||||
for (unsigned i = 0; i != NumPrint; ++i)
|
||||
outs() << " " << Funcs[i]->getName();
|
||||
if (NumPrint < Funcs.size())
|
||||
outs() << "... <" << Funcs.size() << " total>";
|
||||
outs().flush();
|
||||
}
|
||||
|
||||
void llvm::PrintGlobalVariableList(const std::vector<GlobalVariable*> &GVs) {
|
||||
unsigned NumPrint = GVs.size();
|
||||
if (NumPrint > 10) NumPrint = 10;
|
||||
for (unsigned i = 0; i != NumPrint; ++i)
|
||||
outs() << " " << GVs[i]->getName();
|
||||
if (NumPrint < GVs.size())
|
||||
outs() << "... <" << GVs.size() << " total>";
|
||||
outs().flush();
|
||||
}
|
330
contrib/llvm/tools/bugpoint/BugDriver.h
Normal file
330
contrib/llvm/tools/bugpoint/BugDriver.h
Normal file
@ -0,0 +1,330 @@
|
||||
//===- BugDriver.h - Top-Level BugPoint class -------------------*- C++ -*-===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This class contains all of the shared state and information that is used by
|
||||
// the BugPoint tool to track down errors in optimizations. This class is the
|
||||
// main driver class that invokes all sub-functionality.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef BUGDRIVER_H
|
||||
#define BUGDRIVER_H
|
||||
|
||||
#include "llvm/ADT/ValueMap.h"
|
||||
#include "llvm/Transforms/Utils/ValueMapper.h"
|
||||
#include <vector>
|
||||
#include <string>
|
||||
|
||||
namespace llvm {
|
||||
|
||||
class Value;
|
||||
class PassInfo;
|
||||
class Module;
|
||||
class GlobalVariable;
|
||||
class Function;
|
||||
class BasicBlock;
|
||||
class AbstractInterpreter;
|
||||
class Instruction;
|
||||
class LLVMContext;
|
||||
|
||||
class DebugCrashes;
|
||||
|
||||
class GCC;
|
||||
|
||||
extern bool DisableSimplifyCFG;
|
||||
|
||||
/// BugpointIsInterrupted - Set to true when the user presses ctrl-c.
|
||||
///
|
||||
extern bool BugpointIsInterrupted;
|
||||
|
||||
class BugDriver {
|
||||
LLVMContext& Context;
|
||||
const char *ToolName; // argv[0] of bugpoint
|
||||
std::string ReferenceOutputFile; // Name of `good' output file
|
||||
Module *Program; // The raw program, linked together
|
||||
std::vector<std::string> PassesToRun;
|
||||
AbstractInterpreter *Interpreter; // How to run the program
|
||||
AbstractInterpreter *SafeInterpreter; // To generate reference output, etc.
|
||||
GCC *gcc;
|
||||
bool run_find_bugs;
|
||||
unsigned Timeout;
|
||||
unsigned MemoryLimit;
|
||||
bool UseValgrind;
|
||||
|
||||
// FIXME: sort out public/private distinctions...
|
||||
friend class ReducePassList;
|
||||
friend class ReduceMisCodegenFunctions;
|
||||
|
||||
public:
|
||||
BugDriver(const char *toolname, bool find_bugs,
|
||||
unsigned timeout, unsigned memlimit, bool use_valgrind,
|
||||
LLVMContext& ctxt);
|
||||
~BugDriver();
|
||||
|
||||
const char *getToolName() const { return ToolName; }
|
||||
|
||||
LLVMContext& getContext() const { return Context; }
|
||||
|
||||
// Set up methods... these methods are used to copy information about the
|
||||
// command line arguments into instance variables of BugDriver.
|
||||
//
|
||||
bool addSources(const std::vector<std::string> &FileNames);
|
||||
void addPass(std::string p) { PassesToRun.push_back(p); }
|
||||
void setPassesToRun(const std::vector<std::string> &PTR) {
|
||||
PassesToRun = PTR;
|
||||
}
|
||||
const std::vector<std::string> &getPassesToRun() const {
|
||||
return PassesToRun;
|
||||
}
|
||||
|
||||
/// run - The top level method that is invoked after all of the instance
|
||||
/// variables are set up from command line arguments. The \p as_child argument
|
||||
/// indicates whether the driver is to run in parent mode or child mode.
|
||||
///
|
||||
bool run(std::string &ErrMsg);
|
||||
|
||||
/// debugOptimizerCrash - This method is called when some optimizer pass
|
||||
/// crashes on input. It attempts to prune down the testcase to something
|
||||
/// reasonable, and figure out exactly which pass is crashing.
|
||||
///
|
||||
bool debugOptimizerCrash(const std::string &ID = "passes");
|
||||
|
||||
/// debugCodeGeneratorCrash - This method is called when the code generator
|
||||
/// crashes on an input. It attempts to reduce the input as much as possible
|
||||
/// while still causing the code generator to crash.
|
||||
bool debugCodeGeneratorCrash(std::string &Error);
|
||||
|
||||
/// debugMiscompilation - This method is used when the passes selected are not
|
||||
/// crashing, but the generated output is semantically different from the
|
||||
/// input.
|
||||
void debugMiscompilation(std::string *Error);
|
||||
|
||||
/// debugPassMiscompilation - This method is called when the specified pass
|
||||
/// miscompiles Program as input. It tries to reduce the testcase to
|
||||
/// something that smaller that still miscompiles the program.
|
||||
/// ReferenceOutput contains the filename of the file containing the output we
|
||||
/// are to match.
|
||||
///
|
||||
bool debugPassMiscompilation(const PassInfo *ThePass,
|
||||
const std::string &ReferenceOutput);
|
||||
|
||||
/// compileSharedObject - This method creates a SharedObject from a given
|
||||
/// BitcodeFile for debugging a code generator.
|
||||
///
|
||||
std::string compileSharedObject(const std::string &BitcodeFile,
|
||||
std::string &Error);
|
||||
|
||||
/// debugCodeGenerator - This method narrows down a module to a function or
|
||||
/// set of functions, using the CBE as a ``safe'' code generator for other
|
||||
/// functions that are not under consideration.
|
||||
bool debugCodeGenerator(std::string *Error);
|
||||
|
||||
/// isExecutingJIT - Returns true if bugpoint is currently testing the JIT
|
||||
///
|
||||
bool isExecutingJIT();
|
||||
|
||||
/// runPasses - Run all of the passes in the "PassesToRun" list, discard the
|
||||
/// output, and return true if any of the passes crashed.
|
||||
bool runPasses(Module *M) const {
|
||||
return runPasses(M, PassesToRun);
|
||||
}
|
||||
|
||||
Module *getProgram() const { return Program; }
|
||||
|
||||
/// swapProgramIn - Set the current module to the specified module, returning
|
||||
/// the old one.
|
||||
Module *swapProgramIn(Module *M) {
|
||||
Module *OldProgram = Program;
|
||||
Program = M;
|
||||
return OldProgram;
|
||||
}
|
||||
|
||||
AbstractInterpreter *switchToSafeInterpreter() {
|
||||
AbstractInterpreter *Old = Interpreter;
|
||||
Interpreter = (AbstractInterpreter*)SafeInterpreter;
|
||||
return Old;
|
||||
}
|
||||
|
||||
void switchToInterpreter(AbstractInterpreter *AI) {
|
||||
Interpreter = AI;
|
||||
}
|
||||
|
||||
/// setNewProgram - If we reduce or update the program somehow, call this
|
||||
/// method to update bugdriver with it. This deletes the old module and sets
|
||||
/// the specified one as the current program.
|
||||
void setNewProgram(Module *M);
|
||||
|
||||
/// compileProgram - Try to compile the specified module, returning false and
|
||||
/// setting Error if an error occurs. This is used for code generation
|
||||
/// crash testing.
|
||||
///
|
||||
void compileProgram(Module *M, std::string *Error) const;
|
||||
|
||||
/// executeProgram - This method runs "Program", capturing the output of the
|
||||
/// program to a file. A recommended filename may be optionally specified.
|
||||
///
|
||||
std::string executeProgram(const Module *Program,
|
||||
std::string OutputFilename,
|
||||
std::string Bitcode,
|
||||
const std::string &SharedObjects,
|
||||
AbstractInterpreter *AI,
|
||||
std::string *Error) const;
|
||||
|
||||
/// executeProgramSafely - Used to create reference output with the "safe"
|
||||
/// backend, if reference output is not provided. If there is a problem with
|
||||
/// the code generator (e.g., llc crashes), this will return false and set
|
||||
/// Error.
|
||||
///
|
||||
std::string executeProgramSafely(const Module *Program,
|
||||
std::string OutputFile,
|
||||
std::string *Error) const;
|
||||
|
||||
/// createReferenceFile - calls compileProgram and then records the output
|
||||
/// into ReferenceOutputFile. Returns true if reference file created, false
|
||||
/// otherwise. Note: initializeExecutionEnvironment should be called BEFORE
|
||||
/// this function.
|
||||
///
|
||||
bool createReferenceFile(Module *M, const std::string &Filename
|
||||
= "bugpoint.reference.out");
|
||||
|
||||
/// diffProgram - This method executes the specified module and diffs the
|
||||
/// output against the file specified by ReferenceOutputFile. If the output
|
||||
/// is different, 1 is returned. If there is a problem with the code
|
||||
/// generator (e.g., llc crashes), this will return -1 and set Error.
|
||||
///
|
||||
bool diffProgram(const Module *Program,
|
||||
const std::string &BitcodeFile = "",
|
||||
const std::string &SharedObj = "",
|
||||
bool RemoveBitcode = false,
|
||||
std::string *Error = 0) const;
|
||||
|
||||
/// EmitProgressBitcode - This function is used to output M to a file named
|
||||
/// "bugpoint-ID.bc".
|
||||
///
|
||||
void EmitProgressBitcode(const Module *M, const std::string &ID,
|
||||
bool NoFlyer = false) const;
|
||||
|
||||
/// deleteInstructionFromProgram - This method clones the current Program and
|
||||
/// deletes the specified instruction from the cloned module. It then runs a
|
||||
/// series of cleanup passes (ADCE and SimplifyCFG) to eliminate any code
|
||||
/// which depends on the value. The modified module is then returned.
|
||||
///
|
||||
Module *deleteInstructionFromProgram(const Instruction *I, unsigned Simp);
|
||||
|
||||
/// performFinalCleanups - This method clones the current Program and performs
|
||||
/// a series of cleanups intended to get rid of extra cruft on the module. If
|
||||
/// the MayModifySemantics argument is true, then the cleanups is allowed to
|
||||
/// modify how the code behaves.
|
||||
///
|
||||
Module *performFinalCleanups(Module *M, bool MayModifySemantics = false);
|
||||
|
||||
/// ExtractLoop - Given a module, extract up to one loop from it into a new
|
||||
/// function. This returns null if there are no extractable loops in the
|
||||
/// program or if the loop extractor crashes.
|
||||
Module *ExtractLoop(Module *M);
|
||||
|
||||
/// ExtractMappedBlocksFromModule - Extract all but the specified basic blocks
|
||||
/// into their own functions. The only detail is that M is actually a module
|
||||
/// cloned from the one the BBs are in, so some mapping needs to be performed.
|
||||
/// If this operation fails for some reason (ie the implementation is buggy),
|
||||
/// this function should return null, otherwise it returns a new Module.
|
||||
Module *ExtractMappedBlocksFromModule(const std::vector<BasicBlock*> &BBs,
|
||||
Module *M);
|
||||
|
||||
/// runPassesOn - Carefully run the specified set of pass on the specified
|
||||
/// module, returning the transformed module on success, or a null pointer on
|
||||
/// failure. If AutoDebugCrashes is set to true, then bugpoint will
|
||||
/// automatically attempt to track down a crashing pass if one exists, and
|
||||
/// this method will never return null.
|
||||
Module *runPassesOn(Module *M, const std::vector<std::string> &Passes,
|
||||
bool AutoDebugCrashes = false, unsigned NumExtraArgs = 0,
|
||||
const char * const *ExtraArgs = NULL);
|
||||
|
||||
/// runPasses - Run the specified passes on Program, outputting a bitcode
|
||||
/// file and writting the filename into OutputFile if successful. If the
|
||||
/// optimizations fail for some reason (optimizer crashes), return true,
|
||||
/// otherwise return false. If DeleteOutput is set to true, the bitcode is
|
||||
/// deleted on success, and the filename string is undefined. This prints to
|
||||
/// outs() a single line message indicating whether compilation was successful
|
||||
/// or failed, unless Quiet is set. ExtraArgs specifies additional arguments
|
||||
/// to pass to the child bugpoint instance.
|
||||
///
|
||||
bool runPasses(Module *Program,
|
||||
const std::vector<std::string> &PassesToRun,
|
||||
std::string &OutputFilename, bool DeleteOutput = false,
|
||||
bool Quiet = false, unsigned NumExtraArgs = 0,
|
||||
const char * const *ExtraArgs = NULL) const;
|
||||
|
||||
/// runManyPasses - Take the specified pass list and create different
|
||||
/// combinations of passes to compile the program with. Compile the program with
|
||||
/// each set and mark test to see if it compiled correctly. If the passes
|
||||
/// compiled correctly output nothing and rearrange the passes into a new order.
|
||||
/// If the passes did not compile correctly, output the command required to
|
||||
/// recreate the failure. This returns true if a compiler error is found.
|
||||
///
|
||||
bool runManyPasses(const std::vector<std::string> &AllPasses,
|
||||
std::string &ErrMsg);
|
||||
|
||||
/// writeProgramToFile - This writes the current "Program" to the named
|
||||
/// bitcode file. If an error occurs, true is returned.
|
||||
///
|
||||
bool writeProgramToFile(const std::string &Filename, const Module *M) const;
|
||||
|
||||
private:
|
||||
/// runPasses - Just like the method above, but this just returns true or
|
||||
/// false indicating whether or not the optimizer crashed on the specified
|
||||
/// input (true = crashed).
|
||||
///
|
||||
bool runPasses(Module *M,
|
||||
const std::vector<std::string> &PassesToRun,
|
||||
bool DeleteOutput = true) const {
|
||||
std::string Filename;
|
||||
return runPasses(M, PassesToRun, Filename, DeleteOutput);
|
||||
}
|
||||
|
||||
/// initializeExecutionEnvironment - This method is used to set up the
|
||||
/// environment for executing LLVM programs.
|
||||
///
|
||||
bool initializeExecutionEnvironment();
|
||||
};
|
||||
|
||||
/// ParseInputFile - Given a bitcode or assembly input filename, parse and
|
||||
/// return it, or return null if not possible.
|
||||
///
|
||||
Module *ParseInputFile(const std::string &InputFilename,
|
||||
LLVMContext& ctxt);
|
||||
|
||||
|
||||
/// getPassesString - Turn a list of passes into a string which indicates the
|
||||
/// command line options that must be passed to add the passes.
|
||||
///
|
||||
std::string getPassesString(const std::vector<std::string> &Passes);
|
||||
|
||||
/// PrintFunctionList - prints out list of problematic functions
|
||||
///
|
||||
void PrintFunctionList(const std::vector<Function*> &Funcs);
|
||||
|
||||
/// PrintGlobalVariableList - prints out list of problematic global variables
|
||||
///
|
||||
void PrintGlobalVariableList(const std::vector<GlobalVariable*> &GVs);
|
||||
|
||||
// DeleteFunctionBody - "Remove" the function by deleting all of it's basic
|
||||
// blocks, making it external.
|
||||
//
|
||||
void DeleteFunctionBody(Function *F);
|
||||
|
||||
/// SplitFunctionsOutOfModule - Given a module and a list of functions in the
|
||||
/// module, split the functions OUT of the specified module, and place them in
|
||||
/// the new module.
|
||||
Module *SplitFunctionsOutOfModule(Module *M, const std::vector<Function*> &F,
|
||||
ValueToValueMapTy &VMap);
|
||||
|
||||
} // End llvm namespace
|
||||
|
||||
#endif
|
14
contrib/llvm/tools/bugpoint/CMakeLists.txt
Normal file
14
contrib/llvm/tools/bugpoint/CMakeLists.txt
Normal file
@ -0,0 +1,14 @@
|
||||
set(LLVM_LINK_COMPONENTS asmparser instrumentation scalaropts ipo
|
||||
linker bitreader bitwriter)
|
||||
|
||||
add_llvm_tool(bugpoint
|
||||
BugDriver.cpp
|
||||
CrashDebugger.cpp
|
||||
ExecutionDriver.cpp
|
||||
ExtractFunction.cpp
|
||||
FindBugs.cpp
|
||||
Miscompilation.cpp
|
||||
OptimizerDriver.cpp
|
||||
ToolRunner.cpp
|
||||
bugpoint.cpp
|
||||
)
|
667
contrib/llvm/tools/bugpoint/CrashDebugger.cpp
Normal file
667
contrib/llvm/tools/bugpoint/CrashDebugger.cpp
Normal file
@ -0,0 +1,667 @@
|
||||
//===- CrashDebugger.cpp - Debug compilation crashes ----------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file defines the bugpoint internals that narrow down compilation crashes
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "BugDriver.h"
|
||||
#include "ToolRunner.h"
|
||||
#include "ListReducer.h"
|
||||
#include "llvm/Constants.h"
|
||||
#include "llvm/DerivedTypes.h"
|
||||
#include "llvm/Instructions.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/Pass.h"
|
||||
#include "llvm/PassManager.h"
|
||||
#include "llvm/ValueSymbolTable.h"
|
||||
#include "llvm/ADT/SmallPtrSet.h"
|
||||
#include "llvm/Analysis/Verifier.h"
|
||||
#include "llvm/Support/CFG.h"
|
||||
#include "llvm/Transforms/Scalar.h"
|
||||
#include "llvm/Transforms/Utils/Cloning.h"
|
||||
#include "llvm/Support/FileUtilities.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include <set>
|
||||
using namespace llvm;
|
||||
|
||||
namespace {
|
||||
cl::opt<bool>
|
||||
KeepMain("keep-main",
|
||||
cl::desc("Force function reduction to keep main"),
|
||||
cl::init(false));
|
||||
cl::opt<bool>
|
||||
NoGlobalRM ("disable-global-remove",
|
||||
cl::desc("Do not remove global variables"),
|
||||
cl::init(false));
|
||||
}
|
||||
|
||||
namespace llvm {
|
||||
class ReducePassList : public ListReducer<std::string> {
|
||||
BugDriver &BD;
|
||||
public:
|
||||
ReducePassList(BugDriver &bd) : BD(bd) {}
|
||||
|
||||
// doTest - Return true iff running the "removed" passes succeeds, and
|
||||
// running the "Kept" passes fail when run on the output of the "removed"
|
||||
// passes. If we return true, we update the current module of bugpoint.
|
||||
//
|
||||
virtual TestResult doTest(std::vector<std::string> &Removed,
|
||||
std::vector<std::string> &Kept,
|
||||
std::string &Error);
|
||||
};
|
||||
}
|
||||
|
||||
ReducePassList::TestResult
|
||||
ReducePassList::doTest(std::vector<std::string> &Prefix,
|
||||
std::vector<std::string> &Suffix,
|
||||
std::string &Error) {
|
||||
sys::Path PrefixOutput;
|
||||
Module *OrigProgram = 0;
|
||||
if (!Prefix.empty()) {
|
||||
outs() << "Checking to see if these passes crash: "
|
||||
<< getPassesString(Prefix) << ": ";
|
||||
std::string PfxOutput;
|
||||
if (BD.runPasses(BD.getProgram(), Prefix, PfxOutput))
|
||||
return KeepPrefix;
|
||||
|
||||
PrefixOutput.set(PfxOutput);
|
||||
OrigProgram = BD.Program;
|
||||
|
||||
BD.Program = ParseInputFile(PrefixOutput.str(), BD.getContext());
|
||||
if (BD.Program == 0) {
|
||||
errs() << BD.getToolName() << ": Error reading bitcode file '"
|
||||
<< PrefixOutput.str() << "'!\n";
|
||||
exit(1);
|
||||
}
|
||||
PrefixOutput.eraseFromDisk();
|
||||
}
|
||||
|
||||
outs() << "Checking to see if these passes crash: "
|
||||
<< getPassesString(Suffix) << ": ";
|
||||
|
||||
if (BD.runPasses(BD.getProgram(), Suffix)) {
|
||||
delete OrigProgram; // The suffix crashes alone...
|
||||
return KeepSuffix;
|
||||
}
|
||||
|
||||
// Nothing failed, restore state...
|
||||
if (OrigProgram) {
|
||||
delete BD.Program;
|
||||
BD.Program = OrigProgram;
|
||||
}
|
||||
return NoFailure;
|
||||
}
|
||||
|
||||
namespace {
|
||||
/// ReduceCrashingGlobalVariables - This works by removing the global
|
||||
/// variable's initializer and seeing if the program still crashes. If it
|
||||
/// does, then we keep that program and try again.
|
||||
///
|
||||
class ReduceCrashingGlobalVariables : public ListReducer<GlobalVariable*> {
|
||||
BugDriver &BD;
|
||||
bool (*TestFn)(const BugDriver &, Module *);
|
||||
public:
|
||||
ReduceCrashingGlobalVariables(BugDriver &bd,
|
||||
bool (*testFn)(const BugDriver &, Module *))
|
||||
: BD(bd), TestFn(testFn) {}
|
||||
|
||||
virtual TestResult doTest(std::vector<GlobalVariable*> &Prefix,
|
||||
std::vector<GlobalVariable*> &Kept,
|
||||
std::string &Error) {
|
||||
if (!Kept.empty() && TestGlobalVariables(Kept))
|
||||
return KeepSuffix;
|
||||
if (!Prefix.empty() && TestGlobalVariables(Prefix))
|
||||
return KeepPrefix;
|
||||
return NoFailure;
|
||||
}
|
||||
|
||||
bool TestGlobalVariables(std::vector<GlobalVariable*> &GVs);
|
||||
};
|
||||
}
|
||||
|
||||
bool
|
||||
ReduceCrashingGlobalVariables::TestGlobalVariables(
|
||||
std::vector<GlobalVariable*> &GVs) {
|
||||
// Clone the program to try hacking it apart...
|
||||
ValueToValueMapTy VMap;
|
||||
Module *M = CloneModule(BD.getProgram(), VMap);
|
||||
|
||||
// Convert list to set for fast lookup...
|
||||
std::set<GlobalVariable*> GVSet;
|
||||
|
||||
for (unsigned i = 0, e = GVs.size(); i != e; ++i) {
|
||||
GlobalVariable* CMGV = cast<GlobalVariable>(VMap[GVs[i]]);
|
||||
assert(CMGV && "Global Variable not in module?!");
|
||||
GVSet.insert(CMGV);
|
||||
}
|
||||
|
||||
outs() << "Checking for crash with only these global variables: ";
|
||||
PrintGlobalVariableList(GVs);
|
||||
outs() << ": ";
|
||||
|
||||
// Loop over and delete any global variables which we aren't supposed to be
|
||||
// playing with...
|
||||
for (Module::global_iterator I = M->global_begin(), E = M->global_end();
|
||||
I != E; ++I)
|
||||
if (I->hasInitializer() && !GVSet.count(I)) {
|
||||
I->setInitializer(0);
|
||||
I->setLinkage(GlobalValue::ExternalLinkage);
|
||||
}
|
||||
|
||||
// Try running the hacked up program...
|
||||
if (TestFn(BD, M)) {
|
||||
BD.setNewProgram(M); // It crashed, keep the trimmed version...
|
||||
|
||||
// Make sure to use global variable pointers that point into the now-current
|
||||
// module.
|
||||
GVs.assign(GVSet.begin(), GVSet.end());
|
||||
return true;
|
||||
}
|
||||
|
||||
delete M;
|
||||
return false;
|
||||
}
|
||||
|
||||
namespace llvm {
|
||||
/// ReduceCrashingFunctions reducer - This works by removing functions and
|
||||
/// seeing if the program still crashes. If it does, then keep the newer,
|
||||
/// smaller program.
|
||||
///
|
||||
class ReduceCrashingFunctions : public ListReducer<Function*> {
|
||||
BugDriver &BD;
|
||||
bool (*TestFn)(const BugDriver &, Module *);
|
||||
public:
|
||||
ReduceCrashingFunctions(BugDriver &bd,
|
||||
bool (*testFn)(const BugDriver &, Module *))
|
||||
: BD(bd), TestFn(testFn) {}
|
||||
|
||||
virtual TestResult doTest(std::vector<Function*> &Prefix,
|
||||
std::vector<Function*> &Kept,
|
||||
std::string &Error) {
|
||||
if (!Kept.empty() && TestFuncs(Kept))
|
||||
return KeepSuffix;
|
||||
if (!Prefix.empty() && TestFuncs(Prefix))
|
||||
return KeepPrefix;
|
||||
return NoFailure;
|
||||
}
|
||||
|
||||
bool TestFuncs(std::vector<Function*> &Prefix);
|
||||
};
|
||||
}
|
||||
|
||||
bool ReduceCrashingFunctions::TestFuncs(std::vector<Function*> &Funcs) {
|
||||
|
||||
//if main isn't present, claim there is no problem
|
||||
if (KeepMain && find(Funcs.begin(), Funcs.end(),
|
||||
BD.getProgram()->getFunction("main")) == Funcs.end())
|
||||
return false;
|
||||
|
||||
// Clone the program to try hacking it apart...
|
||||
ValueToValueMapTy VMap;
|
||||
Module *M = CloneModule(BD.getProgram(), VMap);
|
||||
|
||||
// Convert list to set for fast lookup...
|
||||
std::set<Function*> Functions;
|
||||
for (unsigned i = 0, e = Funcs.size(); i != e; ++i) {
|
||||
Function *CMF = cast<Function>(VMap[Funcs[i]]);
|
||||
assert(CMF && "Function not in module?!");
|
||||
assert(CMF->getFunctionType() == Funcs[i]->getFunctionType() && "wrong ty");
|
||||
assert(CMF->getName() == Funcs[i]->getName() && "wrong name");
|
||||
Functions.insert(CMF);
|
||||
}
|
||||
|
||||
outs() << "Checking for crash with only these functions: ";
|
||||
PrintFunctionList(Funcs);
|
||||
outs() << ": ";
|
||||
|
||||
// Loop over and delete any functions which we aren't supposed to be playing
|
||||
// with...
|
||||
for (Module::iterator I = M->begin(), E = M->end(); I != E; ++I)
|
||||
if (!I->isDeclaration() && !Functions.count(I))
|
||||
DeleteFunctionBody(I);
|
||||
|
||||
// Try running the hacked up program...
|
||||
if (TestFn(BD, M)) {
|
||||
BD.setNewProgram(M); // It crashed, keep the trimmed version...
|
||||
|
||||
// Make sure to use function pointers that point into the now-current
|
||||
// module.
|
||||
Funcs.assign(Functions.begin(), Functions.end());
|
||||
return true;
|
||||
}
|
||||
delete M;
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
namespace {
|
||||
/// ReduceCrashingBlocks reducer - This works by setting the terminators of
|
||||
/// all terminators except the specified basic blocks to a 'ret' instruction,
|
||||
/// then running the simplify-cfg pass. This has the effect of chopping up
|
||||
/// the CFG really fast which can reduce large functions quickly.
|
||||
///
|
||||
class ReduceCrashingBlocks : public ListReducer<const BasicBlock*> {
|
||||
BugDriver &BD;
|
||||
bool (*TestFn)(const BugDriver &, Module *);
|
||||
public:
|
||||
ReduceCrashingBlocks(BugDriver &bd,
|
||||
bool (*testFn)(const BugDriver &, Module *))
|
||||
: BD(bd), TestFn(testFn) {}
|
||||
|
||||
virtual TestResult doTest(std::vector<const BasicBlock*> &Prefix,
|
||||
std::vector<const BasicBlock*> &Kept,
|
||||
std::string &Error) {
|
||||
if (!Kept.empty() && TestBlocks(Kept))
|
||||
return KeepSuffix;
|
||||
if (!Prefix.empty() && TestBlocks(Prefix))
|
||||
return KeepPrefix;
|
||||
return NoFailure;
|
||||
}
|
||||
|
||||
bool TestBlocks(std::vector<const BasicBlock*> &Prefix);
|
||||
};
|
||||
}
|
||||
|
||||
bool ReduceCrashingBlocks::TestBlocks(std::vector<const BasicBlock*> &BBs) {
|
||||
// Clone the program to try hacking it apart...
|
||||
ValueToValueMapTy VMap;
|
||||
Module *M = CloneModule(BD.getProgram(), VMap);
|
||||
|
||||
// Convert list to set for fast lookup...
|
||||
SmallPtrSet<BasicBlock*, 8> Blocks;
|
||||
for (unsigned i = 0, e = BBs.size(); i != e; ++i)
|
||||
Blocks.insert(cast<BasicBlock>(VMap[BBs[i]]));
|
||||
|
||||
outs() << "Checking for crash with only these blocks:";
|
||||
unsigned NumPrint = Blocks.size();
|
||||
if (NumPrint > 10) NumPrint = 10;
|
||||
for (unsigned i = 0, e = NumPrint; i != e; ++i)
|
||||
outs() << " " << BBs[i]->getName();
|
||||
if (NumPrint < Blocks.size())
|
||||
outs() << "... <" << Blocks.size() << " total>";
|
||||
outs() << ": ";
|
||||
|
||||
// Loop over and delete any hack up any blocks that are not listed...
|
||||
for (Module::iterator I = M->begin(), E = M->end(); I != E; ++I)
|
||||
for (Function::iterator BB = I->begin(), E = I->end(); BB != E; ++BB)
|
||||
if (!Blocks.count(BB) && BB->getTerminator()->getNumSuccessors()) {
|
||||
// Loop over all of the successors of this block, deleting any PHI nodes
|
||||
// that might include it.
|
||||
for (succ_iterator SI = succ_begin(BB), E = succ_end(BB); SI != E; ++SI)
|
||||
(*SI)->removePredecessor(BB);
|
||||
|
||||
TerminatorInst *BBTerm = BB->getTerminator();
|
||||
|
||||
if (!BB->getTerminator()->getType()->isVoidTy())
|
||||
BBTerm->replaceAllUsesWith(Constant::getNullValue(BBTerm->getType()));
|
||||
|
||||
// Replace the old terminator instruction.
|
||||
BB->getInstList().pop_back();
|
||||
new UnreachableInst(BB->getContext(), BB);
|
||||
}
|
||||
|
||||
// The CFG Simplifier pass may delete one of the basic blocks we are
|
||||
// interested in. If it does we need to take the block out of the list. Make
|
||||
// a "persistent mapping" by turning basic blocks into <function, name> pairs.
|
||||
// This won't work well if blocks are unnamed, but that is just the risk we
|
||||
// have to take.
|
||||
std::vector<std::pair<std::string, std::string> > BlockInfo;
|
||||
|
||||
for (SmallPtrSet<BasicBlock*, 8>::iterator I = Blocks.begin(),
|
||||
E = Blocks.end(); I != E; ++I)
|
||||
BlockInfo.push_back(std::make_pair((*I)->getParent()->getName(),
|
||||
(*I)->getName()));
|
||||
|
||||
// Now run the CFG simplify pass on the function...
|
||||
std::vector<std::string> Passes;
|
||||
Passes.push_back("simplifycfg");
|
||||
Passes.push_back("verify");
|
||||
Module *New = BD.runPassesOn(M, Passes);
|
||||
delete M;
|
||||
if (!New) {
|
||||
errs() << "simplifycfg failed!\n";
|
||||
exit(1);
|
||||
}
|
||||
M = New;
|
||||
|
||||
// Try running on the hacked up program...
|
||||
if (TestFn(BD, M)) {
|
||||
BD.setNewProgram(M); // It crashed, keep the trimmed version...
|
||||
|
||||
// Make sure to use basic block pointers that point into the now-current
|
||||
// module, and that they don't include any deleted blocks.
|
||||
BBs.clear();
|
||||
const ValueSymbolTable &GST = M->getValueSymbolTable();
|
||||
for (unsigned i = 0, e = BlockInfo.size(); i != e; ++i) {
|
||||
Function *F = cast<Function>(GST.lookup(BlockInfo[i].first));
|
||||
ValueSymbolTable &ST = F->getValueSymbolTable();
|
||||
Value* V = ST.lookup(BlockInfo[i].second);
|
||||
if (V && V->getType() == Type::getLabelTy(V->getContext()))
|
||||
BBs.push_back(cast<BasicBlock>(V));
|
||||
}
|
||||
return true;
|
||||
}
|
||||
delete M; // It didn't crash, try something else.
|
||||
return false;
|
||||
}
|
||||
|
||||
namespace {
|
||||
/// ReduceCrashingInstructions reducer - This works by removing the specified
|
||||
/// non-terminator instructions and replacing them with undef.
|
||||
///
|
||||
class ReduceCrashingInstructions : public ListReducer<const Instruction*> {
|
||||
BugDriver &BD;
|
||||
bool (*TestFn)(const BugDriver &, Module *);
|
||||
public:
|
||||
ReduceCrashingInstructions(BugDriver &bd,
|
||||
bool (*testFn)(const BugDriver &, Module *))
|
||||
: BD(bd), TestFn(testFn) {}
|
||||
|
||||
virtual TestResult doTest(std::vector<const Instruction*> &Prefix,
|
||||
std::vector<const Instruction*> &Kept,
|
||||
std::string &Error) {
|
||||
if (!Kept.empty() && TestInsts(Kept))
|
||||
return KeepSuffix;
|
||||
if (!Prefix.empty() && TestInsts(Prefix))
|
||||
return KeepPrefix;
|
||||
return NoFailure;
|
||||
}
|
||||
|
||||
bool TestInsts(std::vector<const Instruction*> &Prefix);
|
||||
};
|
||||
}
|
||||
|
||||
bool ReduceCrashingInstructions::TestInsts(std::vector<const Instruction*>
|
||||
&Insts) {
|
||||
// Clone the program to try hacking it apart...
|
||||
ValueToValueMapTy VMap;
|
||||
Module *M = CloneModule(BD.getProgram(), VMap);
|
||||
|
||||
// Convert list to set for fast lookup...
|
||||
SmallPtrSet<Instruction*, 64> Instructions;
|
||||
for (unsigned i = 0, e = Insts.size(); i != e; ++i) {
|
||||
assert(!isa<TerminatorInst>(Insts[i]));
|
||||
Instructions.insert(cast<Instruction>(VMap[Insts[i]]));
|
||||
}
|
||||
|
||||
outs() << "Checking for crash with only " << Instructions.size();
|
||||
if (Instructions.size() == 1)
|
||||
outs() << " instruction: ";
|
||||
else
|
||||
outs() << " instructions: ";
|
||||
|
||||
for (Module::iterator MI = M->begin(), ME = M->end(); MI != ME; ++MI)
|
||||
for (Function::iterator FI = MI->begin(), FE = MI->end(); FI != FE; ++FI)
|
||||
for (BasicBlock::iterator I = FI->begin(), E = FI->end(); I != E;) {
|
||||
Instruction *Inst = I++;
|
||||
if (!Instructions.count(Inst) && !isa<TerminatorInst>(Inst)) {
|
||||
if (!Inst->getType()->isVoidTy())
|
||||
Inst->replaceAllUsesWith(UndefValue::get(Inst->getType()));
|
||||
Inst->eraseFromParent();
|
||||
}
|
||||
}
|
||||
|
||||
// Verify that this is still valid.
|
||||
PassManager Passes;
|
||||
Passes.add(createVerifierPass());
|
||||
Passes.run(*M);
|
||||
|
||||
// Try running on the hacked up program...
|
||||
if (TestFn(BD, M)) {
|
||||
BD.setNewProgram(M); // It crashed, keep the trimmed version...
|
||||
|
||||
// Make sure to use instruction pointers that point into the now-current
|
||||
// module, and that they don't include any deleted blocks.
|
||||
Insts.clear();
|
||||
for (SmallPtrSet<Instruction*, 64>::const_iterator I = Instructions.begin(),
|
||||
E = Instructions.end(); I != E; ++I)
|
||||
Insts.push_back(*I);
|
||||
return true;
|
||||
}
|
||||
delete M; // It didn't crash, try something else.
|
||||
return false;
|
||||
}
|
||||
|
||||
/// DebugACrash - Given a predicate that determines whether a component crashes
|
||||
/// on a program, try to destructively reduce the program while still keeping
|
||||
/// the predicate true.
|
||||
static bool DebugACrash(BugDriver &BD,
|
||||
bool (*TestFn)(const BugDriver &, Module *),
|
||||
std::string &Error) {
|
||||
// See if we can get away with nuking some of the global variable initializers
|
||||
// in the program...
|
||||
if (!NoGlobalRM &&
|
||||
BD.getProgram()->global_begin() != BD.getProgram()->global_end()) {
|
||||
// Now try to reduce the number of global variable initializers in the
|
||||
// module to something small.
|
||||
Module *M = CloneModule(BD.getProgram());
|
||||
bool DeletedInit = false;
|
||||
|
||||
for (Module::global_iterator I = M->global_begin(), E = M->global_end();
|
||||
I != E; ++I)
|
||||
if (I->hasInitializer()) {
|
||||
I->setInitializer(0);
|
||||
I->setLinkage(GlobalValue::ExternalLinkage);
|
||||
DeletedInit = true;
|
||||
}
|
||||
|
||||
if (!DeletedInit) {
|
||||
delete M; // No change made...
|
||||
} else {
|
||||
// See if the program still causes a crash...
|
||||
outs() << "\nChecking to see if we can delete global inits: ";
|
||||
|
||||
if (TestFn(BD, M)) { // Still crashes?
|
||||
BD.setNewProgram(M);
|
||||
outs() << "\n*** Able to remove all global initializers!\n";
|
||||
} else { // No longer crashes?
|
||||
outs() << " - Removing all global inits hides problem!\n";
|
||||
delete M;
|
||||
|
||||
std::vector<GlobalVariable*> GVs;
|
||||
|
||||
for (Module::global_iterator I = BD.getProgram()->global_begin(),
|
||||
E = BD.getProgram()->global_end(); I != E; ++I)
|
||||
if (I->hasInitializer())
|
||||
GVs.push_back(I);
|
||||
|
||||
if (GVs.size() > 1 && !BugpointIsInterrupted) {
|
||||
outs() << "\n*** Attempting to reduce the number of global "
|
||||
<< "variables in the testcase\n";
|
||||
|
||||
unsigned OldSize = GVs.size();
|
||||
ReduceCrashingGlobalVariables(BD, TestFn).reduceList(GVs, Error);
|
||||
if (!Error.empty())
|
||||
return true;
|
||||
|
||||
if (GVs.size() < OldSize)
|
||||
BD.EmitProgressBitcode(BD.getProgram(), "reduced-global-variables");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Now try to reduce the number of functions in the module to something small.
|
||||
std::vector<Function*> Functions;
|
||||
for (Module::iterator I = BD.getProgram()->begin(),
|
||||
E = BD.getProgram()->end(); I != E; ++I)
|
||||
if (!I->isDeclaration())
|
||||
Functions.push_back(I);
|
||||
|
||||
if (Functions.size() > 1 && !BugpointIsInterrupted) {
|
||||
outs() << "\n*** Attempting to reduce the number of functions "
|
||||
"in the testcase\n";
|
||||
|
||||
unsigned OldSize = Functions.size();
|
||||
ReduceCrashingFunctions(BD, TestFn).reduceList(Functions, Error);
|
||||
|
||||
if (Functions.size() < OldSize)
|
||||
BD.EmitProgressBitcode(BD.getProgram(), "reduced-function");
|
||||
}
|
||||
|
||||
// Attempt to delete entire basic blocks at a time to speed up
|
||||
// convergence... this actually works by setting the terminator of the blocks
|
||||
// to a return instruction then running simplifycfg, which can potentially
|
||||
// shrinks the code dramatically quickly
|
||||
//
|
||||
if (!DisableSimplifyCFG && !BugpointIsInterrupted) {
|
||||
std::vector<const BasicBlock*> Blocks;
|
||||
for (Module::const_iterator I = BD.getProgram()->begin(),
|
||||
E = BD.getProgram()->end(); I != E; ++I)
|
||||
for (Function::const_iterator FI = I->begin(), E = I->end(); FI !=E; ++FI)
|
||||
Blocks.push_back(FI);
|
||||
unsigned OldSize = Blocks.size();
|
||||
ReduceCrashingBlocks(BD, TestFn).reduceList(Blocks, Error);
|
||||
if (Blocks.size() < OldSize)
|
||||
BD.EmitProgressBitcode(BD.getProgram(), "reduced-blocks");
|
||||
}
|
||||
|
||||
// Attempt to delete instructions using bisection. This should help out nasty
|
||||
// cases with large basic blocks where the problem is at one end.
|
||||
if (!BugpointIsInterrupted) {
|
||||
std::vector<const Instruction*> Insts;
|
||||
for (Module::const_iterator MI = BD.getProgram()->begin(),
|
||||
ME = BD.getProgram()->end(); MI != ME; ++MI)
|
||||
for (Function::const_iterator FI = MI->begin(), FE = MI->end(); FI != FE;
|
||||
++FI)
|
||||
for (BasicBlock::const_iterator I = FI->begin(), E = FI->end();
|
||||
I != E; ++I)
|
||||
if (!isa<TerminatorInst>(I))
|
||||
Insts.push_back(I);
|
||||
|
||||
ReduceCrashingInstructions(BD, TestFn).reduceList(Insts, Error);
|
||||
}
|
||||
|
||||
// FIXME: This should use the list reducer to converge faster by deleting
|
||||
// larger chunks of instructions at a time!
|
||||
unsigned Simplification = 2;
|
||||
do {
|
||||
if (BugpointIsInterrupted) break;
|
||||
--Simplification;
|
||||
outs() << "\n*** Attempting to reduce testcase by deleting instruc"
|
||||
<< "tions: Simplification Level #" << Simplification << '\n';
|
||||
|
||||
// Now that we have deleted the functions that are unnecessary for the
|
||||
// program, try to remove instructions that are not necessary to cause the
|
||||
// crash. To do this, we loop through all of the instructions in the
|
||||
// remaining functions, deleting them (replacing any values produced with
|
||||
// nulls), and then running ADCE and SimplifyCFG. If the transformed input
|
||||
// still triggers failure, keep deleting until we cannot trigger failure
|
||||
// anymore.
|
||||
//
|
||||
unsigned InstructionsToSkipBeforeDeleting = 0;
|
||||
TryAgain:
|
||||
|
||||
// Loop over all of the (non-terminator) instructions remaining in the
|
||||
// function, attempting to delete them.
|
||||
unsigned CurInstructionNum = 0;
|
||||
for (Module::const_iterator FI = BD.getProgram()->begin(),
|
||||
E = BD.getProgram()->end(); FI != E; ++FI)
|
||||
if (!FI->isDeclaration())
|
||||
for (Function::const_iterator BI = FI->begin(), E = FI->end(); BI != E;
|
||||
++BI)
|
||||
for (BasicBlock::const_iterator I = BI->begin(), E = --BI->end();
|
||||
I != E; ++I, ++CurInstructionNum)
|
||||
if (InstructionsToSkipBeforeDeleting) {
|
||||
--InstructionsToSkipBeforeDeleting;
|
||||
} else {
|
||||
if (BugpointIsInterrupted) goto ExitLoops;
|
||||
|
||||
outs() << "Checking instruction: " << *I;
|
||||
Module *M = BD.deleteInstructionFromProgram(I, Simplification);
|
||||
|
||||
// Find out if the pass still crashes on this pass...
|
||||
if (TestFn(BD, M)) {
|
||||
// Yup, it does, we delete the old module, and continue trying
|
||||
// to reduce the testcase...
|
||||
BD.setNewProgram(M);
|
||||
InstructionsToSkipBeforeDeleting = CurInstructionNum;
|
||||
goto TryAgain; // I wish I had a multi-level break here!
|
||||
}
|
||||
|
||||
// This pass didn't crash without this instruction, try the next
|
||||
// one.
|
||||
delete M;
|
||||
}
|
||||
|
||||
if (InstructionsToSkipBeforeDeleting) {
|
||||
InstructionsToSkipBeforeDeleting = 0;
|
||||
goto TryAgain;
|
||||
}
|
||||
|
||||
} while (Simplification);
|
||||
ExitLoops:
|
||||
|
||||
// Try to clean up the testcase by running funcresolve and globaldce...
|
||||
if (!BugpointIsInterrupted) {
|
||||
outs() << "\n*** Attempting to perform final cleanups: ";
|
||||
Module *M = CloneModule(BD.getProgram());
|
||||
M = BD.performFinalCleanups(M, true);
|
||||
|
||||
// Find out if the pass still crashes on the cleaned up program...
|
||||
if (TestFn(BD, M)) {
|
||||
BD.setNewProgram(M); // Yup, it does, keep the reduced version...
|
||||
} else {
|
||||
delete M;
|
||||
}
|
||||
}
|
||||
|
||||
BD.EmitProgressBitcode(BD.getProgram(), "reduced-simplified");
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static bool TestForOptimizerCrash(const BugDriver &BD, Module *M) {
|
||||
return BD.runPasses(M);
|
||||
}
|
||||
|
||||
/// debugOptimizerCrash - This method is called when some pass crashes on input.
|
||||
/// It attempts to prune down the testcase to something reasonable, and figure
|
||||
/// out exactly which pass is crashing.
|
||||
///
|
||||
bool BugDriver::debugOptimizerCrash(const std::string &ID) {
|
||||
outs() << "\n*** Debugging optimizer crash!\n";
|
||||
|
||||
std::string Error;
|
||||
// Reduce the list of passes which causes the optimizer to crash...
|
||||
if (!BugpointIsInterrupted)
|
||||
ReducePassList(*this).reduceList(PassesToRun, Error);
|
||||
assert(Error.empty());
|
||||
|
||||
outs() << "\n*** Found crashing pass"
|
||||
<< (PassesToRun.size() == 1 ? ": " : "es: ")
|
||||
<< getPassesString(PassesToRun) << '\n';
|
||||
|
||||
EmitProgressBitcode(Program, ID);
|
||||
|
||||
bool Success = DebugACrash(*this, TestForOptimizerCrash, Error);
|
||||
assert(Error.empty());
|
||||
return Success;
|
||||
}
|
||||
|
||||
static bool TestForCodeGenCrash(const BugDriver &BD, Module *M) {
|
||||
std::string Error;
|
||||
BD.compileProgram(M, &Error);
|
||||
if (!Error.empty()) {
|
||||
errs() << "<crash>\n";
|
||||
return true; // Tool is still crashing.
|
||||
}
|
||||
errs() << '\n';
|
||||
return false;
|
||||
}
|
||||
|
||||
/// debugCodeGeneratorCrash - This method is called when the code generator
|
||||
/// crashes on an input. It attempts to reduce the input as much as possible
|
||||
/// while still causing the code generator to crash.
|
||||
bool BugDriver::debugCodeGeneratorCrash(std::string &Error) {
|
||||
errs() << "*** Debugging code generator crash!\n";
|
||||
|
||||
return DebugACrash(*this, TestForCodeGenCrash, Error);
|
||||
}
|
516
contrib/llvm/tools/bugpoint/ExecutionDriver.cpp
Normal file
516
contrib/llvm/tools/bugpoint/ExecutionDriver.cpp
Normal file
@ -0,0 +1,516 @@
|
||||
//===- ExecutionDriver.cpp - Allow execution of LLVM program --------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file contains code used to execute the program utilizing one of the
|
||||
// various ways of running LLVM bitcode.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "BugDriver.h"
|
||||
#include "ToolRunner.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/Debug.h"
|
||||
#include "llvm/Support/FileUtilities.h"
|
||||
#include "llvm/Support/SystemUtils.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include <fstream>
|
||||
|
||||
using namespace llvm;
|
||||
|
||||
namespace {
|
||||
// OutputType - Allow the user to specify the way code should be run, to test
|
||||
// for miscompilation.
|
||||
//
|
||||
enum OutputType {
|
||||
AutoPick, RunLLI, RunJIT, RunLLC, RunLLCIA, RunCBE, CBE_bug, LLC_Safe,
|
||||
CompileCustom, Custom
|
||||
};
|
||||
|
||||
cl::opt<double>
|
||||
AbsTolerance("abs-tolerance", cl::desc("Absolute error tolerated"),
|
||||
cl::init(0.0));
|
||||
cl::opt<double>
|
||||
RelTolerance("rel-tolerance", cl::desc("Relative error tolerated"),
|
||||
cl::init(0.0));
|
||||
|
||||
cl::opt<OutputType>
|
||||
InterpreterSel(cl::desc("Specify the \"test\" i.e. suspect back-end:"),
|
||||
cl::values(clEnumValN(AutoPick, "auto", "Use best guess"),
|
||||
clEnumValN(RunLLI, "run-int",
|
||||
"Execute with the interpreter"),
|
||||
clEnumValN(RunJIT, "run-jit", "Execute with JIT"),
|
||||
clEnumValN(RunLLC, "run-llc", "Compile with LLC"),
|
||||
clEnumValN(RunLLCIA, "run-llc-ia",
|
||||
"Compile with LLC with integrated assembler"),
|
||||
clEnumValN(RunCBE, "run-cbe", "Compile with CBE"),
|
||||
clEnumValN(CBE_bug,"cbe-bug", "Find CBE bugs"),
|
||||
clEnumValN(LLC_Safe, "llc-safe", "Use LLC for all"),
|
||||
clEnumValN(CompileCustom, "compile-custom",
|
||||
"Use -compile-command to define a command to "
|
||||
"compile the bitcode. Useful to avoid linking."),
|
||||
clEnumValN(Custom, "run-custom",
|
||||
"Use -exec-command to define a command to execute "
|
||||
"the bitcode. Useful for cross-compilation."),
|
||||
clEnumValEnd),
|
||||
cl::init(AutoPick));
|
||||
|
||||
cl::opt<OutputType>
|
||||
SafeInterpreterSel(cl::desc("Specify \"safe\" i.e. known-good backend:"),
|
||||
cl::values(clEnumValN(AutoPick, "safe-auto", "Use best guess"),
|
||||
clEnumValN(RunLLC, "safe-run-llc", "Compile with LLC"),
|
||||
clEnumValN(RunCBE, "safe-run-cbe", "Compile with CBE"),
|
||||
clEnumValN(Custom, "safe-run-custom",
|
||||
"Use -exec-command to define a command to execute "
|
||||
"the bitcode. Useful for cross-compilation."),
|
||||
clEnumValEnd),
|
||||
cl::init(AutoPick));
|
||||
|
||||
cl::opt<std::string>
|
||||
SafeInterpreterPath("safe-path",
|
||||
cl::desc("Specify the path to the \"safe\" backend program"),
|
||||
cl::init(""));
|
||||
|
||||
cl::opt<bool>
|
||||
AppendProgramExitCode("append-exit-code",
|
||||
cl::desc("Append the exit code to the output so it gets diff'd too"),
|
||||
cl::init(false));
|
||||
|
||||
cl::opt<std::string>
|
||||
InputFile("input", cl::init("/dev/null"),
|
||||
cl::desc("Filename to pipe in as stdin (default: /dev/null)"));
|
||||
|
||||
cl::list<std::string>
|
||||
AdditionalSOs("additional-so",
|
||||
cl::desc("Additional shared objects to load "
|
||||
"into executing programs"));
|
||||
|
||||
cl::list<std::string>
|
||||
AdditionalLinkerArgs("Xlinker",
|
||||
cl::desc("Additional arguments to pass to the linker"));
|
||||
|
||||
cl::opt<std::string>
|
||||
CustomCompileCommand("compile-command", cl::init("llc"),
|
||||
cl::desc("Command to compile the bitcode (use with -compile-custom) "
|
||||
"(default: llc)"));
|
||||
|
||||
cl::opt<std::string>
|
||||
CustomExecCommand("exec-command", cl::init("simulate"),
|
||||
cl::desc("Command to execute the bitcode (use with -run-custom) "
|
||||
"(default: simulate)"));
|
||||
}
|
||||
|
||||
namespace llvm {
|
||||
// Anything specified after the --args option are taken as arguments to the
|
||||
// program being debugged.
|
||||
cl::list<std::string>
|
||||
InputArgv("args", cl::Positional, cl::desc("<program arguments>..."),
|
||||
cl::ZeroOrMore, cl::PositionalEatsArgs);
|
||||
|
||||
cl::opt<std::string>
|
||||
OutputPrefix("output-prefix", cl::init("bugpoint"),
|
||||
cl::desc("Prefix to use for outputs (default: 'bugpoint')"));
|
||||
}
|
||||
|
||||
namespace {
|
||||
cl::list<std::string>
|
||||
ToolArgv("tool-args", cl::Positional, cl::desc("<tool arguments>..."),
|
||||
cl::ZeroOrMore, cl::PositionalEatsArgs);
|
||||
|
||||
cl::list<std::string>
|
||||
SafeToolArgv("safe-tool-args", cl::Positional,
|
||||
cl::desc("<safe-tool arguments>..."),
|
||||
cl::ZeroOrMore, cl::PositionalEatsArgs);
|
||||
|
||||
cl::opt<std::string>
|
||||
GCCBinary("gcc", cl::init("gcc"),
|
||||
cl::desc("The gcc binary to use. (default 'gcc')"));
|
||||
|
||||
cl::list<std::string>
|
||||
GCCToolArgv("gcc-tool-args", cl::Positional,
|
||||
cl::desc("<gcc-tool arguments>..."),
|
||||
cl::ZeroOrMore, cl::PositionalEatsArgs);
|
||||
}
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
// BugDriver method implementation
|
||||
//
|
||||
|
||||
/// initializeExecutionEnvironment - This method is used to set up the
|
||||
/// environment for executing LLVM programs.
|
||||
///
|
||||
bool BugDriver::initializeExecutionEnvironment() {
|
||||
outs() << "Initializing execution environment: ";
|
||||
|
||||
// Create an instance of the AbstractInterpreter interface as specified on
|
||||
// the command line
|
||||
SafeInterpreter = 0;
|
||||
std::string Message;
|
||||
|
||||
switch (InterpreterSel) {
|
||||
case AutoPick:
|
||||
InterpreterSel = RunCBE;
|
||||
Interpreter =
|
||||
AbstractInterpreter::createCBE(getToolName(), Message, GCCBinary,
|
||||
&ToolArgv, &GCCToolArgv);
|
||||
if (!Interpreter) {
|
||||
InterpreterSel = RunJIT;
|
||||
Interpreter = AbstractInterpreter::createJIT(getToolName(), Message,
|
||||
&ToolArgv);
|
||||
}
|
||||
if (!Interpreter) {
|
||||
InterpreterSel = RunLLC;
|
||||
Interpreter = AbstractInterpreter::createLLC(getToolName(), Message,
|
||||
GCCBinary, &ToolArgv,
|
||||
&GCCToolArgv);
|
||||
}
|
||||
if (!Interpreter) {
|
||||
InterpreterSel = RunLLI;
|
||||
Interpreter = AbstractInterpreter::createLLI(getToolName(), Message,
|
||||
&ToolArgv);
|
||||
}
|
||||
if (!Interpreter) {
|
||||
InterpreterSel = AutoPick;
|
||||
Message = "Sorry, I can't automatically select an interpreter!\n";
|
||||
}
|
||||
break;
|
||||
case RunLLI:
|
||||
Interpreter = AbstractInterpreter::createLLI(getToolName(), Message,
|
||||
&ToolArgv);
|
||||
break;
|
||||
case RunLLC:
|
||||
case RunLLCIA:
|
||||
case LLC_Safe:
|
||||
Interpreter = AbstractInterpreter::createLLC(getToolName(), Message,
|
||||
GCCBinary, &ToolArgv,
|
||||
&GCCToolArgv,
|
||||
InterpreterSel == RunLLCIA);
|
||||
break;
|
||||
case RunJIT:
|
||||
Interpreter = AbstractInterpreter::createJIT(getToolName(), Message,
|
||||
&ToolArgv);
|
||||
break;
|
||||
case RunCBE:
|
||||
case CBE_bug:
|
||||
Interpreter = AbstractInterpreter::createCBE(getToolName(), Message,
|
||||
GCCBinary, &ToolArgv,
|
||||
&GCCToolArgv);
|
||||
break;
|
||||
case CompileCustom:
|
||||
Interpreter =
|
||||
AbstractInterpreter::createCustomCompiler(Message, CustomCompileCommand);
|
||||
break;
|
||||
case Custom:
|
||||
Interpreter =
|
||||
AbstractInterpreter::createCustomExecutor(Message, CustomExecCommand);
|
||||
break;
|
||||
default:
|
||||
Message = "Sorry, this back-end is not supported by bugpoint right now!\n";
|
||||
break;
|
||||
}
|
||||
if (!Interpreter)
|
||||
errs() << Message;
|
||||
else // Display informational messages on stdout instead of stderr
|
||||
outs() << Message;
|
||||
|
||||
std::string Path = SafeInterpreterPath;
|
||||
if (Path.empty())
|
||||
Path = getToolName();
|
||||
std::vector<std::string> SafeToolArgs = SafeToolArgv;
|
||||
switch (SafeInterpreterSel) {
|
||||
case AutoPick:
|
||||
// In "cbe-bug" mode, default to using LLC as the "safe" backend.
|
||||
if (!SafeInterpreter &&
|
||||
InterpreterSel == CBE_bug) {
|
||||
SafeInterpreterSel = RunLLC;
|
||||
SafeToolArgs.push_back("--relocation-model=pic");
|
||||
SafeInterpreter = AbstractInterpreter::createLLC(Path.c_str(), Message,
|
||||
GCCBinary,
|
||||
&SafeToolArgs,
|
||||
&GCCToolArgv);
|
||||
}
|
||||
|
||||
// In "llc-safe" mode, default to using LLC as the "safe" backend.
|
||||
if (!SafeInterpreter &&
|
||||
InterpreterSel == LLC_Safe) {
|
||||
SafeInterpreterSel = RunLLC;
|
||||
SafeToolArgs.push_back("--relocation-model=pic");
|
||||
SafeInterpreter = AbstractInterpreter::createLLC(Path.c_str(), Message,
|
||||
GCCBinary,
|
||||
&SafeToolArgs,
|
||||
&GCCToolArgv);
|
||||
}
|
||||
|
||||
// Pick a backend that's different from the test backend. The JIT and
|
||||
// LLC backends share a lot of code, so prefer to use the CBE as the
|
||||
// safe back-end when testing them.
|
||||
if (!SafeInterpreter &&
|
||||
InterpreterSel != RunCBE) {
|
||||
SafeInterpreterSel = RunCBE;
|
||||
SafeInterpreter = AbstractInterpreter::createCBE(Path.c_str(), Message,
|
||||
GCCBinary,
|
||||
&SafeToolArgs,
|
||||
&GCCToolArgv);
|
||||
}
|
||||
if (!SafeInterpreter &&
|
||||
InterpreterSel != RunLLC &&
|
||||
InterpreterSel != RunJIT) {
|
||||
SafeInterpreterSel = RunLLC;
|
||||
SafeToolArgs.push_back("--relocation-model=pic");
|
||||
SafeInterpreter = AbstractInterpreter::createLLC(Path.c_str(), Message,
|
||||
GCCBinary,
|
||||
&SafeToolArgs,
|
||||
&GCCToolArgv);
|
||||
}
|
||||
if (!SafeInterpreter) {
|
||||
SafeInterpreterSel = AutoPick;
|
||||
Message = "Sorry, I can't automatically select an interpreter!\n";
|
||||
}
|
||||
break;
|
||||
case RunLLC:
|
||||
case RunLLCIA:
|
||||
SafeToolArgs.push_back("--relocation-model=pic");
|
||||
SafeInterpreter = AbstractInterpreter::createLLC(Path.c_str(), Message,
|
||||
GCCBinary, &SafeToolArgs,
|
||||
&GCCToolArgv,
|
||||
SafeInterpreterSel == RunLLCIA);
|
||||
break;
|
||||
case RunCBE:
|
||||
SafeInterpreter = AbstractInterpreter::createCBE(Path.c_str(), Message,
|
||||
GCCBinary, &SafeToolArgs,
|
||||
&GCCToolArgv);
|
||||
break;
|
||||
case Custom:
|
||||
SafeInterpreter =
|
||||
AbstractInterpreter::createCustomExecutor(Message, CustomExecCommand);
|
||||
break;
|
||||
default:
|
||||
Message = "Sorry, this back-end is not supported by bugpoint as the "
|
||||
"\"safe\" backend right now!\n";
|
||||
break;
|
||||
}
|
||||
if (!SafeInterpreter) { outs() << Message << "\nExiting.\n"; exit(1); }
|
||||
|
||||
gcc = GCC::create(Message, GCCBinary, &GCCToolArgv);
|
||||
if (!gcc) { outs() << Message << "\nExiting.\n"; exit(1); }
|
||||
|
||||
// If there was an error creating the selected interpreter, quit with error.
|
||||
return Interpreter == 0;
|
||||
}
|
||||
|
||||
/// compileProgram - Try to compile the specified module, returning false and
|
||||
/// setting Error if an error occurs. This is used for code generation
|
||||
/// crash testing.
|
||||
///
|
||||
void BugDriver::compileProgram(Module *M, std::string *Error) const {
|
||||
// Emit the program to a bitcode file...
|
||||
sys::Path BitcodeFile (OutputPrefix + "-test-program.bc");
|
||||
std::string ErrMsg;
|
||||
if (BitcodeFile.makeUnique(true, &ErrMsg)) {
|
||||
errs() << ToolName << ": Error making unique filename: " << ErrMsg
|
||||
<< "\n";
|
||||
exit(1);
|
||||
}
|
||||
if (writeProgramToFile(BitcodeFile.str(), M)) {
|
||||
errs() << ToolName << ": Error emitting bitcode to file '"
|
||||
<< BitcodeFile.str() << "'!\n";
|
||||
exit(1);
|
||||
}
|
||||
|
||||
// Remove the temporary bitcode file when we are done.
|
||||
FileRemover BitcodeFileRemover(BitcodeFile.str(), !SaveTemps);
|
||||
|
||||
// Actually compile the program!
|
||||
Interpreter->compileProgram(BitcodeFile.str(), Error, Timeout, MemoryLimit);
|
||||
}
|
||||
|
||||
|
||||
/// executeProgram - This method runs "Program", capturing the output of the
|
||||
/// program to a file, returning the filename of the file. A recommended
|
||||
/// filename may be optionally specified.
|
||||
///
|
||||
std::string BugDriver::executeProgram(const Module *Program,
|
||||
std::string OutputFile,
|
||||
std::string BitcodeFile,
|
||||
const std::string &SharedObj,
|
||||
AbstractInterpreter *AI,
|
||||
std::string *Error) const {
|
||||
if (AI == 0) AI = Interpreter;
|
||||
assert(AI && "Interpreter should have been created already!");
|
||||
bool CreatedBitcode = false;
|
||||
std::string ErrMsg;
|
||||
if (BitcodeFile.empty()) {
|
||||
// Emit the program to a bitcode file...
|
||||
sys::Path uniqueFilename(OutputPrefix + "-test-program.bc");
|
||||
if (uniqueFilename.makeUnique(true, &ErrMsg)) {
|
||||
errs() << ToolName << ": Error making unique filename: "
|
||||
<< ErrMsg << "!\n";
|
||||
exit(1);
|
||||
}
|
||||
BitcodeFile = uniqueFilename.str();
|
||||
|
||||
if (writeProgramToFile(BitcodeFile, Program)) {
|
||||
errs() << ToolName << ": Error emitting bitcode to file '"
|
||||
<< BitcodeFile << "'!\n";
|
||||
exit(1);
|
||||
}
|
||||
CreatedBitcode = true;
|
||||
}
|
||||
|
||||
// Remove the temporary bitcode file when we are done.
|
||||
sys::Path BitcodePath(BitcodeFile);
|
||||
FileRemover BitcodeFileRemover(BitcodePath.str(),
|
||||
CreatedBitcode && !SaveTemps);
|
||||
|
||||
if (OutputFile.empty()) OutputFile = OutputPrefix + "-execution-output";
|
||||
|
||||
// Check to see if this is a valid output filename...
|
||||
sys::Path uniqueFile(OutputFile);
|
||||
if (uniqueFile.makeUnique(true, &ErrMsg)) {
|
||||
errs() << ToolName << ": Error making unique filename: "
|
||||
<< ErrMsg << "\n";
|
||||
exit(1);
|
||||
}
|
||||
OutputFile = uniqueFile.str();
|
||||
|
||||
// Figure out which shared objects to run, if any.
|
||||
std::vector<std::string> SharedObjs(AdditionalSOs);
|
||||
if (!SharedObj.empty())
|
||||
SharedObjs.push_back(SharedObj);
|
||||
|
||||
int RetVal = AI->ExecuteProgram(BitcodeFile, InputArgv, InputFile, OutputFile,
|
||||
Error, AdditionalLinkerArgs, SharedObjs,
|
||||
Timeout, MemoryLimit);
|
||||
if (!Error->empty())
|
||||
return OutputFile;
|
||||
|
||||
if (RetVal == -1) {
|
||||
errs() << "<timeout>";
|
||||
static bool FirstTimeout = true;
|
||||
if (FirstTimeout) {
|
||||
outs() << "\n"
|
||||
"*** Program execution timed out! This mechanism is designed to handle\n"
|
||||
" programs stuck in infinite loops gracefully. The -timeout option\n"
|
||||
" can be used to change the timeout threshold or disable it completely\n"
|
||||
" (with -timeout=0). This message is only displayed once.\n";
|
||||
FirstTimeout = false;
|
||||
}
|
||||
}
|
||||
|
||||
if (AppendProgramExitCode) {
|
||||
std::ofstream outFile(OutputFile.c_str(), std::ios_base::app);
|
||||
outFile << "exit " << RetVal << '\n';
|
||||
outFile.close();
|
||||
}
|
||||
|
||||
// Return the filename we captured the output to.
|
||||
return OutputFile;
|
||||
}
|
||||
|
||||
/// executeProgramSafely - Used to create reference output with the "safe"
|
||||
/// backend, if reference output is not provided.
|
||||
///
|
||||
std::string BugDriver::executeProgramSafely(const Module *Program,
|
||||
std::string OutputFile,
|
||||
std::string *Error) const {
|
||||
return executeProgram(Program, OutputFile, "", "", SafeInterpreter, Error);
|
||||
}
|
||||
|
||||
std::string BugDriver::compileSharedObject(const std::string &BitcodeFile,
|
||||
std::string &Error) {
|
||||
assert(Interpreter && "Interpreter should have been created already!");
|
||||
sys::Path OutputFile;
|
||||
|
||||
// Using the known-good backend.
|
||||
GCC::FileType FT = SafeInterpreter->OutputCode(BitcodeFile, OutputFile,
|
||||
Error);
|
||||
if (!Error.empty())
|
||||
return "";
|
||||
|
||||
std::string SharedObjectFile;
|
||||
bool Failure = gcc->MakeSharedObject(OutputFile.str(), FT, SharedObjectFile,
|
||||
AdditionalLinkerArgs, Error);
|
||||
if (!Error.empty())
|
||||
return "";
|
||||
if (Failure)
|
||||
exit(1);
|
||||
|
||||
// Remove the intermediate C file
|
||||
OutputFile.eraseFromDisk();
|
||||
|
||||
return "./" + SharedObjectFile;
|
||||
}
|
||||
|
||||
/// createReferenceFile - calls compileProgram and then records the output
|
||||
/// into ReferenceOutputFile. Returns true if reference file created, false
|
||||
/// otherwise. Note: initializeExecutionEnvironment should be called BEFORE
|
||||
/// this function.
|
||||
///
|
||||
bool BugDriver::createReferenceFile(Module *M, const std::string &Filename) {
|
||||
std::string Error;
|
||||
compileProgram(Program, &Error);
|
||||
if (!Error.empty())
|
||||
return false;
|
||||
|
||||
ReferenceOutputFile = executeProgramSafely(Program, Filename, &Error);
|
||||
if (!Error.empty()) {
|
||||
errs() << Error;
|
||||
if (Interpreter != SafeInterpreter) {
|
||||
errs() << "*** There is a bug running the \"safe\" backend. Either"
|
||||
<< " debug it (for example with the -run-cbe bugpoint option,"
|
||||
<< " if CBE is being used as the \"safe\" backend), or fix the"
|
||||
<< " error some other way.\n";
|
||||
}
|
||||
return false;
|
||||
}
|
||||
outs() << "\nReference output is: " << ReferenceOutputFile << "\n\n";
|
||||
return true;
|
||||
}
|
||||
|
||||
/// diffProgram - This method executes the specified module and diffs the
|
||||
/// output against the file specified by ReferenceOutputFile. If the output
|
||||
/// is different, 1 is returned. If there is a problem with the code
|
||||
/// generator (e.g., llc crashes), this will set ErrMsg.
|
||||
///
|
||||
bool BugDriver::diffProgram(const Module *Program,
|
||||
const std::string &BitcodeFile,
|
||||
const std::string &SharedObject,
|
||||
bool RemoveBitcode,
|
||||
std::string *ErrMsg) const {
|
||||
// Execute the program, generating an output file...
|
||||
sys::Path Output(executeProgram(Program, "", BitcodeFile, SharedObject, 0,
|
||||
ErrMsg));
|
||||
if (!ErrMsg->empty())
|
||||
return false;
|
||||
|
||||
std::string Error;
|
||||
bool FilesDifferent = false;
|
||||
if (int Diff = DiffFilesWithTolerance(sys::Path(ReferenceOutputFile),
|
||||
sys::Path(Output.str()),
|
||||
AbsTolerance, RelTolerance, &Error)) {
|
||||
if (Diff == 2) {
|
||||
errs() << "While diffing output: " << Error << '\n';
|
||||
exit(1);
|
||||
}
|
||||
FilesDifferent = true;
|
||||
}
|
||||
else {
|
||||
// Remove the generated output if there are no differences.
|
||||
Output.eraseFromDisk();
|
||||
}
|
||||
|
||||
// Remove the bitcode file if we are supposed to.
|
||||
if (RemoveBitcode)
|
||||
sys::Path(BitcodeFile).eraseFromDisk();
|
||||
return FilesDifferent;
|
||||
}
|
||||
|
||||
bool BugDriver::isExecutingJIT() {
|
||||
return InterpreterSel == RunJIT;
|
||||
}
|
||||
|
370
contrib/llvm/tools/bugpoint/ExtractFunction.cpp
Normal file
370
contrib/llvm/tools/bugpoint/ExtractFunction.cpp
Normal file
@ -0,0 +1,370 @@
|
||||
//===- ExtractFunction.cpp - Extract a function from Program --------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file implements several methods that are used to extract functions,
|
||||
// loops, or portions of a module from the rest of the module.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "BugDriver.h"
|
||||
#include "llvm/Constants.h"
|
||||
#include "llvm/DerivedTypes.h"
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/PassManager.h"
|
||||
#include "llvm/Pass.h"
|
||||
#include "llvm/Analysis/Verifier.h"
|
||||
#include "llvm/Assembly/Writer.h"
|
||||
#include "llvm/Transforms/IPO.h"
|
||||
#include "llvm/Transforms/Scalar.h"
|
||||
#include "llvm/Transforms/Utils/Cloning.h"
|
||||
#include "llvm/Transforms/Utils/FunctionUtils.h"
|
||||
#include "llvm/Target/TargetData.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/Debug.h"
|
||||
#include "llvm/Support/FileUtilities.h"
|
||||
#include "llvm/Support/ToolOutputFile.h"
|
||||
#include "llvm/Support/Path.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include <set>
|
||||
using namespace llvm;
|
||||
|
||||
namespace llvm {
|
||||
bool DisableSimplifyCFG = false;
|
||||
extern cl::opt<std::string> OutputPrefix;
|
||||
} // End llvm namespace
|
||||
|
||||
namespace {
|
||||
cl::opt<bool>
|
||||
NoDCE ("disable-dce",
|
||||
cl::desc("Do not use the -dce pass to reduce testcases"));
|
||||
cl::opt<bool, true>
|
||||
NoSCFG("disable-simplifycfg", cl::location(DisableSimplifyCFG),
|
||||
cl::desc("Do not use the -simplifycfg pass to reduce testcases"));
|
||||
}
|
||||
|
||||
/// deleteInstructionFromProgram - This method clones the current Program and
|
||||
/// deletes the specified instruction from the cloned module. It then runs a
|
||||
/// series of cleanup passes (ADCE and SimplifyCFG) to eliminate any code which
|
||||
/// depends on the value. The modified module is then returned.
|
||||
///
|
||||
Module *BugDriver::deleteInstructionFromProgram(const Instruction *I,
|
||||
unsigned Simplification) {
|
||||
// FIXME, use vmap?
|
||||
Module *Clone = CloneModule(Program);
|
||||
|
||||
const BasicBlock *PBB = I->getParent();
|
||||
const Function *PF = PBB->getParent();
|
||||
|
||||
Module::iterator RFI = Clone->begin(); // Get iterator to corresponding fn
|
||||
std::advance(RFI, std::distance(PF->getParent()->begin(),
|
||||
Module::const_iterator(PF)));
|
||||
|
||||
Function::iterator RBI = RFI->begin(); // Get iterator to corresponding BB
|
||||
std::advance(RBI, std::distance(PF->begin(), Function::const_iterator(PBB)));
|
||||
|
||||
BasicBlock::iterator RI = RBI->begin(); // Get iterator to corresponding inst
|
||||
std::advance(RI, std::distance(PBB->begin(), BasicBlock::const_iterator(I)));
|
||||
Instruction *TheInst = RI; // Got the corresponding instruction!
|
||||
|
||||
// If this instruction produces a value, replace any users with null values
|
||||
if (!TheInst->getType()->isVoidTy())
|
||||
TheInst->replaceAllUsesWith(Constant::getNullValue(TheInst->getType()));
|
||||
|
||||
// Remove the instruction from the program.
|
||||
TheInst->getParent()->getInstList().erase(TheInst);
|
||||
|
||||
// Spiff up the output a little bit.
|
||||
std::vector<std::string> Passes;
|
||||
|
||||
/// Can we get rid of the -disable-* options?
|
||||
if (Simplification > 1 && !NoDCE)
|
||||
Passes.push_back("dce");
|
||||
if (Simplification && !DisableSimplifyCFG)
|
||||
Passes.push_back("simplifycfg"); // Delete dead control flow
|
||||
|
||||
Passes.push_back("verify");
|
||||
Module *New = runPassesOn(Clone, Passes);
|
||||
delete Clone;
|
||||
if (!New) {
|
||||
errs() << "Instruction removal failed. Sorry. :( Please report a bug!\n";
|
||||
exit(1);
|
||||
}
|
||||
return New;
|
||||
}
|
||||
|
||||
/// performFinalCleanups - This method clones the current Program and performs
|
||||
/// a series of cleanups intended to get rid of extra cruft on the module
|
||||
/// before handing it to the user.
|
||||
///
|
||||
Module *BugDriver::performFinalCleanups(Module *M, bool MayModifySemantics) {
|
||||
// Make all functions external, so GlobalDCE doesn't delete them...
|
||||
for (Module::iterator I = M->begin(), E = M->end(); I != E; ++I)
|
||||
I->setLinkage(GlobalValue::ExternalLinkage);
|
||||
|
||||
std::vector<std::string> CleanupPasses;
|
||||
CleanupPasses.push_back("globaldce");
|
||||
|
||||
if (MayModifySemantics)
|
||||
CleanupPasses.push_back("deadarghaX0r");
|
||||
else
|
||||
CleanupPasses.push_back("deadargelim");
|
||||
|
||||
Module *New = runPassesOn(M, CleanupPasses);
|
||||
if (New == 0) {
|
||||
errs() << "Final cleanups failed. Sorry. :( Please report a bug!\n";
|
||||
return M;
|
||||
}
|
||||
delete M;
|
||||
return New;
|
||||
}
|
||||
|
||||
|
||||
/// ExtractLoop - Given a module, extract up to one loop from it into a new
|
||||
/// function. This returns null if there are no extractable loops in the
|
||||
/// program or if the loop extractor crashes.
|
||||
Module *BugDriver::ExtractLoop(Module *M) {
|
||||
std::vector<std::string> LoopExtractPasses;
|
||||
LoopExtractPasses.push_back("loop-extract-single");
|
||||
|
||||
Module *NewM = runPassesOn(M, LoopExtractPasses);
|
||||
if (NewM == 0) {
|
||||
outs() << "*** Loop extraction failed: ";
|
||||
EmitProgressBitcode(M, "loopextraction", true);
|
||||
outs() << "*** Sorry. :( Please report a bug!\n";
|
||||
return 0;
|
||||
}
|
||||
|
||||
// Check to see if we created any new functions. If not, no loops were
|
||||
// extracted and we should return null. Limit the number of loops we extract
|
||||
// to avoid taking forever.
|
||||
static unsigned NumExtracted = 32;
|
||||
if (M->size() == NewM->size() || --NumExtracted == 0) {
|
||||
delete NewM;
|
||||
return 0;
|
||||
} else {
|
||||
assert(M->size() < NewM->size() && "Loop extract removed functions?");
|
||||
Module::iterator MI = NewM->begin();
|
||||
for (unsigned i = 0, e = M->size(); i != e; ++i)
|
||||
++MI;
|
||||
}
|
||||
|
||||
return NewM;
|
||||
}
|
||||
|
||||
|
||||
// DeleteFunctionBody - "Remove" the function by deleting all of its basic
|
||||
// blocks, making it external.
|
||||
//
|
||||
void llvm::DeleteFunctionBody(Function *F) {
|
||||
// delete the body of the function...
|
||||
F->deleteBody();
|
||||
assert(F->isDeclaration() && "This didn't make the function external!");
|
||||
}
|
||||
|
||||
/// GetTorInit - Given a list of entries for static ctors/dtors, return them
|
||||
/// as a constant array.
|
||||
static Constant *GetTorInit(std::vector<std::pair<Function*, int> > &TorList) {
|
||||
assert(!TorList.empty() && "Don't create empty tor list!");
|
||||
std::vector<Constant*> ArrayElts;
|
||||
Type *Int32Ty = Type::getInt32Ty(TorList[0].first->getContext());
|
||||
|
||||
StructType *STy =
|
||||
StructType::get(Int32Ty, TorList[0].first->getType(), NULL);
|
||||
for (unsigned i = 0, e = TorList.size(); i != e; ++i) {
|
||||
Constant *Elts[] = {
|
||||
ConstantInt::get(Int32Ty, TorList[i].second),
|
||||
TorList[i].first
|
||||
};
|
||||
ArrayElts.push_back(ConstantStruct::get(STy, Elts));
|
||||
}
|
||||
return ConstantArray::get(ArrayType::get(ArrayElts[0]->getType(),
|
||||
ArrayElts.size()),
|
||||
ArrayElts);
|
||||
}
|
||||
|
||||
/// SplitStaticCtorDtor - A module was recently split into two parts, M1/M2, and
|
||||
/// M1 has all of the global variables. If M2 contains any functions that are
|
||||
/// static ctors/dtors, we need to add an llvm.global_[cd]tors global to M2, and
|
||||
/// prune appropriate entries out of M1s list.
|
||||
static void SplitStaticCtorDtor(const char *GlobalName, Module *M1, Module *M2,
|
||||
ValueToValueMapTy &VMap) {
|
||||
GlobalVariable *GV = M1->getNamedGlobal(GlobalName);
|
||||
if (!GV || GV->isDeclaration() || GV->hasLocalLinkage() ||
|
||||
!GV->use_empty()) return;
|
||||
|
||||
std::vector<std::pair<Function*, int> > M1Tors, M2Tors;
|
||||
ConstantArray *InitList = dyn_cast<ConstantArray>(GV->getInitializer());
|
||||
if (!InitList) return;
|
||||
|
||||
for (unsigned i = 0, e = InitList->getNumOperands(); i != e; ++i) {
|
||||
if (ConstantStruct *CS = dyn_cast<ConstantStruct>(InitList->getOperand(i))){
|
||||
if (CS->getNumOperands() != 2) return; // Not array of 2-element structs.
|
||||
|
||||
if (CS->getOperand(1)->isNullValue())
|
||||
break; // Found a null terminator, stop here.
|
||||
|
||||
ConstantInt *CI = dyn_cast<ConstantInt>(CS->getOperand(0));
|
||||
int Priority = CI ? CI->getSExtValue() : 0;
|
||||
|
||||
Constant *FP = CS->getOperand(1);
|
||||
if (ConstantExpr *CE = dyn_cast<ConstantExpr>(FP))
|
||||
if (CE->isCast())
|
||||
FP = CE->getOperand(0);
|
||||
if (Function *F = dyn_cast<Function>(FP)) {
|
||||
if (!F->isDeclaration())
|
||||
M1Tors.push_back(std::make_pair(F, Priority));
|
||||
else {
|
||||
// Map to M2's version of the function.
|
||||
F = cast<Function>(VMap[F]);
|
||||
M2Tors.push_back(std::make_pair(F, Priority));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
GV->eraseFromParent();
|
||||
if (!M1Tors.empty()) {
|
||||
Constant *M1Init = GetTorInit(M1Tors);
|
||||
new GlobalVariable(*M1, M1Init->getType(), false,
|
||||
GlobalValue::AppendingLinkage,
|
||||
M1Init, GlobalName);
|
||||
}
|
||||
|
||||
GV = M2->getNamedGlobal(GlobalName);
|
||||
assert(GV && "Not a clone of M1?");
|
||||
assert(GV->use_empty() && "llvm.ctors shouldn't have uses!");
|
||||
|
||||
GV->eraseFromParent();
|
||||
if (!M2Tors.empty()) {
|
||||
Constant *M2Init = GetTorInit(M2Tors);
|
||||
new GlobalVariable(*M2, M2Init->getType(), false,
|
||||
GlobalValue::AppendingLinkage,
|
||||
M2Init, GlobalName);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/// SplitFunctionsOutOfModule - Given a module and a list of functions in the
|
||||
/// module, split the functions OUT of the specified module, and place them in
|
||||
/// the new module.
|
||||
Module *
|
||||
llvm::SplitFunctionsOutOfModule(Module *M,
|
||||
const std::vector<Function*> &F,
|
||||
ValueToValueMapTy &VMap) {
|
||||
// Make sure functions & globals are all external so that linkage
|
||||
// between the two modules will work.
|
||||
for (Module::iterator I = M->begin(), E = M->end(); I != E; ++I)
|
||||
I->setLinkage(GlobalValue::ExternalLinkage);
|
||||
for (Module::global_iterator I = M->global_begin(), E = M->global_end();
|
||||
I != E; ++I) {
|
||||
if (I->hasName() && I->getName()[0] == '\01')
|
||||
I->setName(I->getName().substr(1));
|
||||
I->setLinkage(GlobalValue::ExternalLinkage);
|
||||
}
|
||||
|
||||
ValueToValueMapTy NewVMap;
|
||||
Module *New = CloneModule(M, NewVMap);
|
||||
|
||||
// Make sure global initializers exist only in the safe module (CBE->.so)
|
||||
for (Module::global_iterator I = New->global_begin(), E = New->global_end();
|
||||
I != E; ++I)
|
||||
I->setInitializer(0); // Delete the initializer to make it external
|
||||
|
||||
// Remove the Test functions from the Safe module
|
||||
std::set<Function *> TestFunctions;
|
||||
for (unsigned i = 0, e = F.size(); i != e; ++i) {
|
||||
Function *TNOF = cast<Function>(VMap[F[i]]);
|
||||
DEBUG(errs() << "Removing function ");
|
||||
DEBUG(WriteAsOperand(errs(), TNOF, false));
|
||||
DEBUG(errs() << "\n");
|
||||
TestFunctions.insert(cast<Function>(NewVMap[TNOF]));
|
||||
DeleteFunctionBody(TNOF); // Function is now external in this module!
|
||||
}
|
||||
|
||||
|
||||
// Remove the Safe functions from the Test module
|
||||
for (Module::iterator I = New->begin(), E = New->end(); I != E; ++I)
|
||||
if (!TestFunctions.count(I))
|
||||
DeleteFunctionBody(I);
|
||||
|
||||
|
||||
// Make sure that there is a global ctor/dtor array in both halves of the
|
||||
// module if they both have static ctor/dtor functions.
|
||||
SplitStaticCtorDtor("llvm.global_ctors", M, New, NewVMap);
|
||||
SplitStaticCtorDtor("llvm.global_dtors", M, New, NewVMap);
|
||||
|
||||
return New;
|
||||
}
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
// Basic Block Extraction Code
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
/// ExtractMappedBlocksFromModule - Extract all but the specified basic blocks
|
||||
/// into their own functions. The only detail is that M is actually a module
|
||||
/// cloned from the one the BBs are in, so some mapping needs to be performed.
|
||||
/// If this operation fails for some reason (ie the implementation is buggy),
|
||||
/// this function should return null, otherwise it returns a new Module.
|
||||
Module *BugDriver::ExtractMappedBlocksFromModule(const
|
||||
std::vector<BasicBlock*> &BBs,
|
||||
Module *M) {
|
||||
sys::Path uniqueFilename(OutputPrefix + "-extractblocks");
|
||||
std::string ErrMsg;
|
||||
if (uniqueFilename.createTemporaryFileOnDisk(true, &ErrMsg)) {
|
||||
outs() << "*** Basic Block extraction failed!\n";
|
||||
errs() << "Error creating temporary file: " << ErrMsg << "\n";
|
||||
EmitProgressBitcode(M, "basicblockextractfail", true);
|
||||
return 0;
|
||||
}
|
||||
sys::RemoveFileOnSignal(uniqueFilename);
|
||||
|
||||
std::string ErrorInfo;
|
||||
tool_output_file BlocksToNotExtractFile(uniqueFilename.c_str(), ErrorInfo);
|
||||
if (!ErrorInfo.empty()) {
|
||||
outs() << "*** Basic Block extraction failed!\n";
|
||||
errs() << "Error writing list of blocks to not extract: " << ErrorInfo
|
||||
<< "\n";
|
||||
EmitProgressBitcode(M, "basicblockextractfail", true);
|
||||
return 0;
|
||||
}
|
||||
for (std::vector<BasicBlock*>::const_iterator I = BBs.begin(), E = BBs.end();
|
||||
I != E; ++I) {
|
||||
BasicBlock *BB = *I;
|
||||
// If the BB doesn't have a name, give it one so we have something to key
|
||||
// off of.
|
||||
if (!BB->hasName()) BB->setName("tmpbb");
|
||||
BlocksToNotExtractFile.os() << BB->getParent()->getNameStr() << " "
|
||||
<< BB->getName() << "\n";
|
||||
}
|
||||
BlocksToNotExtractFile.os().close();
|
||||
if (BlocksToNotExtractFile.os().has_error()) {
|
||||
errs() << "Error writing list of blocks to not extract: " << ErrorInfo
|
||||
<< "\n";
|
||||
EmitProgressBitcode(M, "basicblockextractfail", true);
|
||||
BlocksToNotExtractFile.os().clear_error();
|
||||
return 0;
|
||||
}
|
||||
BlocksToNotExtractFile.keep();
|
||||
|
||||
std::string uniqueFN = "--extract-blocks-file=" + uniqueFilename.str();
|
||||
const char *ExtraArg = uniqueFN.c_str();
|
||||
|
||||
std::vector<std::string> PI;
|
||||
PI.push_back("extract-blocks");
|
||||
Module *Ret = runPassesOn(M, PI, false, 1, &ExtraArg);
|
||||
|
||||
uniqueFilename.eraseFromDisk(); // Free disk space
|
||||
|
||||
if (Ret == 0) {
|
||||
outs() << "*** Basic Block extraction failed, please report a bug!\n";
|
||||
EmitProgressBitcode(M, "basicblockextractfail", true);
|
||||
}
|
||||
return Ret;
|
||||
}
|
113
contrib/llvm/tools/bugpoint/FindBugs.cpp
Normal file
113
contrib/llvm/tools/bugpoint/FindBugs.cpp
Normal file
@ -0,0 +1,113 @@
|
||||
//===-- FindBugs.cpp - Run Many Different Optimizations -------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file defines an interface that allows bugpoint to choose different
|
||||
// combinations of optimizations to run on the selected input. Bugpoint will
|
||||
// run these optimizations and record the success/failure of each. This way
|
||||
// we can hopefully spot bugs in the optimizations.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "BugDriver.h"
|
||||
#include "ToolRunner.h"
|
||||
#include "llvm/Pass.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include <algorithm>
|
||||
#include <ctime>
|
||||
using namespace llvm;
|
||||
|
||||
/// runManyPasses - Take the specified pass list and create different
|
||||
/// combinations of passes to compile the program with. Compile the program with
|
||||
/// each set and mark test to see if it compiled correctly. If the passes
|
||||
/// compiled correctly output nothing and rearrange the passes into a new order.
|
||||
/// If the passes did not compile correctly, output the command required to
|
||||
/// recreate the failure. This returns true if a compiler error is found.
|
||||
///
|
||||
bool BugDriver::runManyPasses(const std::vector<std::string> &AllPasses,
|
||||
std::string &ErrMsg) {
|
||||
setPassesToRun(AllPasses);
|
||||
outs() << "Starting bug finding procedure...\n\n";
|
||||
|
||||
// Creating a reference output if necessary
|
||||
if (initializeExecutionEnvironment()) return false;
|
||||
|
||||
outs() << "\n";
|
||||
if (ReferenceOutputFile.empty()) {
|
||||
outs() << "Generating reference output from raw program: \n";
|
||||
if (!createReferenceFile(Program))
|
||||
return false;
|
||||
}
|
||||
|
||||
srand(time(NULL));
|
||||
|
||||
unsigned num = 1;
|
||||
while(1) {
|
||||
//
|
||||
// Step 1: Randomize the order of the optimizer passes.
|
||||
//
|
||||
std::random_shuffle(PassesToRun.begin(), PassesToRun.end());
|
||||
|
||||
//
|
||||
// Step 2: Run optimizer passes on the program and check for success.
|
||||
//
|
||||
outs() << "Running selected passes on program to test for crash: ";
|
||||
for(int i = 0, e = PassesToRun.size(); i != e; i++) {
|
||||
outs() << "-" << PassesToRun[i] << " ";
|
||||
}
|
||||
|
||||
std::string Filename;
|
||||
if(runPasses(Program, PassesToRun, Filename, false)) {
|
||||
outs() << "\n";
|
||||
outs() << "Optimizer passes caused failure!\n\n";
|
||||
debugOptimizerCrash();
|
||||
return true;
|
||||
} else {
|
||||
outs() << "Combination " << num << " optimized successfully!\n";
|
||||
}
|
||||
|
||||
//
|
||||
// Step 3: Compile the optimized code.
|
||||
//
|
||||
outs() << "Running the code generator to test for a crash: ";
|
||||
std::string Error;
|
||||
compileProgram(Program, &Error);
|
||||
if (!Error.empty()) {
|
||||
outs() << "\n*** compileProgram threw an exception: ";
|
||||
outs() << Error;
|
||||
return debugCodeGeneratorCrash(ErrMsg);
|
||||
}
|
||||
outs() << '\n';
|
||||
|
||||
//
|
||||
// Step 4: Run the program and compare its output to the reference
|
||||
// output (created above).
|
||||
//
|
||||
outs() << "*** Checking if passes caused miscompliation:\n";
|
||||
bool Diff = diffProgram(Program, Filename, "", false, &Error);
|
||||
if (Error.empty() && Diff) {
|
||||
outs() << "\n*** diffProgram returned true!\n";
|
||||
debugMiscompilation(&Error);
|
||||
if (Error.empty())
|
||||
return true;
|
||||
}
|
||||
if (!Error.empty()) {
|
||||
errs() << Error;
|
||||
debugCodeGeneratorCrash(ErrMsg);
|
||||
return true;
|
||||
}
|
||||
outs() << "\n*** diff'd output matches!\n";
|
||||
|
||||
sys::Path(Filename).eraseFromDisk();
|
||||
|
||||
outs() << "\n\n";
|
||||
num++;
|
||||
} //end while
|
||||
|
||||
// Unreachable.
|
||||
}
|
201
contrib/llvm/tools/bugpoint/ListReducer.h
Normal file
201
contrib/llvm/tools/bugpoint/ListReducer.h
Normal file
@ -0,0 +1,201 @@
|
||||
//===- ListReducer.h - Trim down list while retaining property --*- C++ -*-===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This class is to be used as a base class for operations that want to zero in
|
||||
// on a subset of the input which still causes the bug we are tracking.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef BUGPOINT_LIST_REDUCER_H
|
||||
#define BUGPOINT_LIST_REDUCER_H
|
||||
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/ErrorHandling.h"
|
||||
#include <vector>
|
||||
#include <cstdlib>
|
||||
#include <algorithm>
|
||||
|
||||
namespace llvm {
|
||||
|
||||
extern bool BugpointIsInterrupted;
|
||||
|
||||
template<typename ElTy>
|
||||
struct ListReducer {
|
||||
enum TestResult {
|
||||
NoFailure, // No failure of the predicate was detected
|
||||
KeepSuffix, // The suffix alone satisfies the predicate
|
||||
KeepPrefix, // The prefix alone satisfies the predicate
|
||||
InternalError // Encountered an error trying to run the predicate
|
||||
};
|
||||
|
||||
virtual ~ListReducer() {}
|
||||
|
||||
// doTest - This virtual function should be overriden by subclasses to
|
||||
// implement the test desired. The testcase is only required to test to see
|
||||
// if the Kept list still satisfies the property, but if it is going to check
|
||||
// the prefix anyway, it can.
|
||||
//
|
||||
virtual TestResult doTest(std::vector<ElTy> &Prefix,
|
||||
std::vector<ElTy> &Kept,
|
||||
std::string &Error) = 0;
|
||||
|
||||
// reduceList - This function attempts to reduce the length of the specified
|
||||
// list while still maintaining the "test" property. This is the core of the
|
||||
// "work" that bugpoint does.
|
||||
//
|
||||
bool reduceList(std::vector<ElTy> &TheList, std::string &Error) {
|
||||
std::vector<ElTy> empty;
|
||||
std::srand(0x6e5ea738); // Seed the random number generator
|
||||
switch (doTest(TheList, empty, Error)) {
|
||||
case KeepPrefix:
|
||||
if (TheList.size() == 1) // we are done, it's the base case and it fails
|
||||
return true;
|
||||
else
|
||||
break; // there's definitely an error, but we need to narrow it down
|
||||
|
||||
case KeepSuffix:
|
||||
// cannot be reached!
|
||||
llvm_unreachable("bugpoint ListReducer internal error: "
|
||||
"selected empty set.");
|
||||
|
||||
case NoFailure:
|
||||
return false; // there is no failure with the full set of passes/funcs!
|
||||
|
||||
case InternalError:
|
||||
assert(!Error.empty());
|
||||
return true;
|
||||
}
|
||||
|
||||
// Maximal number of allowed splitting iterations,
|
||||
// before the elements are randomly shuffled.
|
||||
const unsigned MaxIterationsWithoutProgress = 3;
|
||||
bool ShufflingEnabled = true;
|
||||
|
||||
Backjump:
|
||||
unsigned MidTop = TheList.size();
|
||||
unsigned MaxIterations = MaxIterationsWithoutProgress;
|
||||
unsigned NumOfIterationsWithoutProgress = 0;
|
||||
while (MidTop > 1) { // Binary split reduction loop
|
||||
// Halt if the user presses ctrl-c.
|
||||
if (BugpointIsInterrupted) {
|
||||
errs() << "\n\n*** Reduction Interrupted, cleaning up...\n\n";
|
||||
return true;
|
||||
}
|
||||
|
||||
// If the loop doesn't make satisfying progress, try shuffling.
|
||||
// The purpose of shuffling is to avoid the heavy tails of the
|
||||
// distribution (improving the speed of convergence).
|
||||
if (ShufflingEnabled &&
|
||||
NumOfIterationsWithoutProgress > MaxIterations) {
|
||||
std::vector<ElTy> ShuffledList(TheList);
|
||||
std::random_shuffle(ShuffledList.begin(), ShuffledList.end());
|
||||
errs() << "\n\n*** Testing shuffled set...\n\n";
|
||||
// Check that random shuffle doesn't loose the bug
|
||||
if (doTest(ShuffledList, empty, Error) == KeepPrefix) {
|
||||
// If the bug is still here, use the shuffled list.
|
||||
TheList.swap(ShuffledList);
|
||||
MidTop = TheList.size();
|
||||
// Must increase the shuffling treshold to avoid the small
|
||||
// probability of inifinite looping without making progress.
|
||||
MaxIterations += 2;
|
||||
errs() << "\n\n*** Shuffling does not hide the bug...\n\n";
|
||||
} else {
|
||||
ShufflingEnabled = false; // Disable shuffling further on
|
||||
errs() << "\n\n*** Shuffling hides the bug...\n\n";
|
||||
}
|
||||
NumOfIterationsWithoutProgress = 0;
|
||||
}
|
||||
|
||||
unsigned Mid = MidTop / 2;
|
||||
std::vector<ElTy> Prefix(TheList.begin(), TheList.begin()+Mid);
|
||||
std::vector<ElTy> Suffix(TheList.begin()+Mid, TheList.end());
|
||||
|
||||
switch (doTest(Prefix, Suffix, Error)) {
|
||||
case KeepSuffix:
|
||||
// The property still holds. We can just drop the prefix elements, and
|
||||
// shorten the list to the "kept" elements.
|
||||
TheList.swap(Suffix);
|
||||
MidTop = TheList.size();
|
||||
// Reset progress treshold and progress counter
|
||||
MaxIterations = MaxIterationsWithoutProgress;
|
||||
NumOfIterationsWithoutProgress = 0;
|
||||
break;
|
||||
case KeepPrefix:
|
||||
// The predicate still holds, shorten the list to the prefix elements.
|
||||
TheList.swap(Prefix);
|
||||
MidTop = TheList.size();
|
||||
// Reset progress treshold and progress counter
|
||||
MaxIterations = MaxIterationsWithoutProgress;
|
||||
NumOfIterationsWithoutProgress = 0;
|
||||
break;
|
||||
case NoFailure:
|
||||
// Otherwise the property doesn't hold. Some of the elements we removed
|
||||
// must be necessary to maintain the property.
|
||||
MidTop = Mid;
|
||||
NumOfIterationsWithoutProgress++;
|
||||
break;
|
||||
case InternalError:
|
||||
return true; // Error was set by doTest.
|
||||
}
|
||||
assert(Error.empty() && "doTest did not return InternalError for error");
|
||||
}
|
||||
|
||||
// Probability of backjumping from the trimming loop back to the binary
|
||||
// split reduction loop.
|
||||
const int BackjumpProbability = 10;
|
||||
|
||||
// Okay, we trimmed as much off the top and the bottom of the list as we
|
||||
// could. If there is more than two elements in the list, try deleting
|
||||
// interior elements and testing that.
|
||||
//
|
||||
if (TheList.size() > 2) {
|
||||
bool Changed = true;
|
||||
std::vector<ElTy> EmptyList;
|
||||
while (Changed) { // Trimming loop.
|
||||
Changed = false;
|
||||
|
||||
// If the binary split reduction loop made an unfortunate sequence of
|
||||
// splits, the trimming loop might be left off with a huge number of
|
||||
// remaining elements (large search space). Backjumping out of that
|
||||
// search space and attempting a different split can significantly
|
||||
// improve the convergence speed.
|
||||
if (std::rand() % 100 < BackjumpProbability)
|
||||
goto Backjump;
|
||||
|
||||
for (unsigned i = 1; i < TheList.size()-1; ++i) { // Check interior elts
|
||||
if (BugpointIsInterrupted) {
|
||||
errs() << "\n\n*** Reduction Interrupted, cleaning up...\n\n";
|
||||
return true;
|
||||
}
|
||||
|
||||
std::vector<ElTy> TestList(TheList);
|
||||
TestList.erase(TestList.begin()+i);
|
||||
|
||||
if (doTest(EmptyList, TestList, Error) == KeepSuffix) {
|
||||
// We can trim down the list!
|
||||
TheList.swap(TestList);
|
||||
--i; // Don't skip an element of the list
|
||||
Changed = true;
|
||||
}
|
||||
if (!Error.empty())
|
||||
return true;
|
||||
}
|
||||
// This can take a long time if left uncontrolled. For now, don't
|
||||
// iterate.
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return true; // there are some failure and we've narrowed them down
|
||||
}
|
||||
};
|
||||
|
||||
} // End llvm namespace
|
||||
|
||||
#endif
|
16
contrib/llvm/tools/bugpoint/Makefile
Normal file
16
contrib/llvm/tools/bugpoint/Makefile
Normal file
@ -0,0 +1,16 @@
|
||||
##===- tools/bugpoint/Makefile -----------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
LEVEL = ../..
|
||||
|
||||
TOOLNAME = bugpoint
|
||||
|
||||
LINK_COMPONENTS := asmparser instrumentation scalaropts ipo \
|
||||
linker bitreader bitwriter
|
||||
|
||||
include $(LEVEL)/Makefile.common
|
1078
contrib/llvm/tools/bugpoint/Miscompilation.cpp
Normal file
1078
contrib/llvm/tools/bugpoint/Miscompilation.cpp
Normal file
File diff suppressed because it is too large
Load Diff
265
contrib/llvm/tools/bugpoint/OptimizerDriver.cpp
Normal file
265
contrib/llvm/tools/bugpoint/OptimizerDriver.cpp
Normal file
@ -0,0 +1,265 @@
|
||||
//===- OptimizerDriver.cpp - Allow BugPoint to run passes safely ----------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file defines an interface that allows bugpoint to run various passes
|
||||
// without the threat of a buggy pass corrupting bugpoint (of course, bugpoint
|
||||
// may have its own bugs, but that's another story...). It achieves this by
|
||||
// forking a copy of itself and having the child process do the optimizations.
|
||||
// If this client dies, we can always fork a new one. :)
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "BugDriver.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/PassManager.h"
|
||||
#include "llvm/Analysis/Verifier.h"
|
||||
#include "llvm/Bitcode/ReaderWriter.h"
|
||||
#include "llvm/Target/TargetData.h"
|
||||
#include "llvm/Support/FileUtilities.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/SystemUtils.h"
|
||||
#include "llvm/Support/Debug.h"
|
||||
#include "llvm/Support/ToolOutputFile.h"
|
||||
#include "llvm/Support/Path.h"
|
||||
#include "llvm/Support/Program.h"
|
||||
|
||||
#define DONT_GET_PLUGIN_LOADER_OPTION
|
||||
#include "llvm/Support/PluginLoader.h"
|
||||
|
||||
#include <fstream>
|
||||
using namespace llvm;
|
||||
|
||||
namespace llvm {
|
||||
extern cl::opt<std::string> OutputPrefix;
|
||||
}
|
||||
|
||||
namespace {
|
||||
// ChildOutput - This option captures the name of the child output file that
|
||||
// is set up by the parent bugpoint process
|
||||
cl::opt<std::string> ChildOutput("child-output", cl::ReallyHidden);
|
||||
}
|
||||
|
||||
/// writeProgramToFile - This writes the current "Program" to the named bitcode
|
||||
/// file. If an error occurs, true is returned.
|
||||
///
|
||||
bool BugDriver::writeProgramToFile(const std::string &Filename,
|
||||
const Module *M) const {
|
||||
std::string ErrInfo;
|
||||
tool_output_file Out(Filename.c_str(), ErrInfo,
|
||||
raw_fd_ostream::F_Binary);
|
||||
if (ErrInfo.empty()) {
|
||||
WriteBitcodeToFile(M, Out.os());
|
||||
Out.os().close();
|
||||
if (!Out.os().has_error()) {
|
||||
Out.keep();
|
||||
return false;
|
||||
}
|
||||
}
|
||||
Out.os().clear_error();
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
/// EmitProgressBitcode - This function is used to output the current Program
|
||||
/// to a file named "bugpoint-ID.bc".
|
||||
///
|
||||
void BugDriver::EmitProgressBitcode(const Module *M,
|
||||
const std::string &ID,
|
||||
bool NoFlyer) const {
|
||||
// Output the input to the current pass to a bitcode file, emit a message
|
||||
// telling the user how to reproduce it: opt -foo blah.bc
|
||||
//
|
||||
std::string Filename = OutputPrefix + "-" + ID + ".bc";
|
||||
if (writeProgramToFile(Filename, M)) {
|
||||
errs() << "Error opening file '" << Filename << "' for writing!\n";
|
||||
return;
|
||||
}
|
||||
|
||||
outs() << "Emitted bitcode to '" << Filename << "'\n";
|
||||
if (NoFlyer || PassesToRun.empty()) return;
|
||||
outs() << "\n*** You can reproduce the problem with: ";
|
||||
if (UseValgrind) outs() << "valgrind ";
|
||||
outs() << "opt " << Filename << " ";
|
||||
outs() << getPassesString(PassesToRun) << "\n";
|
||||
}
|
||||
|
||||
cl::opt<bool> SilencePasses("silence-passes",
|
||||
cl::desc("Suppress output of running passes (both stdout and stderr)"));
|
||||
|
||||
static cl::list<std::string> OptArgs("opt-args", cl::Positional,
|
||||
cl::desc("<opt arguments>..."),
|
||||
cl::ZeroOrMore, cl::PositionalEatsArgs);
|
||||
|
||||
/// runPasses - Run the specified passes on Program, outputting a bitcode file
|
||||
/// and writing the filename into OutputFile if successful. If the
|
||||
/// optimizations fail for some reason (optimizer crashes), return true,
|
||||
/// otherwise return false. If DeleteOutput is set to true, the bitcode is
|
||||
/// deleted on success, and the filename string is undefined. This prints to
|
||||
/// outs() a single line message indicating whether compilation was successful
|
||||
/// or failed.
|
||||
///
|
||||
bool BugDriver::runPasses(Module *Program,
|
||||
const std::vector<std::string> &Passes,
|
||||
std::string &OutputFilename, bool DeleteOutput,
|
||||
bool Quiet, unsigned NumExtraArgs,
|
||||
const char * const *ExtraArgs) const {
|
||||
// setup the output file name
|
||||
outs().flush();
|
||||
sys::Path uniqueFilename(OutputPrefix + "-output.bc");
|
||||
std::string ErrMsg;
|
||||
if (uniqueFilename.makeUnique(true, &ErrMsg)) {
|
||||
errs() << getToolName() << ": Error making unique filename: "
|
||||
<< ErrMsg << "\n";
|
||||
return(1);
|
||||
}
|
||||
OutputFilename = uniqueFilename.str();
|
||||
|
||||
// set up the input file name
|
||||
sys::Path inputFilename(OutputPrefix + "-input.bc");
|
||||
if (inputFilename.makeUnique(true, &ErrMsg)) {
|
||||
errs() << getToolName() << ": Error making unique filename: "
|
||||
<< ErrMsg << "\n";
|
||||
return(1);
|
||||
}
|
||||
|
||||
std::string ErrInfo;
|
||||
tool_output_file InFile(inputFilename.c_str(), ErrInfo,
|
||||
raw_fd_ostream::F_Binary);
|
||||
|
||||
|
||||
if (!ErrInfo.empty()) {
|
||||
errs() << "Error opening bitcode file: " << inputFilename.str() << "\n";
|
||||
return 1;
|
||||
}
|
||||
WriteBitcodeToFile(Program, InFile.os());
|
||||
InFile.os().close();
|
||||
if (InFile.os().has_error()) {
|
||||
errs() << "Error writing bitcode file: " << inputFilename.str() << "\n";
|
||||
InFile.os().clear_error();
|
||||
return 1;
|
||||
}
|
||||
|
||||
sys::Path tool = PrependMainExecutablePath("opt", getToolName(),
|
||||
(void*)"opt");
|
||||
if (tool.empty()) {
|
||||
errs() << "Cannot find `opt' in executable directory!\n";
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Ok, everything that could go wrong before running opt is done.
|
||||
InFile.keep();
|
||||
|
||||
// setup the child process' arguments
|
||||
SmallVector<const char*, 8> Args;
|
||||
std::string Opt = tool.str();
|
||||
if (UseValgrind) {
|
||||
Args.push_back("valgrind");
|
||||
Args.push_back("--error-exitcode=1");
|
||||
Args.push_back("-q");
|
||||
Args.push_back(tool.c_str());
|
||||
} else
|
||||
Args.push_back(Opt.c_str());
|
||||
|
||||
Args.push_back("-o");
|
||||
Args.push_back(OutputFilename.c_str());
|
||||
for (unsigned i = 0, e = OptArgs.size(); i != e; ++i)
|
||||
Args.push_back(OptArgs[i].c_str());
|
||||
std::vector<std::string> pass_args;
|
||||
for (unsigned i = 0, e = PluginLoader::getNumPlugins(); i != e; ++i) {
|
||||
pass_args.push_back( std::string("-load"));
|
||||
pass_args.push_back( PluginLoader::getPlugin(i));
|
||||
}
|
||||
for (std::vector<std::string>::const_iterator I = Passes.begin(),
|
||||
E = Passes.end(); I != E; ++I )
|
||||
pass_args.push_back( std::string("-") + (*I) );
|
||||
for (std::vector<std::string>::const_iterator I = pass_args.begin(),
|
||||
E = pass_args.end(); I != E; ++I )
|
||||
Args.push_back(I->c_str());
|
||||
Args.push_back(inputFilename.c_str());
|
||||
for (unsigned i = 0; i < NumExtraArgs; ++i)
|
||||
Args.push_back(*ExtraArgs);
|
||||
Args.push_back(0);
|
||||
|
||||
DEBUG(errs() << "\nAbout to run:\t";
|
||||
for (unsigned i = 0, e = Args.size()-1; i != e; ++i)
|
||||
errs() << " " << Args[i];
|
||||
errs() << "\n";
|
||||
);
|
||||
|
||||
sys::Path prog;
|
||||
if (UseValgrind)
|
||||
prog = sys::Program::FindProgramByName("valgrind");
|
||||
else
|
||||
prog = tool;
|
||||
|
||||
// Redirect stdout and stderr to nowhere if SilencePasses is given
|
||||
sys::Path Nowhere;
|
||||
const sys::Path *Redirects[3] = {0, &Nowhere, &Nowhere};
|
||||
|
||||
int result = sys::Program::ExecuteAndWait(prog, Args.data(), 0,
|
||||
(SilencePasses ? Redirects : 0),
|
||||
Timeout, MemoryLimit, &ErrMsg);
|
||||
|
||||
// If we are supposed to delete the bitcode file or if the passes crashed,
|
||||
// remove it now. This may fail if the file was never created, but that's ok.
|
||||
if (DeleteOutput || result != 0)
|
||||
sys::Path(OutputFilename).eraseFromDisk();
|
||||
|
||||
// Remove the temporary input file as well
|
||||
inputFilename.eraseFromDisk();
|
||||
|
||||
if (!Quiet) {
|
||||
if (result == 0)
|
||||
outs() << "Success!\n";
|
||||
else if (result > 0)
|
||||
outs() << "Exited with error code '" << result << "'\n";
|
||||
else if (result < 0) {
|
||||
if (result == -1)
|
||||
outs() << "Execute failed: " << ErrMsg << "\n";
|
||||
else
|
||||
outs() << "Crashed: " << ErrMsg << "\n";
|
||||
}
|
||||
if (result & 0x01000000)
|
||||
outs() << "Dumped core\n";
|
||||
}
|
||||
|
||||
// Was the child successful?
|
||||
return result != 0;
|
||||
}
|
||||
|
||||
|
||||
/// runPassesOn - Carefully run the specified set of pass on the specified
|
||||
/// module, returning the transformed module on success, or a null pointer on
|
||||
/// failure.
|
||||
Module *BugDriver::runPassesOn(Module *M,
|
||||
const std::vector<std::string> &Passes,
|
||||
bool AutoDebugCrashes, unsigned NumExtraArgs,
|
||||
const char * const *ExtraArgs) {
|
||||
std::string BitcodeResult;
|
||||
if (runPasses(M, Passes, BitcodeResult, false/*delete*/, true/*quiet*/,
|
||||
NumExtraArgs, ExtraArgs)) {
|
||||
if (AutoDebugCrashes) {
|
||||
errs() << " Error running this sequence of passes"
|
||||
<< " on the input program!\n";
|
||||
delete swapProgramIn(M);
|
||||
EmitProgressBitcode(M, "pass-error", false);
|
||||
exit(debugOptimizerCrash());
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
Module *Ret = ParseInputFile(BitcodeResult, Context);
|
||||
if (Ret == 0) {
|
||||
errs() << getToolName() << ": Error reading bitcode file '"
|
||||
<< BitcodeResult << "'!\n";
|
||||
exit(1);
|
||||
}
|
||||
sys::Path(BitcodeResult).eraseFromDisk(); // No longer need the file on disk
|
||||
return Ret;
|
||||
}
|
977
contrib/llvm/tools/bugpoint/ToolRunner.cpp
Normal file
977
contrib/llvm/tools/bugpoint/ToolRunner.cpp
Normal file
@ -0,0 +1,977 @@
|
||||
//===-- ToolRunner.cpp ----------------------------------------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file implements the interfaces described in the ToolRunner.h file.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#define DEBUG_TYPE "toolrunner"
|
||||
#include "ToolRunner.h"
|
||||
#include "llvm/Support/Program.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/Debug.h"
|
||||
#include "llvm/Support/FileUtilities.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Config/config.h" // for HAVE_LINK_R
|
||||
#include <fstream>
|
||||
#include <sstream>
|
||||
using namespace llvm;
|
||||
|
||||
namespace llvm {
|
||||
cl::opt<bool>
|
||||
SaveTemps("save-temps", cl::init(false), cl::desc("Save temporary files"));
|
||||
}
|
||||
|
||||
namespace {
|
||||
cl::opt<std::string>
|
||||
RemoteClient("remote-client",
|
||||
cl::desc("Remote execution client (rsh/ssh)"));
|
||||
|
||||
cl::opt<std::string>
|
||||
RemoteHost("remote-host",
|
||||
cl::desc("Remote execution (rsh/ssh) host"));
|
||||
|
||||
cl::opt<std::string>
|
||||
RemotePort("remote-port",
|
||||
cl::desc("Remote execution (rsh/ssh) port"));
|
||||
|
||||
cl::opt<std::string>
|
||||
RemoteUser("remote-user",
|
||||
cl::desc("Remote execution (rsh/ssh) user id"));
|
||||
|
||||
cl::opt<std::string>
|
||||
RemoteExtra("remote-extra-options",
|
||||
cl::desc("Remote execution (rsh/ssh) extra options"));
|
||||
}
|
||||
|
||||
/// RunProgramWithTimeout - This function provides an alternate interface
|
||||
/// to the sys::Program::ExecuteAndWait interface.
|
||||
/// @see sys::Program::ExecuteAndWait
|
||||
static int RunProgramWithTimeout(const sys::Path &ProgramPath,
|
||||
const char **Args,
|
||||
const sys::Path &StdInFile,
|
||||
const sys::Path &StdOutFile,
|
||||
const sys::Path &StdErrFile,
|
||||
unsigned NumSeconds = 0,
|
||||
unsigned MemoryLimit = 0,
|
||||
std::string *ErrMsg = 0) {
|
||||
const sys::Path* redirects[3];
|
||||
redirects[0] = &StdInFile;
|
||||
redirects[1] = &StdOutFile;
|
||||
redirects[2] = &StdErrFile;
|
||||
|
||||
#if 0 // For debug purposes
|
||||
{
|
||||
errs() << "RUN:";
|
||||
for (unsigned i = 0; Args[i]; ++i)
|
||||
errs() << " " << Args[i];
|
||||
errs() << "\n";
|
||||
}
|
||||
#endif
|
||||
|
||||
return
|
||||
sys::Program::ExecuteAndWait(ProgramPath, Args, 0, redirects,
|
||||
NumSeconds, MemoryLimit, ErrMsg);
|
||||
}
|
||||
|
||||
/// RunProgramRemotelyWithTimeout - This function runs the given program
|
||||
/// remotely using the given remote client and the sys::Program::ExecuteAndWait.
|
||||
/// Returns the remote program exit code or reports a remote client error if it
|
||||
/// fails. Remote client is required to return 255 if it failed or program exit
|
||||
/// code otherwise.
|
||||
/// @see sys::Program::ExecuteAndWait
|
||||
static int RunProgramRemotelyWithTimeout(const sys::Path &RemoteClientPath,
|
||||
const char **Args,
|
||||
const sys::Path &StdInFile,
|
||||
const sys::Path &StdOutFile,
|
||||
const sys::Path &StdErrFile,
|
||||
unsigned NumSeconds = 0,
|
||||
unsigned MemoryLimit = 0) {
|
||||
const sys::Path* redirects[3];
|
||||
redirects[0] = &StdInFile;
|
||||
redirects[1] = &StdOutFile;
|
||||
redirects[2] = &StdErrFile;
|
||||
|
||||
#if 0 // For debug purposes
|
||||
{
|
||||
errs() << "RUN:";
|
||||
for (unsigned i = 0; Args[i]; ++i)
|
||||
errs() << " " << Args[i];
|
||||
errs() << "\n";
|
||||
}
|
||||
#endif
|
||||
|
||||
// Run the program remotely with the remote client
|
||||
int ReturnCode = sys::Program::ExecuteAndWait(RemoteClientPath, Args,
|
||||
0, redirects, NumSeconds, MemoryLimit);
|
||||
|
||||
// Has the remote client fail?
|
||||
if (255 == ReturnCode) {
|
||||
std::ostringstream OS;
|
||||
OS << "\nError running remote client:\n ";
|
||||
for (const char **Arg = Args; *Arg; ++Arg)
|
||||
OS << " " << *Arg;
|
||||
OS << "\n";
|
||||
|
||||
// The error message is in the output file, let's print it out from there.
|
||||
std::ifstream ErrorFile(StdOutFile.c_str());
|
||||
if (ErrorFile) {
|
||||
std::copy(std::istreambuf_iterator<char>(ErrorFile),
|
||||
std::istreambuf_iterator<char>(),
|
||||
std::ostreambuf_iterator<char>(OS));
|
||||
ErrorFile.close();
|
||||
}
|
||||
|
||||
errs() << OS;
|
||||
}
|
||||
|
||||
return ReturnCode;
|
||||
}
|
||||
|
||||
static std::string ProcessFailure(sys::Path ProgPath, const char** Args,
|
||||
unsigned Timeout = 0,
|
||||
unsigned MemoryLimit = 0) {
|
||||
std::ostringstream OS;
|
||||
OS << "\nError running tool:\n ";
|
||||
for (const char **Arg = Args; *Arg; ++Arg)
|
||||
OS << " " << *Arg;
|
||||
OS << "\n";
|
||||
|
||||
// Rerun the compiler, capturing any error messages to print them.
|
||||
sys::Path ErrorFilename("bugpoint.program_error_messages");
|
||||
std::string ErrMsg;
|
||||
if (ErrorFilename.makeUnique(true, &ErrMsg)) {
|
||||
errs() << "Error making unique filename: " << ErrMsg << "\n";
|
||||
exit(1);
|
||||
}
|
||||
RunProgramWithTimeout(ProgPath, Args, sys::Path(""), ErrorFilename,
|
||||
ErrorFilename, Timeout, MemoryLimit);
|
||||
// FIXME: check return code ?
|
||||
|
||||
// Print out the error messages generated by GCC if possible...
|
||||
std::ifstream ErrorFile(ErrorFilename.c_str());
|
||||
if (ErrorFile) {
|
||||
std::copy(std::istreambuf_iterator<char>(ErrorFile),
|
||||
std::istreambuf_iterator<char>(),
|
||||
std::ostreambuf_iterator<char>(OS));
|
||||
ErrorFile.close();
|
||||
}
|
||||
|
||||
ErrorFilename.eraseFromDisk();
|
||||
return OS.str();
|
||||
}
|
||||
|
||||
//===---------------------------------------------------------------------===//
|
||||
// LLI Implementation of AbstractIntepreter interface
|
||||
//
|
||||
namespace {
|
||||
class LLI : public AbstractInterpreter {
|
||||
std::string LLIPath; // The path to the LLI executable
|
||||
std::vector<std::string> ToolArgs; // Args to pass to LLI
|
||||
public:
|
||||
LLI(const std::string &Path, const std::vector<std::string> *Args)
|
||||
: LLIPath(Path) {
|
||||
ToolArgs.clear ();
|
||||
if (Args) { ToolArgs = *Args; }
|
||||
}
|
||||
|
||||
virtual int ExecuteProgram(const std::string &Bitcode,
|
||||
const std::vector<std::string> &Args,
|
||||
const std::string &InputFile,
|
||||
const std::string &OutputFile,
|
||||
std::string *Error,
|
||||
const std::vector<std::string> &GCCArgs,
|
||||
const std::vector<std::string> &SharedLibs =
|
||||
std::vector<std::string>(),
|
||||
unsigned Timeout = 0,
|
||||
unsigned MemoryLimit = 0);
|
||||
};
|
||||
}
|
||||
|
||||
int LLI::ExecuteProgram(const std::string &Bitcode,
|
||||
const std::vector<std::string> &Args,
|
||||
const std::string &InputFile,
|
||||
const std::string &OutputFile,
|
||||
std::string *Error,
|
||||
const std::vector<std::string> &GCCArgs,
|
||||
const std::vector<std::string> &SharedLibs,
|
||||
unsigned Timeout,
|
||||
unsigned MemoryLimit) {
|
||||
std::vector<const char*> LLIArgs;
|
||||
LLIArgs.push_back(LLIPath.c_str());
|
||||
LLIArgs.push_back("-force-interpreter=true");
|
||||
|
||||
for (std::vector<std::string>::const_iterator i = SharedLibs.begin(),
|
||||
e = SharedLibs.end(); i != e; ++i) {
|
||||
LLIArgs.push_back("-load");
|
||||
LLIArgs.push_back((*i).c_str());
|
||||
}
|
||||
|
||||
// Add any extra LLI args.
|
||||
for (unsigned i = 0, e = ToolArgs.size(); i != e; ++i)
|
||||
LLIArgs.push_back(ToolArgs[i].c_str());
|
||||
|
||||
LLIArgs.push_back(Bitcode.c_str());
|
||||
// Add optional parameters to the running program from Argv
|
||||
for (unsigned i=0, e = Args.size(); i != e; ++i)
|
||||
LLIArgs.push_back(Args[i].c_str());
|
||||
LLIArgs.push_back(0);
|
||||
|
||||
outs() << "<lli>"; outs().flush();
|
||||
DEBUG(errs() << "\nAbout to run:\t";
|
||||
for (unsigned i=0, e = LLIArgs.size()-1; i != e; ++i)
|
||||
errs() << " " << LLIArgs[i];
|
||||
errs() << "\n";
|
||||
);
|
||||
return RunProgramWithTimeout(sys::Path(LLIPath), &LLIArgs[0],
|
||||
sys::Path(InputFile), sys::Path(OutputFile), sys::Path(OutputFile),
|
||||
Timeout, MemoryLimit, Error);
|
||||
}
|
||||
|
||||
// LLI create method - Try to find the LLI executable
|
||||
AbstractInterpreter *AbstractInterpreter::createLLI(const char *Argv0,
|
||||
std::string &Message,
|
||||
const std::vector<std::string> *ToolArgs) {
|
||||
std::string LLIPath =
|
||||
PrependMainExecutablePath("lli", Argv0, (void *)(intptr_t)&createLLI).str();
|
||||
if (!LLIPath.empty()) {
|
||||
Message = "Found lli: " + LLIPath + "\n";
|
||||
return new LLI(LLIPath, ToolArgs);
|
||||
}
|
||||
|
||||
Message = "Cannot find `lli' in executable directory!\n";
|
||||
return 0;
|
||||
}
|
||||
|
||||
//===---------------------------------------------------------------------===//
|
||||
// Custom compiler command implementation of AbstractIntepreter interface
|
||||
//
|
||||
// Allows using a custom command for compiling the bitcode, thus allows, for
|
||||
// example, to compile a bitcode fragment without linking or executing, then
|
||||
// using a custom wrapper script to check for compiler errors.
|
||||
namespace {
|
||||
class CustomCompiler : public AbstractInterpreter {
|
||||
std::string CompilerCommand;
|
||||
std::vector<std::string> CompilerArgs;
|
||||
public:
|
||||
CustomCompiler(
|
||||
const std::string &CompilerCmd, std::vector<std::string> CompArgs) :
|
||||
CompilerCommand(CompilerCmd), CompilerArgs(CompArgs) {}
|
||||
|
||||
virtual void compileProgram(const std::string &Bitcode,
|
||||
std::string *Error,
|
||||
unsigned Timeout = 0,
|
||||
unsigned MemoryLimit = 0);
|
||||
|
||||
virtual int ExecuteProgram(const std::string &Bitcode,
|
||||
const std::vector<std::string> &Args,
|
||||
const std::string &InputFile,
|
||||
const std::string &OutputFile,
|
||||
std::string *Error,
|
||||
const std::vector<std::string> &GCCArgs =
|
||||
std::vector<std::string>(),
|
||||
const std::vector<std::string> &SharedLibs =
|
||||
std::vector<std::string>(),
|
||||
unsigned Timeout = 0,
|
||||
unsigned MemoryLimit = 0) {
|
||||
*Error = "Execution not supported with -compile-custom";
|
||||
return -1;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
void CustomCompiler::compileProgram(const std::string &Bitcode,
|
||||
std::string *Error,
|
||||
unsigned Timeout,
|
||||
unsigned MemoryLimit) {
|
||||
|
||||
std::vector<const char*> ProgramArgs;
|
||||
ProgramArgs.push_back(CompilerCommand.c_str());
|
||||
|
||||
for (std::size_t i = 0; i < CompilerArgs.size(); ++i)
|
||||
ProgramArgs.push_back(CompilerArgs.at(i).c_str());
|
||||
ProgramArgs.push_back(Bitcode.c_str());
|
||||
ProgramArgs.push_back(0);
|
||||
|
||||
// Add optional parameters to the running program from Argv
|
||||
for (unsigned i = 0, e = CompilerArgs.size(); i != e; ++i)
|
||||
ProgramArgs.push_back(CompilerArgs[i].c_str());
|
||||
|
||||
if (RunProgramWithTimeout( sys::Path(CompilerCommand), &ProgramArgs[0],
|
||||
sys::Path(), sys::Path(), sys::Path(),
|
||||
Timeout, MemoryLimit, Error))
|
||||
*Error = ProcessFailure(sys::Path(CompilerCommand), &ProgramArgs[0],
|
||||
Timeout, MemoryLimit);
|
||||
}
|
||||
|
||||
//===---------------------------------------------------------------------===//
|
||||
// Custom execution command implementation of AbstractIntepreter interface
|
||||
//
|
||||
// Allows using a custom command for executing the bitcode, thus allows,
|
||||
// for example, to invoke a cross compiler for code generation followed by
|
||||
// a simulator that executes the generated binary.
|
||||
namespace {
|
||||
class CustomExecutor : public AbstractInterpreter {
|
||||
std::string ExecutionCommand;
|
||||
std::vector<std::string> ExecutorArgs;
|
||||
public:
|
||||
CustomExecutor(
|
||||
const std::string &ExecutionCmd, std::vector<std::string> ExecArgs) :
|
||||
ExecutionCommand(ExecutionCmd), ExecutorArgs(ExecArgs) {}
|
||||
|
||||
virtual int ExecuteProgram(const std::string &Bitcode,
|
||||
const std::vector<std::string> &Args,
|
||||
const std::string &InputFile,
|
||||
const std::string &OutputFile,
|
||||
std::string *Error,
|
||||
const std::vector<std::string> &GCCArgs,
|
||||
const std::vector<std::string> &SharedLibs =
|
||||
std::vector<std::string>(),
|
||||
unsigned Timeout = 0,
|
||||
unsigned MemoryLimit = 0);
|
||||
};
|
||||
}
|
||||
|
||||
int CustomExecutor::ExecuteProgram(const std::string &Bitcode,
|
||||
const std::vector<std::string> &Args,
|
||||
const std::string &InputFile,
|
||||
const std::string &OutputFile,
|
||||
std::string *Error,
|
||||
const std::vector<std::string> &GCCArgs,
|
||||
const std::vector<std::string> &SharedLibs,
|
||||
unsigned Timeout,
|
||||
unsigned MemoryLimit) {
|
||||
|
||||
std::vector<const char*> ProgramArgs;
|
||||
ProgramArgs.push_back(ExecutionCommand.c_str());
|
||||
|
||||
for (std::size_t i = 0; i < ExecutorArgs.size(); ++i)
|
||||
ProgramArgs.push_back(ExecutorArgs.at(i).c_str());
|
||||
ProgramArgs.push_back(Bitcode.c_str());
|
||||
ProgramArgs.push_back(0);
|
||||
|
||||
// Add optional parameters to the running program from Argv
|
||||
for (unsigned i = 0, e = Args.size(); i != e; ++i)
|
||||
ProgramArgs.push_back(Args[i].c_str());
|
||||
|
||||
return RunProgramWithTimeout(
|
||||
sys::Path(ExecutionCommand),
|
||||
&ProgramArgs[0], sys::Path(InputFile), sys::Path(OutputFile),
|
||||
sys::Path(OutputFile), Timeout, MemoryLimit, Error);
|
||||
}
|
||||
|
||||
// Tokenize the CommandLine to the command and the args to allow
|
||||
// defining a full command line as the command instead of just the
|
||||
// executed program. We cannot just pass the whole string after the command
|
||||
// as a single argument because then program sees only a single
|
||||
// command line argument (with spaces in it: "foo bar" instead
|
||||
// of "foo" and "bar").
|
||||
//
|
||||
// code borrowed from:
|
||||
// http://oopweb.com/CPP/Documents/CPPHOWTO/Volume/C++Programming-HOWTO-7.html
|
||||
static void lexCommand(std::string &Message, const std::string &CommandLine,
|
||||
std::string &CmdPath, std::vector<std::string> Args) {
|
||||
|
||||
std::string Command = "";
|
||||
std::string delimiters = " ";
|
||||
|
||||
std::string::size_type lastPos = CommandLine.find_first_not_of(delimiters, 0);
|
||||
std::string::size_type pos = CommandLine.find_first_of(delimiters, lastPos);
|
||||
|
||||
while (std::string::npos != pos || std::string::npos != lastPos) {
|
||||
std::string token = CommandLine.substr(lastPos, pos - lastPos);
|
||||
if (Command == "")
|
||||
Command = token;
|
||||
else
|
||||
Args.push_back(token);
|
||||
// Skip delimiters. Note the "not_of"
|
||||
lastPos = CommandLine.find_first_not_of(delimiters, pos);
|
||||
// Find next "non-delimiter"
|
||||
pos = CommandLine.find_first_of(delimiters, lastPos);
|
||||
}
|
||||
|
||||
CmdPath = sys::Program::FindProgramByName(Command).str();
|
||||
if (CmdPath.empty()) {
|
||||
Message =
|
||||
std::string("Cannot find '") + Command +
|
||||
"' in PATH!\n";
|
||||
return;
|
||||
}
|
||||
|
||||
Message = "Found command in: " + CmdPath + "\n";
|
||||
}
|
||||
|
||||
// Custom execution environment create method, takes the execution command
|
||||
// as arguments
|
||||
AbstractInterpreter *AbstractInterpreter::createCustomCompiler(
|
||||
std::string &Message,
|
||||
const std::string &CompileCommandLine) {
|
||||
|
||||
std::string CmdPath;
|
||||
std::vector<std::string> Args;
|
||||
lexCommand(Message, CompileCommandLine, CmdPath, Args);
|
||||
if (CmdPath.empty())
|
||||
return 0;
|
||||
|
||||
return new CustomCompiler(CmdPath, Args);
|
||||
}
|
||||
|
||||
// Custom execution environment create method, takes the execution command
|
||||
// as arguments
|
||||
AbstractInterpreter *AbstractInterpreter::createCustomExecutor(
|
||||
std::string &Message,
|
||||
const std::string &ExecCommandLine) {
|
||||
|
||||
|
||||
std::string CmdPath;
|
||||
std::vector<std::string> Args;
|
||||
lexCommand(Message, ExecCommandLine, CmdPath, Args);
|
||||
if (CmdPath.empty())
|
||||
return 0;
|
||||
|
||||
return new CustomExecutor(CmdPath, Args);
|
||||
}
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
// LLC Implementation of AbstractIntepreter interface
|
||||
//
|
||||
GCC::FileType LLC::OutputCode(const std::string &Bitcode,
|
||||
sys::Path &OutputAsmFile, std::string &Error,
|
||||
unsigned Timeout, unsigned MemoryLimit) {
|
||||
const char *Suffix = (UseIntegratedAssembler ? ".llc.o" : ".llc.s");
|
||||
sys::Path uniqueFile(Bitcode + Suffix);
|
||||
std::string ErrMsg;
|
||||
if (uniqueFile.makeUnique(true, &ErrMsg)) {
|
||||
errs() << "Error making unique filename: " << ErrMsg << "\n";
|
||||
exit(1);
|
||||
}
|
||||
OutputAsmFile = uniqueFile;
|
||||
std::vector<const char *> LLCArgs;
|
||||
LLCArgs.push_back(LLCPath.c_str());
|
||||
|
||||
// Add any extra LLC args.
|
||||
for (unsigned i = 0, e = ToolArgs.size(); i != e; ++i)
|
||||
LLCArgs.push_back(ToolArgs[i].c_str());
|
||||
|
||||
LLCArgs.push_back("-o");
|
||||
LLCArgs.push_back(OutputAsmFile.c_str()); // Output to the Asm file
|
||||
LLCArgs.push_back(Bitcode.c_str()); // This is the input bitcode
|
||||
|
||||
if (UseIntegratedAssembler)
|
||||
LLCArgs.push_back("-filetype=obj");
|
||||
|
||||
LLCArgs.push_back (0);
|
||||
|
||||
outs() << (UseIntegratedAssembler ? "<llc-ia>" : "<llc>");
|
||||
outs().flush();
|
||||
DEBUG(errs() << "\nAbout to run:\t";
|
||||
for (unsigned i = 0, e = LLCArgs.size()-1; i != e; ++i)
|
||||
errs() << " " << LLCArgs[i];
|
||||
errs() << "\n";
|
||||
);
|
||||
if (RunProgramWithTimeout(sys::Path(LLCPath), &LLCArgs[0],
|
||||
sys::Path(), sys::Path(), sys::Path(),
|
||||
Timeout, MemoryLimit))
|
||||
Error = ProcessFailure(sys::Path(LLCPath), &LLCArgs[0],
|
||||
Timeout, MemoryLimit);
|
||||
return UseIntegratedAssembler ? GCC::ObjectFile : GCC::AsmFile;
|
||||
}
|
||||
|
||||
void LLC::compileProgram(const std::string &Bitcode, std::string *Error,
|
||||
unsigned Timeout, unsigned MemoryLimit) {
|
||||
sys::Path OutputAsmFile;
|
||||
OutputCode(Bitcode, OutputAsmFile, *Error, Timeout, MemoryLimit);
|
||||
OutputAsmFile.eraseFromDisk();
|
||||
}
|
||||
|
||||
int LLC::ExecuteProgram(const std::string &Bitcode,
|
||||
const std::vector<std::string> &Args,
|
||||
const std::string &InputFile,
|
||||
const std::string &OutputFile,
|
||||
std::string *Error,
|
||||
const std::vector<std::string> &ArgsForGCC,
|
||||
const std::vector<std::string> &SharedLibs,
|
||||
unsigned Timeout,
|
||||
unsigned MemoryLimit) {
|
||||
|
||||
sys::Path OutputAsmFile;
|
||||
GCC::FileType FileKind = OutputCode(Bitcode, OutputAsmFile, *Error, Timeout,
|
||||
MemoryLimit);
|
||||
FileRemover OutFileRemover(OutputAsmFile.str(), !SaveTemps);
|
||||
|
||||
std::vector<std::string> GCCArgs(ArgsForGCC);
|
||||
GCCArgs.insert(GCCArgs.end(), SharedLibs.begin(), SharedLibs.end());
|
||||
|
||||
// Assuming LLC worked, compile the result with GCC and run it.
|
||||
return gcc->ExecuteProgram(OutputAsmFile.str(), Args, FileKind,
|
||||
InputFile, OutputFile, Error, GCCArgs,
|
||||
Timeout, MemoryLimit);
|
||||
}
|
||||
|
||||
/// createLLC - Try to find the LLC executable
|
||||
///
|
||||
LLC *AbstractInterpreter::createLLC(const char *Argv0,
|
||||
std::string &Message,
|
||||
const std::string &GCCBinary,
|
||||
const std::vector<std::string> *Args,
|
||||
const std::vector<std::string> *GCCArgs,
|
||||
bool UseIntegratedAssembler) {
|
||||
std::string LLCPath =
|
||||
PrependMainExecutablePath("llc", Argv0, (void *)(intptr_t)&createLLC).str();
|
||||
if (LLCPath.empty()) {
|
||||
Message = "Cannot find `llc' in executable directory!\n";
|
||||
return 0;
|
||||
}
|
||||
|
||||
Message = "Found llc: " + LLCPath + "\n";
|
||||
GCC *gcc = GCC::create(Message, GCCBinary, GCCArgs);
|
||||
if (!gcc) {
|
||||
errs() << Message << "\n";
|
||||
exit(1);
|
||||
}
|
||||
return new LLC(LLCPath, gcc, Args, UseIntegratedAssembler);
|
||||
}
|
||||
|
||||
//===---------------------------------------------------------------------===//
|
||||
// JIT Implementation of AbstractIntepreter interface
|
||||
//
|
||||
namespace {
|
||||
class JIT : public AbstractInterpreter {
|
||||
std::string LLIPath; // The path to the LLI executable
|
||||
std::vector<std::string> ToolArgs; // Args to pass to LLI
|
||||
public:
|
||||
JIT(const std::string &Path, const std::vector<std::string> *Args)
|
||||
: LLIPath(Path) {
|
||||
ToolArgs.clear ();
|
||||
if (Args) { ToolArgs = *Args; }
|
||||
}
|
||||
|
||||
virtual int ExecuteProgram(const std::string &Bitcode,
|
||||
const std::vector<std::string> &Args,
|
||||
const std::string &InputFile,
|
||||
const std::string &OutputFile,
|
||||
std::string *Error,
|
||||
const std::vector<std::string> &GCCArgs =
|
||||
std::vector<std::string>(),
|
||||
const std::vector<std::string> &SharedLibs =
|
||||
std::vector<std::string>(),
|
||||
unsigned Timeout = 0,
|
||||
unsigned MemoryLimit = 0);
|
||||
};
|
||||
}
|
||||
|
||||
int JIT::ExecuteProgram(const std::string &Bitcode,
|
||||
const std::vector<std::string> &Args,
|
||||
const std::string &InputFile,
|
||||
const std::string &OutputFile,
|
||||
std::string *Error,
|
||||
const std::vector<std::string> &GCCArgs,
|
||||
const std::vector<std::string> &SharedLibs,
|
||||
unsigned Timeout,
|
||||
unsigned MemoryLimit) {
|
||||
// Construct a vector of parameters, incorporating those from the command-line
|
||||
std::vector<const char*> JITArgs;
|
||||
JITArgs.push_back(LLIPath.c_str());
|
||||
JITArgs.push_back("-force-interpreter=false");
|
||||
|
||||
// Add any extra LLI args.
|
||||
for (unsigned i = 0, e = ToolArgs.size(); i != e; ++i)
|
||||
JITArgs.push_back(ToolArgs[i].c_str());
|
||||
|
||||
for (unsigned i = 0, e = SharedLibs.size(); i != e; ++i) {
|
||||
JITArgs.push_back("-load");
|
||||
JITArgs.push_back(SharedLibs[i].c_str());
|
||||
}
|
||||
JITArgs.push_back(Bitcode.c_str());
|
||||
// Add optional parameters to the running program from Argv
|
||||
for (unsigned i=0, e = Args.size(); i != e; ++i)
|
||||
JITArgs.push_back(Args[i].c_str());
|
||||
JITArgs.push_back(0);
|
||||
|
||||
outs() << "<jit>"; outs().flush();
|
||||
DEBUG(errs() << "\nAbout to run:\t";
|
||||
for (unsigned i=0, e = JITArgs.size()-1; i != e; ++i)
|
||||
errs() << " " << JITArgs[i];
|
||||
errs() << "\n";
|
||||
);
|
||||
DEBUG(errs() << "\nSending output to " << OutputFile << "\n");
|
||||
return RunProgramWithTimeout(sys::Path(LLIPath), &JITArgs[0],
|
||||
sys::Path(InputFile), sys::Path(OutputFile), sys::Path(OutputFile),
|
||||
Timeout, MemoryLimit, Error);
|
||||
}
|
||||
|
||||
/// createJIT - Try to find the LLI executable
|
||||
///
|
||||
AbstractInterpreter *AbstractInterpreter::createJIT(const char *Argv0,
|
||||
std::string &Message, const std::vector<std::string> *Args) {
|
||||
std::string LLIPath =
|
||||
PrependMainExecutablePath("lli", Argv0, (void *)(intptr_t)&createJIT).str();
|
||||
if (!LLIPath.empty()) {
|
||||
Message = "Found lli: " + LLIPath + "\n";
|
||||
return new JIT(LLIPath, Args);
|
||||
}
|
||||
|
||||
Message = "Cannot find `lli' in executable directory!\n";
|
||||
return 0;
|
||||
}
|
||||
|
||||
GCC::FileType CBE::OutputCode(const std::string &Bitcode,
|
||||
sys::Path &OutputCFile, std::string &Error,
|
||||
unsigned Timeout, unsigned MemoryLimit) {
|
||||
sys::Path uniqueFile(Bitcode+".cbe.c");
|
||||
std::string ErrMsg;
|
||||
if (uniqueFile.makeUnique(true, &ErrMsg)) {
|
||||
errs() << "Error making unique filename: " << ErrMsg << "\n";
|
||||
exit(1);
|
||||
}
|
||||
OutputCFile = uniqueFile;
|
||||
std::vector<const char *> LLCArgs;
|
||||
LLCArgs.push_back(LLCPath.c_str());
|
||||
|
||||
// Add any extra LLC args.
|
||||
for (unsigned i = 0, e = ToolArgs.size(); i != e; ++i)
|
||||
LLCArgs.push_back(ToolArgs[i].c_str());
|
||||
|
||||
LLCArgs.push_back("-o");
|
||||
LLCArgs.push_back(OutputCFile.c_str()); // Output to the C file
|
||||
LLCArgs.push_back("-march=c"); // Output C language
|
||||
LLCArgs.push_back(Bitcode.c_str()); // This is the input bitcode
|
||||
LLCArgs.push_back(0);
|
||||
|
||||
outs() << "<cbe>"; outs().flush();
|
||||
DEBUG(errs() << "\nAbout to run:\t";
|
||||
for (unsigned i = 0, e = LLCArgs.size()-1; i != e; ++i)
|
||||
errs() << " " << LLCArgs[i];
|
||||
errs() << "\n";
|
||||
);
|
||||
if (RunProgramWithTimeout(LLCPath, &LLCArgs[0], sys::Path(), sys::Path(),
|
||||
sys::Path(), Timeout, MemoryLimit))
|
||||
Error = ProcessFailure(LLCPath, &LLCArgs[0], Timeout, MemoryLimit);
|
||||
return GCC::CFile;
|
||||
}
|
||||
|
||||
void CBE::compileProgram(const std::string &Bitcode, std::string *Error,
|
||||
unsigned Timeout, unsigned MemoryLimit) {
|
||||
sys::Path OutputCFile;
|
||||
OutputCode(Bitcode, OutputCFile, *Error, Timeout, MemoryLimit);
|
||||
OutputCFile.eraseFromDisk();
|
||||
}
|
||||
|
||||
int CBE::ExecuteProgram(const std::string &Bitcode,
|
||||
const std::vector<std::string> &Args,
|
||||
const std::string &InputFile,
|
||||
const std::string &OutputFile,
|
||||
std::string *Error,
|
||||
const std::vector<std::string> &ArgsForGCC,
|
||||
const std::vector<std::string> &SharedLibs,
|
||||
unsigned Timeout,
|
||||
unsigned MemoryLimit) {
|
||||
sys::Path OutputCFile;
|
||||
OutputCode(Bitcode, OutputCFile, *Error, Timeout, MemoryLimit);
|
||||
|
||||
FileRemover CFileRemove(OutputCFile.str(), !SaveTemps);
|
||||
|
||||
std::vector<std::string> GCCArgs(ArgsForGCC);
|
||||
GCCArgs.insert(GCCArgs.end(), SharedLibs.begin(), SharedLibs.end());
|
||||
|
||||
return gcc->ExecuteProgram(OutputCFile.str(), Args, GCC::CFile,
|
||||
InputFile, OutputFile, Error, GCCArgs,
|
||||
Timeout, MemoryLimit);
|
||||
}
|
||||
|
||||
/// createCBE - Try to find the 'llc' executable
|
||||
///
|
||||
CBE *AbstractInterpreter::createCBE(const char *Argv0,
|
||||
std::string &Message,
|
||||
const std::string &GCCBinary,
|
||||
const std::vector<std::string> *Args,
|
||||
const std::vector<std::string> *GCCArgs) {
|
||||
sys::Path LLCPath =
|
||||
PrependMainExecutablePath("llc", Argv0, (void *)(intptr_t)&createCBE);
|
||||
if (LLCPath.isEmpty()) {
|
||||
Message =
|
||||
"Cannot find `llc' in executable directory!\n";
|
||||
return 0;
|
||||
}
|
||||
|
||||
Message = "Found llc: " + LLCPath.str() + "\n";
|
||||
GCC *gcc = GCC::create(Message, GCCBinary, GCCArgs);
|
||||
if (!gcc) {
|
||||
errs() << Message << "\n";
|
||||
exit(1);
|
||||
}
|
||||
return new CBE(LLCPath, gcc, Args);
|
||||
}
|
||||
|
||||
//===---------------------------------------------------------------------===//
|
||||
// GCC abstraction
|
||||
//
|
||||
|
||||
static bool IsARMArchitecture(std::vector<const char*> Args) {
|
||||
for (std::vector<const char*>::const_iterator
|
||||
I = Args.begin(), E = Args.end(); I != E; ++I) {
|
||||
if (StringRef(*I).equals_lower("-arch")) {
|
||||
++I;
|
||||
if (I != E && StringRef(*I).substr(0, strlen("arm")).equals_lower("arm"))
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
int GCC::ExecuteProgram(const std::string &ProgramFile,
|
||||
const std::vector<std::string> &Args,
|
||||
FileType fileType,
|
||||
const std::string &InputFile,
|
||||
const std::string &OutputFile,
|
||||
std::string *Error,
|
||||
const std::vector<std::string> &ArgsForGCC,
|
||||
unsigned Timeout,
|
||||
unsigned MemoryLimit) {
|
||||
std::vector<const char*> GCCArgs;
|
||||
|
||||
GCCArgs.push_back(GCCPath.c_str());
|
||||
|
||||
if (TargetTriple.getArch() == Triple::x86)
|
||||
GCCArgs.push_back("-m32");
|
||||
|
||||
for (std::vector<std::string>::const_iterator
|
||||
I = gccArgs.begin(), E = gccArgs.end(); I != E; ++I)
|
||||
GCCArgs.push_back(I->c_str());
|
||||
|
||||
// Specify -x explicitly in case the extension is wonky
|
||||
if (fileType != ObjectFile) {
|
||||
GCCArgs.push_back("-x");
|
||||
if (fileType == CFile) {
|
||||
GCCArgs.push_back("c");
|
||||
GCCArgs.push_back("-fno-strict-aliasing");
|
||||
} else {
|
||||
GCCArgs.push_back("assembler");
|
||||
|
||||
// For ARM architectures we don't want this flag. bugpoint isn't
|
||||
// explicitly told what architecture it is working on, so we get
|
||||
// it from gcc flags
|
||||
if (TargetTriple.isOSDarwin() && !IsARMArchitecture(GCCArgs))
|
||||
GCCArgs.push_back("-force_cpusubtype_ALL");
|
||||
}
|
||||
}
|
||||
|
||||
GCCArgs.push_back(ProgramFile.c_str()); // Specify the input filename.
|
||||
|
||||
GCCArgs.push_back("-x");
|
||||
GCCArgs.push_back("none");
|
||||
GCCArgs.push_back("-o");
|
||||
sys::Path OutputBinary (ProgramFile+".gcc.exe");
|
||||
std::string ErrMsg;
|
||||
if (OutputBinary.makeUnique(true, &ErrMsg)) {
|
||||
errs() << "Error making unique filename: " << ErrMsg << "\n";
|
||||
exit(1);
|
||||
}
|
||||
GCCArgs.push_back(OutputBinary.c_str()); // Output to the right file...
|
||||
|
||||
// Add any arguments intended for GCC. We locate them here because this is
|
||||
// most likely -L and -l options that need to come before other libraries but
|
||||
// after the source. Other options won't be sensitive to placement on the
|
||||
// command line, so this should be safe.
|
||||
for (unsigned i = 0, e = ArgsForGCC.size(); i != e; ++i)
|
||||
GCCArgs.push_back(ArgsForGCC[i].c_str());
|
||||
|
||||
GCCArgs.push_back("-lm"); // Hard-code the math library...
|
||||
GCCArgs.push_back("-O2"); // Optimize the program a bit...
|
||||
#if defined (HAVE_LINK_R)
|
||||
GCCArgs.push_back("-Wl,-R."); // Search this dir for .so files
|
||||
#endif
|
||||
if (TargetTriple.getArch() == Triple::sparc)
|
||||
GCCArgs.push_back("-mcpu=v9");
|
||||
GCCArgs.push_back(0); // NULL terminator
|
||||
|
||||
outs() << "<gcc>"; outs().flush();
|
||||
DEBUG(errs() << "\nAbout to run:\t";
|
||||
for (unsigned i = 0, e = GCCArgs.size()-1; i != e; ++i)
|
||||
errs() << " " << GCCArgs[i];
|
||||
errs() << "\n";
|
||||
);
|
||||
if (RunProgramWithTimeout(GCCPath, &GCCArgs[0], sys::Path(), sys::Path(),
|
||||
sys::Path())) {
|
||||
*Error = ProcessFailure(GCCPath, &GCCArgs[0]);
|
||||
return -1;
|
||||
}
|
||||
|
||||
std::vector<const char*> ProgramArgs;
|
||||
|
||||
// Declared here so that the destructor only runs after
|
||||
// ProgramArgs is used.
|
||||
std::string Exec;
|
||||
|
||||
if (RemoteClientPath.isEmpty())
|
||||
ProgramArgs.push_back(OutputBinary.c_str());
|
||||
else {
|
||||
ProgramArgs.push_back(RemoteClientPath.c_str());
|
||||
ProgramArgs.push_back(RemoteHost.c_str());
|
||||
if (!RemoteUser.empty()) {
|
||||
ProgramArgs.push_back("-l");
|
||||
ProgramArgs.push_back(RemoteUser.c_str());
|
||||
}
|
||||
if (!RemotePort.empty()) {
|
||||
ProgramArgs.push_back("-p");
|
||||
ProgramArgs.push_back(RemotePort.c_str());
|
||||
}
|
||||
if (!RemoteExtra.empty()) {
|
||||
ProgramArgs.push_back(RemoteExtra.c_str());
|
||||
}
|
||||
|
||||
// Full path to the binary. We need to cd to the exec directory because
|
||||
// there is a dylib there that the exec expects to find in the CWD
|
||||
char* env_pwd = getenv("PWD");
|
||||
Exec = "cd ";
|
||||
Exec += env_pwd;
|
||||
Exec += "; ./";
|
||||
Exec += OutputBinary.c_str();
|
||||
ProgramArgs.push_back(Exec.c_str());
|
||||
}
|
||||
|
||||
// Add optional parameters to the running program from Argv
|
||||
for (unsigned i = 0, e = Args.size(); i != e; ++i)
|
||||
ProgramArgs.push_back(Args[i].c_str());
|
||||
ProgramArgs.push_back(0); // NULL terminator
|
||||
|
||||
// Now that we have a binary, run it!
|
||||
outs() << "<program>"; outs().flush();
|
||||
DEBUG(errs() << "\nAbout to run:\t";
|
||||
for (unsigned i = 0, e = ProgramArgs.size()-1; i != e; ++i)
|
||||
errs() << " " << ProgramArgs[i];
|
||||
errs() << "\n";
|
||||
);
|
||||
|
||||
FileRemover OutputBinaryRemover(OutputBinary.str(), !SaveTemps);
|
||||
|
||||
if (RemoteClientPath.isEmpty()) {
|
||||
DEBUG(errs() << "<run locally>");
|
||||
int ExitCode = RunProgramWithTimeout(OutputBinary, &ProgramArgs[0],
|
||||
sys::Path(InputFile), sys::Path(OutputFile), sys::Path(OutputFile),
|
||||
Timeout, MemoryLimit, Error);
|
||||
// Treat a signal (usually SIGSEGV) or timeout as part of the program output
|
||||
// so that crash-causing miscompilation is handled seamlessly.
|
||||
if (ExitCode < -1) {
|
||||
std::ofstream outFile(OutputFile.c_str(), std::ios_base::app);
|
||||
outFile << *Error << '\n';
|
||||
outFile.close();
|
||||
Error->clear();
|
||||
}
|
||||
return ExitCode;
|
||||
} else {
|
||||
outs() << "<run remotely>"; outs().flush();
|
||||
return RunProgramRemotelyWithTimeout(sys::Path(RemoteClientPath),
|
||||
&ProgramArgs[0], sys::Path(InputFile), sys::Path(OutputFile),
|
||||
sys::Path(OutputFile), Timeout, MemoryLimit);
|
||||
}
|
||||
}
|
||||
|
||||
int GCC::MakeSharedObject(const std::string &InputFile, FileType fileType,
|
||||
std::string &OutputFile,
|
||||
const std::vector<std::string> &ArgsForGCC,
|
||||
std::string &Error) {
|
||||
sys::Path uniqueFilename(InputFile+LTDL_SHLIB_EXT);
|
||||
std::string ErrMsg;
|
||||
if (uniqueFilename.makeUnique(true, &ErrMsg)) {
|
||||
errs() << "Error making unique filename: " << ErrMsg << "\n";
|
||||
exit(1);
|
||||
}
|
||||
OutputFile = uniqueFilename.str();
|
||||
|
||||
std::vector<const char*> GCCArgs;
|
||||
|
||||
GCCArgs.push_back(GCCPath.c_str());
|
||||
|
||||
if (TargetTriple.getArch() == Triple::x86)
|
||||
GCCArgs.push_back("-m32");
|
||||
|
||||
for (std::vector<std::string>::const_iterator
|
||||
I = gccArgs.begin(), E = gccArgs.end(); I != E; ++I)
|
||||
GCCArgs.push_back(I->c_str());
|
||||
|
||||
// Compile the C/asm file into a shared object
|
||||
if (fileType != ObjectFile) {
|
||||
GCCArgs.push_back("-x");
|
||||
GCCArgs.push_back(fileType == AsmFile ? "assembler" : "c");
|
||||
}
|
||||
GCCArgs.push_back("-fno-strict-aliasing");
|
||||
GCCArgs.push_back(InputFile.c_str()); // Specify the input filename.
|
||||
GCCArgs.push_back("-x");
|
||||
GCCArgs.push_back("none");
|
||||
if (TargetTriple.getArch() == Triple::sparc)
|
||||
GCCArgs.push_back("-G"); // Compile a shared library, `-G' for Sparc
|
||||
else if (TargetTriple.isOSDarwin()) {
|
||||
// link all source files into a single module in data segment, rather than
|
||||
// generating blocks. dynamic_lookup requires that you set
|
||||
// MACOSX_DEPLOYMENT_TARGET=10.3 in your env. FIXME: it would be better for
|
||||
// bugpoint to just pass that in the environment of GCC.
|
||||
GCCArgs.push_back("-single_module");
|
||||
GCCArgs.push_back("-dynamiclib"); // `-dynamiclib' for MacOS X/PowerPC
|
||||
GCCArgs.push_back("-undefined");
|
||||
GCCArgs.push_back("dynamic_lookup");
|
||||
} else
|
||||
GCCArgs.push_back("-shared"); // `-shared' for Linux/X86, maybe others
|
||||
|
||||
if ((TargetTriple.getArch() == Triple::alpha) ||
|
||||
(TargetTriple.getArch() == Triple::x86_64))
|
||||
GCCArgs.push_back("-fPIC"); // Requires shared objs to contain PIC
|
||||
|
||||
if (TargetTriple.getArch() == Triple::sparc)
|
||||
GCCArgs.push_back("-mcpu=v9");
|
||||
|
||||
GCCArgs.push_back("-o");
|
||||
GCCArgs.push_back(OutputFile.c_str()); // Output to the right filename.
|
||||
GCCArgs.push_back("-O2"); // Optimize the program a bit.
|
||||
|
||||
|
||||
|
||||
// Add any arguments intended for GCC. We locate them here because this is
|
||||
// most likely -L and -l options that need to come before other libraries but
|
||||
// after the source. Other options won't be sensitive to placement on the
|
||||
// command line, so this should be safe.
|
||||
for (unsigned i = 0, e = ArgsForGCC.size(); i != e; ++i)
|
||||
GCCArgs.push_back(ArgsForGCC[i].c_str());
|
||||
GCCArgs.push_back(0); // NULL terminator
|
||||
|
||||
|
||||
|
||||
outs() << "<gcc>"; outs().flush();
|
||||
DEBUG(errs() << "\nAbout to run:\t";
|
||||
for (unsigned i = 0, e = GCCArgs.size()-1; i != e; ++i)
|
||||
errs() << " " << GCCArgs[i];
|
||||
errs() << "\n";
|
||||
);
|
||||
if (RunProgramWithTimeout(GCCPath, &GCCArgs[0], sys::Path(), sys::Path(),
|
||||
sys::Path())) {
|
||||
Error = ProcessFailure(GCCPath, &GCCArgs[0]);
|
||||
return 1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
/// create - Try to find the `gcc' executable
|
||||
///
|
||||
GCC *GCC::create(std::string &Message,
|
||||
const std::string &GCCBinary,
|
||||
const std::vector<std::string> *Args) {
|
||||
sys::Path GCCPath = sys::Program::FindProgramByName(GCCBinary);
|
||||
if (GCCPath.isEmpty()) {
|
||||
Message = "Cannot find `"+ GCCBinary +"' in PATH!\n";
|
||||
return 0;
|
||||
}
|
||||
|
||||
sys::Path RemoteClientPath;
|
||||
if (!RemoteClient.empty())
|
||||
RemoteClientPath = sys::Program::FindProgramByName(RemoteClient);
|
||||
|
||||
Message = "Found gcc: " + GCCPath.str() + "\n";
|
||||
return new GCC(GCCPath, RemoteClientPath, Args);
|
||||
}
|
247
contrib/llvm/tools/bugpoint/ToolRunner.h
Normal file
247
contrib/llvm/tools/bugpoint/ToolRunner.h
Normal file
@ -0,0 +1,247 @@
|
||||
//===-- tools/bugpoint/ToolRunner.h -----------------------------*- C++ -*-===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file exposes an abstraction around a platform C compiler, used to
|
||||
// compile C and assembly code. It also exposes an "AbstractIntepreter"
|
||||
// interface, which is used to execute code using one of the LLVM execution
|
||||
// engines.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef BUGPOINT_TOOLRUNNER_H
|
||||
#define BUGPOINT_TOOLRUNNER_H
|
||||
|
||||
#include "llvm/ADT/Triple.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/ErrorHandling.h"
|
||||
#include "llvm/Support/SystemUtils.h"
|
||||
#include "llvm/Support/Path.h"
|
||||
#include <exception>
|
||||
#include <vector>
|
||||
|
||||
namespace llvm {
|
||||
|
||||
extern cl::opt<bool> SaveTemps;
|
||||
extern Triple TargetTriple;
|
||||
|
||||
class CBE;
|
||||
class LLC;
|
||||
|
||||
//===---------------------------------------------------------------------===//
|
||||
// GCC abstraction
|
||||
//
|
||||
class GCC {
|
||||
sys::Path GCCPath; // The path to the gcc executable.
|
||||
sys::Path RemoteClientPath; // The path to the rsh / ssh executable.
|
||||
std::vector<std::string> gccArgs; // GCC-specific arguments.
|
||||
GCC(const sys::Path &gccPath, const sys::Path &RemotePath,
|
||||
const std::vector<std::string> *GCCArgs)
|
||||
: GCCPath(gccPath), RemoteClientPath(RemotePath) {
|
||||
if (GCCArgs) gccArgs = *GCCArgs;
|
||||
}
|
||||
public:
|
||||
enum FileType { AsmFile, ObjectFile, CFile };
|
||||
|
||||
static GCC *create(std::string &Message,
|
||||
const std::string &GCCBinary,
|
||||
const std::vector<std::string> *Args);
|
||||
|
||||
/// ExecuteProgram - Execute the program specified by "ProgramFile" (which is
|
||||
/// either a .s file, or a .c file, specified by FileType), with the specified
|
||||
/// arguments. Standard input is specified with InputFile, and standard
|
||||
/// Output is captured to the specified OutputFile location. The SharedLibs
|
||||
/// option specifies optional native shared objects that can be loaded into
|
||||
/// the program for execution.
|
||||
///
|
||||
int ExecuteProgram(const std::string &ProgramFile,
|
||||
const std::vector<std::string> &Args,
|
||||
FileType fileType,
|
||||
const std::string &InputFile,
|
||||
const std::string &OutputFile,
|
||||
std::string *Error = 0,
|
||||
const std::vector<std::string> &GCCArgs =
|
||||
std::vector<std::string>(),
|
||||
unsigned Timeout = 0,
|
||||
unsigned MemoryLimit = 0);
|
||||
|
||||
/// MakeSharedObject - This compiles the specified file (which is either a .c
|
||||
/// file or a .s file) into a shared object.
|
||||
///
|
||||
int MakeSharedObject(const std::string &InputFile, FileType fileType,
|
||||
std::string &OutputFile,
|
||||
const std::vector<std::string> &ArgsForGCC,
|
||||
std::string &Error);
|
||||
};
|
||||
|
||||
|
||||
//===---------------------------------------------------------------------===//
|
||||
/// AbstractInterpreter Class - Subclasses of this class are used to execute
|
||||
/// LLVM bitcode in a variety of ways. This abstract interface hides this
|
||||
/// complexity behind a simple interface.
|
||||
///
|
||||
class AbstractInterpreter {
|
||||
public:
|
||||
static CBE *createCBE(const char *Argv0, std::string &Message,
|
||||
const std::string &GCCBinary,
|
||||
const std::vector<std::string> *Args = 0,
|
||||
const std::vector<std::string> *GCCArgs = 0);
|
||||
static LLC *createLLC(const char *Argv0, std::string &Message,
|
||||
const std::string &GCCBinary,
|
||||
const std::vector<std::string> *Args = 0,
|
||||
const std::vector<std::string> *GCCArgs = 0,
|
||||
bool UseIntegratedAssembler = false);
|
||||
|
||||
static AbstractInterpreter* createLLI(const char *Argv0, std::string &Message,
|
||||
const std::vector<std::string> *Args=0);
|
||||
|
||||
static AbstractInterpreter* createJIT(const char *Argv0, std::string &Message,
|
||||
const std::vector<std::string> *Args=0);
|
||||
|
||||
static AbstractInterpreter*
|
||||
createCustomCompiler(std::string &Message,
|
||||
const std::string &CompileCommandLine);
|
||||
|
||||
static AbstractInterpreter*
|
||||
createCustomExecutor(std::string &Message,
|
||||
const std::string &ExecCommandLine);
|
||||
|
||||
|
||||
virtual ~AbstractInterpreter() {}
|
||||
|
||||
/// compileProgram - Compile the specified program from bitcode to executable
|
||||
/// code. This does not produce any output, it is only used when debugging
|
||||
/// the code generator. It returns false if the code generator fails.
|
||||
virtual void compileProgram(const std::string &Bitcode, std::string *Error,
|
||||
unsigned Timeout = 0, unsigned MemoryLimit = 0) {}
|
||||
|
||||
/// OutputCode - Compile the specified program from bitcode to code
|
||||
/// understood by the GCC driver (either C or asm). If the code generator
|
||||
/// fails, it sets Error, otherwise, this function returns the type of code
|
||||
/// emitted.
|
||||
virtual GCC::FileType OutputCode(const std::string &Bitcode,
|
||||
sys::Path &OutFile, std::string &Error,
|
||||
unsigned Timeout = 0,
|
||||
unsigned MemoryLimit = 0) {
|
||||
Error = "OutputCode not supported by this AbstractInterpreter!";
|
||||
return GCC::AsmFile;
|
||||
}
|
||||
|
||||
/// ExecuteProgram - Run the specified bitcode file, emitting output to the
|
||||
/// specified filename. This sets RetVal to the exit code of the program or
|
||||
/// returns false if a problem was encountered that prevented execution of
|
||||
/// the program.
|
||||
///
|
||||
virtual int ExecuteProgram(const std::string &Bitcode,
|
||||
const std::vector<std::string> &Args,
|
||||
const std::string &InputFile,
|
||||
const std::string &OutputFile,
|
||||
std::string *Error,
|
||||
const std::vector<std::string> &GCCArgs =
|
||||
std::vector<std::string>(),
|
||||
const std::vector<std::string> &SharedLibs =
|
||||
std::vector<std::string>(),
|
||||
unsigned Timeout = 0,
|
||||
unsigned MemoryLimit = 0) = 0;
|
||||
};
|
||||
|
||||
//===---------------------------------------------------------------------===//
|
||||
// CBE Implementation of AbstractIntepreter interface
|
||||
//
|
||||
class CBE : public AbstractInterpreter {
|
||||
sys::Path LLCPath; // The path to the `llc' executable.
|
||||
std::vector<std::string> ToolArgs; // Extra args to pass to LLC.
|
||||
GCC *gcc;
|
||||
public:
|
||||
CBE(const sys::Path &llcPath, GCC *Gcc,
|
||||
const std::vector<std::string> *Args)
|
||||
: LLCPath(llcPath), gcc(Gcc) {
|
||||
ToolArgs.clear ();
|
||||
if (Args) ToolArgs = *Args;
|
||||
}
|
||||
~CBE() { delete gcc; }
|
||||
|
||||
/// compileProgram - Compile the specified program from bitcode to executable
|
||||
/// code. This does not produce any output, it is only used when debugging
|
||||
/// the code generator. Returns false if the code generator fails.
|
||||
virtual void compileProgram(const std::string &Bitcode, std::string *Error,
|
||||
unsigned Timeout = 0, unsigned MemoryLimit = 0);
|
||||
|
||||
virtual int ExecuteProgram(const std::string &Bitcode,
|
||||
const std::vector<std::string> &Args,
|
||||
const std::string &InputFile,
|
||||
const std::string &OutputFile,
|
||||
std::string *Error,
|
||||
const std::vector<std::string> &GCCArgs =
|
||||
std::vector<std::string>(),
|
||||
const std::vector<std::string> &SharedLibs =
|
||||
std::vector<std::string>(),
|
||||
unsigned Timeout = 0,
|
||||
unsigned MemoryLimit = 0);
|
||||
|
||||
/// OutputCode - Compile the specified program from bitcode to code
|
||||
/// understood by the GCC driver (either C or asm). If the code generator
|
||||
/// fails, it sets Error, otherwise, this function returns the type of code
|
||||
/// emitted.
|
||||
virtual GCC::FileType OutputCode(const std::string &Bitcode,
|
||||
sys::Path &OutFile, std::string &Error,
|
||||
unsigned Timeout = 0,
|
||||
unsigned MemoryLimit = 0);
|
||||
};
|
||||
|
||||
|
||||
//===---------------------------------------------------------------------===//
|
||||
// LLC Implementation of AbstractIntepreter interface
|
||||
//
|
||||
class LLC : public AbstractInterpreter {
|
||||
std::string LLCPath; // The path to the LLC executable.
|
||||
std::vector<std::string> ToolArgs; // Extra args to pass to LLC.
|
||||
GCC *gcc;
|
||||
bool UseIntegratedAssembler;
|
||||
public:
|
||||
LLC(const std::string &llcPath, GCC *Gcc,
|
||||
const std::vector<std::string> *Args,
|
||||
bool useIntegratedAssembler)
|
||||
: LLCPath(llcPath), gcc(Gcc),
|
||||
UseIntegratedAssembler(useIntegratedAssembler) {
|
||||
ToolArgs.clear();
|
||||
if (Args) ToolArgs = *Args;
|
||||
}
|
||||
~LLC() { delete gcc; }
|
||||
|
||||
/// compileProgram - Compile the specified program from bitcode to executable
|
||||
/// code. This does not produce any output, it is only used when debugging
|
||||
/// the code generator. Returns false if the code generator fails.
|
||||
virtual void compileProgram(const std::string &Bitcode, std::string *Error,
|
||||
unsigned Timeout = 0, unsigned MemoryLimit = 0);
|
||||
|
||||
virtual int ExecuteProgram(const std::string &Bitcode,
|
||||
const std::vector<std::string> &Args,
|
||||
const std::string &InputFile,
|
||||
const std::string &OutputFile,
|
||||
std::string *Error,
|
||||
const std::vector<std::string> &GCCArgs =
|
||||
std::vector<std::string>(),
|
||||
const std::vector<std::string> &SharedLibs =
|
||||
std::vector<std::string>(),
|
||||
unsigned Timeout = 0,
|
||||
unsigned MemoryLimit = 0);
|
||||
|
||||
/// OutputCode - Compile the specified program from bitcode to code
|
||||
/// understood by the GCC driver (either C or asm). If the code generator
|
||||
/// fails, it sets Error, otherwise, this function returns the type of code
|
||||
/// emitted.
|
||||
virtual GCC::FileType OutputCode(const std::string &Bitcode,
|
||||
sys::Path &OutFile, std::string &Error,
|
||||
unsigned Timeout = 0,
|
||||
unsigned MemoryLimit = 0);
|
||||
};
|
||||
|
||||
} // End llvm namespace
|
||||
|
||||
#endif
|
209
contrib/llvm/tools/bugpoint/bugpoint.cpp
Normal file
209
contrib/llvm/tools/bugpoint/bugpoint.cpp
Normal file
@ -0,0 +1,209 @@
|
||||
//===- bugpoint.cpp - The LLVM Bugpoint utility ---------------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This program is an automated compiler debugger tool. It is used to narrow
|
||||
// down miscompilations and crash problems to a specific pass in the compiler,
|
||||
// and the specific Module or Function input that is causing the problem.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "BugDriver.h"
|
||||
#include "ToolRunner.h"
|
||||
#include "llvm/LinkAllPasses.h"
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/PassManager.h"
|
||||
#include "llvm/Support/PassNameParser.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/PluginLoader.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/Process.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include "llvm/Support/Valgrind.h"
|
||||
#include "llvm/LinkAllVMCore.h"
|
||||
#include "llvm/Transforms/IPO/PassManagerBuilder.h"
|
||||
|
||||
//Enable this macro to debug bugpoint itself.
|
||||
//#define DEBUG_BUGPOINT 1
|
||||
|
||||
using namespace llvm;
|
||||
|
||||
static cl::opt<bool>
|
||||
FindBugs("find-bugs", cl::desc("Run many different optimization sequences "
|
||||
"on program to find bugs"), cl::init(false));
|
||||
|
||||
static cl::list<std::string>
|
||||
InputFilenames(cl::Positional, cl::OneOrMore,
|
||||
cl::desc("<input llvm ll/bc files>"));
|
||||
|
||||
static cl::opt<unsigned>
|
||||
TimeoutValue("timeout", cl::init(300), cl::value_desc("seconds"),
|
||||
cl::desc("Number of seconds program is allowed to run before it "
|
||||
"is killed (default is 300s), 0 disables timeout"));
|
||||
|
||||
static cl::opt<int>
|
||||
MemoryLimit("mlimit", cl::init(-1), cl::value_desc("MBytes"),
|
||||
cl::desc("Maximum amount of memory to use. 0 disables check."
|
||||
" Defaults to 100MB (800MB under valgrind)."));
|
||||
|
||||
static cl::opt<bool>
|
||||
UseValgrind("enable-valgrind",
|
||||
cl::desc("Run optimizations through valgrind"));
|
||||
|
||||
// The AnalysesList is automatically populated with registered Passes by the
|
||||
// PassNameParser.
|
||||
//
|
||||
static cl::list<const PassInfo*, bool, PassNameParser>
|
||||
PassList(cl::desc("Passes available:"), cl::ZeroOrMore);
|
||||
|
||||
static cl::opt<bool>
|
||||
StandardCompileOpts("std-compile-opts",
|
||||
cl::desc("Include the standard compile time optimizations"));
|
||||
|
||||
static cl::opt<bool>
|
||||
StandardLinkOpts("std-link-opts",
|
||||
cl::desc("Include the standard link time optimizations"));
|
||||
|
||||
static cl::opt<bool>
|
||||
OptLevelO1("O1",
|
||||
cl::desc("Optimization level 1. Similar to llvm-gcc -O1"));
|
||||
|
||||
static cl::opt<bool>
|
||||
OptLevelO2("O2",
|
||||
cl::desc("Optimization level 2. Similar to llvm-gcc -O2"));
|
||||
|
||||
static cl::opt<bool>
|
||||
OptLevelO3("O3",
|
||||
cl::desc("Optimization level 3. Similar to llvm-gcc -O3"));
|
||||
|
||||
static cl::opt<std::string>
|
||||
OverrideTriple("mtriple", cl::desc("Override target triple for module"));
|
||||
|
||||
/// BugpointIsInterrupted - Set to true when the user presses ctrl-c.
|
||||
bool llvm::BugpointIsInterrupted = false;
|
||||
|
||||
#ifndef DEBUG_BUGPOINT
|
||||
static void BugpointInterruptFunction() {
|
||||
BugpointIsInterrupted = true;
|
||||
}
|
||||
#endif
|
||||
|
||||
// Hack to capture a pass list.
|
||||
namespace {
|
||||
class AddToDriver : public FunctionPassManager {
|
||||
BugDriver &D;
|
||||
public:
|
||||
AddToDriver(BugDriver &_D) : FunctionPassManager(0), D(_D) {}
|
||||
|
||||
virtual void add(Pass *P) {
|
||||
const void *ID = P->getPassID();
|
||||
const PassInfo *PI = PassRegistry::getPassRegistry()->getPassInfo(ID);
|
||||
D.addPass(PI->getPassArgument());
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
#ifndef DEBUG_BUGPOINT
|
||||
llvm::sys::PrintStackTraceOnErrorSignal();
|
||||
llvm::PrettyStackTraceProgram X(argc, argv);
|
||||
llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
|
||||
#endif
|
||||
|
||||
// Initialize passes
|
||||
PassRegistry &Registry = *PassRegistry::getPassRegistry();
|
||||
initializeCore(Registry);
|
||||
initializeScalarOpts(Registry);
|
||||
initializeIPO(Registry);
|
||||
initializeAnalysis(Registry);
|
||||
initializeIPA(Registry);
|
||||
initializeTransformUtils(Registry);
|
||||
initializeInstCombine(Registry);
|
||||
initializeInstrumentation(Registry);
|
||||
initializeTarget(Registry);
|
||||
|
||||
cl::ParseCommandLineOptions(argc, argv,
|
||||
"LLVM automatic testcase reducer. See\nhttp://"
|
||||
"llvm.org/cmds/bugpoint.html"
|
||||
" for more information.\n");
|
||||
#ifndef DEBUG_BUGPOINT
|
||||
sys::SetInterruptFunction(BugpointInterruptFunction);
|
||||
#endif
|
||||
|
||||
LLVMContext& Context = getGlobalContext();
|
||||
// If we have an override, set it and then track the triple we want Modules
|
||||
// to use.
|
||||
if (!OverrideTriple.empty()) {
|
||||
TargetTriple.setTriple(Triple::normalize(OverrideTriple));
|
||||
outs() << "Override triple set to '" << TargetTriple.getTriple() << "'\n";
|
||||
}
|
||||
|
||||
if (MemoryLimit < 0) {
|
||||
// Set the default MemoryLimit. Be sure to update the flag's description if
|
||||
// you change this.
|
||||
if (sys::RunningOnValgrind() || UseValgrind)
|
||||
MemoryLimit = 800;
|
||||
else
|
||||
MemoryLimit = 100;
|
||||
}
|
||||
|
||||
BugDriver D(argv[0], FindBugs, TimeoutValue, MemoryLimit,
|
||||
UseValgrind, Context);
|
||||
if (D.addSources(InputFilenames)) return 1;
|
||||
|
||||
AddToDriver PM(D);
|
||||
if (StandardCompileOpts) {
|
||||
PassManagerBuilder Builder;
|
||||
Builder.OptLevel = 3;
|
||||
Builder.Inliner = createFunctionInliningPass();
|
||||
Builder.populateModulePassManager(PM);
|
||||
}
|
||||
|
||||
if (StandardLinkOpts) {
|
||||
PassManagerBuilder Builder;
|
||||
Builder.populateLTOPassManager(PM, /*Internalize=*/true,
|
||||
/*RunInliner=*/true);
|
||||
}
|
||||
|
||||
if (OptLevelO1 || OptLevelO2 || OptLevelO3) {
|
||||
PassManagerBuilder Builder;
|
||||
if (OptLevelO1)
|
||||
Builder.Inliner = createAlwaysInlinerPass();
|
||||
else if (OptLevelO2)
|
||||
Builder.Inliner = createFunctionInliningPass(225);
|
||||
else
|
||||
Builder.Inliner = createFunctionInliningPass(275);
|
||||
|
||||
// Note that although clang/llvm-gcc use two separate passmanagers
|
||||
// here, it shouldn't normally make a difference.
|
||||
Builder.populateFunctionPassManager(PM);
|
||||
Builder.populateModulePassManager(PM);
|
||||
}
|
||||
|
||||
for (std::vector<const PassInfo*>::iterator I = PassList.begin(),
|
||||
E = PassList.end();
|
||||
I != E; ++I) {
|
||||
const PassInfo* PI = *I;
|
||||
D.addPass(PI->getPassArgument());
|
||||
}
|
||||
|
||||
// Bugpoint has the ability of generating a plethora of core files, so to
|
||||
// avoid filling up the disk, we prevent it
|
||||
#ifndef DEBUG_BUGPOINT
|
||||
sys::Process::PreventCoreFiles();
|
||||
#endif
|
||||
|
||||
std::string Error;
|
||||
bool Failure = D.run(Error);
|
||||
if (!Error.empty()) {
|
||||
errs() << Error;
|
||||
return 1;
|
||||
}
|
||||
return Failure;
|
||||
}
|
5
contrib/llvm/tools/llc/CMakeLists.txt
Normal file
5
contrib/llvm/tools/llc/CMakeLists.txt
Normal file
@ -0,0 +1,5 @@
|
||||
set(LLVM_LINK_COMPONENTS ${LLVM_TARGETS_TO_BUILD} bitreader asmparser)
|
||||
|
||||
add_llvm_tool(llc
|
||||
llc.cpp
|
||||
)
|
21
contrib/llvm/tools/llc/Makefile
Normal file
21
contrib/llvm/tools/llc/Makefile
Normal file
@ -0,0 +1,21 @@
|
||||
#===- tools/llc/Makefile -----------------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
|
||||
LEVEL = ../..
|
||||
TOOLNAME = llc
|
||||
|
||||
# Include this here so we can get the configuration of the targets
|
||||
# that have been configured for construction. We have to do this
|
||||
# early so we can set up LINK_COMPONENTS before including Makefile.rules
|
||||
include $(LEVEL)/Makefile.config
|
||||
|
||||
LINK_COMPONENTS := $(TARGETS_TO_BUILD) bitreader asmparser
|
||||
|
||||
include $(LLVM_SRC_ROOT)/Makefile.rules
|
||||
|
381
contrib/llvm/tools/llc/llc.cpp
Normal file
381
contrib/llvm/tools/llc/llc.cpp
Normal file
@ -0,0 +1,381 @@
|
||||
//===-- llc.cpp - Implement the LLVM Native Code Generator ----------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This is the llc code generator driver. It provides a convenient
|
||||
// command-line interface for generating native assembly-language code
|
||||
// or C code, given LLVM bitcode.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/PassManager.h"
|
||||
#include "llvm/Pass.h"
|
||||
#include "llvm/ADT/Triple.h"
|
||||
#include "llvm/Support/IRReader.h"
|
||||
#include "llvm/CodeGen/LinkAllAsmWriterComponents.h"
|
||||
#include "llvm/CodeGen/LinkAllCodegenComponents.h"
|
||||
#include "llvm/Config/config.h"
|
||||
#include "llvm/MC/SubtargetFeature.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/Debug.h"
|
||||
#include "llvm/Support/FormattedStream.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/PluginLoader.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/ToolOutputFile.h"
|
||||
#include "llvm/Support/Host.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include "llvm/Support/TargetRegistry.h"
|
||||
#include "llvm/Support/TargetSelect.h"
|
||||
#include "llvm/Target/TargetData.h"
|
||||
#include "llvm/Target/TargetMachine.h"
|
||||
#include <memory>
|
||||
using namespace llvm;
|
||||
|
||||
// General options for llc. Other pass-specific options are specified
|
||||
// within the corresponding llc passes, and target-specific options
|
||||
// and back-end code generation options are specified with the target machine.
|
||||
//
|
||||
static cl::opt<std::string>
|
||||
InputFilename(cl::Positional, cl::desc("<input bitcode>"), cl::init("-"));
|
||||
|
||||
static cl::opt<std::string>
|
||||
OutputFilename("o", cl::desc("Output filename"), cl::value_desc("filename"));
|
||||
|
||||
// Determine optimization level.
|
||||
static cl::opt<char>
|
||||
OptLevel("O",
|
||||
cl::desc("Optimization level. [-O0, -O1, -O2, or -O3] "
|
||||
"(default = '-O2')"),
|
||||
cl::Prefix,
|
||||
cl::ZeroOrMore,
|
||||
cl::init(' '));
|
||||
|
||||
static cl::opt<std::string>
|
||||
TargetTriple("mtriple", cl::desc("Override target triple for module"));
|
||||
|
||||
static cl::opt<std::string>
|
||||
MArch("march", cl::desc("Architecture to generate code for (see --version)"));
|
||||
|
||||
static cl::opt<std::string>
|
||||
MCPU("mcpu",
|
||||
cl::desc("Target a specific cpu type (-mcpu=help for details)"),
|
||||
cl::value_desc("cpu-name"),
|
||||
cl::init(""));
|
||||
|
||||
static cl::list<std::string>
|
||||
MAttrs("mattr",
|
||||
cl::CommaSeparated,
|
||||
cl::desc("Target specific attributes (-mattr=help for details)"),
|
||||
cl::value_desc("a1,+a2,-a3,..."));
|
||||
|
||||
static cl::opt<Reloc::Model>
|
||||
RelocModel("relocation-model",
|
||||
cl::desc("Choose relocation model"),
|
||||
cl::init(Reloc::Default),
|
||||
cl::values(
|
||||
clEnumValN(Reloc::Default, "default",
|
||||
"Target default relocation model"),
|
||||
clEnumValN(Reloc::Static, "static",
|
||||
"Non-relocatable code"),
|
||||
clEnumValN(Reloc::PIC_, "pic",
|
||||
"Fully relocatable, position independent code"),
|
||||
clEnumValN(Reloc::DynamicNoPIC, "dynamic-no-pic",
|
||||
"Relocatable external references, non-relocatable code"),
|
||||
clEnumValEnd));
|
||||
|
||||
static cl::opt<llvm::CodeModel::Model>
|
||||
CMModel("code-model",
|
||||
cl::desc("Choose code model"),
|
||||
cl::init(CodeModel::Default),
|
||||
cl::values(clEnumValN(CodeModel::Default, "default",
|
||||
"Target default code model"),
|
||||
clEnumValN(CodeModel::Small, "small",
|
||||
"Small code model"),
|
||||
clEnumValN(CodeModel::Kernel, "kernel",
|
||||
"Kernel code model"),
|
||||
clEnumValN(CodeModel::Medium, "medium",
|
||||
"Medium code model"),
|
||||
clEnumValN(CodeModel::Large, "large",
|
||||
"Large code model"),
|
||||
clEnumValEnd));
|
||||
|
||||
static cl::opt<bool>
|
||||
RelaxAll("mc-relax-all",
|
||||
cl::desc("When used with filetype=obj, "
|
||||
"relax all fixups in the emitted object file"));
|
||||
|
||||
cl::opt<TargetMachine::CodeGenFileType>
|
||||
FileType("filetype", cl::init(TargetMachine::CGFT_AssemblyFile),
|
||||
cl::desc("Choose a file type (not all types are supported by all targets):"),
|
||||
cl::values(
|
||||
clEnumValN(TargetMachine::CGFT_AssemblyFile, "asm",
|
||||
"Emit an assembly ('.s') file"),
|
||||
clEnumValN(TargetMachine::CGFT_ObjectFile, "obj",
|
||||
"Emit a native object ('.o') file [experimental]"),
|
||||
clEnumValN(TargetMachine::CGFT_Null, "null",
|
||||
"Emit nothing, for performance testing"),
|
||||
clEnumValEnd));
|
||||
|
||||
cl::opt<bool> NoVerify("disable-verify", cl::Hidden,
|
||||
cl::desc("Do not verify input module"));
|
||||
|
||||
cl::opt<bool> DisableDotLoc("disable-dot-loc", cl::Hidden,
|
||||
cl::desc("Do not use .loc entries"));
|
||||
|
||||
cl::opt<bool> DisableCFI("disable-cfi", cl::Hidden,
|
||||
cl::desc("Do not use .cfi_* directives"));
|
||||
|
||||
static cl::opt<bool>
|
||||
DisableRedZone("disable-red-zone",
|
||||
cl::desc("Do not emit code that uses the red zone."),
|
||||
cl::init(false));
|
||||
|
||||
// GetFileNameRoot - Helper function to get the basename of a filename.
|
||||
static inline std::string
|
||||
GetFileNameRoot(const std::string &InputFilename) {
|
||||
std::string IFN = InputFilename;
|
||||
std::string outputFilename;
|
||||
int Len = IFN.length();
|
||||
if ((Len > 2) &&
|
||||
IFN[Len-3] == '.' &&
|
||||
((IFN[Len-2] == 'b' && IFN[Len-1] == 'c') ||
|
||||
(IFN[Len-2] == 'l' && IFN[Len-1] == 'l'))) {
|
||||
outputFilename = std::string(IFN.begin(), IFN.end()-3); // s/.bc/.s/
|
||||
} else {
|
||||
outputFilename = IFN;
|
||||
}
|
||||
return outputFilename;
|
||||
}
|
||||
|
||||
static tool_output_file *GetOutputStream(const char *TargetName,
|
||||
Triple::OSType OS,
|
||||
const char *ProgName) {
|
||||
// If we don't yet have an output filename, make one.
|
||||
if (OutputFilename.empty()) {
|
||||
if (InputFilename == "-")
|
||||
OutputFilename = "-";
|
||||
else {
|
||||
OutputFilename = GetFileNameRoot(InputFilename);
|
||||
|
||||
switch (FileType) {
|
||||
default: assert(0 && "Unknown file type");
|
||||
case TargetMachine::CGFT_AssemblyFile:
|
||||
if (TargetName[0] == 'c') {
|
||||
if (TargetName[1] == 0)
|
||||
OutputFilename += ".cbe.c";
|
||||
else if (TargetName[1] == 'p' && TargetName[2] == 'p')
|
||||
OutputFilename += ".cpp";
|
||||
else
|
||||
OutputFilename += ".s";
|
||||
} else
|
||||
OutputFilename += ".s";
|
||||
break;
|
||||
case TargetMachine::CGFT_ObjectFile:
|
||||
if (OS == Triple::Win32)
|
||||
OutputFilename += ".obj";
|
||||
else
|
||||
OutputFilename += ".o";
|
||||
break;
|
||||
case TargetMachine::CGFT_Null:
|
||||
OutputFilename += ".null";
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Decide if we need "binary" output.
|
||||
bool Binary = false;
|
||||
switch (FileType) {
|
||||
default: assert(0 && "Unknown file type");
|
||||
case TargetMachine::CGFT_AssemblyFile:
|
||||
break;
|
||||
case TargetMachine::CGFT_ObjectFile:
|
||||
case TargetMachine::CGFT_Null:
|
||||
Binary = true;
|
||||
break;
|
||||
}
|
||||
|
||||
// Open the file.
|
||||
std::string error;
|
||||
unsigned OpenFlags = 0;
|
||||
if (Binary) OpenFlags |= raw_fd_ostream::F_Binary;
|
||||
tool_output_file *FDOut = new tool_output_file(OutputFilename.c_str(), error,
|
||||
OpenFlags);
|
||||
if (!error.empty()) {
|
||||
errs() << error << '\n';
|
||||
delete FDOut;
|
||||
return 0;
|
||||
}
|
||||
|
||||
return FDOut;
|
||||
}
|
||||
|
||||
// main - Entry point for the llc compiler.
|
||||
//
|
||||
int main(int argc, char **argv) {
|
||||
sys::PrintStackTraceOnErrorSignal();
|
||||
PrettyStackTraceProgram X(argc, argv);
|
||||
|
||||
// Enable debug stream buffering.
|
||||
EnableDebugBuffering = true;
|
||||
|
||||
LLVMContext &Context = getGlobalContext();
|
||||
llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
|
||||
|
||||
// Initialize targets first, so that --version shows registered targets.
|
||||
InitializeAllTargets();
|
||||
InitializeAllTargetMCs();
|
||||
InitializeAllAsmPrinters();
|
||||
InitializeAllAsmParsers();
|
||||
|
||||
// Register the target printer for --version.
|
||||
cl::AddExtraVersionPrinter(TargetRegistry::printRegisteredTargetsForVersion);
|
||||
|
||||
cl::ParseCommandLineOptions(argc, argv, "llvm system compiler\n");
|
||||
|
||||
// Load the module to be compiled...
|
||||
SMDiagnostic Err;
|
||||
std::auto_ptr<Module> M;
|
||||
|
||||
M.reset(ParseIRFile(InputFilename, Err, Context));
|
||||
if (M.get() == 0) {
|
||||
Err.Print(argv[0], errs());
|
||||
return 1;
|
||||
}
|
||||
Module &mod = *M.get();
|
||||
|
||||
// If we are supposed to override the target triple, do so now.
|
||||
if (!TargetTriple.empty())
|
||||
mod.setTargetTriple(Triple::normalize(TargetTriple));
|
||||
|
||||
Triple TheTriple(mod.getTargetTriple());
|
||||
if (TheTriple.getTriple().empty())
|
||||
TheTriple.setTriple(sys::getHostTriple());
|
||||
|
||||
// Allocate target machine. First, check whether the user has explicitly
|
||||
// specified an architecture to compile for. If so we have to look it up by
|
||||
// name, because it might be a backend that has no mapping to a target triple.
|
||||
const Target *TheTarget = 0;
|
||||
if (!MArch.empty()) {
|
||||
for (TargetRegistry::iterator it = TargetRegistry::begin(),
|
||||
ie = TargetRegistry::end(); it != ie; ++it) {
|
||||
if (MArch == it->getName()) {
|
||||
TheTarget = &*it;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!TheTarget) {
|
||||
errs() << argv[0] << ": error: invalid target '" << MArch << "'.\n";
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Adjust the triple to match (if known), otherwise stick with the
|
||||
// module/host triple.
|
||||
Triple::ArchType Type = Triple::getArchTypeForLLVMName(MArch);
|
||||
if (Type != Triple::UnknownArch)
|
||||
TheTriple.setArch(Type);
|
||||
} else {
|
||||
std::string Err;
|
||||
TheTarget = TargetRegistry::lookupTarget(TheTriple.getTriple(), Err);
|
||||
if (TheTarget == 0) {
|
||||
errs() << argv[0] << ": error auto-selecting target for module '"
|
||||
<< Err << "'. Please use the -march option to explicitly "
|
||||
<< "pick a target.\n";
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
|
||||
// Package up features to be passed to target/subtarget
|
||||
std::string FeaturesStr;
|
||||
if (MAttrs.size()) {
|
||||
SubtargetFeatures Features;
|
||||
for (unsigned i = 0; i != MAttrs.size(); ++i)
|
||||
Features.AddFeature(MAttrs[i]);
|
||||
FeaturesStr = Features.getString();
|
||||
}
|
||||
|
||||
std::auto_ptr<TargetMachine>
|
||||
target(TheTarget->createTargetMachine(TheTriple.getTriple(),
|
||||
MCPU, FeaturesStr,
|
||||
RelocModel, CMModel));
|
||||
assert(target.get() && "Could not allocate target machine!");
|
||||
TargetMachine &Target = *target.get();
|
||||
|
||||
if (DisableDotLoc)
|
||||
Target.setMCUseLoc(false);
|
||||
|
||||
if (DisableCFI)
|
||||
Target.setMCUseCFI(false);
|
||||
|
||||
// Disable .loc support for older OS X versions.
|
||||
if (TheTriple.isMacOSX() &&
|
||||
TheTriple.isMacOSXVersionLT(10, 6))
|
||||
Target.setMCUseLoc(false);
|
||||
|
||||
// Figure out where we are going to send the output...
|
||||
OwningPtr<tool_output_file> Out
|
||||
(GetOutputStream(TheTarget->getName(), TheTriple.getOS(), argv[0]));
|
||||
if (!Out) return 1;
|
||||
|
||||
CodeGenOpt::Level OLvl = CodeGenOpt::Default;
|
||||
switch (OptLevel) {
|
||||
default:
|
||||
errs() << argv[0] << ": invalid optimization level.\n";
|
||||
return 1;
|
||||
case ' ': break;
|
||||
case '0': OLvl = CodeGenOpt::None; break;
|
||||
case '1': OLvl = CodeGenOpt::Less; break;
|
||||
case '2': OLvl = CodeGenOpt::Default; break;
|
||||
case '3': OLvl = CodeGenOpt::Aggressive; break;
|
||||
}
|
||||
|
||||
// Build up all of the passes that we want to do to the module.
|
||||
PassManager PM;
|
||||
|
||||
// Add the target data from the target machine, if it exists, or the module.
|
||||
if (const TargetData *TD = Target.getTargetData())
|
||||
PM.add(new TargetData(*TD));
|
||||
else
|
||||
PM.add(new TargetData(&mod));
|
||||
|
||||
// Override default to generate verbose assembly.
|
||||
Target.setAsmVerbosityDefault(true);
|
||||
|
||||
if (RelaxAll) {
|
||||
if (FileType != TargetMachine::CGFT_ObjectFile)
|
||||
errs() << argv[0]
|
||||
<< ": warning: ignoring -mc-relax-all because filetype != obj";
|
||||
else
|
||||
Target.setMCRelaxAll(true);
|
||||
}
|
||||
|
||||
{
|
||||
formatted_raw_ostream FOS(Out->os());
|
||||
|
||||
// Ask the target to add backend passes as necessary.
|
||||
if (Target.addPassesToEmitFile(PM, FOS, FileType, OLvl, NoVerify)) {
|
||||
errs() << argv[0] << ": target does not support generation of this"
|
||||
<< " file type!\n";
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Before executing passes, print the final values of the LLVM options.
|
||||
cl::PrintOptionValues();
|
||||
|
||||
PM.run(mod);
|
||||
}
|
||||
|
||||
// Declare success.
|
||||
Out->keep();
|
||||
|
||||
return 0;
|
||||
}
|
5
contrib/llvm/tools/lli/CMakeLists.txt
Normal file
5
contrib/llvm/tools/lli/CMakeLists.txt
Normal file
@ -0,0 +1,5 @@
|
||||
set(LLVM_LINK_COMPONENTS mcjit jit interpreter nativecodegen bitreader asmparser selectiondag)
|
||||
|
||||
add_llvm_tool(lli
|
||||
lli.cpp
|
||||
)
|
15
contrib/llvm/tools/lli/Makefile
Normal file
15
contrib/llvm/tools/lli/Makefile
Normal file
@ -0,0 +1,15 @@
|
||||
##===- tools/lli/Makefile ------------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
|
||||
LEVEL := ../..
|
||||
TOOLNAME := lli
|
||||
LINK_COMPONENTS := mcjit jit interpreter nativecodegen bitreader asmparser selectiondag
|
||||
|
||||
# Enable JIT support
|
||||
include $(LEVEL)/Makefile.common
|
305
contrib/llvm/tools/lli/lli.cpp
Normal file
305
contrib/llvm/tools/lli/lli.cpp
Normal file
@ -0,0 +1,305 @@
|
||||
//===- lli.cpp - LLVM Interpreter / Dynamic compiler ----------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This utility provides a simple wrapper around the LLVM Execution Engines,
|
||||
// which allow the direct execution of LLVM programs through a Just-In-Time
|
||||
// compiler, or through an interpreter if no JIT is available for this platform.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/Type.h"
|
||||
#include "llvm/ADT/Triple.h"
|
||||
#include "llvm/Bitcode/ReaderWriter.h"
|
||||
#include "llvm/CodeGen/LinkAllCodegenComponents.h"
|
||||
#include "llvm/ExecutionEngine/GenericValue.h"
|
||||
#include "llvm/ExecutionEngine/Interpreter.h"
|
||||
#include "llvm/ExecutionEngine/JIT.h"
|
||||
#include "llvm/ExecutionEngine/JITEventListener.h"
|
||||
#include "llvm/ExecutionEngine/MCJIT.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/IRReader.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/MemoryBuffer.h"
|
||||
#include "llvm/Support/PluginLoader.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/Process.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include "llvm/Support/TargetSelect.h"
|
||||
#include <cerrno>
|
||||
|
||||
#ifdef __CYGWIN__
|
||||
#include <cygwin/version.h>
|
||||
#if defined(CYGWIN_VERSION_DLL_MAJOR) && CYGWIN_VERSION_DLL_MAJOR<1007
|
||||
#define DO_NOTHING_ATEXIT 1
|
||||
#endif
|
||||
#endif
|
||||
|
||||
using namespace llvm;
|
||||
|
||||
namespace {
|
||||
cl::opt<std::string>
|
||||
InputFile(cl::desc("<input bitcode>"), cl::Positional, cl::init("-"));
|
||||
|
||||
cl::list<std::string>
|
||||
InputArgv(cl::ConsumeAfter, cl::desc("<program arguments>..."));
|
||||
|
||||
cl::opt<bool> ForceInterpreter("force-interpreter",
|
||||
cl::desc("Force interpretation: disable JIT"),
|
||||
cl::init(false));
|
||||
|
||||
cl::opt<bool> UseMCJIT(
|
||||
"use-mcjit", cl::desc("Enable use of the MC-based JIT (if available)"),
|
||||
cl::init(false));
|
||||
|
||||
// Determine optimization level.
|
||||
cl::opt<char>
|
||||
OptLevel("O",
|
||||
cl::desc("Optimization level. [-O0, -O1, -O2, or -O3] "
|
||||
"(default = '-O2')"),
|
||||
cl::Prefix,
|
||||
cl::ZeroOrMore,
|
||||
cl::init(' '));
|
||||
|
||||
cl::opt<std::string>
|
||||
TargetTriple("mtriple", cl::desc("Override target triple for module"));
|
||||
|
||||
cl::opt<std::string>
|
||||
MArch("march",
|
||||
cl::desc("Architecture to generate assembly for (see --version)"));
|
||||
|
||||
cl::opt<std::string>
|
||||
MCPU("mcpu",
|
||||
cl::desc("Target a specific cpu type (-mcpu=help for details)"),
|
||||
cl::value_desc("cpu-name"),
|
||||
cl::init(""));
|
||||
|
||||
cl::list<std::string>
|
||||
MAttrs("mattr",
|
||||
cl::CommaSeparated,
|
||||
cl::desc("Target specific attributes (-mattr=help for details)"),
|
||||
cl::value_desc("a1,+a2,-a3,..."));
|
||||
|
||||
cl::opt<std::string>
|
||||
EntryFunc("entry-function",
|
||||
cl::desc("Specify the entry function (default = 'main') "
|
||||
"of the executable"),
|
||||
cl::value_desc("function"),
|
||||
cl::init("main"));
|
||||
|
||||
cl::opt<std::string>
|
||||
FakeArgv0("fake-argv0",
|
||||
cl::desc("Override the 'argv[0]' value passed into the executing"
|
||||
" program"), cl::value_desc("executable"));
|
||||
|
||||
cl::opt<bool>
|
||||
DisableCoreFiles("disable-core-files", cl::Hidden,
|
||||
cl::desc("Disable emission of core files if possible"));
|
||||
|
||||
cl::opt<bool>
|
||||
NoLazyCompilation("disable-lazy-compilation",
|
||||
cl::desc("Disable JIT lazy compilation"),
|
||||
cl::init(false));
|
||||
|
||||
cl::opt<Reloc::Model>
|
||||
RelocModel("relocation-model",
|
||||
cl::desc("Choose relocation model"),
|
||||
cl::init(Reloc::Default),
|
||||
cl::values(
|
||||
clEnumValN(Reloc::Default, "default",
|
||||
"Target default relocation model"),
|
||||
clEnumValN(Reloc::Static, "static",
|
||||
"Non-relocatable code"),
|
||||
clEnumValN(Reloc::PIC_, "pic",
|
||||
"Fully relocatable, position independent code"),
|
||||
clEnumValN(Reloc::DynamicNoPIC, "dynamic-no-pic",
|
||||
"Relocatable external references, non-relocatable code"),
|
||||
clEnumValEnd));
|
||||
|
||||
cl::opt<llvm::CodeModel::Model>
|
||||
CMModel("code-model",
|
||||
cl::desc("Choose code model"),
|
||||
cl::init(CodeModel::JITDefault),
|
||||
cl::values(clEnumValN(CodeModel::JITDefault, "default",
|
||||
"Target default JIT code model"),
|
||||
clEnumValN(CodeModel::Small, "small",
|
||||
"Small code model"),
|
||||
clEnumValN(CodeModel::Kernel, "kernel",
|
||||
"Kernel code model"),
|
||||
clEnumValN(CodeModel::Medium, "medium",
|
||||
"Medium code model"),
|
||||
clEnumValN(CodeModel::Large, "large",
|
||||
"Large code model"),
|
||||
clEnumValEnd));
|
||||
|
||||
}
|
||||
|
||||
static ExecutionEngine *EE = 0;
|
||||
|
||||
static void do_shutdown() {
|
||||
// Cygwin-1.5 invokes DLL's dtors before atexit handler.
|
||||
#ifndef DO_NOTHING_ATEXIT
|
||||
delete EE;
|
||||
llvm_shutdown();
|
||||
#endif
|
||||
}
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
// main Driver function
|
||||
//
|
||||
int main(int argc, char **argv, char * const *envp) {
|
||||
sys::PrintStackTraceOnErrorSignal();
|
||||
PrettyStackTraceProgram X(argc, argv);
|
||||
|
||||
LLVMContext &Context = getGlobalContext();
|
||||
atexit(do_shutdown); // Call llvm_shutdown() on exit.
|
||||
|
||||
// If we have a native target, initialize it to ensure it is linked in and
|
||||
// usable by the JIT.
|
||||
InitializeNativeTarget();
|
||||
InitializeNativeTargetAsmPrinter();
|
||||
|
||||
cl::ParseCommandLineOptions(argc, argv,
|
||||
"llvm interpreter & dynamic compiler\n");
|
||||
|
||||
// If the user doesn't want core files, disable them.
|
||||
if (DisableCoreFiles)
|
||||
sys::Process::PreventCoreFiles();
|
||||
|
||||
// Load the bitcode...
|
||||
SMDiagnostic Err;
|
||||
Module *Mod = ParseIRFile(InputFile, Err, Context);
|
||||
if (!Mod) {
|
||||
Err.Print(argv[0], errs());
|
||||
return 1;
|
||||
}
|
||||
|
||||
// If not jitting lazily, load the whole bitcode file eagerly too.
|
||||
std::string ErrorMsg;
|
||||
if (NoLazyCompilation) {
|
||||
if (Mod->MaterializeAllPermanently(&ErrorMsg)) {
|
||||
errs() << argv[0] << ": bitcode didn't read correctly.\n";
|
||||
errs() << "Reason: " << ErrorMsg << "\n";
|
||||
exit(1);
|
||||
}
|
||||
}
|
||||
|
||||
EngineBuilder builder(Mod);
|
||||
builder.setMArch(MArch);
|
||||
builder.setMCPU(MCPU);
|
||||
builder.setMAttrs(MAttrs);
|
||||
builder.setRelocationModel(RelocModel);
|
||||
builder.setCodeModel(CMModel);
|
||||
builder.setErrorStr(&ErrorMsg);
|
||||
builder.setEngineKind(ForceInterpreter
|
||||
? EngineKind::Interpreter
|
||||
: EngineKind::JIT);
|
||||
|
||||
// If we are supposed to override the target triple, do so now.
|
||||
if (!TargetTriple.empty())
|
||||
Mod->setTargetTriple(Triple::normalize(TargetTriple));
|
||||
|
||||
// Enable MCJIT, if desired.
|
||||
if (UseMCJIT)
|
||||
builder.setUseMCJIT(true);
|
||||
|
||||
CodeGenOpt::Level OLvl = CodeGenOpt::Default;
|
||||
switch (OptLevel) {
|
||||
default:
|
||||
errs() << argv[0] << ": invalid optimization level.\n";
|
||||
return 1;
|
||||
case ' ': break;
|
||||
case '0': OLvl = CodeGenOpt::None; break;
|
||||
case '1': OLvl = CodeGenOpt::Less; break;
|
||||
case '2': OLvl = CodeGenOpt::Default; break;
|
||||
case '3': OLvl = CodeGenOpt::Aggressive; break;
|
||||
}
|
||||
builder.setOptLevel(OLvl);
|
||||
|
||||
EE = builder.create();
|
||||
if (!EE) {
|
||||
if (!ErrorMsg.empty())
|
||||
errs() << argv[0] << ": error creating EE: " << ErrorMsg << "\n";
|
||||
else
|
||||
errs() << argv[0] << ": unknown error creating EE!\n";
|
||||
exit(1);
|
||||
}
|
||||
|
||||
EE->RegisterJITEventListener(createOProfileJITEventListener());
|
||||
|
||||
EE->DisableLazyCompilation(NoLazyCompilation);
|
||||
|
||||
// If the user specifically requested an argv[0] to pass into the program,
|
||||
// do it now.
|
||||
if (!FakeArgv0.empty()) {
|
||||
InputFile = FakeArgv0;
|
||||
} else {
|
||||
// Otherwise, if there is a .bc suffix on the executable strip it off, it
|
||||
// might confuse the program.
|
||||
if (StringRef(InputFile).endswith(".bc"))
|
||||
InputFile.erase(InputFile.length() - 3);
|
||||
}
|
||||
|
||||
// Add the module's name to the start of the vector of arguments to main().
|
||||
InputArgv.insert(InputArgv.begin(), InputFile);
|
||||
|
||||
// Call the main function from M as if its signature were:
|
||||
// int main (int argc, char **argv, const char **envp)
|
||||
// using the contents of Args to determine argc & argv, and the contents of
|
||||
// EnvVars to determine envp.
|
||||
//
|
||||
Function *EntryFn = Mod->getFunction(EntryFunc);
|
||||
if (!EntryFn) {
|
||||
errs() << '\'' << EntryFunc << "\' function not found in module.\n";
|
||||
return -1;
|
||||
}
|
||||
|
||||
// If the program doesn't explicitly call exit, we will need the Exit
|
||||
// function later on to make an explicit call, so get the function now.
|
||||
Constant *Exit = Mod->getOrInsertFunction("exit", Type::getVoidTy(Context),
|
||||
Type::getInt32Ty(Context),
|
||||
NULL);
|
||||
|
||||
// Reset errno to zero on entry to main.
|
||||
errno = 0;
|
||||
|
||||
// Run static constructors.
|
||||
EE->runStaticConstructorsDestructors(false);
|
||||
|
||||
if (NoLazyCompilation) {
|
||||
for (Module::iterator I = Mod->begin(), E = Mod->end(); I != E; ++I) {
|
||||
Function *Fn = &*I;
|
||||
if (Fn != EntryFn && !Fn->isDeclaration())
|
||||
EE->getPointerToFunction(Fn);
|
||||
}
|
||||
}
|
||||
|
||||
// Run main.
|
||||
int Result = EE->runFunctionAsMain(EntryFn, InputArgv, envp);
|
||||
|
||||
// Run static destructors.
|
||||
EE->runStaticConstructorsDestructors(true);
|
||||
|
||||
// If the program didn't call exit explicitly, we should call it now.
|
||||
// This ensures that any atexit handlers get called correctly.
|
||||
if (Function *ExitF = dyn_cast<Function>(Exit)) {
|
||||
std::vector<GenericValue> Args;
|
||||
GenericValue ResultGV;
|
||||
ResultGV.IntVal = APInt(32, Result);
|
||||
Args.push_back(ResultGV);
|
||||
EE->runFunction(ExitF, Args);
|
||||
errs() << "ERROR: exit(" << Result << ") returned!\n";
|
||||
abort();
|
||||
} else {
|
||||
errs() << "ERROR: exit defined with wrong prototype!\n";
|
||||
abort();
|
||||
}
|
||||
}
|
8
contrib/llvm/tools/llvm-ar/CMakeLists.txt
Normal file
8
contrib/llvm/tools/llvm-ar/CMakeLists.txt
Normal file
@ -0,0 +1,8 @@
|
||||
set(LLVM_LINK_COMPONENTS archive)
|
||||
set(LLVM_REQUIRES_EH 1)
|
||||
|
||||
add_llvm_tool(llvm-ar
|
||||
llvm-ar.cpp
|
||||
)
|
||||
|
||||
# TODO: Support check-local.
|
25
contrib/llvm/tools/llvm-ar/Makefile
Normal file
25
contrib/llvm/tools/llvm-ar/Makefile
Normal file
@ -0,0 +1,25 @@
|
||||
##===- tools/llvm-ar/Makefile ------------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
LEVEL = ../..
|
||||
|
||||
TOOLNAME = llvm-ar
|
||||
LINK_COMPONENTS = archive
|
||||
REQUIRES_EH := 1
|
||||
|
||||
# This tool has no plugins, optimize startup time.
|
||||
TOOL_NO_EXPORTS = 1
|
||||
|
||||
include $(LEVEL)/Makefile.common
|
||||
|
||||
check-local::
|
||||
$(Echo) Checking llvm-ar
|
||||
$(Verb) $(ToolDir)/llvm-ar zRrS nada.a .
|
||||
$(Verb) $(ToolDir)/llvm-ar tv nada.a | \
|
||||
grep Debug/llvm-ar.d >/dev/null 2>&1
|
||||
$(Verb) $(RM) -f nada.a
|
781
contrib/llvm/tools/llvm-ar/llvm-ar.cpp
Normal file
781
contrib/llvm/tools/llvm-ar/llvm-ar.cpp
Normal file
@ -0,0 +1,781 @@
|
||||
//===-- llvm-ar.cpp - LLVM archive librarian utility ----------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// Builds up (relatively) standard unix archive files (.a) containing LLVM
|
||||
// bitcode or other files.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/Bitcode/Archive.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/FileSystem.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/Format.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include <algorithm>
|
||||
#include <memory>
|
||||
#include <fstream>
|
||||
using namespace llvm;
|
||||
|
||||
// Option for compatibility with AIX, not used but must allow it to be present.
|
||||
static cl::opt<bool>
|
||||
X32Option ("X32_64", cl::Hidden,
|
||||
cl::desc("Ignored option for compatibility with AIX"));
|
||||
|
||||
// llvm-ar operation code and modifier flags. This must come first.
|
||||
static cl::opt<std::string>
|
||||
Options(cl::Positional, cl::Required, cl::desc("{operation}[modifiers]..."));
|
||||
|
||||
// llvm-ar remaining positional arguments.
|
||||
static cl::list<std::string>
|
||||
RestOfArgs(cl::Positional, cl::OneOrMore,
|
||||
cl::desc("[relpos] [count] <archive-file> [members]..."));
|
||||
|
||||
// MoreHelp - Provide additional help output explaining the operations and
|
||||
// modifiers of llvm-ar. This object instructs the CommandLine library
|
||||
// to print the text of the constructor when the --help option is given.
|
||||
static cl::extrahelp MoreHelp(
|
||||
"\nOPERATIONS:\n"
|
||||
" d[NsS] - delete file(s) from the archive\n"
|
||||
" m[abiSs] - move file(s) in the archive\n"
|
||||
" p[kN] - print file(s) found in the archive\n"
|
||||
" q[ufsS] - quick append file(s) to the archive\n"
|
||||
" r[abfiuzRsS] - replace or insert file(s) into the archive\n"
|
||||
" t - display contents of archive\n"
|
||||
" x[No] - extract file(s) from the archive\n"
|
||||
"\nMODIFIERS (operation specific):\n"
|
||||
" [a] - put file(s) after [relpos]\n"
|
||||
" [b] - put file(s) before [relpos] (same as [i])\n"
|
||||
" [f] - truncate inserted file names\n"
|
||||
" [i] - put file(s) before [relpos] (same as [b])\n"
|
||||
" [k] - always print bitcode files (default is to skip them)\n"
|
||||
" [N] - use instance [count] of name\n"
|
||||
" [o] - preserve original dates\n"
|
||||
" [P] - use full path names when matching\n"
|
||||
" [R] - recurse through directories when inserting\n"
|
||||
" [s] - create an archive index (cf. ranlib)\n"
|
||||
" [S] - do not build a symbol table\n"
|
||||
" [u] - update only files newer than archive contents\n"
|
||||
" [z] - compress files before inserting/extracting\n"
|
||||
"\nMODIFIERS (generic):\n"
|
||||
" [c] - do not warn if the library had to be created\n"
|
||||
" [v] - be verbose about actions taken\n"
|
||||
" [V] - be *really* verbose about actions taken\n"
|
||||
);
|
||||
|
||||
// This enumeration delineates the kinds of operations on an archive
|
||||
// that are permitted.
|
||||
enum ArchiveOperation {
|
||||
NoOperation, ///< An operation hasn't been specified
|
||||
Print, ///< Print the contents of the archive
|
||||
Delete, ///< Delete the specified members
|
||||
Move, ///< Move members to end or as given by {a,b,i} modifiers
|
||||
QuickAppend, ///< Quickly append to end of archive
|
||||
ReplaceOrInsert, ///< Replace or Insert members
|
||||
DisplayTable, ///< Display the table of contents
|
||||
Extract ///< Extract files back to file system
|
||||
};
|
||||
|
||||
// Modifiers to follow operation to vary behavior
|
||||
bool AddAfter = false; ///< 'a' modifier
|
||||
bool AddBefore = false; ///< 'b' modifier
|
||||
bool Create = false; ///< 'c' modifier
|
||||
bool TruncateNames = false; ///< 'f' modifier
|
||||
bool InsertBefore = false; ///< 'i' modifier
|
||||
bool DontSkipBitcode = false; ///< 'k' modifier
|
||||
bool UseCount = false; ///< 'N' modifier
|
||||
bool OriginalDates = false; ///< 'o' modifier
|
||||
bool FullPath = false; ///< 'P' modifier
|
||||
bool RecurseDirectories = false; ///< 'R' modifier
|
||||
bool SymTable = true; ///< 's' & 'S' modifiers
|
||||
bool OnlyUpdate = false; ///< 'u' modifier
|
||||
bool Verbose = false; ///< 'v' modifier
|
||||
bool ReallyVerbose = false; ///< 'V' modifier
|
||||
bool Compression = false; ///< 'z' modifier
|
||||
|
||||
// Relative Positional Argument (for insert/move). This variable holds
|
||||
// the name of the archive member to which the 'a', 'b' or 'i' modifier
|
||||
// refers. Only one of 'a', 'b' or 'i' can be specified so we only need
|
||||
// one variable.
|
||||
std::string RelPos;
|
||||
|
||||
// Select which of multiple entries in the archive with the same name should be
|
||||
// used (specified with -N) for the delete and extract operations.
|
||||
int Count = 1;
|
||||
|
||||
// This variable holds the name of the archive file as given on the
|
||||
// command line.
|
||||
std::string ArchiveName;
|
||||
|
||||
// This variable holds the list of member files to proecess, as given
|
||||
// on the command line.
|
||||
std::vector<std::string> Members;
|
||||
|
||||
// This variable holds the (possibly expanded) list of path objects that
|
||||
// correspond to files we will
|
||||
std::set<sys::Path> Paths;
|
||||
|
||||
// The Archive object to which all the editing operations will be sent.
|
||||
Archive* TheArchive = 0;
|
||||
|
||||
// getRelPos - Extract the member filename from the command line for
|
||||
// the [relpos] argument associated with a, b, and i modifiers
|
||||
void getRelPos() {
|
||||
if(RestOfArgs.size() > 0) {
|
||||
RelPos = RestOfArgs[0];
|
||||
RestOfArgs.erase(RestOfArgs.begin());
|
||||
}
|
||||
else
|
||||
throw "Expected [relpos] for a, b, or i modifier";
|
||||
}
|
||||
|
||||
// getCount - Extract the [count] argument associated with the N modifier
|
||||
// from the command line and check its value.
|
||||
void getCount() {
|
||||
if(RestOfArgs.size() > 0) {
|
||||
Count = atoi(RestOfArgs[0].c_str());
|
||||
RestOfArgs.erase(RestOfArgs.begin());
|
||||
}
|
||||
else
|
||||
throw "Expected [count] value with N modifier";
|
||||
|
||||
// Non-positive counts are not allowed
|
||||
if (Count < 1)
|
||||
throw "Invalid [count] value (not a positive integer)";
|
||||
}
|
||||
|
||||
// getArchive - Get the archive file name from the command line
|
||||
void getArchive() {
|
||||
if(RestOfArgs.size() > 0) {
|
||||
ArchiveName = RestOfArgs[0];
|
||||
RestOfArgs.erase(RestOfArgs.begin());
|
||||
}
|
||||
else
|
||||
throw "An archive name must be specified.";
|
||||
}
|
||||
|
||||
// getMembers - Copy over remaining items in RestOfArgs to our Members vector
|
||||
// This is just for clarity.
|
||||
void getMembers() {
|
||||
if(RestOfArgs.size() > 0)
|
||||
Members = std::vector<std::string>(RestOfArgs);
|
||||
}
|
||||
|
||||
// parseCommandLine - Parse the command line options as presented and return the
|
||||
// operation specified. Process all modifiers and check to make sure that
|
||||
// constraints on modifier/operation pairs have not been violated.
|
||||
ArchiveOperation parseCommandLine() {
|
||||
|
||||
// Keep track of number of operations. We can only specify one
|
||||
// per execution.
|
||||
unsigned NumOperations = 0;
|
||||
|
||||
// Keep track of the number of positional modifiers (a,b,i). Only
|
||||
// one can be specified.
|
||||
unsigned NumPositional = 0;
|
||||
|
||||
// Keep track of which operation was requested
|
||||
ArchiveOperation Operation = NoOperation;
|
||||
|
||||
for(unsigned i=0; i<Options.size(); ++i) {
|
||||
switch(Options[i]) {
|
||||
case 'd': ++NumOperations; Operation = Delete; break;
|
||||
case 'm': ++NumOperations; Operation = Move ; break;
|
||||
case 'p': ++NumOperations; Operation = Print; break;
|
||||
case 'q': ++NumOperations; Operation = QuickAppend; break;
|
||||
case 'r': ++NumOperations; Operation = ReplaceOrInsert; break;
|
||||
case 't': ++NumOperations; Operation = DisplayTable; break;
|
||||
case 'x': ++NumOperations; Operation = Extract; break;
|
||||
case 'c': Create = true; break;
|
||||
case 'f': TruncateNames = true; break;
|
||||
case 'k': DontSkipBitcode = true; break;
|
||||
case 'l': /* accepted but unused */ break;
|
||||
case 'o': OriginalDates = true; break;
|
||||
case 'P': FullPath = true; break;
|
||||
case 'R': RecurseDirectories = true; break;
|
||||
case 's': SymTable = true; break;
|
||||
case 'S': SymTable = false; break;
|
||||
case 'u': OnlyUpdate = true; break;
|
||||
case 'v': Verbose = true; break;
|
||||
case 'V': Verbose = ReallyVerbose = true; break;
|
||||
case 'z': Compression = true; break;
|
||||
case 'a':
|
||||
getRelPos();
|
||||
AddAfter = true;
|
||||
NumPositional++;
|
||||
break;
|
||||
case 'b':
|
||||
getRelPos();
|
||||
AddBefore = true;
|
||||
NumPositional++;
|
||||
break;
|
||||
case 'i':
|
||||
getRelPos();
|
||||
InsertBefore = true;
|
||||
NumPositional++;
|
||||
break;
|
||||
case 'N':
|
||||
getCount();
|
||||
UseCount = true;
|
||||
break;
|
||||
default:
|
||||
cl::PrintHelpMessage();
|
||||
}
|
||||
}
|
||||
|
||||
// At this point, the next thing on the command line must be
|
||||
// the archive name.
|
||||
getArchive();
|
||||
|
||||
// Everything on the command line at this point is a member.
|
||||
getMembers();
|
||||
|
||||
// Perform various checks on the operation/modifier specification
|
||||
// to make sure we are dealing with a legal request.
|
||||
if (NumOperations == 0)
|
||||
throw "You must specify at least one of the operations";
|
||||
if (NumOperations > 1)
|
||||
throw "Only one operation may be specified";
|
||||
if (NumPositional > 1)
|
||||
throw "You may only specify one of a, b, and i modifiers";
|
||||
if (AddAfter || AddBefore || InsertBefore)
|
||||
if (Operation != Move && Operation != ReplaceOrInsert)
|
||||
throw "The 'a', 'b' and 'i' modifiers can only be specified with "
|
||||
"the 'm' or 'r' operations";
|
||||
if (RecurseDirectories && Operation != ReplaceOrInsert)
|
||||
throw "The 'R' modifiers is only applicabe to the 'r' operation";
|
||||
if (OriginalDates && Operation != Extract)
|
||||
throw "The 'o' modifier is only applicable to the 'x' operation";
|
||||
if (TruncateNames && Operation!=QuickAppend && Operation!=ReplaceOrInsert)
|
||||
throw "The 'f' modifier is only applicable to the 'q' and 'r' operations";
|
||||
if (OnlyUpdate && Operation != ReplaceOrInsert)
|
||||
throw "The 'u' modifier is only applicable to the 'r' operation";
|
||||
if (Compression && Operation!=ReplaceOrInsert && Operation!=Extract)
|
||||
throw "The 'z' modifier is only applicable to the 'r' and 'x' operations";
|
||||
if (Count > 1 && Members.size() > 1)
|
||||
throw "Only one member name may be specified with the 'N' modifier";
|
||||
|
||||
// Return the parsed operation to the caller
|
||||
return Operation;
|
||||
}
|
||||
|
||||
// recurseDirectories - Implements the "R" modifier. This function scans through
|
||||
// the Paths vector (built by buildPaths, below) and replaces any directories it
|
||||
// finds with all the files in that directory (recursively). It uses the
|
||||
// sys::Path::getDirectoryContent method to perform the actual directory scans.
|
||||
bool
|
||||
recurseDirectories(const sys::Path& path,
|
||||
std::set<sys::Path>& result, std::string* ErrMsg) {
|
||||
result.clear();
|
||||
if (RecurseDirectories) {
|
||||
std::set<sys::Path> content;
|
||||
if (path.getDirectoryContents(content, ErrMsg))
|
||||
return true;
|
||||
|
||||
for (std::set<sys::Path>::iterator I = content.begin(), E = content.end();
|
||||
I != E; ++I) {
|
||||
// Make sure it exists and is a directory
|
||||
sys::PathWithStatus PwS(*I);
|
||||
const sys::FileStatus *Status = PwS.getFileStatus(false, ErrMsg);
|
||||
if (!Status)
|
||||
return true;
|
||||
if (Status->isDir) {
|
||||
std::set<sys::Path> moreResults;
|
||||
if (recurseDirectories(*I, moreResults, ErrMsg))
|
||||
return true;
|
||||
result.insert(moreResults.begin(), moreResults.end());
|
||||
} else {
|
||||
result.insert(*I);
|
||||
}
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// buildPaths - Convert the strings in the Members vector to sys::Path objects
|
||||
// and make sure they are valid and exist exist. This check is only needed for
|
||||
// the operations that add/replace files to the archive ('q' and 'r')
|
||||
bool buildPaths(bool checkExistence, std::string* ErrMsg) {
|
||||
for (unsigned i = 0; i < Members.size(); i++) {
|
||||
sys::Path aPath;
|
||||
if (!aPath.set(Members[i]))
|
||||
throw std::string("File member name invalid: ") + Members[i];
|
||||
if (checkExistence) {
|
||||
bool Exists;
|
||||
if (sys::fs::exists(aPath.str(), Exists) || !Exists)
|
||||
throw std::string("File does not exist: ") + Members[i];
|
||||
std::string Err;
|
||||
sys::PathWithStatus PwS(aPath);
|
||||
const sys::FileStatus *si = PwS.getFileStatus(false, &Err);
|
||||
if (!si)
|
||||
throw Err;
|
||||
if (si->isDir) {
|
||||
std::set<sys::Path> dirpaths;
|
||||
if (recurseDirectories(aPath, dirpaths, ErrMsg))
|
||||
return true;
|
||||
Paths.insert(dirpaths.begin(),dirpaths.end());
|
||||
} else {
|
||||
Paths.insert(aPath);
|
||||
}
|
||||
} else {
|
||||
Paths.insert(aPath);
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// printSymbolTable - print out the archive's symbol table.
|
||||
void printSymbolTable() {
|
||||
outs() << "\nArchive Symbol Table:\n";
|
||||
const Archive::SymTabType& symtab = TheArchive->getSymbolTable();
|
||||
for (Archive::SymTabType::const_iterator I=symtab.begin(), E=symtab.end();
|
||||
I != E; ++I ) {
|
||||
unsigned offset = TheArchive->getFirstFileOffset() + I->second;
|
||||
outs() << " " << format("%9u", offset) << "\t" << I->first <<"\n";
|
||||
}
|
||||
}
|
||||
|
||||
// doPrint - Implements the 'p' operation. This function traverses the archive
|
||||
// looking for members that match the path list. It is careful to uncompress
|
||||
// things that should be and to skip bitcode files unless the 'k' modifier was
|
||||
// given.
|
||||
bool doPrint(std::string* ErrMsg) {
|
||||
if (buildPaths(false, ErrMsg))
|
||||
return true;
|
||||
unsigned countDown = Count;
|
||||
for (Archive::iterator I = TheArchive->begin(), E = TheArchive->end();
|
||||
I != E; ++I ) {
|
||||
if (Paths.empty() ||
|
||||
(std::find(Paths.begin(), Paths.end(), I->getPath()) != Paths.end())) {
|
||||
if (countDown == 1) {
|
||||
const char* data = reinterpret_cast<const char*>(I->getData());
|
||||
|
||||
// Skip things that don't make sense to print
|
||||
if (I->isLLVMSymbolTable() || I->isSVR4SymbolTable() ||
|
||||
I->isBSD4SymbolTable() || (!DontSkipBitcode && I->isBitcode()))
|
||||
continue;
|
||||
|
||||
if (Verbose)
|
||||
outs() << "Printing " << I->getPath().str() << "\n";
|
||||
|
||||
unsigned len = I->getSize();
|
||||
outs().write(data, len);
|
||||
} else {
|
||||
countDown--;
|
||||
}
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// putMode - utility function for printing out the file mode when the 't'
|
||||
// operation is in verbose mode.
|
||||
void
|
||||
printMode(unsigned mode) {
|
||||
if (mode & 004)
|
||||
outs() << "r";
|
||||
else
|
||||
outs() << "-";
|
||||
if (mode & 002)
|
||||
outs() << "w";
|
||||
else
|
||||
outs() << "-";
|
||||
if (mode & 001)
|
||||
outs() << "x";
|
||||
else
|
||||
outs() << "-";
|
||||
}
|
||||
|
||||
// doDisplayTable - Implement the 't' operation. This function prints out just
|
||||
// the file names of each of the members. However, if verbose mode is requested
|
||||
// ('v' modifier) then the file type, permission mode, user, group, size, and
|
||||
// modification time are also printed.
|
||||
bool
|
||||
doDisplayTable(std::string* ErrMsg) {
|
||||
if (buildPaths(false, ErrMsg))
|
||||
return true;
|
||||
for (Archive::iterator I = TheArchive->begin(), E = TheArchive->end();
|
||||
I != E; ++I ) {
|
||||
if (Paths.empty() ||
|
||||
(std::find(Paths.begin(), Paths.end(), I->getPath()) != Paths.end())) {
|
||||
if (Verbose) {
|
||||
// FIXME: Output should be this format:
|
||||
// Zrw-r--r-- 500/ 500 525 Nov 8 17:42 2004 Makefile
|
||||
if (I->isBitcode())
|
||||
outs() << "b";
|
||||
else if (I->isCompressed())
|
||||
outs() << "Z";
|
||||
else
|
||||
outs() << " ";
|
||||
unsigned mode = I->getMode();
|
||||
printMode((mode >> 6) & 007);
|
||||
printMode((mode >> 3) & 007);
|
||||
printMode(mode & 007);
|
||||
outs() << " " << format("%4u", I->getUser());
|
||||
outs() << "/" << format("%4u", I->getGroup());
|
||||
outs() << " " << format("%8u", I->getSize());
|
||||
outs() << " " << format("%20s", I->getModTime().str().substr(4).c_str());
|
||||
outs() << " " << I->getPath().str() << "\n";
|
||||
} else {
|
||||
outs() << I->getPath().str() << "\n";
|
||||
}
|
||||
}
|
||||
}
|
||||
if (ReallyVerbose)
|
||||
printSymbolTable();
|
||||
return false;
|
||||
}
|
||||
|
||||
// doExtract - Implement the 'x' operation. This function extracts files back to
|
||||
// the file system, making sure to uncompress any that were compressed
|
||||
bool
|
||||
doExtract(std::string* ErrMsg) {
|
||||
if (buildPaths(false, ErrMsg))
|
||||
return true;
|
||||
for (Archive::iterator I = TheArchive->begin(), E = TheArchive->end();
|
||||
I != E; ++I ) {
|
||||
if (Paths.empty() ||
|
||||
(std::find(Paths.begin(), Paths.end(), I->getPath()) != Paths.end())) {
|
||||
|
||||
// Make sure the intervening directories are created
|
||||
if (I->hasPath()) {
|
||||
sys::Path dirs(I->getPath());
|
||||
dirs.eraseComponent();
|
||||
if (dirs.createDirectoryOnDisk(/*create_parents=*/true, ErrMsg))
|
||||
return true;
|
||||
}
|
||||
|
||||
// Open up a file stream for writing
|
||||
std::ios::openmode io_mode = std::ios::out | std::ios::trunc |
|
||||
std::ios::binary;
|
||||
std::ofstream file(I->getPath().c_str(), io_mode);
|
||||
|
||||
// Get the data and its length
|
||||
const char* data = reinterpret_cast<const char*>(I->getData());
|
||||
unsigned len = I->getSize();
|
||||
|
||||
// Write the data.
|
||||
file.write(data,len);
|
||||
file.close();
|
||||
|
||||
// If we're supposed to retain the original modification times, etc. do so
|
||||
// now.
|
||||
if (OriginalDates)
|
||||
I->getPath().setStatusInfoOnDisk(I->getFileStatus());
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// doDelete - Implement the delete operation. This function deletes zero or more
|
||||
// members from the archive. Note that if the count is specified, there should
|
||||
// be no more than one path in the Paths list or else this algorithm breaks.
|
||||
// That check is enforced in parseCommandLine (above).
|
||||
bool
|
||||
doDelete(std::string* ErrMsg) {
|
||||
if (buildPaths(false, ErrMsg))
|
||||
return true;
|
||||
if (Paths.empty())
|
||||
return false;
|
||||
unsigned countDown = Count;
|
||||
for (Archive::iterator I = TheArchive->begin(), E = TheArchive->end();
|
||||
I != E; ) {
|
||||
if (std::find(Paths.begin(), Paths.end(), I->getPath()) != Paths.end()) {
|
||||
if (countDown == 1) {
|
||||
Archive::iterator J = I;
|
||||
++I;
|
||||
TheArchive->erase(J);
|
||||
} else
|
||||
countDown--;
|
||||
} else {
|
||||
++I;
|
||||
}
|
||||
}
|
||||
|
||||
// We're done editting, reconstruct the archive.
|
||||
if (TheArchive->writeToDisk(SymTable,TruncateNames,Compression,ErrMsg))
|
||||
return true;
|
||||
if (ReallyVerbose)
|
||||
printSymbolTable();
|
||||
return false;
|
||||
}
|
||||
|
||||
// doMore - Implement the move operation. This function re-arranges just the
|
||||
// order of the archive members so that when the archive is written the move
|
||||
// of the members is accomplished. Note the use of the RelPos variable to
|
||||
// determine where the items should be moved to.
|
||||
bool
|
||||
doMove(std::string* ErrMsg) {
|
||||
if (buildPaths(false, ErrMsg))
|
||||
return true;
|
||||
|
||||
// By default and convention the place to move members to is the end of the
|
||||
// archive.
|
||||
Archive::iterator moveto_spot = TheArchive->end();
|
||||
|
||||
// However, if the relative positioning modifiers were used, we need to scan
|
||||
// the archive to find the member in question. If we don't find it, its no
|
||||
// crime, we just move to the end.
|
||||
if (AddBefore || InsertBefore || AddAfter) {
|
||||
for (Archive::iterator I = TheArchive->begin(), E= TheArchive->end();
|
||||
I != E; ++I ) {
|
||||
if (RelPos == I->getPath().str()) {
|
||||
if (AddAfter) {
|
||||
moveto_spot = I;
|
||||
moveto_spot++;
|
||||
} else {
|
||||
moveto_spot = I;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Keep a list of the paths remaining to be moved
|
||||
std::set<sys::Path> remaining(Paths);
|
||||
|
||||
// Scan the archive again, this time looking for the members to move to the
|
||||
// moveto_spot.
|
||||
for (Archive::iterator I = TheArchive->begin(), E= TheArchive->end();
|
||||
I != E && !remaining.empty(); ++I ) {
|
||||
std::set<sys::Path>::iterator found =
|
||||
std::find(remaining.begin(),remaining.end(),I->getPath());
|
||||
if (found != remaining.end()) {
|
||||
if (I != moveto_spot)
|
||||
TheArchive->splice(moveto_spot,*TheArchive,I);
|
||||
remaining.erase(found);
|
||||
}
|
||||
}
|
||||
|
||||
// We're done editting, reconstruct the archive.
|
||||
if (TheArchive->writeToDisk(SymTable,TruncateNames,Compression,ErrMsg))
|
||||
return true;
|
||||
if (ReallyVerbose)
|
||||
printSymbolTable();
|
||||
return false;
|
||||
}
|
||||
|
||||
// doQuickAppend - Implements the 'q' operation. This function just
|
||||
// indiscriminantly adds the members to the archive and rebuilds it.
|
||||
bool
|
||||
doQuickAppend(std::string* ErrMsg) {
|
||||
// Get the list of paths to append.
|
||||
if (buildPaths(true, ErrMsg))
|
||||
return true;
|
||||
if (Paths.empty())
|
||||
return false;
|
||||
|
||||
// Append them quickly.
|
||||
for (std::set<sys::Path>::iterator PI = Paths.begin(), PE = Paths.end();
|
||||
PI != PE; ++PI) {
|
||||
if (TheArchive->addFileBefore(*PI,TheArchive->end(),ErrMsg))
|
||||
return true;
|
||||
}
|
||||
|
||||
// We're done editting, reconstruct the archive.
|
||||
if (TheArchive->writeToDisk(SymTable,TruncateNames,Compression,ErrMsg))
|
||||
return true;
|
||||
if (ReallyVerbose)
|
||||
printSymbolTable();
|
||||
return false;
|
||||
}
|
||||
|
||||
// doReplaceOrInsert - Implements the 'r' operation. This function will replace
|
||||
// any existing files or insert new ones into the archive.
|
||||
bool
|
||||
doReplaceOrInsert(std::string* ErrMsg) {
|
||||
|
||||
// Build the list of files to be added/replaced.
|
||||
if (buildPaths(true, ErrMsg))
|
||||
return true;
|
||||
if (Paths.empty())
|
||||
return false;
|
||||
|
||||
// Keep track of the paths that remain to be inserted.
|
||||
std::set<sys::Path> remaining(Paths);
|
||||
|
||||
// Default the insertion spot to the end of the archive
|
||||
Archive::iterator insert_spot = TheArchive->end();
|
||||
|
||||
// Iterate over the archive contents
|
||||
for (Archive::iterator I = TheArchive->begin(), E = TheArchive->end();
|
||||
I != E && !remaining.empty(); ++I ) {
|
||||
|
||||
// Determine if this archive member matches one of the paths we're trying
|
||||
// to replace.
|
||||
|
||||
std::set<sys::Path>::iterator found = remaining.end();
|
||||
for (std::set<sys::Path>::iterator RI = remaining.begin(),
|
||||
RE = remaining.end(); RI != RE; ++RI ) {
|
||||
std::string compare(RI->str());
|
||||
if (TruncateNames && compare.length() > 15) {
|
||||
const char* nm = compare.c_str();
|
||||
unsigned len = compare.length();
|
||||
size_t slashpos = compare.rfind('/');
|
||||
if (slashpos != std::string::npos) {
|
||||
nm += slashpos + 1;
|
||||
len -= slashpos +1;
|
||||
}
|
||||
if (len > 15)
|
||||
len = 15;
|
||||
compare.assign(nm,len);
|
||||
}
|
||||
if (compare == I->getPath().str()) {
|
||||
found = RI;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (found != remaining.end()) {
|
||||
std::string Err;
|
||||
sys::PathWithStatus PwS(*found);
|
||||
const sys::FileStatus *si = PwS.getFileStatus(false, &Err);
|
||||
if (!si)
|
||||
return true;
|
||||
if (!si->isDir) {
|
||||
if (OnlyUpdate) {
|
||||
// Replace the item only if it is newer.
|
||||
if (si->modTime > I->getModTime())
|
||||
if (I->replaceWith(*found, ErrMsg))
|
||||
return true;
|
||||
} else {
|
||||
// Replace the item regardless of time stamp
|
||||
if (I->replaceWith(*found, ErrMsg))
|
||||
return true;
|
||||
}
|
||||
} else {
|
||||
// We purposefully ignore directories.
|
||||
}
|
||||
|
||||
// Remove it from our "to do" list
|
||||
remaining.erase(found);
|
||||
}
|
||||
|
||||
// Determine if this is the place where we should insert
|
||||
if ((AddBefore || InsertBefore) && RelPos == I->getPath().str())
|
||||
insert_spot = I;
|
||||
else if (AddAfter && RelPos == I->getPath().str()) {
|
||||
insert_spot = I;
|
||||
insert_spot++;
|
||||
}
|
||||
}
|
||||
|
||||
// If we didn't replace all the members, some will remain and need to be
|
||||
// inserted at the previously computed insert-spot.
|
||||
if (!remaining.empty()) {
|
||||
for (std::set<sys::Path>::iterator PI = remaining.begin(),
|
||||
PE = remaining.end(); PI != PE; ++PI) {
|
||||
if (TheArchive->addFileBefore(*PI,insert_spot, ErrMsg))
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
// We're done editting, reconstruct the archive.
|
||||
if (TheArchive->writeToDisk(SymTable,TruncateNames,Compression,ErrMsg))
|
||||
return true;
|
||||
if (ReallyVerbose)
|
||||
printSymbolTable();
|
||||
return false;
|
||||
}
|
||||
|
||||
// main - main program for llvm-ar .. see comments in the code
|
||||
int main(int argc, char **argv) {
|
||||
// Print a stack trace if we signal out.
|
||||
sys::PrintStackTraceOnErrorSignal();
|
||||
PrettyStackTraceProgram X(argc, argv);
|
||||
LLVMContext &Context = getGlobalContext();
|
||||
llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
|
||||
|
||||
// Have the command line options parsed and handle things
|
||||
// like --help and --version.
|
||||
cl::ParseCommandLineOptions(argc, argv,
|
||||
"LLVM Archiver (llvm-ar)\n\n"
|
||||
" This program archives bitcode files into single libraries\n"
|
||||
);
|
||||
|
||||
int exitCode = 0;
|
||||
|
||||
// Make sure we don't exit with "unhandled exception".
|
||||
try {
|
||||
// Do our own parsing of the command line because the CommandLine utility
|
||||
// can't handle the grouped positional parameters without a dash.
|
||||
ArchiveOperation Operation = parseCommandLine();
|
||||
|
||||
// Check the path name of the archive
|
||||
sys::Path ArchivePath;
|
||||
if (!ArchivePath.set(ArchiveName))
|
||||
throw std::string("Archive name invalid: ") + ArchiveName;
|
||||
|
||||
// Create or open the archive object.
|
||||
bool Exists;
|
||||
if (llvm::sys::fs::exists(ArchivePath.str(), Exists) || !Exists) {
|
||||
// Produce a warning if we should and we're creating the archive
|
||||
if (!Create)
|
||||
errs() << argv[0] << ": creating " << ArchivePath.str() << "\n";
|
||||
TheArchive = Archive::CreateEmpty(ArchivePath, Context);
|
||||
TheArchive->writeToDisk();
|
||||
} else {
|
||||
std::string Error;
|
||||
TheArchive = Archive::OpenAndLoad(ArchivePath, Context, &Error);
|
||||
if (TheArchive == 0) {
|
||||
errs() << argv[0] << ": error loading '" << ArchivePath.str() << "': "
|
||||
<< Error << "!\n";
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
|
||||
// Make sure we're not fooling ourselves.
|
||||
assert(TheArchive && "Unable to instantiate the archive");
|
||||
|
||||
// Make sure we clean up the archive even on failure.
|
||||
std::auto_ptr<Archive> AutoArchive(TheArchive);
|
||||
|
||||
// Perform the operation
|
||||
std::string ErrMsg;
|
||||
bool haveError = false;
|
||||
switch (Operation) {
|
||||
case Print: haveError = doPrint(&ErrMsg); break;
|
||||
case Delete: haveError = doDelete(&ErrMsg); break;
|
||||
case Move: haveError = doMove(&ErrMsg); break;
|
||||
case QuickAppend: haveError = doQuickAppend(&ErrMsg); break;
|
||||
case ReplaceOrInsert: haveError = doReplaceOrInsert(&ErrMsg); break;
|
||||
case DisplayTable: haveError = doDisplayTable(&ErrMsg); break;
|
||||
case Extract: haveError = doExtract(&ErrMsg); break;
|
||||
case NoOperation:
|
||||
errs() << argv[0] << ": No operation was selected.\n";
|
||||
break;
|
||||
}
|
||||
if (haveError) {
|
||||
errs() << argv[0] << ": " << ErrMsg << "\n";
|
||||
return 1;
|
||||
}
|
||||
} catch (const char*msg) {
|
||||
// These errors are usage errors, thrown only by the various checks in the
|
||||
// code above.
|
||||
errs() << argv[0] << ": " << msg << "\n\n";
|
||||
cl::PrintHelpMessage();
|
||||
exitCode = 1;
|
||||
} catch (const std::string& msg) {
|
||||
// These errors are thrown by LLVM libraries (e.g. lib System) and represent
|
||||
// a more serious error so we bump the exitCode and don't print the usage.
|
||||
errs() << argv[0] << ": " << msg << "\n";
|
||||
exitCode = 2;
|
||||
} catch (...) {
|
||||
// This really shouldn't happen, but just in case ....
|
||||
errs() << argv[0] << ": An unexpected unknown exception occurred.\n";
|
||||
exitCode = 3;
|
||||
}
|
||||
|
||||
// Return result code back to operating system.
|
||||
return exitCode;
|
||||
}
|
6
contrib/llvm/tools/llvm-as/CMakeLists.txt
Normal file
6
contrib/llvm/tools/llvm-as/CMakeLists.txt
Normal file
@ -0,0 +1,6 @@
|
||||
set(LLVM_LINK_COMPONENTS asmparser bitwriter)
|
||||
set(LLVM_REQUIRES_EH 1)
|
||||
|
||||
add_llvm_tool(llvm-as
|
||||
llvm-as.cpp
|
||||
)
|
17
contrib/llvm/tools/llvm-as/Makefile
Normal file
17
contrib/llvm/tools/llvm-as/Makefile
Normal file
@ -0,0 +1,17 @@
|
||||
##===- tools/llvm-as/Makefile ------------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
|
||||
LEVEL = ../..
|
||||
TOOLNAME = llvm-as
|
||||
LINK_COMPONENTS := asmparser bitwriter
|
||||
|
||||
# This tool has no plugins, optimize startup time.
|
||||
TOOL_NO_EXPORTS = 1
|
||||
|
||||
include $(LEVEL)/Makefile.common
|
119
contrib/llvm/tools/llvm-as/llvm-as.cpp
Normal file
119
contrib/llvm/tools/llvm-as/llvm-as.cpp
Normal file
@ -0,0 +1,119 @@
|
||||
//===--- llvm-as.cpp - The low-level LLVM assembler -----------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This utility may be invoked in the following manner:
|
||||
// llvm-as --help - Output information about command line switches
|
||||
// llvm-as [options] - Read LLVM asm from stdin, write bitcode to stdout
|
||||
// llvm-as [options] x.ll - Read LLVM asm from the x.ll file, write bitcode
|
||||
// to the x.bc file.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/Assembly/Parser.h"
|
||||
#include "llvm/Analysis/Verifier.h"
|
||||
#include "llvm/Bitcode/ReaderWriter.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/SourceMgr.h"
|
||||
#include "llvm/Support/SystemUtils.h"
|
||||
#include "llvm/Support/ToolOutputFile.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include <memory>
|
||||
using namespace llvm;
|
||||
|
||||
static cl::opt<std::string>
|
||||
InputFilename(cl::Positional, cl::desc("<input .llvm file>"), cl::init("-"));
|
||||
|
||||
static cl::opt<std::string>
|
||||
OutputFilename("o", cl::desc("Override output filename"),
|
||||
cl::value_desc("filename"));
|
||||
|
||||
static cl::opt<bool>
|
||||
Force("f", cl::desc("Enable binary output on terminals"));
|
||||
|
||||
static cl::opt<bool>
|
||||
DisableOutput("disable-output", cl::desc("Disable output"), cl::init(false));
|
||||
|
||||
static cl::opt<bool>
|
||||
DumpAsm("d", cl::desc("Print assembly as parsed"), cl::Hidden);
|
||||
|
||||
static cl::opt<bool>
|
||||
DisableVerify("disable-verify", cl::Hidden,
|
||||
cl::desc("Do not run verifier on input LLVM (dangerous!)"));
|
||||
|
||||
static void WriteOutputFile(const Module *M) {
|
||||
// Infer the output filename if needed.
|
||||
if (OutputFilename.empty()) {
|
||||
if (InputFilename == "-") {
|
||||
OutputFilename = "-";
|
||||
} else {
|
||||
std::string IFN = InputFilename;
|
||||
int Len = IFN.length();
|
||||
if (IFN[Len-3] == '.' && IFN[Len-2] == 'l' && IFN[Len-1] == 'l') {
|
||||
// Source ends in .ll
|
||||
OutputFilename = std::string(IFN.begin(), IFN.end()-3);
|
||||
} else {
|
||||
OutputFilename = IFN; // Append a .bc to it
|
||||
}
|
||||
OutputFilename += ".bc";
|
||||
}
|
||||
}
|
||||
|
||||
std::string ErrorInfo;
|
||||
OwningPtr<tool_output_file> Out
|
||||
(new tool_output_file(OutputFilename.c_str(), ErrorInfo,
|
||||
raw_fd_ostream::F_Binary));
|
||||
if (!ErrorInfo.empty()) {
|
||||
errs() << ErrorInfo << '\n';
|
||||
exit(1);
|
||||
}
|
||||
|
||||
if (Force || !CheckBitcodeOutputToConsole(Out->os(), true))
|
||||
WriteBitcodeToFile(M, Out->os());
|
||||
|
||||
// Declare success.
|
||||
Out->keep();
|
||||
}
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
// Print a stack trace if we signal out.
|
||||
sys::PrintStackTraceOnErrorSignal();
|
||||
PrettyStackTraceProgram X(argc, argv);
|
||||
LLVMContext &Context = getGlobalContext();
|
||||
llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
|
||||
cl::ParseCommandLineOptions(argc, argv, "llvm .ll -> .bc assembler\n");
|
||||
|
||||
// Parse the file now...
|
||||
SMDiagnostic Err;
|
||||
std::auto_ptr<Module> M(ParseAssemblyFile(InputFilename, Err, Context));
|
||||
if (M.get() == 0) {
|
||||
Err.Print(argv[0], errs());
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (!DisableVerify) {
|
||||
std::string Err;
|
||||
if (verifyModule(*M.get(), ReturnStatusAction, &Err)) {
|
||||
errs() << argv[0]
|
||||
<< ": assembly parsed, but does not verify as correct!\n";
|
||||
errs() << Err;
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
|
||||
if (DumpAsm) errs() << "Here's the assembly:\n" << *M.get();
|
||||
|
||||
if (!DisableOutput)
|
||||
WriteOutputFile(M.get());
|
||||
|
||||
return 0;
|
||||
}
|
6
contrib/llvm/tools/llvm-bcanalyzer/CMakeLists.txt
Normal file
6
contrib/llvm/tools/llvm-bcanalyzer/CMakeLists.txt
Normal file
@ -0,0 +1,6 @@
|
||||
set(LLVM_LINK_COMPONENTS bitreader)
|
||||
set(LLVM_REQUIRES_EH 1)
|
||||
|
||||
add_llvm_tool(llvm-bcanalyzer
|
||||
llvm-bcanalyzer.cpp
|
||||
)
|
17
contrib/llvm/tools/llvm-bcanalyzer/Makefile
Normal file
17
contrib/llvm/tools/llvm-bcanalyzer/Makefile
Normal file
@ -0,0 +1,17 @@
|
||||
##===- tools/llvm-bcanalyzer/Makefile ----------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
LEVEL = ../..
|
||||
|
||||
TOOLNAME = llvm-bcanalyzer
|
||||
LINK_COMPONENTS := bitreader
|
||||
|
||||
# This tool has no plugins, optimize startup time.
|
||||
TOOL_NO_EXPORTS = 1
|
||||
|
||||
include $(LEVEL)/Makefile.common
|
629
contrib/llvm/tools/llvm-bcanalyzer/llvm-bcanalyzer.cpp
Normal file
629
contrib/llvm/tools/llvm-bcanalyzer/llvm-bcanalyzer.cpp
Normal file
@ -0,0 +1,629 @@
|
||||
//===-- llvm-bcanalyzer.cpp - Bitcode Analyzer --------------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This tool may be invoked in the following manner:
|
||||
// llvm-bcanalyzer [options] - Read LLVM bitcode from stdin
|
||||
// llvm-bcanalyzer [options] x.bc - Read LLVM bitcode from the x.bc file
|
||||
//
|
||||
// Options:
|
||||
// --help - Output information about command line switches
|
||||
// --dump - Dump low-level bitcode structure in readable format
|
||||
//
|
||||
// This tool provides analytical information about a bitcode file. It is
|
||||
// intended as an aid to developers of bitcode reading and writing software. It
|
||||
// produces on std::out a summary of the bitcode file that shows various
|
||||
// statistics about the contents of the file. By default this information is
|
||||
// detailed and contains information about individual bitcode blocks and the
|
||||
// functions in the module.
|
||||
// The tool is also able to print a bitcode file in a straight forward text
|
||||
// format that shows the containment and relationships of the information in
|
||||
// the bitcode file (-dump option).
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/ADT/OwningPtr.h"
|
||||
#include "llvm/Analysis/Verifier.h"
|
||||
#include "llvm/Bitcode/BitstreamReader.h"
|
||||
#include "llvm/Bitcode/LLVMBitCodes.h"
|
||||
#include "llvm/Bitcode/ReaderWriter.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/Format.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/MemoryBuffer.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include "llvm/Support/system_error.h"
|
||||
#include <cstdio>
|
||||
#include <map>
|
||||
#include <algorithm>
|
||||
using namespace llvm;
|
||||
|
||||
static cl::opt<std::string>
|
||||
InputFilename(cl::Positional, cl::desc("<input bitcode>"), cl::init("-"));
|
||||
|
||||
static cl::opt<bool> Dump("dump", cl::desc("Dump low level bitcode trace"));
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
// Bitcode specific analysis.
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
static cl::opt<bool> NoHistogram("disable-histogram",
|
||||
cl::desc("Do not print per-code histogram"));
|
||||
|
||||
static cl::opt<bool>
|
||||
NonSymbolic("non-symbolic",
|
||||
cl::desc("Emit numeric info in dump even if"
|
||||
" symbolic info is available"));
|
||||
|
||||
namespace {
|
||||
|
||||
/// CurStreamTypeType - A type for CurStreamType
|
||||
enum CurStreamTypeType {
|
||||
UnknownBitstream,
|
||||
LLVMIRBitstream
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
/// CurStreamType - If we can sniff the flavor of this stream, we can produce
|
||||
/// better dump info.
|
||||
static CurStreamTypeType CurStreamType;
|
||||
|
||||
|
||||
/// GetBlockName - Return a symbolic block name if known, otherwise return
|
||||
/// null.
|
||||
static const char *GetBlockName(unsigned BlockID,
|
||||
const BitstreamReader &StreamFile) {
|
||||
// Standard blocks for all bitcode files.
|
||||
if (BlockID < bitc::FIRST_APPLICATION_BLOCKID) {
|
||||
if (BlockID == bitc::BLOCKINFO_BLOCK_ID)
|
||||
return "BLOCKINFO_BLOCK";
|
||||
return 0;
|
||||
}
|
||||
|
||||
// Check to see if we have a blockinfo record for this block, with a name.
|
||||
if (const BitstreamReader::BlockInfo *Info =
|
||||
StreamFile.getBlockInfo(BlockID)) {
|
||||
if (!Info->Name.empty())
|
||||
return Info->Name.c_str();
|
||||
}
|
||||
|
||||
|
||||
if (CurStreamType != LLVMIRBitstream) return 0;
|
||||
|
||||
switch (BlockID) {
|
||||
default: return 0;
|
||||
case bitc::MODULE_BLOCK_ID: return "MODULE_BLOCK";
|
||||
case bitc::PARAMATTR_BLOCK_ID: return "PARAMATTR_BLOCK";
|
||||
case bitc::TYPE_BLOCK_ID_OLD: return "TYPE_BLOCK_ID_OLD";
|
||||
case bitc::TYPE_BLOCK_ID_NEW: return "TYPE_BLOCK_ID";
|
||||
case bitc::CONSTANTS_BLOCK_ID: return "CONSTANTS_BLOCK";
|
||||
case bitc::FUNCTION_BLOCK_ID: return "FUNCTION_BLOCK";
|
||||
case bitc::TYPE_SYMTAB_BLOCK_ID_OLD: return "TYPE_SYMTAB_OLD";
|
||||
case bitc::VALUE_SYMTAB_BLOCK_ID: return "VALUE_SYMTAB";
|
||||
case bitc::METADATA_BLOCK_ID: return "METADATA_BLOCK";
|
||||
case bitc::METADATA_ATTACHMENT_ID: return "METADATA_ATTACHMENT_BLOCK";
|
||||
}
|
||||
}
|
||||
|
||||
/// GetCodeName - Return a symbolic code name if known, otherwise return
|
||||
/// null.
|
||||
static const char *GetCodeName(unsigned CodeID, unsigned BlockID,
|
||||
const BitstreamReader &StreamFile) {
|
||||
// Standard blocks for all bitcode files.
|
||||
if (BlockID < bitc::FIRST_APPLICATION_BLOCKID) {
|
||||
if (BlockID == bitc::BLOCKINFO_BLOCK_ID) {
|
||||
switch (CodeID) {
|
||||
default: return 0;
|
||||
case bitc::BLOCKINFO_CODE_SETBID: return "SETBID";
|
||||
case bitc::BLOCKINFO_CODE_BLOCKNAME: return "BLOCKNAME";
|
||||
case bitc::BLOCKINFO_CODE_SETRECORDNAME: return "SETRECORDNAME";
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
// Check to see if we have a blockinfo record for this record, with a name.
|
||||
if (const BitstreamReader::BlockInfo *Info =
|
||||
StreamFile.getBlockInfo(BlockID)) {
|
||||
for (unsigned i = 0, e = Info->RecordNames.size(); i != e; ++i)
|
||||
if (Info->RecordNames[i].first == CodeID)
|
||||
return Info->RecordNames[i].second.c_str();
|
||||
}
|
||||
|
||||
|
||||
if (CurStreamType != LLVMIRBitstream) return 0;
|
||||
|
||||
switch (BlockID) {
|
||||
default: return 0;
|
||||
case bitc::MODULE_BLOCK_ID:
|
||||
switch (CodeID) {
|
||||
default: return 0;
|
||||
case bitc::MODULE_CODE_VERSION: return "VERSION";
|
||||
case bitc::MODULE_CODE_TRIPLE: return "TRIPLE";
|
||||
case bitc::MODULE_CODE_DATALAYOUT: return "DATALAYOUT";
|
||||
case bitc::MODULE_CODE_ASM: return "ASM";
|
||||
case bitc::MODULE_CODE_SECTIONNAME: return "SECTIONNAME";
|
||||
case bitc::MODULE_CODE_DEPLIB: return "DEPLIB";
|
||||
case bitc::MODULE_CODE_GLOBALVAR: return "GLOBALVAR";
|
||||
case bitc::MODULE_CODE_FUNCTION: return "FUNCTION";
|
||||
case bitc::MODULE_CODE_ALIAS: return "ALIAS";
|
||||
case bitc::MODULE_CODE_PURGEVALS: return "PURGEVALS";
|
||||
case bitc::MODULE_CODE_GCNAME: return "GCNAME";
|
||||
}
|
||||
case bitc::PARAMATTR_BLOCK_ID:
|
||||
switch (CodeID) {
|
||||
default: return 0;
|
||||
case bitc::PARAMATTR_CODE_ENTRY: return "ENTRY";
|
||||
}
|
||||
case bitc::TYPE_BLOCK_ID_OLD:
|
||||
case bitc::TYPE_BLOCK_ID_NEW:
|
||||
switch (CodeID) {
|
||||
default: return 0;
|
||||
case bitc::TYPE_CODE_NUMENTRY: return "NUMENTRY";
|
||||
case bitc::TYPE_CODE_VOID: return "VOID";
|
||||
case bitc::TYPE_CODE_FLOAT: return "FLOAT";
|
||||
case bitc::TYPE_CODE_DOUBLE: return "DOUBLE";
|
||||
case bitc::TYPE_CODE_LABEL: return "LABEL";
|
||||
case bitc::TYPE_CODE_OPAQUE: return "OPAQUE";
|
||||
case bitc::TYPE_CODE_INTEGER: return "INTEGER";
|
||||
case bitc::TYPE_CODE_POINTER: return "POINTER";
|
||||
case bitc::TYPE_CODE_FUNCTION: return "FUNCTION";
|
||||
case bitc::TYPE_CODE_STRUCT_OLD: return "STRUCT_OLD";
|
||||
case bitc::TYPE_CODE_ARRAY: return "ARRAY";
|
||||
case bitc::TYPE_CODE_VECTOR: return "VECTOR";
|
||||
case bitc::TYPE_CODE_X86_FP80: return "X86_FP80";
|
||||
case bitc::TYPE_CODE_FP128: return "FP128";
|
||||
case bitc::TYPE_CODE_PPC_FP128: return "PPC_FP128";
|
||||
case bitc::TYPE_CODE_METADATA: return "METADATA";
|
||||
case bitc::TYPE_CODE_STRUCT_ANON: return "STRUCT_ANON";
|
||||
case bitc::TYPE_CODE_STRUCT_NAME: return "STRUCT_NAME";
|
||||
case bitc::TYPE_CODE_STRUCT_NAMED: return "STRUCT_NAMED";
|
||||
}
|
||||
|
||||
case bitc::CONSTANTS_BLOCK_ID:
|
||||
switch (CodeID) {
|
||||
default: return 0;
|
||||
case bitc::CST_CODE_SETTYPE: return "SETTYPE";
|
||||
case bitc::CST_CODE_NULL: return "NULL";
|
||||
case bitc::CST_CODE_UNDEF: return "UNDEF";
|
||||
case bitc::CST_CODE_INTEGER: return "INTEGER";
|
||||
case bitc::CST_CODE_WIDE_INTEGER: return "WIDE_INTEGER";
|
||||
case bitc::CST_CODE_FLOAT: return "FLOAT";
|
||||
case bitc::CST_CODE_AGGREGATE: return "AGGREGATE";
|
||||
case bitc::CST_CODE_STRING: return "STRING";
|
||||
case bitc::CST_CODE_CSTRING: return "CSTRING";
|
||||
case bitc::CST_CODE_CE_BINOP: return "CE_BINOP";
|
||||
case bitc::CST_CODE_CE_CAST: return "CE_CAST";
|
||||
case bitc::CST_CODE_CE_GEP: return "CE_GEP";
|
||||
case bitc::CST_CODE_CE_INBOUNDS_GEP: return "CE_INBOUNDS_GEP";
|
||||
case bitc::CST_CODE_CE_SELECT: return "CE_SELECT";
|
||||
case bitc::CST_CODE_CE_EXTRACTELT: return "CE_EXTRACTELT";
|
||||
case bitc::CST_CODE_CE_INSERTELT: return "CE_INSERTELT";
|
||||
case bitc::CST_CODE_CE_SHUFFLEVEC: return "CE_SHUFFLEVEC";
|
||||
case bitc::CST_CODE_CE_CMP: return "CE_CMP";
|
||||
case bitc::CST_CODE_INLINEASM: return "INLINEASM";
|
||||
case bitc::CST_CODE_CE_SHUFVEC_EX: return "CE_SHUFVEC_EX";
|
||||
}
|
||||
case bitc::FUNCTION_BLOCK_ID:
|
||||
switch (CodeID) {
|
||||
default: return 0;
|
||||
case bitc::FUNC_CODE_DECLAREBLOCKS: return "DECLAREBLOCKS";
|
||||
|
||||
case bitc::FUNC_CODE_INST_BINOP: return "INST_BINOP";
|
||||
case bitc::FUNC_CODE_INST_CAST: return "INST_CAST";
|
||||
case bitc::FUNC_CODE_INST_GEP: return "INST_GEP";
|
||||
case bitc::FUNC_CODE_INST_INBOUNDS_GEP: return "INST_INBOUNDS_GEP";
|
||||
case bitc::FUNC_CODE_INST_SELECT: return "INST_SELECT";
|
||||
case bitc::FUNC_CODE_INST_EXTRACTELT: return "INST_EXTRACTELT";
|
||||
case bitc::FUNC_CODE_INST_INSERTELT: return "INST_INSERTELT";
|
||||
case bitc::FUNC_CODE_INST_SHUFFLEVEC: return "INST_SHUFFLEVEC";
|
||||
case bitc::FUNC_CODE_INST_CMP: return "INST_CMP";
|
||||
|
||||
case bitc::FUNC_CODE_INST_RET: return "INST_RET";
|
||||
case bitc::FUNC_CODE_INST_BR: return "INST_BR";
|
||||
case bitc::FUNC_CODE_INST_SWITCH: return "INST_SWITCH";
|
||||
case bitc::FUNC_CODE_INST_INVOKE: return "INST_INVOKE";
|
||||
case bitc::FUNC_CODE_INST_UNWIND: return "INST_UNWIND";
|
||||
case bitc::FUNC_CODE_INST_UNREACHABLE: return "INST_UNREACHABLE";
|
||||
|
||||
case bitc::FUNC_CODE_INST_PHI: return "INST_PHI";
|
||||
case bitc::FUNC_CODE_INST_ALLOCA: return "INST_ALLOCA";
|
||||
case bitc::FUNC_CODE_INST_LOAD: return "INST_LOAD";
|
||||
case bitc::FUNC_CODE_INST_VAARG: return "INST_VAARG";
|
||||
case bitc::FUNC_CODE_INST_STORE: return "INST_STORE";
|
||||
case bitc::FUNC_CODE_INST_EXTRACTVAL: return "INST_EXTRACTVAL";
|
||||
case bitc::FUNC_CODE_INST_INSERTVAL: return "INST_INSERTVAL";
|
||||
case bitc::FUNC_CODE_INST_CMP2: return "INST_CMP2";
|
||||
case bitc::FUNC_CODE_INST_VSELECT: return "INST_VSELECT";
|
||||
case bitc::FUNC_CODE_DEBUG_LOC_AGAIN: return "DEBUG_LOC_AGAIN";
|
||||
case bitc::FUNC_CODE_INST_CALL: return "INST_CALL";
|
||||
case bitc::FUNC_CODE_DEBUG_LOC: return "DEBUG_LOC";
|
||||
}
|
||||
case bitc::TYPE_SYMTAB_BLOCK_ID_OLD:
|
||||
switch (CodeID) {
|
||||
default: return 0;
|
||||
case bitc::TST_CODE_ENTRY: return "ENTRY";
|
||||
}
|
||||
case bitc::VALUE_SYMTAB_BLOCK_ID:
|
||||
switch (CodeID) {
|
||||
default: return 0;
|
||||
case bitc::VST_CODE_ENTRY: return "ENTRY";
|
||||
case bitc::VST_CODE_BBENTRY: return "BBENTRY";
|
||||
}
|
||||
case bitc::METADATA_ATTACHMENT_ID:
|
||||
switch(CodeID) {
|
||||
default:return 0;
|
||||
case bitc::METADATA_ATTACHMENT: return "METADATA_ATTACHMENT";
|
||||
}
|
||||
case bitc::METADATA_BLOCK_ID:
|
||||
switch(CodeID) {
|
||||
default:return 0;
|
||||
case bitc::METADATA_STRING: return "METADATA_STRING";
|
||||
case bitc::METADATA_NAME: return "METADATA_NAME";
|
||||
case bitc::METADATA_KIND: return "METADATA_KIND";
|
||||
case bitc::METADATA_NODE: return "METADATA_NODE";
|
||||
case bitc::METADATA_FN_NODE: return "METADATA_FN_NODE";
|
||||
case bitc::METADATA_NAMED_NODE: return "METADATA_NAMED_NODE";
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
struct PerRecordStats {
|
||||
unsigned NumInstances;
|
||||
unsigned NumAbbrev;
|
||||
uint64_t TotalBits;
|
||||
|
||||
PerRecordStats() : NumInstances(0), NumAbbrev(0), TotalBits(0) {}
|
||||
};
|
||||
|
||||
struct PerBlockIDStats {
|
||||
/// NumInstances - This the number of times this block ID has been seen.
|
||||
unsigned NumInstances;
|
||||
|
||||
/// NumBits - The total size in bits of all of these blocks.
|
||||
uint64_t NumBits;
|
||||
|
||||
/// NumSubBlocks - The total number of blocks these blocks contain.
|
||||
unsigned NumSubBlocks;
|
||||
|
||||
/// NumAbbrevs - The total number of abbreviations.
|
||||
unsigned NumAbbrevs;
|
||||
|
||||
/// NumRecords - The total number of records these blocks contain, and the
|
||||
/// number that are abbreviated.
|
||||
unsigned NumRecords, NumAbbreviatedRecords;
|
||||
|
||||
/// CodeFreq - Keep track of the number of times we see each code.
|
||||
std::vector<PerRecordStats> CodeFreq;
|
||||
|
||||
PerBlockIDStats()
|
||||
: NumInstances(0), NumBits(0),
|
||||
NumSubBlocks(0), NumAbbrevs(0), NumRecords(0), NumAbbreviatedRecords(0) {}
|
||||
};
|
||||
|
||||
static std::map<unsigned, PerBlockIDStats> BlockIDStats;
|
||||
|
||||
|
||||
|
||||
/// Error - All bitcode analysis errors go through this function, making this a
|
||||
/// good place to breakpoint if debugging.
|
||||
static bool Error(const std::string &Err) {
|
||||
errs() << Err << "\n";
|
||||
return true;
|
||||
}
|
||||
|
||||
/// ParseBlock - Read a block, updating statistics, etc.
|
||||
static bool ParseBlock(BitstreamCursor &Stream, unsigned IndentLevel) {
|
||||
std::string Indent(IndentLevel*2, ' ');
|
||||
uint64_t BlockBitStart = Stream.GetCurrentBitNo();
|
||||
unsigned BlockID = Stream.ReadSubBlockID();
|
||||
|
||||
// Get the statistics for this BlockID.
|
||||
PerBlockIDStats &BlockStats = BlockIDStats[BlockID];
|
||||
|
||||
BlockStats.NumInstances++;
|
||||
|
||||
// BLOCKINFO is a special part of the stream.
|
||||
if (BlockID == bitc::BLOCKINFO_BLOCK_ID) {
|
||||
if (Dump) errs() << Indent << "<BLOCKINFO_BLOCK/>\n";
|
||||
if (Stream.ReadBlockInfoBlock())
|
||||
return Error("Malformed BlockInfoBlock");
|
||||
uint64_t BlockBitEnd = Stream.GetCurrentBitNo();
|
||||
BlockStats.NumBits += BlockBitEnd-BlockBitStart;
|
||||
return false;
|
||||
}
|
||||
|
||||
unsigned NumWords = 0;
|
||||
if (Stream.EnterSubBlock(BlockID, &NumWords))
|
||||
return Error("Malformed block record");
|
||||
|
||||
const char *BlockName = 0;
|
||||
if (Dump) {
|
||||
errs() << Indent << "<";
|
||||
if ((BlockName = GetBlockName(BlockID, *Stream.getBitStreamReader())))
|
||||
errs() << BlockName;
|
||||
else
|
||||
errs() << "UnknownBlock" << BlockID;
|
||||
|
||||
if (NonSymbolic && BlockName)
|
||||
errs() << " BlockID=" << BlockID;
|
||||
|
||||
errs() << " NumWords=" << NumWords
|
||||
<< " BlockCodeSize=" << Stream.GetAbbrevIDWidth() << ">\n";
|
||||
}
|
||||
|
||||
SmallVector<uint64_t, 64> Record;
|
||||
|
||||
// Read all the records for this block.
|
||||
while (1) {
|
||||
if (Stream.AtEndOfStream())
|
||||
return Error("Premature end of bitstream");
|
||||
|
||||
uint64_t RecordStartBit = Stream.GetCurrentBitNo();
|
||||
|
||||
// Read the code for this record.
|
||||
unsigned AbbrevID = Stream.ReadCode();
|
||||
switch (AbbrevID) {
|
||||
case bitc::END_BLOCK: {
|
||||
if (Stream.ReadBlockEnd())
|
||||
return Error("Error at end of block");
|
||||
uint64_t BlockBitEnd = Stream.GetCurrentBitNo();
|
||||
BlockStats.NumBits += BlockBitEnd-BlockBitStart;
|
||||
if (Dump) {
|
||||
errs() << Indent << "</";
|
||||
if (BlockName)
|
||||
errs() << BlockName << ">\n";
|
||||
else
|
||||
errs() << "UnknownBlock" << BlockID << ">\n";
|
||||
}
|
||||
return false;
|
||||
}
|
||||
case bitc::ENTER_SUBBLOCK: {
|
||||
uint64_t SubBlockBitStart = Stream.GetCurrentBitNo();
|
||||
if (ParseBlock(Stream, IndentLevel+1))
|
||||
return true;
|
||||
++BlockStats.NumSubBlocks;
|
||||
uint64_t SubBlockBitEnd = Stream.GetCurrentBitNo();
|
||||
|
||||
// Don't include subblock sizes in the size of this block.
|
||||
BlockBitStart += SubBlockBitEnd-SubBlockBitStart;
|
||||
break;
|
||||
}
|
||||
case bitc::DEFINE_ABBREV:
|
||||
Stream.ReadAbbrevRecord();
|
||||
++BlockStats.NumAbbrevs;
|
||||
break;
|
||||
default:
|
||||
Record.clear();
|
||||
|
||||
++BlockStats.NumRecords;
|
||||
if (AbbrevID != bitc::UNABBREV_RECORD)
|
||||
++BlockStats.NumAbbreviatedRecords;
|
||||
|
||||
const char *BlobStart = 0;
|
||||
unsigned BlobLen = 0;
|
||||
unsigned Code = Stream.ReadRecord(AbbrevID, Record, BlobStart, BlobLen);
|
||||
|
||||
|
||||
|
||||
// Increment the # occurrences of this code.
|
||||
if (BlockStats.CodeFreq.size() <= Code)
|
||||
BlockStats.CodeFreq.resize(Code+1);
|
||||
BlockStats.CodeFreq[Code].NumInstances++;
|
||||
BlockStats.CodeFreq[Code].TotalBits +=
|
||||
Stream.GetCurrentBitNo()-RecordStartBit;
|
||||
if (AbbrevID != bitc::UNABBREV_RECORD)
|
||||
BlockStats.CodeFreq[Code].NumAbbrev++;
|
||||
|
||||
if (Dump) {
|
||||
errs() << Indent << " <";
|
||||
if (const char *CodeName =
|
||||
GetCodeName(Code, BlockID, *Stream.getBitStreamReader()))
|
||||
errs() << CodeName;
|
||||
else
|
||||
errs() << "UnknownCode" << Code;
|
||||
if (NonSymbolic &&
|
||||
GetCodeName(Code, BlockID, *Stream.getBitStreamReader()))
|
||||
errs() << " codeid=" << Code;
|
||||
if (AbbrevID != bitc::UNABBREV_RECORD)
|
||||
errs() << " abbrevid=" << AbbrevID;
|
||||
|
||||
for (unsigned i = 0, e = Record.size(); i != e; ++i)
|
||||
errs() << " op" << i << "=" << (int64_t)Record[i];
|
||||
|
||||
errs() << "/>";
|
||||
|
||||
if (BlobStart) {
|
||||
errs() << " blob data = ";
|
||||
bool BlobIsPrintable = true;
|
||||
for (unsigned i = 0; i != BlobLen; ++i)
|
||||
if (!isprint(BlobStart[i])) {
|
||||
BlobIsPrintable = false;
|
||||
break;
|
||||
}
|
||||
|
||||
if (BlobIsPrintable)
|
||||
errs() << "'" << std::string(BlobStart, BlobStart+BlobLen) <<"'";
|
||||
else
|
||||
errs() << "unprintable, " << BlobLen << " bytes.";
|
||||
}
|
||||
|
||||
errs() << "\n";
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void PrintSize(double Bits) {
|
||||
fprintf(stderr, "%.2f/%.2fB/%luW", Bits, Bits/8,(unsigned long)(Bits/32));
|
||||
}
|
||||
static void PrintSize(uint64_t Bits) {
|
||||
fprintf(stderr, "%lub/%.2fB/%luW", (unsigned long)Bits,
|
||||
(double)Bits/8, (unsigned long)(Bits/32));
|
||||
}
|
||||
|
||||
|
||||
/// AnalyzeBitcode - Analyze the bitcode file specified by InputFilename.
|
||||
static int AnalyzeBitcode() {
|
||||
// Read the input file.
|
||||
OwningPtr<MemoryBuffer> MemBuf;
|
||||
|
||||
if (error_code ec =
|
||||
MemoryBuffer::getFileOrSTDIN(InputFilename.c_str(), MemBuf))
|
||||
return Error("Error reading '" + InputFilename + "': " + ec.message());
|
||||
|
||||
if (MemBuf->getBufferSize() & 3)
|
||||
return Error("Bitcode stream should be a multiple of 4 bytes in length");
|
||||
|
||||
unsigned char *BufPtr = (unsigned char *)MemBuf->getBufferStart();
|
||||
unsigned char *EndBufPtr = BufPtr+MemBuf->getBufferSize();
|
||||
|
||||
// If we have a wrapper header, parse it and ignore the non-bc file contents.
|
||||
// The magic number is 0x0B17C0DE stored in little endian.
|
||||
if (isBitcodeWrapper(BufPtr, EndBufPtr))
|
||||
if (SkipBitcodeWrapperHeader(BufPtr, EndBufPtr))
|
||||
return Error("Invalid bitcode wrapper header");
|
||||
|
||||
BitstreamReader StreamFile(BufPtr, EndBufPtr);
|
||||
BitstreamCursor Stream(StreamFile);
|
||||
StreamFile.CollectBlockInfoNames();
|
||||
|
||||
// Read the stream signature.
|
||||
char Signature[6];
|
||||
Signature[0] = Stream.Read(8);
|
||||
Signature[1] = Stream.Read(8);
|
||||
Signature[2] = Stream.Read(4);
|
||||
Signature[3] = Stream.Read(4);
|
||||
Signature[4] = Stream.Read(4);
|
||||
Signature[5] = Stream.Read(4);
|
||||
|
||||
// Autodetect the file contents, if it is one we know.
|
||||
CurStreamType = UnknownBitstream;
|
||||
if (Signature[0] == 'B' && Signature[1] == 'C' &&
|
||||
Signature[2] == 0x0 && Signature[3] == 0xC &&
|
||||
Signature[4] == 0xE && Signature[5] == 0xD)
|
||||
CurStreamType = LLVMIRBitstream;
|
||||
|
||||
unsigned NumTopBlocks = 0;
|
||||
|
||||
// Parse the top-level structure. We only allow blocks at the top-level.
|
||||
while (!Stream.AtEndOfStream()) {
|
||||
unsigned Code = Stream.ReadCode();
|
||||
if (Code != bitc::ENTER_SUBBLOCK)
|
||||
return Error("Invalid record at top-level");
|
||||
|
||||
if (ParseBlock(Stream, 0))
|
||||
return true;
|
||||
++NumTopBlocks;
|
||||
}
|
||||
|
||||
if (Dump) errs() << "\n\n";
|
||||
|
||||
uint64_t BufferSizeBits = (EndBufPtr-BufPtr)*CHAR_BIT;
|
||||
// Print a summary of the read file.
|
||||
errs() << "Summary of " << InputFilename << ":\n";
|
||||
errs() << " Total size: ";
|
||||
PrintSize(BufferSizeBits);
|
||||
errs() << "\n";
|
||||
errs() << " Stream type: ";
|
||||
switch (CurStreamType) {
|
||||
default: assert(0 && "Unknown bitstream type");
|
||||
case UnknownBitstream: errs() << "unknown\n"; break;
|
||||
case LLVMIRBitstream: errs() << "LLVM IR\n"; break;
|
||||
}
|
||||
errs() << " # Toplevel Blocks: " << NumTopBlocks << "\n";
|
||||
errs() << "\n";
|
||||
|
||||
// Emit per-block stats.
|
||||
errs() << "Per-block Summary:\n";
|
||||
for (std::map<unsigned, PerBlockIDStats>::iterator I = BlockIDStats.begin(),
|
||||
E = BlockIDStats.end(); I != E; ++I) {
|
||||
errs() << " Block ID #" << I->first;
|
||||
if (const char *BlockName = GetBlockName(I->first, StreamFile))
|
||||
errs() << " (" << BlockName << ")";
|
||||
errs() << ":\n";
|
||||
|
||||
const PerBlockIDStats &Stats = I->second;
|
||||
errs() << " Num Instances: " << Stats.NumInstances << "\n";
|
||||
errs() << " Total Size: ";
|
||||
PrintSize(Stats.NumBits);
|
||||
errs() << "\n";
|
||||
double pct = (Stats.NumBits * 100.0) / BufferSizeBits;
|
||||
errs() << " Percent of file: " << format("%2.4f%%", pct) << "\n";
|
||||
if (Stats.NumInstances > 1) {
|
||||
errs() << " Average Size: ";
|
||||
PrintSize(Stats.NumBits/(double)Stats.NumInstances);
|
||||
errs() << "\n";
|
||||
errs() << " Tot/Avg SubBlocks: " << Stats.NumSubBlocks << "/"
|
||||
<< Stats.NumSubBlocks/(double)Stats.NumInstances << "\n";
|
||||
errs() << " Tot/Avg Abbrevs: " << Stats.NumAbbrevs << "/"
|
||||
<< Stats.NumAbbrevs/(double)Stats.NumInstances << "\n";
|
||||
errs() << " Tot/Avg Records: " << Stats.NumRecords << "/"
|
||||
<< Stats.NumRecords/(double)Stats.NumInstances << "\n";
|
||||
} else {
|
||||
errs() << " Num SubBlocks: " << Stats.NumSubBlocks << "\n";
|
||||
errs() << " Num Abbrevs: " << Stats.NumAbbrevs << "\n";
|
||||
errs() << " Num Records: " << Stats.NumRecords << "\n";
|
||||
}
|
||||
if (Stats.NumRecords) {
|
||||
double pct = (Stats.NumAbbreviatedRecords * 100.0) / Stats.NumRecords;
|
||||
errs() << " Percent Abbrevs: " << format("%2.4f%%", pct) << "\n";
|
||||
}
|
||||
errs() << "\n";
|
||||
|
||||
// Print a histogram of the codes we see.
|
||||
if (!NoHistogram && !Stats.CodeFreq.empty()) {
|
||||
std::vector<std::pair<unsigned, unsigned> > FreqPairs; // <freq,code>
|
||||
for (unsigned i = 0, e = Stats.CodeFreq.size(); i != e; ++i)
|
||||
if (unsigned Freq = Stats.CodeFreq[i].NumInstances)
|
||||
FreqPairs.push_back(std::make_pair(Freq, i));
|
||||
std::stable_sort(FreqPairs.begin(), FreqPairs.end());
|
||||
std::reverse(FreqPairs.begin(), FreqPairs.end());
|
||||
|
||||
errs() << "\tRecord Histogram:\n";
|
||||
fprintf(stderr, "\t\t Count # Bits %% Abv Record Kind\n");
|
||||
for (unsigned i = 0, e = FreqPairs.size(); i != e; ++i) {
|
||||
const PerRecordStats &RecStats = Stats.CodeFreq[FreqPairs[i].second];
|
||||
|
||||
fprintf(stderr, "\t\t%7d %9lu ", RecStats.NumInstances,
|
||||
(unsigned long)RecStats.TotalBits);
|
||||
|
||||
if (RecStats.NumAbbrev)
|
||||
fprintf(stderr, "%7.2f ",
|
||||
(double)RecStats.NumAbbrev/RecStats.NumInstances*100);
|
||||
else
|
||||
fprintf(stderr, " ");
|
||||
|
||||
if (const char *CodeName =
|
||||
GetCodeName(FreqPairs[i].second, I->first, StreamFile))
|
||||
fprintf(stderr, "%s\n", CodeName);
|
||||
else
|
||||
fprintf(stderr, "UnknownCode%d\n", FreqPairs[i].second);
|
||||
}
|
||||
errs() << "\n";
|
||||
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
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-bcanalyzer file analyzer\n");
|
||||
|
||||
return AnalyzeBitcode();
|
||||
}
|
8
contrib/llvm/tools/llvm-diff/CMakeLists.txt
Normal file
8
contrib/llvm/tools/llvm-diff/CMakeLists.txt
Normal file
@ -0,0 +1,8 @@
|
||||
set(LLVM_LINK_COMPONENTS support asmparser bitreader)
|
||||
|
||||
add_llvm_tool(llvm-diff
|
||||
llvm-diff.cpp
|
||||
DiffConsumer.cpp
|
||||
DiffLog.cpp
|
||||
DifferenceEngine.cpp
|
||||
)
|
209
contrib/llvm/tools/llvm-diff/DiffConsumer.cpp
Normal file
209
contrib/llvm/tools/llvm-diff/DiffConsumer.cpp
Normal file
@ -0,0 +1,209 @@
|
||||
//===-- DiffConsumer.cpp - Difference Consumer ------------------*- C++ -*-===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This files implements the the LLVM difference Consumer
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "DiffConsumer.h"
|
||||
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/Instructions.h"
|
||||
#include "llvm/Support/ErrorHandling.h"
|
||||
|
||||
using namespace llvm;
|
||||
|
||||
static void ComputeNumbering(Function *F, DenseMap<Value*,unsigned> &Numbering){
|
||||
unsigned IN = 0;
|
||||
|
||||
// Arguments get the first numbers.
|
||||
for (Function::arg_iterator
|
||||
AI = F->arg_begin(), AE = F->arg_end(); AI != AE; ++AI)
|
||||
if (!AI->hasName())
|
||||
Numbering[&*AI] = IN++;
|
||||
|
||||
// Walk the basic blocks in order.
|
||||
for (Function::iterator FI = F->begin(), FE = F->end(); FI != FE; ++FI) {
|
||||
if (!FI->hasName())
|
||||
Numbering[&*FI] = IN++;
|
||||
|
||||
// Walk the instructions in order.
|
||||
for (BasicBlock::iterator BI = FI->begin(), BE = FI->end(); BI != BE; ++BI)
|
||||
// void instructions don't get numbers.
|
||||
if (!BI->hasName() && !BI->getType()->isVoidTy())
|
||||
Numbering[&*BI] = IN++;
|
||||
}
|
||||
|
||||
assert(!Numbering.empty() && "asked for numbering but numbering was no-op");
|
||||
}
|
||||
|
||||
|
||||
void DiffConsumer::printValue(Value *V, bool isL) {
|
||||
if (V->hasName()) {
|
||||
out << (isa<GlobalValue>(V) ? '@' : '%') << V->getName();
|
||||
return;
|
||||
}
|
||||
if (V->getType()->isVoidTy()) {
|
||||
if (isa<StoreInst>(V)) {
|
||||
out << "store to ";
|
||||
printValue(cast<StoreInst>(V)->getPointerOperand(), isL);
|
||||
} else if (isa<CallInst>(V)) {
|
||||
out << "call to ";
|
||||
printValue(cast<CallInst>(V)->getCalledValue(), isL);
|
||||
} else if (isa<InvokeInst>(V)) {
|
||||
out << "invoke to ";
|
||||
printValue(cast<InvokeInst>(V)->getCalledValue(), isL);
|
||||
} else {
|
||||
out << *V;
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
unsigned N = contexts.size();
|
||||
while (N > 0) {
|
||||
--N;
|
||||
DiffContext &ctxt = contexts[N];
|
||||
if (!ctxt.IsFunction) continue;
|
||||
if (isL) {
|
||||
if (ctxt.LNumbering.empty())
|
||||
ComputeNumbering(cast<Function>(ctxt.L), ctxt.LNumbering);
|
||||
out << '%' << ctxt.LNumbering[V];
|
||||
return;
|
||||
} else {
|
||||
if (ctxt.RNumbering.empty())
|
||||
ComputeNumbering(cast<Function>(ctxt.R), ctxt.RNumbering);
|
||||
out << '%' << ctxt.RNumbering[V];
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
out << "<anonymous>";
|
||||
}
|
||||
|
||||
void DiffConsumer::header() {
|
||||
if (contexts.empty()) return;
|
||||
for (SmallVectorImpl<DiffContext>::iterator
|
||||
I = contexts.begin(), E = contexts.end(); I != E; ++I) {
|
||||
if (I->Differences) continue;
|
||||
if (isa<Function>(I->L)) {
|
||||
// Extra newline between functions.
|
||||
if (Differences) out << "\n";
|
||||
|
||||
Function *L = cast<Function>(I->L);
|
||||
Function *R = cast<Function>(I->R);
|
||||
if (L->getName() != R->getName())
|
||||
out << "in function " << L->getName()
|
||||
<< " / " << R->getName() << ":\n";
|
||||
else
|
||||
out << "in function " << L->getName() << ":\n";
|
||||
} else if (isa<BasicBlock>(I->L)) {
|
||||
BasicBlock *L = cast<BasicBlock>(I->L);
|
||||
BasicBlock *R = cast<BasicBlock>(I->R);
|
||||
if (L->hasName() && R->hasName() && L->getName() == R->getName())
|
||||
out << " in block %" << L->getName() << ":\n";
|
||||
else {
|
||||
out << " in block ";
|
||||
printValue(L, true);
|
||||
out << " / ";
|
||||
printValue(R, false);
|
||||
out << ":\n";
|
||||
}
|
||||
} else if (isa<Instruction>(I->L)) {
|
||||
out << " in instruction ";
|
||||
printValue(I->L, true);
|
||||
out << " / ";
|
||||
printValue(I->R, false);
|
||||
out << ":\n";
|
||||
}
|
||||
|
||||
I->Differences = true;
|
||||
}
|
||||
}
|
||||
|
||||
void DiffConsumer::indent() {
|
||||
unsigned N = Indent;
|
||||
while (N--) out << ' ';
|
||||
}
|
||||
|
||||
bool DiffConsumer::hadDifferences() const {
|
||||
return Differences;
|
||||
}
|
||||
|
||||
void DiffConsumer::enterContext(Value *L, Value *R) {
|
||||
contexts.push_back(DiffContext(L, R));
|
||||
Indent += 2;
|
||||
}
|
||||
|
||||
void DiffConsumer::exitContext() {
|
||||
Differences |= contexts.back().Differences;
|
||||
contexts.pop_back();
|
||||
Indent -= 2;
|
||||
}
|
||||
|
||||
void DiffConsumer::log(StringRef text) {
|
||||
header();
|
||||
indent();
|
||||
out << text << '\n';
|
||||
}
|
||||
|
||||
void DiffConsumer::logf(const LogBuilder &Log) {
|
||||
header();
|
||||
indent();
|
||||
|
||||
unsigned arg = 0;
|
||||
|
||||
StringRef format = Log.getFormat();
|
||||
while (true) {
|
||||
size_t percent = format.find('%');
|
||||
if (percent == StringRef::npos) {
|
||||
out << format;
|
||||
break;
|
||||
}
|
||||
assert(format[percent] == '%');
|
||||
|
||||
if (percent > 0) out << format.substr(0, percent);
|
||||
|
||||
switch (format[percent+1]) {
|
||||
case '%': out << '%'; break;
|
||||
case 'l': printValue(Log.getArgument(arg++), true); break;
|
||||
case 'r': printValue(Log.getArgument(arg++), false); break;
|
||||
default: llvm_unreachable("unknown format character");
|
||||
}
|
||||
|
||||
format = format.substr(percent+2);
|
||||
}
|
||||
|
||||
out << '\n';
|
||||
}
|
||||
|
||||
void DiffConsumer::logd(const DiffLogBuilder &Log) {
|
||||
header();
|
||||
|
||||
for (unsigned I = 0, E = Log.getNumLines(); I != E; ++I) {
|
||||
indent();
|
||||
switch (Log.getLineKind(I)) {
|
||||
case DC_match:
|
||||
out << " ";
|
||||
Log.getLeft(I)->dump();
|
||||
//printValue(Log.getLeft(I), true);
|
||||
break;
|
||||
case DC_left:
|
||||
out << "< ";
|
||||
Log.getLeft(I)->dump();
|
||||
//printValue(Log.getLeft(I), true);
|
||||
break;
|
||||
case DC_right:
|
||||
out << "> ";
|
||||
Log.getRight(I)->dump();
|
||||
//printValue(Log.getRight(I), false);
|
||||
break;
|
||||
}
|
||||
//out << "\n";
|
||||
}
|
||||
}
|
92
contrib/llvm/tools/llvm-diff/DiffConsumer.h
Normal file
92
contrib/llvm/tools/llvm-diff/DiffConsumer.h
Normal file
@ -0,0 +1,92 @@
|
||||
//===-- DiffConsumer.h - Difference Consumer --------------------*- C++ -*-===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This header defines the interface to the LLVM difference Consumer
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef _LLVM_DIFFCONSUMER_H_
|
||||
#define _LLVM_DIFFCONSUMER_H_
|
||||
|
||||
#include "DiffLog.h"
|
||||
|
||||
#include "llvm/ADT/SmallVector.h"
|
||||
#include "llvm/ADT/DenseMap.h"
|
||||
#include "llvm/ADT/StringRef.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/Casting.h"
|
||||
|
||||
namespace llvm {
|
||||
class Module;
|
||||
class Value;
|
||||
class Function;
|
||||
|
||||
/// The interface for consumers of difference data.
|
||||
class Consumer {
|
||||
public:
|
||||
/// Record that a local context has been entered. Left and
|
||||
/// Right are IR "containers" of some sort which are being
|
||||
/// considered for structural equivalence: global variables,
|
||||
/// functions, blocks, instructions, etc.
|
||||
virtual void enterContext(Value *Left, Value *Right) = 0;
|
||||
|
||||
/// Record that a local context has been exited.
|
||||
virtual void exitContext() = 0;
|
||||
|
||||
/// Record a difference within the current context.
|
||||
virtual void log(StringRef Text) = 0;
|
||||
|
||||
/// Record a formatted difference within the current context.
|
||||
virtual void logf(const LogBuilder &Log) = 0;
|
||||
|
||||
/// Record a line-by-line instruction diff.
|
||||
virtual void logd(const DiffLogBuilder &Log) = 0;
|
||||
|
||||
protected:
|
||||
virtual ~Consumer() {}
|
||||
};
|
||||
|
||||
class DiffConsumer : public Consumer {
|
||||
private:
|
||||
struct DiffContext {
|
||||
DiffContext(Value *L, Value *R)
|
||||
: L(L), R(R), Differences(false), IsFunction(isa<Function>(L)) {}
|
||||
Value *L;
|
||||
Value *R;
|
||||
bool Differences;
|
||||
bool IsFunction;
|
||||
DenseMap<Value*,unsigned> LNumbering;
|
||||
DenseMap<Value*,unsigned> RNumbering;
|
||||
};
|
||||
|
||||
raw_ostream &out;
|
||||
Module *LModule;
|
||||
Module *RModule;
|
||||
SmallVector<DiffContext, 5> contexts;
|
||||
bool Differences;
|
||||
unsigned Indent;
|
||||
|
||||
void printValue(Value *V, bool isL);
|
||||
void header();
|
||||
void indent();
|
||||
|
||||
public:
|
||||
DiffConsumer(Module *L, Module *R)
|
||||
: out(errs()), LModule(L), RModule(R), Differences(false), Indent(0) {}
|
||||
|
||||
bool hadDifferences() const;
|
||||
void enterContext(Value *L, Value *R);
|
||||
void exitContext();
|
||||
void log(StringRef text);
|
||||
void logf(const LogBuilder &Log);
|
||||
void logd(const DiffLogBuilder &Log);
|
||||
};
|
||||
}
|
||||
|
||||
#endif
|
53
contrib/llvm/tools/llvm-diff/DiffLog.cpp
Normal file
53
contrib/llvm/tools/llvm-diff/DiffLog.cpp
Normal file
@ -0,0 +1,53 @@
|
||||
//===-- DiffLog.h - Difference Log Builder and accessories ------*- C++ -*-===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This header defines the interface to the LLVM difference log builder.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "DiffLog.h"
|
||||
#include "DiffConsumer.h"
|
||||
|
||||
#include "llvm/Instructions.h"
|
||||
#include "llvm/ADT/SmallVector.h"
|
||||
#include "llvm/ADT/StringRef.h"
|
||||
|
||||
using namespace llvm;
|
||||
|
||||
LogBuilder::~LogBuilder() {
|
||||
consumer.logf(*this);
|
||||
}
|
||||
|
||||
StringRef LogBuilder::getFormat() const { return Format; }
|
||||
|
||||
unsigned LogBuilder::getNumArguments() const { return Arguments.size(); }
|
||||
Value *LogBuilder::getArgument(unsigned I) const { return Arguments[I]; }
|
||||
|
||||
DiffLogBuilder::~DiffLogBuilder() { consumer.logd(*this); }
|
||||
|
||||
void DiffLogBuilder::addMatch(Instruction *L, Instruction *R) {
|
||||
Diff.push_back(DiffRecord(L, R));
|
||||
}
|
||||
void DiffLogBuilder::addLeft(Instruction *L) {
|
||||
// HACK: VS 2010 has a bug in the stdlib that requires this.
|
||||
Diff.push_back(DiffRecord(L, DiffRecord::second_type(0)));
|
||||
}
|
||||
void DiffLogBuilder::addRight(Instruction *R) {
|
||||
// HACK: VS 2010 has a bug in the stdlib that requires this.
|
||||
Diff.push_back(DiffRecord(DiffRecord::first_type(0), R));
|
||||
}
|
||||
|
||||
unsigned DiffLogBuilder::getNumLines() const { return Diff.size(); }
|
||||
|
||||
DiffChange DiffLogBuilder::getLineKind(unsigned I) const {
|
||||
return (Diff[I].first ? (Diff[I].second ? DC_match : DC_left)
|
||||
: DC_right);
|
||||
}
|
||||
Instruction *DiffLogBuilder::getLeft(unsigned I) const { return Diff[I].first; }
|
||||
Instruction *DiffLogBuilder::getRight(unsigned I) const { return Diff[I].second; }
|
80
contrib/llvm/tools/llvm-diff/DiffLog.h
Normal file
80
contrib/llvm/tools/llvm-diff/DiffLog.h
Normal file
@ -0,0 +1,80 @@
|
||||
//===-- DiffLog.h - Difference Log Builder and accessories ------*- C++ -*-===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This header defines the interface to the LLVM difference log builder.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef _LLVM_DIFFLOG_H_
|
||||
#define _LLVM_DIFFLOG_H_
|
||||
|
||||
#include "llvm/ADT/SmallVector.h"
|
||||
#include "llvm/ADT/StringRef.h"
|
||||
|
||||
namespace llvm {
|
||||
class Instruction;
|
||||
class Value;
|
||||
class Consumer;
|
||||
|
||||
/// Trichotomy assumption
|
||||
enum DiffChange { DC_match, DC_left, DC_right };
|
||||
|
||||
/// A temporary-object class for building up log messages.
|
||||
class LogBuilder {
|
||||
Consumer &consumer;
|
||||
|
||||
/// The use of a stored StringRef here is okay because
|
||||
/// LogBuilder should be used only as a temporary, and as a
|
||||
/// temporary it will be destructed before whatever temporary
|
||||
/// might be initializing this format.
|
||||
StringRef Format;
|
||||
|
||||
SmallVector<Value*, 4> Arguments;
|
||||
|
||||
public:
|
||||
LogBuilder(Consumer &c, StringRef Format)
|
||||
: consumer(c), Format(Format) {}
|
||||
|
||||
LogBuilder &operator<<(Value *V) {
|
||||
Arguments.push_back(V);
|
||||
return *this;
|
||||
}
|
||||
|
||||
~LogBuilder();
|
||||
|
||||
StringRef getFormat() const;
|
||||
unsigned getNumArguments() const;
|
||||
Value *getArgument(unsigned I) const;
|
||||
};
|
||||
|
||||
/// A temporary-object class for building up diff messages.
|
||||
class DiffLogBuilder {
|
||||
typedef std::pair<Instruction*,Instruction*> DiffRecord;
|
||||
SmallVector<DiffRecord, 20> Diff;
|
||||
|
||||
Consumer &consumer;
|
||||
|
||||
public:
|
||||
DiffLogBuilder(Consumer &c) : consumer(c) {}
|
||||
~DiffLogBuilder();
|
||||
|
||||
void addMatch(Instruction *L, Instruction *R);
|
||||
// HACK: VS 2010 has a bug in the stdlib that requires this.
|
||||
void addLeft(Instruction *L);
|
||||
void addRight(Instruction *R);
|
||||
|
||||
unsigned getNumLines() const;
|
||||
DiffChange getLineKind(unsigned I) const;
|
||||
Instruction *getLeft(unsigned I) const;
|
||||
Instruction *getRight(unsigned I) const;
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
#endif
|
677
contrib/llvm/tools/llvm-diff/DifferenceEngine.cpp
Normal file
677
contrib/llvm/tools/llvm-diff/DifferenceEngine.cpp
Normal file
@ -0,0 +1,677 @@
|
||||
//===-- DifferenceEngine.cpp - Structural function/module comparison ------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This header defines the implementation of the LLVM difference
|
||||
// engine, which structurally compares global values within a module.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "DifferenceEngine.h"
|
||||
|
||||
#include "llvm/Constants.h"
|
||||
#include "llvm/Function.h"
|
||||
#include "llvm/Instructions.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/ADT/DenseMap.h"
|
||||
#include "llvm/ADT/DenseSet.h"
|
||||
#include "llvm/ADT/SmallVector.h"
|
||||
#include "llvm/ADT/StringRef.h"
|
||||
#include "llvm/ADT/StringSet.h"
|
||||
#include "llvm/Support/CallSite.h"
|
||||
#include "llvm/Support/CFG.h"
|
||||
#include "llvm/Support/ErrorHandling.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/type_traits.h"
|
||||
|
||||
#include <utility>
|
||||
|
||||
using namespace llvm;
|
||||
|
||||
namespace {
|
||||
|
||||
/// A priority queue, implemented as a heap.
|
||||
template <class T, class Sorter, unsigned InlineCapacity>
|
||||
class PriorityQueue {
|
||||
Sorter Precedes;
|
||||
llvm::SmallVector<T, InlineCapacity> Storage;
|
||||
|
||||
public:
|
||||
PriorityQueue(const Sorter &Precedes) : Precedes(Precedes) {}
|
||||
|
||||
/// Checks whether the heap is empty.
|
||||
bool empty() const { return Storage.empty(); }
|
||||
|
||||
/// Insert a new value on the heap.
|
||||
void insert(const T &V) {
|
||||
unsigned Index = Storage.size();
|
||||
Storage.push_back(V);
|
||||
if (Index == 0) return;
|
||||
|
||||
T *data = Storage.data();
|
||||
while (true) {
|
||||
unsigned Target = (Index + 1) / 2 - 1;
|
||||
if (!Precedes(data[Index], data[Target])) return;
|
||||
std::swap(data[Index], data[Target]);
|
||||
if (Target == 0) return;
|
||||
Index = Target;
|
||||
}
|
||||
}
|
||||
|
||||
/// Remove the minimum value in the heap. Only valid on a non-empty heap.
|
||||
T remove_min() {
|
||||
assert(!empty());
|
||||
T tmp = Storage[0];
|
||||
|
||||
unsigned NewSize = Storage.size() - 1;
|
||||
if (NewSize) {
|
||||
// Move the slot at the end to the beginning.
|
||||
if (isPodLike<T>::value)
|
||||
Storage[0] = Storage[NewSize];
|
||||
else
|
||||
std::swap(Storage[0], Storage[NewSize]);
|
||||
|
||||
// Bubble the root up as necessary.
|
||||
unsigned Index = 0;
|
||||
while (true) {
|
||||
// With a 1-based index, the children would be Index*2 and Index*2+1.
|
||||
unsigned R = (Index + 1) * 2;
|
||||
unsigned L = R - 1;
|
||||
|
||||
// If R is out of bounds, we're done after this in any case.
|
||||
if (R >= NewSize) {
|
||||
// If L is also out of bounds, we're done immediately.
|
||||
if (L >= NewSize) break;
|
||||
|
||||
// Otherwise, test whether we should swap L and Index.
|
||||
if (Precedes(Storage[L], Storage[Index]))
|
||||
std::swap(Storage[L], Storage[Index]);
|
||||
break;
|
||||
}
|
||||
|
||||
// Otherwise, we need to compare with the smaller of L and R.
|
||||
// Prefer R because it's closer to the end of the array.
|
||||
unsigned IndexToTest = (Precedes(Storage[L], Storage[R]) ? L : R);
|
||||
|
||||
// If Index is >= the min of L and R, then heap ordering is restored.
|
||||
if (!Precedes(Storage[IndexToTest], Storage[Index]))
|
||||
break;
|
||||
|
||||
// Otherwise, keep bubbling up.
|
||||
std::swap(Storage[IndexToTest], Storage[Index]);
|
||||
Index = IndexToTest;
|
||||
}
|
||||
}
|
||||
Storage.pop_back();
|
||||
|
||||
return tmp;
|
||||
}
|
||||
};
|
||||
|
||||
/// A function-scope difference engine.
|
||||
class FunctionDifferenceEngine {
|
||||
DifferenceEngine &Engine;
|
||||
|
||||
/// The current mapping from old local values to new local values.
|
||||
DenseMap<Value*, Value*> Values;
|
||||
|
||||
/// The current mapping from old blocks to new blocks.
|
||||
DenseMap<BasicBlock*, BasicBlock*> Blocks;
|
||||
|
||||
DenseSet<std::pair<Value*, Value*> > TentativeValues;
|
||||
|
||||
unsigned getUnprocPredCount(BasicBlock *Block) const {
|
||||
unsigned Count = 0;
|
||||
for (pred_iterator I = pred_begin(Block), E = pred_end(Block); I != E; ++I)
|
||||
if (!Blocks.count(*I)) Count++;
|
||||
return Count;
|
||||
}
|
||||
|
||||
typedef std::pair<BasicBlock*, BasicBlock*> BlockPair;
|
||||
|
||||
/// A type which sorts a priority queue by the number of unprocessed
|
||||
/// predecessor blocks it has remaining.
|
||||
///
|
||||
/// This is actually really expensive to calculate.
|
||||
struct QueueSorter {
|
||||
const FunctionDifferenceEngine &fde;
|
||||
explicit QueueSorter(const FunctionDifferenceEngine &fde) : fde(fde) {}
|
||||
|
||||
bool operator()(const BlockPair &Old, const BlockPair &New) {
|
||||
return fde.getUnprocPredCount(Old.first)
|
||||
< fde.getUnprocPredCount(New.first);
|
||||
}
|
||||
};
|
||||
|
||||
/// A queue of unified blocks to process.
|
||||
PriorityQueue<BlockPair, QueueSorter, 20> Queue;
|
||||
|
||||
/// Try to unify the given two blocks. Enqueues them for processing
|
||||
/// if they haven't already been processed.
|
||||
///
|
||||
/// Returns true if there was a problem unifying them.
|
||||
bool tryUnify(BasicBlock *L, BasicBlock *R) {
|
||||
BasicBlock *&Ref = Blocks[L];
|
||||
|
||||
if (Ref) {
|
||||
if (Ref == R) return false;
|
||||
|
||||
Engine.logf("successor %l cannot be equivalent to %r; "
|
||||
"it's already equivalent to %r")
|
||||
<< L << R << Ref;
|
||||
return true;
|
||||
}
|
||||
|
||||
Ref = R;
|
||||
Queue.insert(BlockPair(L, R));
|
||||
return false;
|
||||
}
|
||||
|
||||
/// Unifies two instructions, given that they're known not to have
|
||||
/// structural differences.
|
||||
void unify(Instruction *L, Instruction *R) {
|
||||
DifferenceEngine::Context C(Engine, L, R);
|
||||
|
||||
bool Result = diff(L, R, true, true);
|
||||
assert(!Result && "structural differences second time around?");
|
||||
(void) Result;
|
||||
if (!L->use_empty())
|
||||
Values[L] = R;
|
||||
}
|
||||
|
||||
void processQueue() {
|
||||
while (!Queue.empty()) {
|
||||
BlockPair Pair = Queue.remove_min();
|
||||
diff(Pair.first, Pair.second);
|
||||
}
|
||||
}
|
||||
|
||||
void diff(BasicBlock *L, BasicBlock *R) {
|
||||
DifferenceEngine::Context C(Engine, L, R);
|
||||
|
||||
BasicBlock::iterator LI = L->begin(), LE = L->end();
|
||||
BasicBlock::iterator RI = R->begin();
|
||||
|
||||
llvm::SmallVector<std::pair<Instruction*,Instruction*>, 20> TentativePairs;
|
||||
|
||||
do {
|
||||
assert(LI != LE && RI != R->end());
|
||||
Instruction *LeftI = &*LI, *RightI = &*RI;
|
||||
|
||||
// If the instructions differ, start the more sophisticated diff
|
||||
// algorithm at the start of the block.
|
||||
if (diff(LeftI, RightI, false, false)) {
|
||||
TentativeValues.clear();
|
||||
return runBlockDiff(L->begin(), R->begin());
|
||||
}
|
||||
|
||||
// Otherwise, tentatively unify them.
|
||||
if (!LeftI->use_empty())
|
||||
TentativeValues.insert(std::make_pair(LeftI, RightI));
|
||||
|
||||
++LI, ++RI;
|
||||
} while (LI != LE); // This is sufficient: we can't get equality of
|
||||
// terminators if there are residual instructions.
|
||||
|
||||
// Unify everything in the block, non-tentatively this time.
|
||||
TentativeValues.clear();
|
||||
for (LI = L->begin(), RI = R->begin(); LI != LE; ++LI, ++RI)
|
||||
unify(&*LI, &*RI);
|
||||
}
|
||||
|
||||
bool matchForBlockDiff(Instruction *L, Instruction *R);
|
||||
void runBlockDiff(BasicBlock::iterator LI, BasicBlock::iterator RI);
|
||||
|
||||
bool diffCallSites(CallSite L, CallSite R, bool Complain) {
|
||||
// FIXME: call attributes
|
||||
if (!equivalentAsOperands(L.getCalledValue(), R.getCalledValue())) {
|
||||
if (Complain) Engine.log("called functions differ");
|
||||
return true;
|
||||
}
|
||||
if (L.arg_size() != R.arg_size()) {
|
||||
if (Complain) Engine.log("argument counts differ");
|
||||
return true;
|
||||
}
|
||||
for (unsigned I = 0, E = L.arg_size(); I != E; ++I)
|
||||
if (!equivalentAsOperands(L.getArgument(I), R.getArgument(I))) {
|
||||
if (Complain)
|
||||
Engine.logf("arguments %l and %r differ")
|
||||
<< L.getArgument(I) << R.getArgument(I);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
bool diff(Instruction *L, Instruction *R, bool Complain, bool TryUnify) {
|
||||
// FIXME: metadata (if Complain is set)
|
||||
|
||||
// Different opcodes always imply different operations.
|
||||
if (L->getOpcode() != R->getOpcode()) {
|
||||
if (Complain) Engine.log("different instruction types");
|
||||
return true;
|
||||
}
|
||||
|
||||
if (isa<CmpInst>(L)) {
|
||||
if (cast<CmpInst>(L)->getPredicate()
|
||||
!= cast<CmpInst>(R)->getPredicate()) {
|
||||
if (Complain) Engine.log("different predicates");
|
||||
return true;
|
||||
}
|
||||
} else if (isa<CallInst>(L)) {
|
||||
return diffCallSites(CallSite(L), CallSite(R), Complain);
|
||||
} else if (isa<PHINode>(L)) {
|
||||
// FIXME: implement.
|
||||
|
||||
// This is really weird; type uniquing is broken?
|
||||
if (L->getType() != R->getType()) {
|
||||
if (!L->getType()->isPointerTy() || !R->getType()->isPointerTy()) {
|
||||
if (Complain) Engine.log("different phi types");
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
|
||||
// Terminators.
|
||||
} else if (isa<InvokeInst>(L)) {
|
||||
InvokeInst *LI = cast<InvokeInst>(L);
|
||||
InvokeInst *RI = cast<InvokeInst>(R);
|
||||
if (diffCallSites(CallSite(LI), CallSite(RI), Complain))
|
||||
return true;
|
||||
|
||||
if (TryUnify) {
|
||||
tryUnify(LI->getNormalDest(), RI->getNormalDest());
|
||||
tryUnify(LI->getUnwindDest(), RI->getUnwindDest());
|
||||
}
|
||||
return false;
|
||||
|
||||
} else if (isa<BranchInst>(L)) {
|
||||
BranchInst *LI = cast<BranchInst>(L);
|
||||
BranchInst *RI = cast<BranchInst>(R);
|
||||
if (LI->isConditional() != RI->isConditional()) {
|
||||
if (Complain) Engine.log("branch conditionality differs");
|
||||
return true;
|
||||
}
|
||||
|
||||
if (LI->isConditional()) {
|
||||
if (!equivalentAsOperands(LI->getCondition(), RI->getCondition())) {
|
||||
if (Complain) Engine.log("branch conditions differ");
|
||||
return true;
|
||||
}
|
||||
if (TryUnify) tryUnify(LI->getSuccessor(1), RI->getSuccessor(1));
|
||||
}
|
||||
if (TryUnify) tryUnify(LI->getSuccessor(0), RI->getSuccessor(0));
|
||||
return false;
|
||||
|
||||
} else if (isa<SwitchInst>(L)) {
|
||||
SwitchInst *LI = cast<SwitchInst>(L);
|
||||
SwitchInst *RI = cast<SwitchInst>(R);
|
||||
if (!equivalentAsOperands(LI->getCondition(), RI->getCondition())) {
|
||||
if (Complain) Engine.log("switch conditions differ");
|
||||
return true;
|
||||
}
|
||||
if (TryUnify) tryUnify(LI->getDefaultDest(), RI->getDefaultDest());
|
||||
|
||||
bool Difference = false;
|
||||
|
||||
DenseMap<ConstantInt*,BasicBlock*> LCases;
|
||||
for (unsigned I = 1, E = LI->getNumCases(); I != E; ++I)
|
||||
LCases[LI->getCaseValue(I)] = LI->getSuccessor(I);
|
||||
for (unsigned I = 1, E = RI->getNumCases(); I != E; ++I) {
|
||||
ConstantInt *CaseValue = RI->getCaseValue(I);
|
||||
BasicBlock *LCase = LCases[CaseValue];
|
||||
if (LCase) {
|
||||
if (TryUnify) tryUnify(LCase, RI->getSuccessor(I));
|
||||
LCases.erase(CaseValue);
|
||||
} else if (!Difference) {
|
||||
if (Complain)
|
||||
Engine.logf("right switch has extra case %r") << CaseValue;
|
||||
Difference = true;
|
||||
}
|
||||
}
|
||||
if (!Difference)
|
||||
for (DenseMap<ConstantInt*,BasicBlock*>::iterator
|
||||
I = LCases.begin(), E = LCases.end(); I != E; ++I) {
|
||||
if (Complain)
|
||||
Engine.logf("left switch has extra case %l") << I->first;
|
||||
Difference = true;
|
||||
}
|
||||
return Difference;
|
||||
} else if (isa<UnreachableInst>(L)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if (L->getNumOperands() != R->getNumOperands()) {
|
||||
if (Complain) Engine.log("instructions have different operand counts");
|
||||
return true;
|
||||
}
|
||||
|
||||
for (unsigned I = 0, E = L->getNumOperands(); I != E; ++I) {
|
||||
Value *LO = L->getOperand(I), *RO = R->getOperand(I);
|
||||
if (!equivalentAsOperands(LO, RO)) {
|
||||
if (Complain) Engine.logf("operands %l and %r differ") << LO << RO;
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
bool equivalentAsOperands(Constant *L, Constant *R) {
|
||||
// Use equality as a preliminary filter.
|
||||
if (L == R)
|
||||
return true;
|
||||
|
||||
if (L->getValueID() != R->getValueID())
|
||||
return false;
|
||||
|
||||
// Ask the engine about global values.
|
||||
if (isa<GlobalValue>(L))
|
||||
return Engine.equivalentAsOperands(cast<GlobalValue>(L),
|
||||
cast<GlobalValue>(R));
|
||||
|
||||
// Compare constant expressions structurally.
|
||||
if (isa<ConstantExpr>(L))
|
||||
return equivalentAsOperands(cast<ConstantExpr>(L),
|
||||
cast<ConstantExpr>(R));
|
||||
|
||||
// Nulls of the "same type" don't always actually have the same
|
||||
// type; I don't know why. Just white-list them.
|
||||
if (isa<ConstantPointerNull>(L))
|
||||
return true;
|
||||
|
||||
// Block addresses only match if we've already encountered the
|
||||
// block. FIXME: tentative matches?
|
||||
if (isa<BlockAddress>(L))
|
||||
return Blocks[cast<BlockAddress>(L)->getBasicBlock()]
|
||||
== cast<BlockAddress>(R)->getBasicBlock();
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
bool equivalentAsOperands(ConstantExpr *L, ConstantExpr *R) {
|
||||
if (L == R)
|
||||
return true;
|
||||
if (L->getOpcode() != R->getOpcode())
|
||||
return false;
|
||||
|
||||
switch (L->getOpcode()) {
|
||||
case Instruction::ICmp:
|
||||
case Instruction::FCmp:
|
||||
if (L->getPredicate() != R->getPredicate())
|
||||
return false;
|
||||
break;
|
||||
|
||||
case Instruction::GetElementPtr:
|
||||
// FIXME: inbounds?
|
||||
break;
|
||||
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
if (L->getNumOperands() != R->getNumOperands())
|
||||
return false;
|
||||
|
||||
for (unsigned I = 0, E = L->getNumOperands(); I != E; ++I)
|
||||
if (!equivalentAsOperands(L->getOperand(I), R->getOperand(I)))
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool equivalentAsOperands(Value *L, Value *R) {
|
||||
// Fall out if the values have different kind.
|
||||
// This possibly shouldn't take priority over oracles.
|
||||
if (L->getValueID() != R->getValueID())
|
||||
return false;
|
||||
|
||||
// Value subtypes: Argument, Constant, Instruction, BasicBlock,
|
||||
// InlineAsm, MDNode, MDString, PseudoSourceValue
|
||||
|
||||
if (isa<Constant>(L))
|
||||
return equivalentAsOperands(cast<Constant>(L), cast<Constant>(R));
|
||||
|
||||
if (isa<Instruction>(L))
|
||||
return Values[L] == R || TentativeValues.count(std::make_pair(L, R));
|
||||
|
||||
if (isa<Argument>(L))
|
||||
return Values[L] == R;
|
||||
|
||||
if (isa<BasicBlock>(L))
|
||||
return Blocks[cast<BasicBlock>(L)] != R;
|
||||
|
||||
// Pretend everything else is identical.
|
||||
return true;
|
||||
}
|
||||
|
||||
// Avoid a gcc warning about accessing 'this' in an initializer.
|
||||
FunctionDifferenceEngine *this_() { return this; }
|
||||
|
||||
public:
|
||||
FunctionDifferenceEngine(DifferenceEngine &Engine) :
|
||||
Engine(Engine), Queue(QueueSorter(*this_())) {}
|
||||
|
||||
void diff(Function *L, Function *R) {
|
||||
if (L->arg_size() != R->arg_size())
|
||||
Engine.log("different argument counts");
|
||||
|
||||
// Map the arguments.
|
||||
for (Function::arg_iterator
|
||||
LI = L->arg_begin(), LE = L->arg_end(),
|
||||
RI = R->arg_begin(), RE = R->arg_end();
|
||||
LI != LE && RI != RE; ++LI, ++RI)
|
||||
Values[&*LI] = &*RI;
|
||||
|
||||
tryUnify(&*L->begin(), &*R->begin());
|
||||
processQueue();
|
||||
}
|
||||
};
|
||||
|
||||
struct DiffEntry {
|
||||
DiffEntry() : Cost(0) {}
|
||||
|
||||
unsigned Cost;
|
||||
llvm::SmallVector<char, 8> Path; // actually of DifferenceEngine::DiffChange
|
||||
};
|
||||
|
||||
bool FunctionDifferenceEngine::matchForBlockDiff(Instruction *L,
|
||||
Instruction *R) {
|
||||
return !diff(L, R, false, false);
|
||||
}
|
||||
|
||||
void FunctionDifferenceEngine::runBlockDiff(BasicBlock::iterator LStart,
|
||||
BasicBlock::iterator RStart) {
|
||||
BasicBlock::iterator LE = LStart->getParent()->end();
|
||||
BasicBlock::iterator RE = RStart->getParent()->end();
|
||||
|
||||
unsigned NL = std::distance(LStart, LE);
|
||||
|
||||
SmallVector<DiffEntry, 20> Paths1(NL+1);
|
||||
SmallVector<DiffEntry, 20> Paths2(NL+1);
|
||||
|
||||
DiffEntry *Cur = Paths1.data();
|
||||
DiffEntry *Next = Paths2.data();
|
||||
|
||||
const unsigned LeftCost = 2;
|
||||
const unsigned RightCost = 2;
|
||||
const unsigned MatchCost = 0;
|
||||
|
||||
assert(TentativeValues.empty());
|
||||
|
||||
// Initialize the first column.
|
||||
for (unsigned I = 0; I != NL+1; ++I) {
|
||||
Cur[I].Cost = I * LeftCost;
|
||||
for (unsigned J = 0; J != I; ++J)
|
||||
Cur[I].Path.push_back(DC_left);
|
||||
}
|
||||
|
||||
for (BasicBlock::iterator RI = RStart; RI != RE; ++RI) {
|
||||
// Initialize the first row.
|
||||
Next[0] = Cur[0];
|
||||
Next[0].Cost += RightCost;
|
||||
Next[0].Path.push_back(DC_right);
|
||||
|
||||
unsigned Index = 1;
|
||||
for (BasicBlock::iterator LI = LStart; LI != LE; ++LI, ++Index) {
|
||||
if (matchForBlockDiff(&*LI, &*RI)) {
|
||||
Next[Index] = Cur[Index-1];
|
||||
Next[Index].Cost += MatchCost;
|
||||
Next[Index].Path.push_back(DC_match);
|
||||
TentativeValues.insert(std::make_pair(&*LI, &*RI));
|
||||
} else if (Next[Index-1].Cost <= Cur[Index].Cost) {
|
||||
Next[Index] = Next[Index-1];
|
||||
Next[Index].Cost += LeftCost;
|
||||
Next[Index].Path.push_back(DC_left);
|
||||
} else {
|
||||
Next[Index] = Cur[Index];
|
||||
Next[Index].Cost += RightCost;
|
||||
Next[Index].Path.push_back(DC_right);
|
||||
}
|
||||
}
|
||||
|
||||
std::swap(Cur, Next);
|
||||
}
|
||||
|
||||
// We don't need the tentative values anymore; everything from here
|
||||
// on out should be non-tentative.
|
||||
TentativeValues.clear();
|
||||
|
||||
SmallVectorImpl<char> &Path = Cur[NL].Path;
|
||||
BasicBlock::iterator LI = LStart, RI = RStart;
|
||||
|
||||
DiffLogBuilder Diff(Engine.getConsumer());
|
||||
|
||||
// Drop trailing matches.
|
||||
while (Path.back() == DC_match)
|
||||
Path.pop_back();
|
||||
|
||||
// Skip leading matches.
|
||||
SmallVectorImpl<char>::iterator
|
||||
PI = Path.begin(), PE = Path.end();
|
||||
while (PI != PE && *PI == DC_match) {
|
||||
unify(&*LI, &*RI);
|
||||
++PI, ++LI, ++RI;
|
||||
}
|
||||
|
||||
for (; PI != PE; ++PI) {
|
||||
switch (static_cast<DiffChange>(*PI)) {
|
||||
case DC_match:
|
||||
assert(LI != LE && RI != RE);
|
||||
{
|
||||
Instruction *L = &*LI, *R = &*RI;
|
||||
unify(L, R);
|
||||
Diff.addMatch(L, R);
|
||||
}
|
||||
++LI; ++RI;
|
||||
break;
|
||||
|
||||
case DC_left:
|
||||
assert(LI != LE);
|
||||
Diff.addLeft(&*LI);
|
||||
++LI;
|
||||
break;
|
||||
|
||||
case DC_right:
|
||||
assert(RI != RE);
|
||||
Diff.addRight(&*RI);
|
||||
++RI;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Finishing unifying and complaining about the tails of the block,
|
||||
// which should be matches all the way through.
|
||||
while (LI != LE) {
|
||||
assert(RI != RE);
|
||||
unify(&*LI, &*RI);
|
||||
++LI, ++RI;
|
||||
}
|
||||
|
||||
// If the terminators have different kinds, but one is an invoke and the
|
||||
// other is an unconditional branch immediately following a call, unify
|
||||
// the results and the destinations.
|
||||
TerminatorInst *LTerm = LStart->getParent()->getTerminator();
|
||||
TerminatorInst *RTerm = RStart->getParent()->getTerminator();
|
||||
if (isa<BranchInst>(LTerm) && isa<InvokeInst>(RTerm)) {
|
||||
if (cast<BranchInst>(LTerm)->isConditional()) return;
|
||||
BasicBlock::iterator I = LTerm;
|
||||
if (I == LStart->getParent()->begin()) return;
|
||||
--I;
|
||||
if (!isa<CallInst>(*I)) return;
|
||||
CallInst *LCall = cast<CallInst>(&*I);
|
||||
InvokeInst *RInvoke = cast<InvokeInst>(RTerm);
|
||||
if (!equivalentAsOperands(LCall->getCalledValue(), RInvoke->getCalledValue()))
|
||||
return;
|
||||
if (!LCall->use_empty())
|
||||
Values[LCall] = RInvoke;
|
||||
tryUnify(LTerm->getSuccessor(0), RInvoke->getNormalDest());
|
||||
} else if (isa<InvokeInst>(LTerm) && isa<BranchInst>(RTerm)) {
|
||||
if (cast<BranchInst>(RTerm)->isConditional()) return;
|
||||
BasicBlock::iterator I = RTerm;
|
||||
if (I == RStart->getParent()->begin()) return;
|
||||
--I;
|
||||
if (!isa<CallInst>(*I)) return;
|
||||
CallInst *RCall = cast<CallInst>(I);
|
||||
InvokeInst *LInvoke = cast<InvokeInst>(LTerm);
|
||||
if (!equivalentAsOperands(LInvoke->getCalledValue(), RCall->getCalledValue()))
|
||||
return;
|
||||
if (!LInvoke->use_empty())
|
||||
Values[LInvoke] = RCall;
|
||||
tryUnify(LInvoke->getNormalDest(), RTerm->getSuccessor(0));
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void DifferenceEngine::diff(Function *L, Function *R) {
|
||||
Context C(*this, L, R);
|
||||
|
||||
// FIXME: types
|
||||
// FIXME: attributes and CC
|
||||
// FIXME: parameter attributes
|
||||
|
||||
// If both are declarations, we're done.
|
||||
if (L->empty() && R->empty())
|
||||
return;
|
||||
else if (L->empty())
|
||||
log("left function is declaration, right function is definition");
|
||||
else if (R->empty())
|
||||
log("right function is declaration, left function is definition");
|
||||
else
|
||||
FunctionDifferenceEngine(*this).diff(L, R);
|
||||
}
|
||||
|
||||
void DifferenceEngine::diff(Module *L, Module *R) {
|
||||
StringSet<> LNames;
|
||||
SmallVector<std::pair<Function*,Function*>, 20> Queue;
|
||||
|
||||
for (Module::iterator I = L->begin(), E = L->end(); I != E; ++I) {
|
||||
Function *LFn = &*I;
|
||||
LNames.insert(LFn->getName());
|
||||
|
||||
if (Function *RFn = R->getFunction(LFn->getName()))
|
||||
Queue.push_back(std::make_pair(LFn, RFn));
|
||||
else
|
||||
logf("function %l exists only in left module") << LFn;
|
||||
}
|
||||
|
||||
for (Module::iterator I = R->begin(), E = R->end(); I != E; ++I) {
|
||||
Function *RFn = &*I;
|
||||
if (!LNames.count(RFn->getName()))
|
||||
logf("function %r exists only in right module") << RFn;
|
||||
}
|
||||
|
||||
for (SmallVectorImpl<std::pair<Function*,Function*> >::iterator
|
||||
I = Queue.begin(), E = Queue.end(); I != E; ++I)
|
||||
diff(I->first, I->second);
|
||||
}
|
||||
|
||||
bool DifferenceEngine::equivalentAsOperands(GlobalValue *L, GlobalValue *R) {
|
||||
if (globalValueOracle) return (*globalValueOracle)(L, R);
|
||||
return L->getName() == R->getName();
|
||||
}
|
91
contrib/llvm/tools/llvm-diff/DifferenceEngine.h
Normal file
91
contrib/llvm/tools/llvm-diff/DifferenceEngine.h
Normal file
@ -0,0 +1,91 @@
|
||||
//===-- DifferenceEngine.h - Module comparator ------------------*- C++ -*-===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This header defines the interface to the LLVM difference engine,
|
||||
// which structurally compares functions within a module.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef _LLVM_DIFFERENCE_ENGINE_H_
|
||||
#define _LLVM_DIFFERENCE_ENGINE_H_
|
||||
|
||||
#include "llvm/ADT/SmallVector.h"
|
||||
#include "llvm/ADT/StringRef.h"
|
||||
#include "DiffLog.h"
|
||||
#include "DiffConsumer.h"
|
||||
|
||||
#include <utility>
|
||||
|
||||
namespace llvm {
|
||||
class Function;
|
||||
class GlobalValue;
|
||||
class Instruction;
|
||||
class LLVMContext;
|
||||
class Module;
|
||||
class Twine;
|
||||
class Value;
|
||||
|
||||
/// A class for performing structural comparisons of LLVM assembly.
|
||||
class DifferenceEngine {
|
||||
public:
|
||||
/// A RAII object for recording the current context.
|
||||
struct Context {
|
||||
Context(DifferenceEngine &Engine, Value *L, Value *R) : Engine(Engine) {
|
||||
Engine.consumer.enterContext(L, R);
|
||||
}
|
||||
|
||||
~Context() {
|
||||
Engine.consumer.exitContext();
|
||||
}
|
||||
|
||||
private:
|
||||
DifferenceEngine &Engine;
|
||||
};
|
||||
|
||||
/// An oracle for answering whether two values are equivalent as
|
||||
/// operands.
|
||||
struct Oracle {
|
||||
virtual bool operator()(Value *L, Value *R) = 0;
|
||||
|
||||
protected:
|
||||
virtual ~Oracle() {}
|
||||
};
|
||||
|
||||
DifferenceEngine(LLVMContext &context, Consumer &consumer)
|
||||
: context(context), consumer(consumer), globalValueOracle(0) {}
|
||||
|
||||
void diff(Module *L, Module *R);
|
||||
void diff(Function *L, Function *R);
|
||||
void log(StringRef text) {
|
||||
consumer.log(text);
|
||||
}
|
||||
LogBuilder logf(StringRef text) {
|
||||
return LogBuilder(consumer, text);
|
||||
}
|
||||
Consumer& getConsumer() const { return consumer; }
|
||||
|
||||
/// Installs an oracle to decide whether two global values are
|
||||
/// equivalent as operands. Without an oracle, global values are
|
||||
/// considered equivalent as operands precisely when they have the
|
||||
/// same name.
|
||||
void setGlobalValueOracle(Oracle *oracle) {
|
||||
globalValueOracle = oracle;
|
||||
}
|
||||
|
||||
/// Determines whether two global values are equivalent.
|
||||
bool equivalentAsOperands(GlobalValue *L, GlobalValue *R);
|
||||
|
||||
private:
|
||||
LLVMContext &context;
|
||||
Consumer &consumer;
|
||||
Oracle *globalValueOracle;
|
||||
};
|
||||
}
|
||||
|
||||
#endif
|
17
contrib/llvm/tools/llvm-diff/Makefile
Normal file
17
contrib/llvm/tools/llvm-diff/Makefile
Normal file
@ -0,0 +1,17 @@
|
||||
##===- tools/llvm-diff/Makefile ----------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
|
||||
LEVEL = ../..
|
||||
TOOLNAME = llvm-diff
|
||||
LINK_COMPONENTS := asmparser bitreader
|
||||
|
||||
# This tool has no plugins, optimize startup time.
|
||||
TOOL_NO_EXPORTS = 1
|
||||
|
||||
include $(LEVEL)/Makefile.common
|
98
contrib/llvm/tools/llvm-diff/llvm-diff.cpp
Normal file
98
contrib/llvm/tools/llvm-diff/llvm-diff.cpp
Normal file
@ -0,0 +1,98 @@
|
||||
//===-- llvm-diff.cpp - Module comparator command-line driver ---*- C++ -*-===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file defines the command-line driver for the difference engine.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "DiffLog.h"
|
||||
#include "DifferenceEngine.h"
|
||||
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/Type.h"
|
||||
#include "llvm/ADT/DenseMap.h"
|
||||
#include "llvm/ADT/SmallVector.h"
|
||||
#include "llvm/ADT/StringRef.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/IRReader.h"
|
||||
#include "llvm/Support/MemoryBuffer.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/SourceMgr.h"
|
||||
|
||||
#include <string>
|
||||
#include <utility>
|
||||
|
||||
|
||||
using namespace llvm;
|
||||
|
||||
/// Reads a module from a file. On error, messages are written to stderr
|
||||
/// and null is returned.
|
||||
static Module *ReadModule(LLVMContext &Context, StringRef Name) {
|
||||
SMDiagnostic Diag;
|
||||
Module *M = ParseIRFile(Name, Diag, Context);
|
||||
if (!M)
|
||||
Diag.Print("llvmdiff", errs());
|
||||
return M;
|
||||
}
|
||||
|
||||
static void diffGlobal(DifferenceEngine &Engine, Module *L, Module *R,
|
||||
StringRef Name) {
|
||||
// Drop leading sigils from the global name.
|
||||
if (Name.startswith("@")) Name = Name.substr(1);
|
||||
|
||||
Function *LFn = L->getFunction(Name);
|
||||
Function *RFn = R->getFunction(Name);
|
||||
if (LFn && RFn)
|
||||
Engine.diff(LFn, RFn);
|
||||
else if (!LFn && !RFn)
|
||||
errs() << "No function named @" << Name << " in either module\n";
|
||||
else if (!LFn)
|
||||
errs() << "No function named @" << Name << " in left module\n";
|
||||
else
|
||||
errs() << "No function named @" << Name << " in right module\n";
|
||||
}
|
||||
|
||||
static cl::opt<std::string> LeftFilename(cl::Positional,
|
||||
cl::desc("<first file>"),
|
||||
cl::Required);
|
||||
static cl::opt<std::string> RightFilename(cl::Positional,
|
||||
cl::desc("<second file>"),
|
||||
cl::Required);
|
||||
static cl::list<std::string> GlobalsToCompare(cl::Positional,
|
||||
cl::desc("<globals to compare>"));
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
cl::ParseCommandLineOptions(argc, argv);
|
||||
|
||||
LLVMContext Context;
|
||||
|
||||
// Load both modules. Die if that fails.
|
||||
Module *LModule = ReadModule(Context, LeftFilename);
|
||||
Module *RModule = ReadModule(Context, RightFilename);
|
||||
if (!LModule || !RModule) return 1;
|
||||
|
||||
DiffConsumer Consumer(LModule, RModule);
|
||||
DifferenceEngine Engine(Context, Consumer);
|
||||
|
||||
// If any global names were given, just diff those.
|
||||
if (!GlobalsToCompare.empty()) {
|
||||
for (unsigned I = 0, E = GlobalsToCompare.size(); I != E; ++I)
|
||||
diffGlobal(Engine, LModule, RModule, GlobalsToCompare[I]);
|
||||
|
||||
// Otherwise, diff everything in the module.
|
||||
} else {
|
||||
Engine.diff(LModule, RModule);
|
||||
}
|
||||
|
||||
delete LModule;
|
||||
delete RModule;
|
||||
|
||||
return Consumer.hadDifferences();
|
||||
}
|
6
contrib/llvm/tools/llvm-dis/CMakeLists.txt
Normal file
6
contrib/llvm/tools/llvm-dis/CMakeLists.txt
Normal file
@ -0,0 +1,6 @@
|
||||
set(LLVM_LINK_COMPONENTS bitreader analysis)
|
||||
set(LLVM_REQUIRES_EH 1)
|
||||
|
||||
add_llvm_tool(llvm-dis
|
||||
llvm-dis.cpp
|
||||
)
|
17
contrib/llvm/tools/llvm-dis/Makefile
Normal file
17
contrib/llvm/tools/llvm-dis/Makefile
Normal file
@ -0,0 +1,17 @@
|
||||
##===- tools/llvm-dis/Makefile ------------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
LEVEL = ../..
|
||||
|
||||
TOOLNAME = llvm-dis
|
||||
LINK_COMPONENTS := bitreader analysis
|
||||
|
||||
# This tool has no plugins, optimize startup time.
|
||||
TOOL_NO_EXPORTS = 1
|
||||
|
||||
include $(LEVEL)/Makefile.common
|
186
contrib/llvm/tools/llvm-dis/llvm-dis.cpp
Normal file
186
contrib/llvm/tools/llvm-dis/llvm-dis.cpp
Normal file
@ -0,0 +1,186 @@
|
||||
//===-- llvm-dis.cpp - The low-level LLVM disassembler --------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This utility may be invoked in the following manner:
|
||||
// llvm-dis [options] - Read LLVM bitcode from stdin, write asm to stdout
|
||||
// llvm-dis [options] x.bc - Read LLVM bitcode from the x.bc file, write asm
|
||||
// to the x.ll file.
|
||||
// Options:
|
||||
// --help - Output information about command line switches
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/Type.h"
|
||||
#include "llvm/IntrinsicInst.h"
|
||||
#include "llvm/Bitcode/ReaderWriter.h"
|
||||
#include "llvm/Analysis/DebugInfo.h"
|
||||
#include "llvm/Assembly/AssemblyAnnotationWriter.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/FormattedStream.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/MemoryBuffer.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/ToolOutputFile.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include "llvm/Support/system_error.h"
|
||||
using namespace llvm;
|
||||
|
||||
static cl::opt<std::string>
|
||||
InputFilename(cl::Positional, cl::desc("<input bitcode>"), cl::init("-"));
|
||||
|
||||
static cl::opt<std::string>
|
||||
OutputFilename("o", cl::desc("Override output filename"),
|
||||
cl::value_desc("filename"));
|
||||
|
||||
static cl::opt<bool>
|
||||
Force("f", cl::desc("Enable binary output on terminals"));
|
||||
|
||||
static cl::opt<bool>
|
||||
DontPrint("disable-output", cl::desc("Don't output the .ll file"), cl::Hidden);
|
||||
|
||||
static cl::opt<bool>
|
||||
ShowAnnotations("show-annotations",
|
||||
cl::desc("Add informational comments to the .ll file"));
|
||||
|
||||
namespace {
|
||||
|
||||
static void printDebugLoc(const DebugLoc &DL, formatted_raw_ostream &OS) {
|
||||
OS << DL.getLine() << ":" << DL.getCol();
|
||||
if (MDNode *N = DL.getInlinedAt(getGlobalContext())) {
|
||||
DebugLoc IDL = DebugLoc::getFromDILocation(N);
|
||||
if (!IDL.isUnknown()) {
|
||||
OS << "@";
|
||||
printDebugLoc(IDL,OS);
|
||||
}
|
||||
}
|
||||
}
|
||||
class CommentWriter : public AssemblyAnnotationWriter {
|
||||
public:
|
||||
void emitFunctionAnnot(const Function *F,
|
||||
formatted_raw_ostream &OS) {
|
||||
OS << "; [#uses=" << F->getNumUses() << ']'; // Output # uses
|
||||
OS << '\n';
|
||||
}
|
||||
void printInfoComment(const Value &V, formatted_raw_ostream &OS) {
|
||||
bool Padded = false;
|
||||
if (!V.getType()->isVoidTy()) {
|
||||
OS.PadToColumn(50);
|
||||
Padded = true;
|
||||
OS << "; [#uses=" << V.getNumUses() << " type=" << *V.getType() << "]"; // Output # uses and type
|
||||
}
|
||||
if (const Instruction *I = dyn_cast<Instruction>(&V)) {
|
||||
const DebugLoc &DL = I->getDebugLoc();
|
||||
if (!DL.isUnknown()) {
|
||||
if (!Padded) {
|
||||
OS.PadToColumn(50);
|
||||
Padded = true;
|
||||
OS << ";";
|
||||
}
|
||||
OS << " [debug line = ";
|
||||
printDebugLoc(DL,OS);
|
||||
OS << "]";
|
||||
}
|
||||
if (const DbgDeclareInst *DDI = dyn_cast<DbgDeclareInst>(I)) {
|
||||
DIVariable Var(DDI->getVariable());
|
||||
if (!Padded) {
|
||||
OS.PadToColumn(50);
|
||||
Padded = true;
|
||||
OS << ";";
|
||||
}
|
||||
OS << " [debug variable = " << Var.getName() << "]";
|
||||
}
|
||||
else if (const DbgValueInst *DVI = dyn_cast<DbgValueInst>(I)) {
|
||||
DIVariable Var(DVI->getVariable());
|
||||
if (!Padded) {
|
||||
OS.PadToColumn(50);
|
||||
Padded = true;
|
||||
OS << ";";
|
||||
}
|
||||
OS << " [debug variable = " << Var.getName() << "]";
|
||||
}
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
} // end anon namespace
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
// Print a stack trace if we signal out.
|
||||
sys::PrintStackTraceOnErrorSignal();
|
||||
PrettyStackTraceProgram X(argc, argv);
|
||||
|
||||
LLVMContext &Context = getGlobalContext();
|
||||
llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
|
||||
|
||||
|
||||
cl::ParseCommandLineOptions(argc, argv, "llvm .bc -> .ll disassembler\n");
|
||||
|
||||
std::string ErrorMessage;
|
||||
std::auto_ptr<Module> M;
|
||||
|
||||
{
|
||||
OwningPtr<MemoryBuffer> BufferPtr;
|
||||
if (error_code ec = MemoryBuffer::getFileOrSTDIN(InputFilename, BufferPtr))
|
||||
ErrorMessage = ec.message();
|
||||
else
|
||||
M.reset(ParseBitcodeFile(BufferPtr.get(), Context, &ErrorMessage));
|
||||
}
|
||||
|
||||
if (M.get() == 0) {
|
||||
errs() << argv[0] << ": ";
|
||||
if (ErrorMessage.size())
|
||||
errs() << ErrorMessage << "\n";
|
||||
else
|
||||
errs() << "bitcode didn't read correctly.\n";
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Just use stdout. We won't actually print anything on it.
|
||||
if (DontPrint)
|
||||
OutputFilename = "-";
|
||||
|
||||
if (OutputFilename.empty()) { // Unspecified output, infer it.
|
||||
if (InputFilename == "-") {
|
||||
OutputFilename = "-";
|
||||
} else {
|
||||
const std::string &IFN = InputFilename;
|
||||
int Len = IFN.length();
|
||||
// If the source ends in .bc, strip it off.
|
||||
if (IFN[Len-3] == '.' && IFN[Len-2] == 'b' && IFN[Len-1] == 'c')
|
||||
OutputFilename = std::string(IFN.begin(), IFN.end()-3)+".ll";
|
||||
else
|
||||
OutputFilename = IFN+".ll";
|
||||
}
|
||||
}
|
||||
|
||||
std::string ErrorInfo;
|
||||
OwningPtr<tool_output_file>
|
||||
Out(new tool_output_file(OutputFilename.c_str(), ErrorInfo,
|
||||
raw_fd_ostream::F_Binary));
|
||||
if (!ErrorInfo.empty()) {
|
||||
errs() << ErrorInfo << '\n';
|
||||
return 1;
|
||||
}
|
||||
|
||||
OwningPtr<AssemblyAnnotationWriter> Annotator;
|
||||
if (ShowAnnotations)
|
||||
Annotator.reset(new CommentWriter());
|
||||
|
||||
// All that llvm-dis does is write the assembly to a file.
|
||||
if (!DontPrint)
|
||||
M->print(Out->os(), Annotator.get());
|
||||
|
||||
// Declare success.
|
||||
Out->keep();
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
5
contrib/llvm/tools/llvm-extract/CMakeLists.txt
Normal file
5
contrib/llvm/tools/llvm-extract/CMakeLists.txt
Normal file
@ -0,0 +1,5 @@
|
||||
set(LLVM_LINK_COMPONENTS asmparser ipo bitreader bitwriter)
|
||||
|
||||
add_llvm_tool(llvm-extract
|
||||
llvm-extract.cpp
|
||||
)
|
18
contrib/llvm/tools/llvm-extract/Makefile
Normal file
18
contrib/llvm/tools/llvm-extract/Makefile
Normal file
@ -0,0 +1,18 @@
|
||||
##===- tools/llvm-extract/Makefile -------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
|
||||
LEVEL = ../..
|
||||
|
||||
TOOLNAME = llvm-extract
|
||||
LINK_COMPONENTS := ipo bitreader bitwriter asmparser
|
||||
|
||||
# This tool has no plugins, optimize startup time.
|
||||
TOOL_NO_EXPORTS = 1
|
||||
|
||||
include $(LEVEL)/Makefile.common
|
238
contrib/llvm/tools/llvm-extract/llvm-extract.cpp
Normal file
238
contrib/llvm/tools/llvm-extract/llvm-extract.cpp
Normal file
@ -0,0 +1,238 @@
|
||||
//===- llvm-extract.cpp - LLVM function extraction utility ----------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This utility changes the input module to only contain a single function,
|
||||
// which is primarily used for debugging transformations.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/PassManager.h"
|
||||
#include "llvm/Assembly/PrintModulePass.h"
|
||||
#include "llvm/Bitcode/ReaderWriter.h"
|
||||
#include "llvm/Transforms/IPO.h"
|
||||
#include "llvm/Target/TargetData.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/IRReader.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/ToolOutputFile.h"
|
||||
#include "llvm/Support/SystemUtils.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include "llvm/Support/Regex.h"
|
||||
#include "llvm/ADT/SmallPtrSet.h"
|
||||
#include "llvm/ADT/SetVector.h"
|
||||
#include <memory>
|
||||
using namespace llvm;
|
||||
|
||||
// InputFilename - The filename to read from.
|
||||
static cl::opt<std::string>
|
||||
InputFilename(cl::Positional, cl::desc("<input bitcode file>"),
|
||||
cl::init("-"), cl::value_desc("filename"));
|
||||
|
||||
static cl::opt<std::string>
|
||||
OutputFilename("o", cl::desc("Specify output filename"),
|
||||
cl::value_desc("filename"), cl::init("-"));
|
||||
|
||||
static cl::opt<bool>
|
||||
Force("f", cl::desc("Enable binary output on terminals"));
|
||||
|
||||
static cl::opt<bool>
|
||||
DeleteFn("delete", cl::desc("Delete specified Globals from Module"));
|
||||
|
||||
// ExtractFuncs - The functions to extract from the module.
|
||||
static cl::list<std::string>
|
||||
ExtractFuncs("func", cl::desc("Specify function to extract"),
|
||||
cl::ZeroOrMore, cl::value_desc("function"));
|
||||
|
||||
// ExtractRegExpFuncs - The functions, matched via regular expression, to
|
||||
// extract from the module.
|
||||
static cl::list<std::string>
|
||||
ExtractRegExpFuncs("rfunc", cl::desc("Specify function(s) to extract using a "
|
||||
"regular expression"),
|
||||
cl::ZeroOrMore, cl::value_desc("rfunction"));
|
||||
|
||||
// ExtractGlobals - The globals to extract from the module.
|
||||
static cl::list<std::string>
|
||||
ExtractGlobals("glob", cl::desc("Specify global to extract"),
|
||||
cl::ZeroOrMore, cl::value_desc("global"));
|
||||
|
||||
// ExtractRegExpGlobals - The globals, matched via regular expression, to
|
||||
// extract from the module...
|
||||
static cl::list<std::string>
|
||||
ExtractRegExpGlobals("rglob", cl::desc("Specify global(s) to extract using a "
|
||||
"regular expression"),
|
||||
cl::ZeroOrMore, cl::value_desc("rglobal"));
|
||||
|
||||
static cl::opt<bool>
|
||||
OutputAssembly("S",
|
||||
cl::desc("Write output as LLVM assembly"), cl::Hidden);
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
// Print a stack trace if we signal out.
|
||||
sys::PrintStackTraceOnErrorSignal();
|
||||
PrettyStackTraceProgram X(argc, argv);
|
||||
|
||||
LLVMContext &Context = getGlobalContext();
|
||||
llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
|
||||
cl::ParseCommandLineOptions(argc, argv, "llvm extractor\n");
|
||||
|
||||
// Use lazy loading, since we only care about selected global values.
|
||||
SMDiagnostic Err;
|
||||
std::auto_ptr<Module> M;
|
||||
M.reset(getLazyIRFileModule(InputFilename, Err, Context));
|
||||
|
||||
if (M.get() == 0) {
|
||||
Err.Print(argv[0], errs());
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Use SetVector to avoid duplicates.
|
||||
SetVector<GlobalValue *> GVs;
|
||||
|
||||
// Figure out which globals we should extract.
|
||||
for (size_t i = 0, e = ExtractGlobals.size(); i != e; ++i) {
|
||||
GlobalValue *GV = M.get()->getNamedGlobal(ExtractGlobals[i]);
|
||||
if (!GV) {
|
||||
errs() << argv[0] << ": program doesn't contain global named '"
|
||||
<< ExtractGlobals[i] << "'!\n";
|
||||
return 1;
|
||||
}
|
||||
GVs.insert(GV);
|
||||
}
|
||||
|
||||
// Extract globals via regular expression matching.
|
||||
for (size_t i = 0, e = ExtractRegExpGlobals.size(); i != e; ++i) {
|
||||
std::string Error;
|
||||
Regex RegEx(ExtractRegExpGlobals[i]);
|
||||
if (!RegEx.isValid(Error)) {
|
||||
errs() << argv[0] << ": '" << ExtractRegExpGlobals[i] << "' "
|
||||
"invalid regex: " << Error;
|
||||
}
|
||||
bool match = false;
|
||||
for (Module::global_iterator GV = M.get()->global_begin(),
|
||||
E = M.get()->global_end(); GV != E; GV++) {
|
||||
if (RegEx.match(GV->getName())) {
|
||||
GVs.insert(&*GV);
|
||||
match = true;
|
||||
}
|
||||
}
|
||||
if (!match) {
|
||||
errs() << argv[0] << ": program doesn't contain global named '"
|
||||
<< ExtractRegExpGlobals[i] << "'!\n";
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
|
||||
// Figure out which functions we should extract.
|
||||
for (size_t i = 0, e = ExtractFuncs.size(); i != e; ++i) {
|
||||
GlobalValue *GV = M.get()->getFunction(ExtractFuncs[i]);
|
||||
if (!GV) {
|
||||
errs() << argv[0] << ": program doesn't contain function named '"
|
||||
<< ExtractFuncs[i] << "'!\n";
|
||||
return 1;
|
||||
}
|
||||
GVs.insert(GV);
|
||||
}
|
||||
// Extract functions via regular expression matching.
|
||||
for (size_t i = 0, e = ExtractRegExpFuncs.size(); i != e; ++i) {
|
||||
std::string Error;
|
||||
StringRef RegExStr = ExtractRegExpFuncs[i];
|
||||
Regex RegEx(RegExStr);
|
||||
if (!RegEx.isValid(Error)) {
|
||||
errs() << argv[0] << ": '" << ExtractRegExpFuncs[i] << "' "
|
||||
"invalid regex: " << Error;
|
||||
}
|
||||
bool match = false;
|
||||
for (Module::iterator F = M.get()->begin(), E = M.get()->end(); F != E;
|
||||
F++) {
|
||||
if (RegEx.match(F->getName())) {
|
||||
GVs.insert(&*F);
|
||||
match = true;
|
||||
}
|
||||
}
|
||||
if (!match) {
|
||||
errs() << argv[0] << ": program doesn't contain global named '"
|
||||
<< ExtractRegExpFuncs[i] << "'!\n";
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
|
||||
// Materialize requisite global values.
|
||||
if (!DeleteFn)
|
||||
for (size_t i = 0, e = GVs.size(); i != e; ++i) {
|
||||
GlobalValue *GV = GVs[i];
|
||||
if (GV->isMaterializable()) {
|
||||
std::string ErrInfo;
|
||||
if (GV->Materialize(&ErrInfo)) {
|
||||
errs() << argv[0] << ": error reading input: " << ErrInfo << "\n";
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
else {
|
||||
// Deleting. Materialize every GV that's *not* in GVs.
|
||||
SmallPtrSet<GlobalValue *, 8> GVSet(GVs.begin(), GVs.end());
|
||||
for (Module::global_iterator I = M->global_begin(), E = M->global_end();
|
||||
I != E; ++I) {
|
||||
GlobalVariable *G = I;
|
||||
if (!GVSet.count(G) && G->isMaterializable()) {
|
||||
std::string ErrInfo;
|
||||
if (G->Materialize(&ErrInfo)) {
|
||||
errs() << argv[0] << ": error reading input: " << ErrInfo << "\n";
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
for (Module::iterator I = M->begin(), E = M->end(); I != E; ++I) {
|
||||
Function *F = I;
|
||||
if (!GVSet.count(F) && F->isMaterializable()) {
|
||||
std::string ErrInfo;
|
||||
if (F->Materialize(&ErrInfo)) {
|
||||
errs() << argv[0] << ": error reading input: " << ErrInfo << "\n";
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// In addition to deleting all other functions, we also want to spiff it
|
||||
// up a little bit. Do this now.
|
||||
PassManager Passes;
|
||||
Passes.add(new TargetData(M.get())); // Use correct TargetData
|
||||
|
||||
std::vector<GlobalValue*> Gvs(GVs.begin(), GVs.end());
|
||||
|
||||
Passes.add(createGVExtractionPass(Gvs, DeleteFn));
|
||||
if (!DeleteFn)
|
||||
Passes.add(createGlobalDCEPass()); // Delete unreachable globals
|
||||
Passes.add(createStripDeadDebugInfoPass()); // Remove dead debug info
|
||||
Passes.add(createStripDeadPrototypesPass()); // Remove dead func decls
|
||||
|
||||
std::string ErrorInfo;
|
||||
tool_output_file Out(OutputFilename.c_str(), ErrorInfo,
|
||||
raw_fd_ostream::F_Binary);
|
||||
if (!ErrorInfo.empty()) {
|
||||
errs() << ErrorInfo << '\n';
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (OutputAssembly)
|
||||
Passes.add(createPrintModulePass(&Out.os()));
|
||||
else if (Force || !CheckBitcodeOutputToConsole(Out.os(), true))
|
||||
Passes.add(createBitcodeWriterPass(Out.os()));
|
||||
|
||||
Passes.run(*M.get());
|
||||
|
||||
// Declare success.
|
||||
Out.keep();
|
||||
|
||||
return 0;
|
||||
}
|
8
contrib/llvm/tools/llvm-ld/CMakeLists.txt
Normal file
8
contrib/llvm/tools/llvm-ld/CMakeLists.txt
Normal file
@ -0,0 +1,8 @@
|
||||
set(LLVM_LINK_COMPONENTS ipo scalaropts linker archive bitwriter)
|
||||
|
||||
add_llvm_tool(llvm-ld
|
||||
Optimize.cpp
|
||||
llvm-ld.cpp
|
||||
)
|
||||
|
||||
add_dependencies(llvm-ld llvm-stub)
|
15
contrib/llvm/tools/llvm-ld/Makefile
Normal file
15
contrib/llvm/tools/llvm-ld/Makefile
Normal file
@ -0,0 +1,15 @@
|
||||
##===- tools/llvm-ld/Makefile ------------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
|
||||
LEVEL = ../..
|
||||
|
||||
TOOLNAME = llvm-ld
|
||||
LINK_COMPONENTS = ipo scalaropts linker archive bitwriter
|
||||
|
||||
include $(LEVEL)/Makefile.common
|
130
contrib/llvm/tools/llvm-ld/Optimize.cpp
Normal file
130
contrib/llvm/tools/llvm-ld/Optimize.cpp
Normal file
@ -0,0 +1,130 @@
|
||||
//===- Optimize.cpp - Optimize a complete program -------------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file implements all optimization of the linked module for llvm-ld.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/PassManager.h"
|
||||
#include "llvm/Analysis/Verifier.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/DynamicLibrary.h"
|
||||
#include "llvm/Target/TargetData.h"
|
||||
#include "llvm/Target/TargetMachine.h"
|
||||
#include "llvm/Support/PassNameParser.h"
|
||||
#include "llvm/Support/PluginLoader.h"
|
||||
#include "llvm/Transforms/IPO.h"
|
||||
#include "llvm/Transforms/IPO/PassManagerBuilder.h"
|
||||
#include "llvm/Transforms/Scalar.h"
|
||||
using namespace llvm;
|
||||
|
||||
// Pass Name Options as generated by the PassNameParser
|
||||
static cl::list<const PassInfo*, bool, PassNameParser>
|
||||
OptimizationList(cl::desc("Optimizations available:"));
|
||||
|
||||
//Don't verify at the end
|
||||
static cl::opt<bool> DontVerify("disable-verify", cl::ReallyHidden);
|
||||
|
||||
static cl::opt<bool> DisableInline("disable-inlining",
|
||||
cl::desc("Do not run the inliner pass"));
|
||||
|
||||
static cl::opt<bool>
|
||||
DisableOptimizations("disable-opt",
|
||||
cl::desc("Do not run any optimization passes"));
|
||||
|
||||
static cl::opt<bool> DisableInternalize("disable-internalize",
|
||||
cl::desc("Do not mark all symbols as internal"));
|
||||
|
||||
static cl::opt<bool> VerifyEach("verify-each",
|
||||
cl::desc("Verify intermediate results of all passes"));
|
||||
|
||||
static cl::alias ExportDynamic("export-dynamic",
|
||||
cl::aliasopt(DisableInternalize),
|
||||
cl::desc("Alias for -disable-internalize"));
|
||||
|
||||
static cl::opt<bool> Strip("strip-all",
|
||||
cl::desc("Strip all symbol info from executable"));
|
||||
|
||||
static cl::alias A0("s", cl::desc("Alias for --strip-all"),
|
||||
cl::aliasopt(Strip));
|
||||
|
||||
static cl::opt<bool> StripDebug("strip-debug",
|
||||
cl::desc("Strip debugger symbol info from executable"));
|
||||
|
||||
static cl::alias A1("S", cl::desc("Alias for --strip-debug"),
|
||||
cl::aliasopt(StripDebug));
|
||||
|
||||
// A utility function that adds a pass to the pass manager but will also add
|
||||
// a verifier pass after if we're supposed to verify.
|
||||
static inline void addPass(PassManager &PM, Pass *P) {
|
||||
// Add the pass to the pass manager...
|
||||
PM.add(P);
|
||||
|
||||
// If we are verifying all of the intermediate steps, add the verifier...
|
||||
if (VerifyEach)
|
||||
PM.add(createVerifierPass());
|
||||
}
|
||||
|
||||
namespace llvm {
|
||||
/// Optimize - Perform link time optimizations. This will run the scalar
|
||||
/// optimizations, any loaded plugin-optimization modules, and then the
|
||||
/// inter-procedural optimizations if applicable.
|
||||
void Optimize(Module *M) {
|
||||
|
||||
// Instantiate the pass manager to organize the passes.
|
||||
PassManager Passes;
|
||||
|
||||
// If we're verifying, start off with a verification pass.
|
||||
if (VerifyEach)
|
||||
Passes.add(createVerifierPass());
|
||||
|
||||
// Add an appropriate TargetData instance for this module...
|
||||
addPass(Passes, new TargetData(M));
|
||||
|
||||
if (!DisableOptimizations)
|
||||
PassManagerBuilder().populateLTOPassManager(Passes, !DisableInternalize,
|
||||
!DisableInline);
|
||||
|
||||
// If the -s or -S command line options were specified, strip the symbols out
|
||||
// of the resulting program to make it smaller. -s and -S are GNU ld options
|
||||
// that we are supporting; they alias -strip-all and -strip-debug.
|
||||
if (Strip || StripDebug)
|
||||
addPass(Passes, createStripSymbolsPass(StripDebug && !Strip));
|
||||
|
||||
// Create a new optimization pass for each one specified on the command line
|
||||
std::auto_ptr<TargetMachine> target;
|
||||
for (unsigned i = 0; i < OptimizationList.size(); ++i) {
|
||||
const PassInfo *Opt = OptimizationList[i];
|
||||
if (Opt->getNormalCtor())
|
||||
addPass(Passes, Opt->getNormalCtor()());
|
||||
else
|
||||
errs() << "llvm-ld: cannot create pass: " << Opt->getPassName()
|
||||
<< "\n";
|
||||
}
|
||||
|
||||
// The user's passes may leave cruft around. Clean up after them them but
|
||||
// only if we haven't got DisableOptimizations set
|
||||
if (!DisableOptimizations) {
|
||||
addPass(Passes, createInstructionCombiningPass());
|
||||
addPass(Passes, createCFGSimplificationPass());
|
||||
addPass(Passes, createAggressiveDCEPass());
|
||||
addPass(Passes, createGlobalDCEPass());
|
||||
}
|
||||
|
||||
// Make sure everything is still good.
|
||||
if (!DontVerify)
|
||||
Passes.add(createVerifierPass());
|
||||
|
||||
// Run our queue of passes all at once now, efficiently.
|
||||
Passes.run(*M);
|
||||
}
|
||||
|
||||
}
|
732
contrib/llvm/tools/llvm-ld/llvm-ld.cpp
Normal file
732
contrib/llvm/tools/llvm-ld/llvm-ld.cpp
Normal file
@ -0,0 +1,732 @@
|
||||
//===- llvm-ld.cpp - LLVM 'ld' compatible linker --------------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This utility is intended to be compatible with GCC, and follows standard
|
||||
// system 'ld' conventions. As such, the default output file is ./a.out.
|
||||
// Additionally, this program outputs a shell script that is used to invoke LLI
|
||||
// to execute the program. In this manner, the generated executable (a.out for
|
||||
// example), is directly executable, whereas the bitcode file actually lives in
|
||||
// the a.out.bc file generated by this program.
|
||||
//
|
||||
// Note that if someone (or a script) deletes the executable program generated,
|
||||
// the .bc file will be left around. Considering that this is a temporary hack,
|
||||
// I'm not too worried about this.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/LinkAllVMCore.h"
|
||||
#include "llvm/Linker.h"
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/Support/Program.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/PassManager.h"
|
||||
#include "llvm/Bitcode/ReaderWriter.h"
|
||||
#include "llvm/Target/TargetData.h"
|
||||
#include "llvm/Target/TargetMachine.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/FileUtilities.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/MemoryBuffer.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/SystemUtils.h"
|
||||
#include "llvm/Support/ToolOutputFile.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include "llvm/Config/config.h"
|
||||
#include <memory>
|
||||
#include <cstring>
|
||||
using namespace llvm;
|
||||
|
||||
// Rightly this should go in a header file but it just seems such a waste.
|
||||
namespace llvm {
|
||||
extern void Optimize(Module*);
|
||||
}
|
||||
|
||||
// Input/Output Options
|
||||
static cl::list<std::string> InputFilenames(cl::Positional, cl::OneOrMore,
|
||||
cl::desc("<input bitcode files>"));
|
||||
|
||||
static cl::opt<std::string> OutputFilename("o", cl::init("a.out"),
|
||||
cl::desc("Override output filename"),
|
||||
cl::value_desc("filename"));
|
||||
|
||||
static cl::opt<std::string> BitcodeOutputFilename("b", cl::init(""),
|
||||
cl::desc("Override bitcode output filename"),
|
||||
cl::value_desc("filename"));
|
||||
|
||||
static cl::opt<bool> Verbose("v",
|
||||
cl::desc("Print information about actions taken"));
|
||||
|
||||
static cl::list<std::string> LibPaths("L", cl::Prefix,
|
||||
cl::desc("Specify a library search path"),
|
||||
cl::value_desc("directory"));
|
||||
|
||||
static cl::list<std::string> FrameworkPaths("F", cl::Prefix,
|
||||
cl::desc("Specify a framework search path"),
|
||||
cl::value_desc("directory"));
|
||||
|
||||
static cl::list<std::string> Libraries("l", cl::Prefix,
|
||||
cl::desc("Specify libraries to link to"),
|
||||
cl::value_desc("library prefix"));
|
||||
|
||||
static cl::list<std::string> Frameworks("framework",
|
||||
cl::desc("Specify frameworks to link to"),
|
||||
cl::value_desc("framework"));
|
||||
|
||||
// Options to control the linking, optimization, and code gen processes
|
||||
static cl::opt<bool> LinkAsLibrary("link-as-library",
|
||||
cl::desc("Link the .bc files together as a library, not an executable"));
|
||||
|
||||
static cl::alias Relink("r", cl::aliasopt(LinkAsLibrary),
|
||||
cl::desc("Alias for -link-as-library"));
|
||||
|
||||
static cl::opt<bool> Native("native",
|
||||
cl::desc("Generate a native binary instead of a shell script"));
|
||||
|
||||
static cl::opt<bool>NativeCBE("native-cbe",
|
||||
cl::desc("Generate a native binary with the C backend and GCC"));
|
||||
|
||||
static cl::list<std::string> PostLinkOpts("post-link-opts",
|
||||
cl::value_desc("path"),
|
||||
cl::desc("Run one or more optimization programs after linking"));
|
||||
|
||||
static cl::list<std::string> XLinker("Xlinker", cl::value_desc("option"),
|
||||
cl::desc("Pass options to the system linker"));
|
||||
|
||||
// Compatibility options that llvm-ld ignores but are supported for
|
||||
// compatibility with LD
|
||||
static cl::opt<std::string> CO3("soname", cl::Hidden,
|
||||
cl::desc("Compatibility option: ignored"));
|
||||
|
||||
static cl::opt<std::string> CO4("version-script", cl::Hidden,
|
||||
cl::desc("Compatibility option: ignored"));
|
||||
|
||||
static cl::opt<bool> CO5("eh-frame-hdr", cl::Hidden,
|
||||
cl::desc("Compatibility option: ignored"));
|
||||
|
||||
static cl::opt<std::string> CO6("h", cl::Hidden,
|
||||
cl::desc("Compatibility option: ignored"));
|
||||
|
||||
static cl::opt<bool> CO7("start-group", cl::Hidden,
|
||||
cl::desc("Compatibility option: ignored"));
|
||||
|
||||
static cl::opt<bool> CO8("end-group", cl::Hidden,
|
||||
cl::desc("Compatibility option: ignored"));
|
||||
|
||||
static cl::opt<std::string> CO9("m", cl::Hidden,
|
||||
cl::desc("Compatibility option: ignored"));
|
||||
|
||||
/// This is just for convenience so it doesn't have to be passed around
|
||||
/// everywhere.
|
||||
static std::string progname;
|
||||
|
||||
/// FileRemover objects to clean up output files in the event of an error.
|
||||
static FileRemover OutputRemover;
|
||||
static FileRemover BitcodeOutputRemover;
|
||||
|
||||
/// PrintAndExit - Prints a message to standard error and exits with error code
|
||||
///
|
||||
/// Inputs:
|
||||
/// Message - The message to print to standard error.
|
||||
///
|
||||
static void PrintAndExit(const std::string &Message, Module *M, int errcode = 1) {
|
||||
errs() << progname << ": " << Message << "\n";
|
||||
delete M;
|
||||
llvm_shutdown();
|
||||
exit(errcode);
|
||||
}
|
||||
|
||||
static void PrintCommand(const std::vector<const char*> &args) {
|
||||
std::vector<const char*>::const_iterator I = args.begin(), E = args.end();
|
||||
for (; I != E; ++I)
|
||||
if (*I)
|
||||
errs() << "'" << *I << "'" << " ";
|
||||
errs() << "\n";
|
||||
}
|
||||
|
||||
/// CopyEnv - This function takes an array of environment variables and makes a
|
||||
/// copy of it. This copy can then be manipulated any way the caller likes
|
||||
/// without affecting the process's real environment.
|
||||
///
|
||||
/// Inputs:
|
||||
/// envp - An array of C strings containing an environment.
|
||||
///
|
||||
/// Return value:
|
||||
/// NULL - An error occurred.
|
||||
///
|
||||
/// Otherwise, a pointer to a new array of C strings is returned. Every string
|
||||
/// in the array is a duplicate of the one in the original array (i.e. we do
|
||||
/// not copy the char *'s from one array to another).
|
||||
///
|
||||
static char ** CopyEnv(char ** const envp) {
|
||||
// Count the number of entries in the old list;
|
||||
unsigned entries; // The number of entries in the old environment list
|
||||
for (entries = 0; envp[entries] != NULL; entries++)
|
||||
/*empty*/;
|
||||
|
||||
// Add one more entry for the NULL pointer that ends the list.
|
||||
++entries;
|
||||
|
||||
// If there are no entries at all, just return NULL.
|
||||
if (entries == 0)
|
||||
return NULL;
|
||||
|
||||
// Allocate a new environment list.
|
||||
char **newenv = new char* [entries];
|
||||
if (newenv == NULL)
|
||||
return NULL;
|
||||
|
||||
// Make a copy of the list. Don't forget the NULL that ends the list.
|
||||
entries = 0;
|
||||
while (envp[entries] != NULL) {
|
||||
size_t len = strlen(envp[entries]) + 1;
|
||||
newenv[entries] = new char[len];
|
||||
memcpy(newenv[entries], envp[entries], len);
|
||||
++entries;
|
||||
}
|
||||
newenv[entries] = NULL;
|
||||
|
||||
return newenv;
|
||||
}
|
||||
|
||||
|
||||
/// RemoveEnv - Remove the specified environment variable from the environment
|
||||
/// array.
|
||||
///
|
||||
/// Inputs:
|
||||
/// name - The name of the variable to remove. It cannot be NULL.
|
||||
/// envp - The array of environment variables. It cannot be NULL.
|
||||
///
|
||||
/// Notes:
|
||||
/// This is mainly done because functions to remove items from the environment
|
||||
/// are not available across all platforms. In particular, Solaris does not
|
||||
/// seem to have an unsetenv() function or a setenv() function (or they are
|
||||
/// undocumented if they do exist).
|
||||
///
|
||||
static void RemoveEnv(const char * name, char ** const envp) {
|
||||
for (unsigned index=0; envp[index] != NULL; index++) {
|
||||
// Find the first equals sign in the array and make it an EOS character.
|
||||
char *p = strchr (envp[index], '=');
|
||||
if (p == NULL)
|
||||
continue;
|
||||
else
|
||||
*p = '\0';
|
||||
|
||||
// Compare the two strings. If they are equal, zap this string.
|
||||
// Otherwise, restore it.
|
||||
if (!strcmp(name, envp[index]))
|
||||
*envp[index] = '\0';
|
||||
else
|
||||
*p = '=';
|
||||
}
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
/// GenerateBitcode - generates a bitcode file from the module provided
|
||||
void GenerateBitcode(Module* M, const std::string& FileName) {
|
||||
|
||||
if (Verbose)
|
||||
errs() << "Generating Bitcode To " << FileName << '\n';
|
||||
|
||||
// Create the output file.
|
||||
std::string ErrorInfo;
|
||||
tool_output_file Out(FileName.c_str(), ErrorInfo,
|
||||
raw_fd_ostream::F_Binary);
|
||||
if (!ErrorInfo.empty()) {
|
||||
PrintAndExit(ErrorInfo, M);
|
||||
return;
|
||||
}
|
||||
|
||||
// Write it out
|
||||
WriteBitcodeToFile(M, Out.os());
|
||||
Out.keep();
|
||||
}
|
||||
|
||||
/// GenerateAssembly - generates a native assembly language source file from the
|
||||
/// specified bitcode file.
|
||||
///
|
||||
/// Inputs:
|
||||
/// InputFilename - The name of the input bitcode file.
|
||||
/// OutputFilename - The name of the file to generate.
|
||||
/// llc - The pathname to use for LLC.
|
||||
/// envp - The environment to use when running LLC.
|
||||
///
|
||||
/// Return non-zero value on error.
|
||||
///
|
||||
static int GenerateAssembly(const std::string &OutputFilename,
|
||||
const std::string &InputFilename,
|
||||
const sys::Path &llc,
|
||||
std::string &ErrMsg ) {
|
||||
// Run LLC to convert the bitcode file into assembly code.
|
||||
std::vector<const char*> args;
|
||||
args.push_back(llc.c_str());
|
||||
// We will use GCC to assemble the program so set the assembly syntax to AT&T,
|
||||
// regardless of what the target in the bitcode file is.
|
||||
args.push_back("-x86-asm-syntax=att");
|
||||
args.push_back("-o");
|
||||
args.push_back(OutputFilename.c_str());
|
||||
args.push_back(InputFilename.c_str());
|
||||
args.push_back(0);
|
||||
|
||||
if (Verbose) {
|
||||
errs() << "Generating Assembly With: \n";
|
||||
PrintCommand(args);
|
||||
}
|
||||
|
||||
return sys::Program::ExecuteAndWait(llc, &args[0], 0, 0, 0, 0, &ErrMsg);
|
||||
}
|
||||
|
||||
/// GenerateCFile - generates a C source file from the specified bitcode file.
|
||||
static int GenerateCFile(const std::string &OutputFile,
|
||||
const std::string &InputFile,
|
||||
const sys::Path &llc,
|
||||
std::string& ErrMsg) {
|
||||
// Run LLC to convert the bitcode file into C.
|
||||
std::vector<const char*> args;
|
||||
args.push_back(llc.c_str());
|
||||
args.push_back("-march=c");
|
||||
args.push_back("-o");
|
||||
args.push_back(OutputFile.c_str());
|
||||
args.push_back(InputFile.c_str());
|
||||
args.push_back(0);
|
||||
|
||||
if (Verbose) {
|
||||
errs() << "Generating C Source With: \n";
|
||||
PrintCommand(args);
|
||||
}
|
||||
|
||||
return sys::Program::ExecuteAndWait(llc, &args[0], 0, 0, 0, 0, &ErrMsg);
|
||||
}
|
||||
|
||||
/// GenerateNative - generates a native object file from the
|
||||
/// specified bitcode file.
|
||||
///
|
||||
/// Inputs:
|
||||
/// InputFilename - The name of the input bitcode file.
|
||||
/// OutputFilename - The name of the file to generate.
|
||||
/// NativeLinkItems - The native libraries, files, code with which to link
|
||||
/// LibPaths - The list of directories in which to find libraries.
|
||||
/// FrameworksPaths - The list of directories in which to find frameworks.
|
||||
/// Frameworks - The list of frameworks (dynamic libraries)
|
||||
/// gcc - The pathname to use for GGC.
|
||||
/// envp - A copy of the process's current environment.
|
||||
///
|
||||
/// Outputs:
|
||||
/// None.
|
||||
///
|
||||
/// Returns non-zero value on error.
|
||||
///
|
||||
static int GenerateNative(const std::string &OutputFilename,
|
||||
const std::string &InputFilename,
|
||||
const Linker::ItemList &LinkItems,
|
||||
const sys::Path &gcc, char ** const envp,
|
||||
std::string& ErrMsg) {
|
||||
// Remove these environment variables from the environment of the
|
||||
// programs that we will execute. It appears that GCC sets these
|
||||
// environment variables so that the programs it uses can configure
|
||||
// themselves identically.
|
||||
//
|
||||
// However, when we invoke GCC below, we want it to use its normal
|
||||
// configuration. Hence, we must sanitize its environment.
|
||||
char ** clean_env = CopyEnv(envp);
|
||||
if (clean_env == NULL)
|
||||
return 1;
|
||||
RemoveEnv("LIBRARY_PATH", clean_env);
|
||||
RemoveEnv("COLLECT_GCC_OPTIONS", clean_env);
|
||||
RemoveEnv("GCC_EXEC_PREFIX", clean_env);
|
||||
RemoveEnv("COMPILER_PATH", clean_env);
|
||||
RemoveEnv("COLLECT_GCC", clean_env);
|
||||
|
||||
|
||||
// Run GCC to assemble and link the program into native code.
|
||||
//
|
||||
// Note:
|
||||
// We can't just assemble and link the file with the system assembler
|
||||
// and linker because we don't know where to put the _start symbol.
|
||||
// GCC mysteriously knows how to do it.
|
||||
std::vector<std::string> args;
|
||||
args.push_back(gcc.c_str());
|
||||
args.push_back("-fno-strict-aliasing");
|
||||
args.push_back("-O3");
|
||||
args.push_back("-o");
|
||||
args.push_back(OutputFilename);
|
||||
args.push_back(InputFilename);
|
||||
|
||||
// Add in the library and framework paths
|
||||
for (unsigned index = 0; index < LibPaths.size(); index++) {
|
||||
args.push_back("-L" + LibPaths[index]);
|
||||
}
|
||||
for (unsigned index = 0; index < FrameworkPaths.size(); index++) {
|
||||
args.push_back("-F" + FrameworkPaths[index]);
|
||||
}
|
||||
|
||||
// Add the requested options
|
||||
for (unsigned index = 0; index < XLinker.size(); index++)
|
||||
args.push_back(XLinker[index]);
|
||||
|
||||
// Add in the libraries to link.
|
||||
for (unsigned index = 0; index < LinkItems.size(); index++)
|
||||
if (LinkItems[index].first != "crtend") {
|
||||
if (LinkItems[index].second)
|
||||
args.push_back("-l" + LinkItems[index].first);
|
||||
else
|
||||
args.push_back(LinkItems[index].first);
|
||||
}
|
||||
|
||||
// Add in frameworks to link.
|
||||
for (unsigned index = 0; index < Frameworks.size(); index++) {
|
||||
args.push_back("-framework");
|
||||
args.push_back(Frameworks[index]);
|
||||
}
|
||||
|
||||
// Now that "args" owns all the std::strings for the arguments, call the c_str
|
||||
// method to get the underlying string array. We do this game so that the
|
||||
// std::string array is guaranteed to outlive the const char* array.
|
||||
std::vector<const char *> Args;
|
||||
for (unsigned i = 0, e = args.size(); i != e; ++i)
|
||||
Args.push_back(args[i].c_str());
|
||||
Args.push_back(0);
|
||||
|
||||
if (Verbose) {
|
||||
errs() << "Generating Native Executable With:\n";
|
||||
PrintCommand(Args);
|
||||
}
|
||||
|
||||
// Run the compiler to assembly and link together the program.
|
||||
int R = sys::Program::ExecuteAndWait(
|
||||
gcc, &Args[0], const_cast<const char **>(clean_env), 0, 0, 0, &ErrMsg);
|
||||
delete [] clean_env;
|
||||
return R;
|
||||
}
|
||||
|
||||
/// EmitShellScript - Output the wrapper file that invokes the JIT on the LLVM
|
||||
/// bitcode file for the program.
|
||||
static void EmitShellScript(char **argv, Module *M) {
|
||||
if (Verbose)
|
||||
errs() << "Emitting Shell Script\n";
|
||||
#if defined(_WIN32)
|
||||
// Windows doesn't support #!/bin/sh style shell scripts in .exe files. To
|
||||
// support windows systems, we copy the llvm-stub.exe executable from the
|
||||
// build tree to the destination file.
|
||||
std::string ErrMsg;
|
||||
sys::Path llvmstub = PrependMainExecutablePath("llvm-stub", argv[0],
|
||||
(void *)(intptr_t)&Optimize);
|
||||
if (llvmstub.isEmpty())
|
||||
PrintAndExit("Could not find llvm-stub.exe executable!", M);
|
||||
|
||||
if (0 != sys::CopyFile(sys::Path(OutputFilename), llvmstub, &ErrMsg))
|
||||
PrintAndExit(ErrMsg, M);
|
||||
|
||||
return;
|
||||
#endif
|
||||
|
||||
// Output the script to start the program...
|
||||
std::string ErrorInfo;
|
||||
tool_output_file Out2(OutputFilename.c_str(), ErrorInfo);
|
||||
if (!ErrorInfo.empty())
|
||||
PrintAndExit(ErrorInfo, M);
|
||||
|
||||
Out2.os() << "#!/bin/sh\n";
|
||||
// Allow user to setenv LLVMINTERP if lli is not in their PATH.
|
||||
Out2.os() << "lli=${LLVMINTERP-lli}\n";
|
||||
Out2.os() << "exec $lli \\\n";
|
||||
// gcc accepts -l<lib> and implicitly searches /lib and /usr/lib.
|
||||
LibPaths.push_back("/lib");
|
||||
LibPaths.push_back("/usr/lib");
|
||||
LibPaths.push_back("/usr/X11R6/lib");
|
||||
// We don't need to link in libc! In fact, /usr/lib/libc.so may not be a
|
||||
// shared object at all! See RH 8: plain text.
|
||||
std::vector<std::string>::iterator libc =
|
||||
std::find(Libraries.begin(), Libraries.end(), "c");
|
||||
if (libc != Libraries.end()) Libraries.erase(libc);
|
||||
// List all the shared object (native) libraries this executable will need
|
||||
// on the command line, so that we don't have to do this manually!
|
||||
for (std::vector<std::string>::iterator i = Libraries.begin(),
|
||||
e = Libraries.end(); i != e; ++i) {
|
||||
// try explicit -L arguments first:
|
||||
sys::Path FullLibraryPath;
|
||||
for (cl::list<std::string>::const_iterator P = LibPaths.begin(),
|
||||
E = LibPaths.end(); P != E; ++P) {
|
||||
FullLibraryPath = *P;
|
||||
FullLibraryPath.appendComponent("lib" + *i);
|
||||
FullLibraryPath.appendSuffix(sys::Path::GetDLLSuffix());
|
||||
if (!FullLibraryPath.isEmpty()) {
|
||||
if (!FullLibraryPath.isDynamicLibrary()) {
|
||||
// Not a native shared library; mark as invalid
|
||||
FullLibraryPath = sys::Path();
|
||||
} else break;
|
||||
}
|
||||
}
|
||||
if (FullLibraryPath.isEmpty())
|
||||
FullLibraryPath = sys::Path::FindLibrary(*i);
|
||||
if (!FullLibraryPath.isEmpty())
|
||||
Out2.os() << " -load=" << FullLibraryPath.str() << " \\\n";
|
||||
}
|
||||
Out2.os() << " " << BitcodeOutputFilename << " ${1+\"$@\"}\n";
|
||||
Out2.keep();
|
||||
}
|
||||
|
||||
// BuildLinkItems -- This function generates a LinkItemList for the LinkItems
|
||||
// linker function by combining the Files and Libraries in the order they were
|
||||
// declared on the command line.
|
||||
static void BuildLinkItems(
|
||||
Linker::ItemList& Items,
|
||||
const cl::list<std::string>& Files,
|
||||
const cl::list<std::string>& Libraries) {
|
||||
|
||||
// Build the list of linkage items for LinkItems.
|
||||
|
||||
cl::list<std::string>::const_iterator fileIt = Files.begin();
|
||||
cl::list<std::string>::const_iterator libIt = Libraries.begin();
|
||||
|
||||
int libPos = -1, filePos = -1;
|
||||
while ( libIt != Libraries.end() || fileIt != Files.end() ) {
|
||||
if (libIt != Libraries.end())
|
||||
libPos = Libraries.getPosition(libIt - Libraries.begin());
|
||||
else
|
||||
libPos = -1;
|
||||
if (fileIt != Files.end())
|
||||
filePos = Files.getPosition(fileIt - Files.begin());
|
||||
else
|
||||
filePos = -1;
|
||||
|
||||
if (filePos != -1 && (libPos == -1 || filePos < libPos)) {
|
||||
// Add a source file
|
||||
Items.push_back(std::make_pair(*fileIt++, false));
|
||||
} else if (libPos != -1 && (filePos == -1 || libPos < filePos)) {
|
||||
// Add a library
|
||||
Items.push_back(std::make_pair(*libIt++, true));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
int main(int argc, char **argv, char **envp) {
|
||||
// Print a stack trace if we signal out.
|
||||
sys::PrintStackTraceOnErrorSignal();
|
||||
PrettyStackTraceProgram X(argc, argv);
|
||||
|
||||
LLVMContext &Context = getGlobalContext();
|
||||
llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
|
||||
|
||||
// Initialize passes
|
||||
PassRegistry &Registry = *PassRegistry::getPassRegistry();
|
||||
initializeCore(Registry);
|
||||
initializeScalarOpts(Registry);
|
||||
initializeIPO(Registry);
|
||||
initializeAnalysis(Registry);
|
||||
initializeIPA(Registry);
|
||||
initializeTransformUtils(Registry);
|
||||
initializeInstCombine(Registry);
|
||||
initializeTarget(Registry);
|
||||
|
||||
// Initial global variable above for convenience printing of program name.
|
||||
progname = sys::path::stem(argv[0]);
|
||||
|
||||
// Parse the command line options
|
||||
cl::ParseCommandLineOptions(argc, argv, "llvm linker\n");
|
||||
|
||||
#if defined(_WIN32) || defined(__CYGWIN__)
|
||||
if (!LinkAsLibrary) {
|
||||
// Default to "a.exe" instead of "a.out".
|
||||
if (OutputFilename.getNumOccurrences() == 0)
|
||||
OutputFilename = "a.exe";
|
||||
|
||||
// If there is no suffix add an "exe" one.
|
||||
if (sys::path::extension(OutputFilename).empty())
|
||||
OutputFilename.append(".exe");
|
||||
}
|
||||
#endif
|
||||
|
||||
// Generate the bitcode for the optimized module.
|
||||
// If -b wasn't specified, use the name specified
|
||||
// with -o to construct BitcodeOutputFilename.
|
||||
if (BitcodeOutputFilename.empty()) {
|
||||
BitcodeOutputFilename = OutputFilename;
|
||||
if (!LinkAsLibrary) BitcodeOutputFilename += ".bc";
|
||||
}
|
||||
|
||||
// Arrange for the bitcode output file to be deleted on any errors.
|
||||
BitcodeOutputRemover.setFile(BitcodeOutputFilename);
|
||||
sys::RemoveFileOnSignal(sys::Path(BitcodeOutputFilename));
|
||||
|
||||
// Arrange for the output file to be deleted on any errors.
|
||||
if (!LinkAsLibrary) {
|
||||
OutputRemover.setFile(OutputFilename);
|
||||
sys::RemoveFileOnSignal(sys::Path(OutputFilename));
|
||||
}
|
||||
|
||||
// Construct a Linker (now that Verbose is set)
|
||||
Linker TheLinker(progname, OutputFilename, Context, Verbose);
|
||||
|
||||
// Keep track of the native link items (versus the bitcode items)
|
||||
Linker::ItemList NativeLinkItems;
|
||||
|
||||
// Add library paths to the linker
|
||||
TheLinker.addPaths(LibPaths);
|
||||
TheLinker.addSystemPaths();
|
||||
|
||||
// Remove any consecutive duplicates of the same library...
|
||||
Libraries.erase(std::unique(Libraries.begin(), Libraries.end()),
|
||||
Libraries.end());
|
||||
|
||||
if (LinkAsLibrary) {
|
||||
std::vector<sys::Path> Files;
|
||||
for (unsigned i = 0; i < InputFilenames.size(); ++i )
|
||||
Files.push_back(sys::Path(InputFilenames[i]));
|
||||
if (TheLinker.LinkInFiles(Files))
|
||||
return 1; // Error already printed
|
||||
|
||||
// The libraries aren't linked in but are noted as "dependent" in the
|
||||
// module.
|
||||
for (cl::list<std::string>::const_iterator I = Libraries.begin(),
|
||||
E = Libraries.end(); I != E ; ++I) {
|
||||
TheLinker.getModule()->addLibrary(*I);
|
||||
}
|
||||
} else {
|
||||
// Build a list of the items from our command line
|
||||
Linker::ItemList Items;
|
||||
BuildLinkItems(Items, InputFilenames, Libraries);
|
||||
|
||||
// Link all the items together
|
||||
if (TheLinker.LinkInItems(Items, NativeLinkItems) )
|
||||
return 1; // Error already printed
|
||||
}
|
||||
|
||||
std::auto_ptr<Module> Composite(TheLinker.releaseModule());
|
||||
|
||||
// Optimize the module
|
||||
Optimize(Composite.get());
|
||||
|
||||
// Generate the bitcode output.
|
||||
GenerateBitcode(Composite.get(), BitcodeOutputFilename);
|
||||
|
||||
// If we are not linking a library, generate either a native executable
|
||||
// or a JIT shell script, depending upon what the user wants.
|
||||
if (!LinkAsLibrary) {
|
||||
// If the user wants to run a post-link optimization, run it now.
|
||||
if (!PostLinkOpts.empty()) {
|
||||
std::vector<std::string> opts = PostLinkOpts;
|
||||
for (std::vector<std::string>::iterator I = opts.begin(),
|
||||
E = opts.end(); I != E; ++I) {
|
||||
sys::Path prog(*I);
|
||||
if (!prog.canExecute()) {
|
||||
prog = sys::Program::FindProgramByName(*I);
|
||||
if (prog.isEmpty())
|
||||
PrintAndExit(std::string("Optimization program '") + *I +
|
||||
"' is not found or not executable.", Composite.get());
|
||||
}
|
||||
// Get the program arguments
|
||||
sys::Path tmp_output("opt_result");
|
||||
std::string ErrMsg;
|
||||
if (tmp_output.createTemporaryFileOnDisk(true, &ErrMsg))
|
||||
PrintAndExit(ErrMsg, Composite.get());
|
||||
|
||||
const char* args[4];
|
||||
args[0] = I->c_str();
|
||||
args[1] = BitcodeOutputFilename.c_str();
|
||||
args[2] = tmp_output.c_str();
|
||||
args[3] = 0;
|
||||
if (0 == sys::Program::ExecuteAndWait(prog, args, 0,0,0,0, &ErrMsg)) {
|
||||
if (tmp_output.isBitcodeFile()) {
|
||||
sys::Path target(BitcodeOutputFilename);
|
||||
target.eraseFromDisk();
|
||||
if (tmp_output.renamePathOnDisk(target, &ErrMsg))
|
||||
PrintAndExit(ErrMsg, Composite.get(), 2);
|
||||
} else
|
||||
PrintAndExit("Post-link optimization output is not bitcode",
|
||||
Composite.get());
|
||||
} else {
|
||||
PrintAndExit(ErrMsg, Composite.get());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// If the user wants to generate a native executable, compile it from the
|
||||
// bitcode file.
|
||||
//
|
||||
// Otherwise, create a script that will run the bitcode through the JIT.
|
||||
if (Native) {
|
||||
// Name of the Assembly Language output file
|
||||
sys::Path AssemblyFile ( OutputFilename);
|
||||
AssemblyFile.appendSuffix("s");
|
||||
|
||||
// Mark the output files for removal.
|
||||
FileRemover AssemblyFileRemover(AssemblyFile.str());
|
||||
sys::RemoveFileOnSignal(AssemblyFile);
|
||||
|
||||
// Determine the locations of the llc and gcc programs.
|
||||
sys::Path llc = PrependMainExecutablePath("llc", argv[0],
|
||||
(void *)(intptr_t)&Optimize);
|
||||
if (llc.isEmpty())
|
||||
PrintAndExit("Failed to find llc", Composite.get());
|
||||
|
||||
sys::Path gcc = sys::Program::FindProgramByName("gcc");
|
||||
if (gcc.isEmpty())
|
||||
PrintAndExit("Failed to find gcc", Composite.get());
|
||||
|
||||
// Generate an assembly language file for the bitcode.
|
||||
std::string ErrMsg;
|
||||
if (0 != GenerateAssembly(AssemblyFile.str(), BitcodeOutputFilename,
|
||||
llc, ErrMsg))
|
||||
PrintAndExit(ErrMsg, Composite.get());
|
||||
|
||||
if (0 != GenerateNative(OutputFilename, AssemblyFile.str(),
|
||||
NativeLinkItems, gcc, envp, ErrMsg))
|
||||
PrintAndExit(ErrMsg, Composite.get());
|
||||
} else if (NativeCBE) {
|
||||
sys::Path CFile (OutputFilename);
|
||||
CFile.appendSuffix("cbe.c");
|
||||
|
||||
// Mark the output files for removal.
|
||||
FileRemover CFileRemover(CFile.str());
|
||||
sys::RemoveFileOnSignal(CFile);
|
||||
|
||||
// Determine the locations of the llc and gcc programs.
|
||||
sys::Path llc = PrependMainExecutablePath("llc", argv[0],
|
||||
(void *)(intptr_t)&Optimize);
|
||||
if (llc.isEmpty())
|
||||
PrintAndExit("Failed to find llc", Composite.get());
|
||||
|
||||
sys::Path gcc = sys::Program::FindProgramByName("gcc");
|
||||
if (gcc.isEmpty())
|
||||
PrintAndExit("Failed to find gcc", Composite.get());
|
||||
|
||||
// Generate an assembly language file for the bitcode.
|
||||
std::string ErrMsg;
|
||||
if (GenerateCFile(CFile.str(), BitcodeOutputFilename, llc, ErrMsg))
|
||||
PrintAndExit(ErrMsg, Composite.get());
|
||||
|
||||
if (GenerateNative(OutputFilename, CFile.str(),
|
||||
NativeLinkItems, gcc, envp, ErrMsg))
|
||||
PrintAndExit(ErrMsg, Composite.get());
|
||||
} else {
|
||||
EmitShellScript(argv, Composite.get());
|
||||
}
|
||||
|
||||
// Make the script executable...
|
||||
std::string ErrMsg;
|
||||
if (sys::Path(OutputFilename).makeExecutableOnDisk(&ErrMsg))
|
||||
PrintAndExit(ErrMsg, Composite.get());
|
||||
|
||||
// Make the bitcode file readable and directly executable in LLEE as well
|
||||
if (sys::Path(BitcodeOutputFilename).makeExecutableOnDisk(&ErrMsg))
|
||||
PrintAndExit(ErrMsg, Composite.get());
|
||||
|
||||
if (sys::Path(BitcodeOutputFilename).makeReadableOnDisk(&ErrMsg))
|
||||
PrintAndExit(ErrMsg, Composite.get());
|
||||
}
|
||||
|
||||
// Operations which may fail are now complete.
|
||||
BitcodeOutputRemover.releaseFile();
|
||||
if (!LinkAsLibrary)
|
||||
OutputRemover.releaseFile();
|
||||
|
||||
// Graceful exit
|
||||
return 0;
|
||||
}
|
5
contrib/llvm/tools/llvm-link/CMakeLists.txt
Normal file
5
contrib/llvm/tools/llvm-link/CMakeLists.txt
Normal file
@ -0,0 +1,5 @@
|
||||
set(LLVM_LINK_COMPONENTS linker bitreader bitwriter asmparser)
|
||||
|
||||
add_llvm_tool(llvm-link
|
||||
llvm-link.cpp
|
||||
)
|
17
contrib/llvm/tools/llvm-link/Makefile
Normal file
17
contrib/llvm/tools/llvm-link/Makefile
Normal file
@ -0,0 +1,17 @@
|
||||
##===- tools/llvm-link/Makefile ----------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
LEVEL = ../..
|
||||
|
||||
TOOLNAME = llvm-link
|
||||
LINK_COMPONENTS = linker bitreader bitwriter asmparser
|
||||
|
||||
# This tool has no plugins, optimize startup time.
|
||||
TOOL_NO_EXPORTS = 1
|
||||
|
||||
include $(LEVEL)/Makefile.common
|
142
contrib/llvm/tools/llvm-link/llvm-link.cpp
Normal file
142
contrib/llvm/tools/llvm-link/llvm-link.cpp
Normal file
@ -0,0 +1,142 @@
|
||||
//===- llvm-link.cpp - Low-level LLVM linker ------------------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This utility may be invoked in the following manner:
|
||||
// llvm-link a.bc b.bc c.bc -o x.bc
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/Linker.h"
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/Analysis/Verifier.h"
|
||||
#include "llvm/Bitcode/ReaderWriter.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/ToolOutputFile.h"
|
||||
#include "llvm/Support/SystemUtils.h"
|
||||
#include "llvm/Support/IRReader.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include "llvm/Support/Path.h"
|
||||
#include <memory>
|
||||
using namespace llvm;
|
||||
|
||||
static cl::list<std::string>
|
||||
InputFilenames(cl::Positional, cl::OneOrMore,
|
||||
cl::desc("<input bitcode files>"));
|
||||
|
||||
static cl::opt<std::string>
|
||||
OutputFilename("o", cl::desc("Override output filename"), cl::init("-"),
|
||||
cl::value_desc("filename"));
|
||||
|
||||
static cl::opt<bool>
|
||||
Force("f", cl::desc("Enable binary output on terminals"));
|
||||
|
||||
static cl::opt<bool>
|
||||
OutputAssembly("S",
|
||||
cl::desc("Write output as LLVM assembly"), cl::Hidden);
|
||||
|
||||
static cl::opt<bool>
|
||||
Verbose("v", cl::desc("Print information about actions taken"));
|
||||
|
||||
static cl::opt<bool>
|
||||
DumpAsm("d", cl::desc("Print assembly as linked"), cl::Hidden);
|
||||
|
||||
// LoadFile - Read the specified bitcode file in and return it. This routine
|
||||
// searches the link path for the specified file to try to find it...
|
||||
//
|
||||
static inline std::auto_ptr<Module> LoadFile(const char *argv0,
|
||||
const std::string &FN,
|
||||
LLVMContext& Context) {
|
||||
sys::Path Filename;
|
||||
if (!Filename.set(FN)) {
|
||||
errs() << "Invalid file name: '" << FN << "'\n";
|
||||
return std::auto_ptr<Module>();
|
||||
}
|
||||
|
||||
SMDiagnostic Err;
|
||||
if (Verbose) errs() << "Loading '" << Filename.c_str() << "'\n";
|
||||
Module* Result = 0;
|
||||
|
||||
const std::string &FNStr = Filename.str();
|
||||
Result = ParseIRFile(FNStr, Err, Context);
|
||||
if (Result) return std::auto_ptr<Module>(Result); // Load successful!
|
||||
|
||||
Err.Print(argv0, errs());
|
||||
return std::auto_ptr<Module>();
|
||||
}
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
// Print a stack trace if we signal out.
|
||||
sys::PrintStackTraceOnErrorSignal();
|
||||
PrettyStackTraceProgram X(argc, argv);
|
||||
|
||||
LLVMContext &Context = getGlobalContext();
|
||||
llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
|
||||
cl::ParseCommandLineOptions(argc, argv, "llvm linker\n");
|
||||
|
||||
unsigned BaseArg = 0;
|
||||
std::string ErrorMessage;
|
||||
|
||||
std::auto_ptr<Module> Composite(LoadFile(argv[0],
|
||||
InputFilenames[BaseArg], Context));
|
||||
if (Composite.get() == 0) {
|
||||
errs() << argv[0] << ": error loading file '"
|
||||
<< InputFilenames[BaseArg] << "'\n";
|
||||
return 1;
|
||||
}
|
||||
|
||||
for (unsigned i = BaseArg+1; i < InputFilenames.size(); ++i) {
|
||||
std::auto_ptr<Module> M(LoadFile(argv[0],
|
||||
InputFilenames[i], Context));
|
||||
if (M.get() == 0) {
|
||||
errs() << argv[0] << ": error loading file '" <<InputFilenames[i]<< "'\n";
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (Verbose) errs() << "Linking in '" << InputFilenames[i] << "'\n";
|
||||
|
||||
if (Linker::LinkModules(Composite.get(), M.get(), Linker::DestroySource,
|
||||
&ErrorMessage)) {
|
||||
errs() << argv[0] << ": link error in '" << InputFilenames[i]
|
||||
<< "': " << ErrorMessage << "\n";
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
|
||||
// TODO: Iterate over the -l list and link in any modules containing
|
||||
// global symbols that have not been resolved so far.
|
||||
|
||||
if (DumpAsm) errs() << "Here's the assembly:\n" << *Composite;
|
||||
|
||||
std::string ErrorInfo;
|
||||
tool_output_file Out(OutputFilename.c_str(), ErrorInfo,
|
||||
raw_fd_ostream::F_Binary);
|
||||
if (!ErrorInfo.empty()) {
|
||||
errs() << ErrorInfo << '\n';
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (verifyModule(*Composite)) {
|
||||
errs() << argv[0] << ": linked module is broken!\n";
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (Verbose) errs() << "Writing bitcode...\n";
|
||||
if (OutputAssembly) {
|
||||
Out.os() << *Composite;
|
||||
} else if (Force || !CheckBitcodeOutputToConsole(Out.os(), true))
|
||||
WriteBitcodeToFile(Composite.get(), Out.os());
|
||||
|
||||
// Declare success.
|
||||
Out.keep();
|
||||
|
||||
return 0;
|
||||
}
|
6
contrib/llvm/tools/llvm-mc/CMakeLists.txt
Normal file
6
contrib/llvm/tools/llvm-mc/CMakeLists.txt
Normal file
@ -0,0 +1,6 @@
|
||||
set(LLVM_LINK_COMPONENTS ${LLVM_TARGETS_TO_BUILD} support MC MCParser MCDisassembler)
|
||||
|
||||
add_llvm_tool(llvm-mc
|
||||
llvm-mc.cpp
|
||||
Disassembler.cpp
|
||||
)
|
368
contrib/llvm/tools/llvm-mc/Disassembler.cpp
Normal file
368
contrib/llvm/tools/llvm-mc/Disassembler.cpp
Normal file
@ -0,0 +1,368 @@
|
||||
//===- Disassembler.cpp - Disassembler for hex strings --------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This class implements the disassembler of strings of bytes written in
|
||||
// hexadecimal, from standard input or from a file.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "Disassembler.h"
|
||||
#include "../../lib/MC/MCDisassembler/EDDisassembler.h"
|
||||
#include "../../lib/MC/MCDisassembler/EDInst.h"
|
||||
#include "../../lib/MC/MCDisassembler/EDOperand.h"
|
||||
#include "../../lib/MC/MCDisassembler/EDToken.h"
|
||||
#include "llvm/MC/MCAsmInfo.h"
|
||||
#include "llvm/MC/MCDisassembler.h"
|
||||
#include "llvm/MC/MCInst.h"
|
||||
#include "llvm/MC/MCInstPrinter.h"
|
||||
#include "llvm/MC/MCSubtargetInfo.h"
|
||||
#include "llvm/ADT/OwningPtr.h"
|
||||
#include "llvm/ADT/Triple.h"
|
||||
#include "llvm/ADT/Twine.h"
|
||||
#include "llvm/Support/MemoryBuffer.h"
|
||||
#include "llvm/Support/MemoryObject.h"
|
||||
#include "llvm/Support/SourceMgr.h"
|
||||
#include "llvm/Support/TargetRegistry.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
using namespace llvm;
|
||||
|
||||
typedef std::vector<std::pair<unsigned char, const char*> > ByteArrayTy;
|
||||
|
||||
namespace {
|
||||
class VectorMemoryObject : public MemoryObject {
|
||||
private:
|
||||
const ByteArrayTy &Bytes;
|
||||
public:
|
||||
VectorMemoryObject(const ByteArrayTy &bytes) : Bytes(bytes) {}
|
||||
|
||||
uint64_t getBase() const { return 0; }
|
||||
uint64_t getExtent() const { return Bytes.size(); }
|
||||
|
||||
int readByte(uint64_t Addr, uint8_t *Byte) const {
|
||||
if (Addr >= getExtent())
|
||||
return -1;
|
||||
*Byte = Bytes[Addr].first;
|
||||
return 0;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
static bool PrintInsts(const MCDisassembler &DisAsm,
|
||||
MCInstPrinter &Printer, const ByteArrayTy &Bytes,
|
||||
SourceMgr &SM, raw_ostream &Out) {
|
||||
// Wrap the vector in a MemoryObject.
|
||||
VectorMemoryObject memoryObject(Bytes);
|
||||
|
||||
// Disassemble it to strings.
|
||||
uint64_t Size;
|
||||
uint64_t Index;
|
||||
|
||||
for (Index = 0; Index < Bytes.size(); Index += Size) {
|
||||
MCInst Inst;
|
||||
|
||||
MCDisassembler::DecodeStatus S;
|
||||
S = DisAsm.getInstruction(Inst, Size, memoryObject, Index,
|
||||
/*REMOVE*/ nulls(), nulls());
|
||||
switch (S) {
|
||||
case MCDisassembler::Fail:
|
||||
SM.PrintMessage(SMLoc::getFromPointer(Bytes[Index].second),
|
||||
"invalid instruction encoding", "warning");
|
||||
if (Size == 0)
|
||||
Size = 1; // skip illegible bytes
|
||||
break;
|
||||
|
||||
case MCDisassembler::SoftFail:
|
||||
SM.PrintMessage(SMLoc::getFromPointer(Bytes[Index].second),
|
||||
"potentially undefined instruction encoding", "warning");
|
||||
// Fall through
|
||||
|
||||
case MCDisassembler::Success:
|
||||
Printer.printInst(&Inst, Out, "");
|
||||
Out << "\n";
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static bool ByteArrayFromString(ByteArrayTy &ByteArray,
|
||||
StringRef &Str,
|
||||
SourceMgr &SM) {
|
||||
while (!Str.empty()) {
|
||||
// Strip horizontal whitespace.
|
||||
if (size_t Pos = Str.find_first_not_of(" \t\r")) {
|
||||
Str = Str.substr(Pos);
|
||||
continue;
|
||||
}
|
||||
|
||||
// If this is the end of a line or start of a comment, remove the rest of
|
||||
// the line.
|
||||
if (Str[0] == '\n' || Str[0] == '#') {
|
||||
// Strip to the end of line if we already processed any bytes on this
|
||||
// line. This strips the comment and/or the \n.
|
||||
if (Str[0] == '\n') {
|
||||
Str = Str.substr(1);
|
||||
} else {
|
||||
Str = Str.substr(Str.find_first_of('\n'));
|
||||
if (!Str.empty())
|
||||
Str = Str.substr(1);
|
||||
}
|
||||
continue;
|
||||
}
|
||||
|
||||
// Get the current token.
|
||||
size_t Next = Str.find_first_of(" \t\n\r#");
|
||||
StringRef Value = Str.substr(0, Next);
|
||||
|
||||
// Convert to a byte and add to the byte vector.
|
||||
unsigned ByteVal;
|
||||
if (Value.getAsInteger(0, ByteVal) || ByteVal > 255) {
|
||||
// If we have an error, print it and skip to the end of line.
|
||||
SM.PrintMessage(SMLoc::getFromPointer(Value.data()),
|
||||
"invalid input token", "error");
|
||||
Str = Str.substr(Str.find('\n'));
|
||||
ByteArray.clear();
|
||||
continue;
|
||||
}
|
||||
|
||||
ByteArray.push_back(std::make_pair((unsigned char)ByteVal, Value.data()));
|
||||
Str = Str.substr(Next);
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
int Disassembler::disassemble(const Target &T,
|
||||
const std::string &Triple,
|
||||
const std::string &Cpu,
|
||||
const std::string &FeaturesStr,
|
||||
MemoryBuffer &Buffer,
|
||||
raw_ostream &Out) {
|
||||
// Set up disassembler.
|
||||
OwningPtr<const MCAsmInfo> AsmInfo(T.createMCAsmInfo(Triple));
|
||||
|
||||
if (!AsmInfo) {
|
||||
errs() << "error: no assembly info for target " << Triple << "\n";
|
||||
return -1;
|
||||
}
|
||||
|
||||
OwningPtr<const MCSubtargetInfo> STI(T.createMCSubtargetInfo(Triple, Cpu, FeaturesStr));
|
||||
if (!STI) {
|
||||
errs() << "error: no subtarget info for target " << Triple << "\n";
|
||||
return -1;
|
||||
}
|
||||
|
||||
OwningPtr<const MCDisassembler> DisAsm(T.createMCDisassembler(*STI));
|
||||
if (!DisAsm) {
|
||||
errs() << "error: no disassembler for target " << Triple << "\n";
|
||||
return -1;
|
||||
}
|
||||
|
||||
int AsmPrinterVariant = AsmInfo->getAssemblerDialect();
|
||||
OwningPtr<MCInstPrinter> IP(T.createMCInstPrinter(AsmPrinterVariant,
|
||||
*AsmInfo, *STI));
|
||||
if (!IP) {
|
||||
errs() << "error: no instruction printer for target " << Triple << '\n';
|
||||
return -1;
|
||||
}
|
||||
|
||||
bool ErrorOccurred = false;
|
||||
|
||||
SourceMgr SM;
|
||||
SM.AddNewSourceBuffer(&Buffer, SMLoc());
|
||||
|
||||
// Convert the input to a vector for disassembly.
|
||||
ByteArrayTy ByteArray;
|
||||
StringRef Str = Buffer.getBuffer();
|
||||
|
||||
ErrorOccurred |= ByteArrayFromString(ByteArray, Str, SM);
|
||||
|
||||
if (!ByteArray.empty())
|
||||
ErrorOccurred |= PrintInsts(*DisAsm, *IP, ByteArray, SM, Out);
|
||||
|
||||
return ErrorOccurred;
|
||||
}
|
||||
|
||||
static int byteArrayReader(uint8_t *B, uint64_t A, void *Arg) {
|
||||
ByteArrayTy &ByteArray = *((ByteArrayTy*)Arg);
|
||||
|
||||
if (A >= ByteArray.size())
|
||||
return -1;
|
||||
|
||||
*B = ByteArray[A].first;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int verboseEvaluator(uint64_t *V, unsigned R, void *Arg) {
|
||||
EDDisassembler &disassembler = *(EDDisassembler *)((void **)Arg)[0];
|
||||
raw_ostream &Out = *(raw_ostream *)((void **)Arg)[1];
|
||||
|
||||
if (const char *regName = disassembler.nameWithRegisterID(R))
|
||||
Out << "[" << regName << "/" << R << "]";
|
||||
|
||||
if (disassembler.registerIsStackPointer(R))
|
||||
Out << "(sp)";
|
||||
if (disassembler.registerIsProgramCounter(R))
|
||||
Out << "(pc)";
|
||||
|
||||
*V = 0;
|
||||
return 0;
|
||||
}
|
||||
|
||||
int Disassembler::disassembleEnhanced(const std::string &TS,
|
||||
MemoryBuffer &Buffer,
|
||||
raw_ostream &Out) {
|
||||
ByteArrayTy ByteArray;
|
||||
StringRef Str = Buffer.getBuffer();
|
||||
SourceMgr SM;
|
||||
|
||||
SM.AddNewSourceBuffer(&Buffer, SMLoc());
|
||||
|
||||
if (ByteArrayFromString(ByteArray, Str, SM)) {
|
||||
return -1;
|
||||
}
|
||||
|
||||
Triple T(TS);
|
||||
EDDisassembler::AssemblySyntax AS;
|
||||
|
||||
switch (T.getArch()) {
|
||||
default:
|
||||
errs() << "error: no default assembly syntax for " << TS.c_str() << "\n";
|
||||
return -1;
|
||||
case Triple::arm:
|
||||
case Triple::thumb:
|
||||
AS = EDDisassembler::kEDAssemblySyntaxARMUAL;
|
||||
break;
|
||||
case Triple::x86:
|
||||
case Triple::x86_64:
|
||||
AS = EDDisassembler::kEDAssemblySyntaxX86ATT;
|
||||
break;
|
||||
}
|
||||
|
||||
EDDisassembler::initialize();
|
||||
OwningPtr<EDDisassembler>
|
||||
disassembler(EDDisassembler::getDisassembler(TS.c_str(), AS));
|
||||
|
||||
if (disassembler == 0) {
|
||||
errs() << "error: couldn't get disassembler for " << TS << '\n';
|
||||
return -1;
|
||||
}
|
||||
|
||||
while (ByteArray.size()) {
|
||||
OwningPtr<EDInst>
|
||||
inst(disassembler->createInst(byteArrayReader, 0, &ByteArray));
|
||||
|
||||
if (inst == 0) {
|
||||
errs() << "error: Didn't get an instruction\n";
|
||||
return -1;
|
||||
}
|
||||
|
||||
ByteArray.erase (ByteArray.begin(), ByteArray.begin() + inst->byteSize());
|
||||
|
||||
unsigned numTokens = inst->numTokens();
|
||||
if ((int)numTokens < 0) {
|
||||
errs() << "error: couldn't count the instruction's tokens\n";
|
||||
return -1;
|
||||
}
|
||||
|
||||
for (unsigned tokenIndex = 0; tokenIndex != numTokens; ++tokenIndex) {
|
||||
EDToken *token;
|
||||
|
||||
if (inst->getToken(token, tokenIndex)) {
|
||||
errs() << "error: Couldn't get token\n";
|
||||
return -1;
|
||||
}
|
||||
|
||||
const char *buf;
|
||||
if (token->getString(buf)) {
|
||||
errs() << "error: Couldn't get string for token\n";
|
||||
return -1;
|
||||
}
|
||||
|
||||
Out << '[';
|
||||
int operandIndex = token->operandID();
|
||||
|
||||
if (operandIndex >= 0)
|
||||
Out << operandIndex << "-";
|
||||
|
||||
switch (token->type()) {
|
||||
default: Out << "?"; break;
|
||||
case EDToken::kTokenWhitespace: Out << "w"; break;
|
||||
case EDToken::kTokenPunctuation: Out << "p"; break;
|
||||
case EDToken::kTokenOpcode: Out << "o"; break;
|
||||
case EDToken::kTokenLiteral: Out << "l"; break;
|
||||
case EDToken::kTokenRegister: Out << "r"; break;
|
||||
}
|
||||
|
||||
Out << ":" << buf;
|
||||
|
||||
if (token->type() == EDToken::kTokenLiteral) {
|
||||
Out << "=";
|
||||
if (token->literalSign())
|
||||
Out << "-";
|
||||
uint64_t absoluteValue;
|
||||
if (token->literalAbsoluteValue(absoluteValue)) {
|
||||
errs() << "error: Couldn't get the value of a literal token\n";
|
||||
return -1;
|
||||
}
|
||||
Out << absoluteValue;
|
||||
} else if (token->type() == EDToken::kTokenRegister) {
|
||||
Out << "=";
|
||||
unsigned regID;
|
||||
if (token->registerID(regID)) {
|
||||
errs() << "error: Couldn't get the ID of a register token\n";
|
||||
return -1;
|
||||
}
|
||||
Out << "r" << regID;
|
||||
}
|
||||
|
||||
Out << "]";
|
||||
}
|
||||
|
||||
Out << " ";
|
||||
|
||||
if (inst->isBranch())
|
||||
Out << "<br> ";
|
||||
if (inst->isMove())
|
||||
Out << "<mov> ";
|
||||
|
||||
unsigned numOperands = inst->numOperands();
|
||||
|
||||
if ((int)numOperands < 0) {
|
||||
errs() << "error: Couldn't count operands\n";
|
||||
return -1;
|
||||
}
|
||||
|
||||
for (unsigned operandIndex = 0; operandIndex != numOperands;
|
||||
++operandIndex) {
|
||||
Out << operandIndex << ":";
|
||||
|
||||
EDOperand *operand;
|
||||
if (inst->getOperand(operand, operandIndex)) {
|
||||
errs() << "error: couldn't get operand\n";
|
||||
return -1;
|
||||
}
|
||||
|
||||
uint64_t evaluatedResult;
|
||||
void *Arg[] = { disassembler.get(), &Out };
|
||||
if (operand->evaluate(evaluatedResult, verboseEvaluator, Arg)) {
|
||||
errs() << "error: Couldn't evaluate an operand\n";
|
||||
return -1;
|
||||
}
|
||||
Out << "=" << evaluatedResult << " ";
|
||||
}
|
||||
|
||||
Out << '\n';
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
42
contrib/llvm/tools/llvm-mc/Disassembler.h
Normal file
42
contrib/llvm/tools/llvm-mc/Disassembler.h
Normal file
@ -0,0 +1,42 @@
|
||||
//===- Disassembler.h - Text File Disassembler ----------------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This class implements the disassembler of strings of bytes written in
|
||||
// hexadecimal, from standard input or from a file.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef DISASSEMBLER_H
|
||||
#define DISASSEMBLER_H
|
||||
|
||||
#include <string>
|
||||
|
||||
namespace llvm {
|
||||
|
||||
class MemoryBuffer;
|
||||
class Target;
|
||||
class raw_ostream;
|
||||
|
||||
class Disassembler {
|
||||
public:
|
||||
static int disassemble(const Target &target,
|
||||
const std::string &tripleString,
|
||||
const std::string &Cpu,
|
||||
const std::string &FeaturesStr,
|
||||
MemoryBuffer &buffer,
|
||||
raw_ostream &Out);
|
||||
|
||||
static int disassembleEnhanced(const std::string &tripleString,
|
||||
MemoryBuffer &buffer,
|
||||
raw_ostream &Out);
|
||||
};
|
||||
|
||||
} // namespace llvm
|
||||
|
||||
#endif
|
24
contrib/llvm/tools/llvm-mc/Makefile
Normal file
24
contrib/llvm/tools/llvm-mc/Makefile
Normal file
@ -0,0 +1,24 @@
|
||||
##===- tools/llvm-mc/Makefile ------------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
|
||||
LEVEL = ../..
|
||||
TOOLNAME = llvm-mc
|
||||
|
||||
# This tool has no plugins, optimize startup time.
|
||||
TOOL_NO_EXPORTS = 1
|
||||
|
||||
# Include this here so we can get the configuration of the targets
|
||||
# that have been configured for construction. We have to do this
|
||||
# early so we can set up LINK_COMPONENTS before including Makefile.rules
|
||||
include $(LEVEL)/Makefile.config
|
||||
|
||||
LINK_COMPONENTS := $(TARGETS_TO_BUILD) MCDisassembler MCParser MC support
|
||||
|
||||
include $(LLVM_SRC_ROOT)/Makefile.rules
|
||||
|
517
contrib/llvm/tools/llvm-mc/llvm-mc.cpp
Normal file
517
contrib/llvm/tools/llvm-mc/llvm-mc.cpp
Normal file
@ -0,0 +1,517 @@
|
||||
//===-- llvm-mc.cpp - Machine Code Hacking Driver -------------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This utility is a simple driver that allows command line hacking on machine
|
||||
// code.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/MC/MCParser/AsmLexer.h"
|
||||
#include "llvm/MC/MCParser/MCAsmLexer.h"
|
||||
#include "llvm/MC/MCAsmBackend.h"
|
||||
#include "llvm/MC/MCContext.h"
|
||||
#include "llvm/MC/MCCodeEmitter.h"
|
||||
#include "llvm/MC/MCInstPrinter.h"
|
||||
#include "llvm/MC/MCInstrInfo.h"
|
||||
#include "llvm/MC/MCObjectFileInfo.h"
|
||||
#include "llvm/MC/MCRegisterInfo.h"
|
||||
#include "llvm/MC/MCSectionMachO.h"
|
||||
#include "llvm/MC/MCStreamer.h"
|
||||
#include "llvm/MC/MCSubtargetInfo.h"
|
||||
#include "llvm/MC/MCTargetAsmParser.h"
|
||||
#include "llvm/MC/SubtargetFeature.h"
|
||||
#include "llvm/ADT/OwningPtr.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/FileUtilities.h"
|
||||
#include "llvm/Support/FormattedStream.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/MemoryBuffer.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/SourceMgr.h"
|
||||
#include "llvm/Support/ToolOutputFile.h"
|
||||
#include "llvm/Support/Host.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include "llvm/Support/TargetRegistry.h"
|
||||
#include "llvm/Support/TargetSelect.h"
|
||||
#include "llvm/Support/system_error.h"
|
||||
#include "Disassembler.h"
|
||||
using namespace llvm;
|
||||
|
||||
static cl::opt<std::string>
|
||||
InputFilename(cl::Positional, cl::desc("<input file>"), cl::init("-"));
|
||||
|
||||
static cl::opt<std::string>
|
||||
OutputFilename("o", cl::desc("Output filename"),
|
||||
cl::value_desc("filename"));
|
||||
|
||||
static cl::opt<bool>
|
||||
ShowEncoding("show-encoding", cl::desc("Show instruction encodings"));
|
||||
|
||||
static cl::opt<bool>
|
||||
ShowInst("show-inst", cl::desc("Show internal instruction representation"));
|
||||
|
||||
static cl::opt<bool>
|
||||
ShowInstOperands("show-inst-operands",
|
||||
cl::desc("Show instructions operands as parsed"));
|
||||
|
||||
static cl::opt<unsigned>
|
||||
OutputAsmVariant("output-asm-variant",
|
||||
cl::desc("Syntax variant to use for output printing"));
|
||||
|
||||
static cl::opt<bool>
|
||||
RelaxAll("mc-relax-all", cl::desc("Relax all fixups"));
|
||||
|
||||
static cl::opt<bool>
|
||||
NoExecStack("mc-no-exec-stack", cl::desc("File doesn't need an exec stack"));
|
||||
|
||||
static cl::opt<bool>
|
||||
EnableLogging("enable-api-logging", cl::desc("Enable MC API logging"));
|
||||
|
||||
enum OutputFileType {
|
||||
OFT_Null,
|
||||
OFT_AssemblyFile,
|
||||
OFT_ObjectFile
|
||||
};
|
||||
static cl::opt<OutputFileType>
|
||||
FileType("filetype", cl::init(OFT_AssemblyFile),
|
||||
cl::desc("Choose an output file type:"),
|
||||
cl::values(
|
||||
clEnumValN(OFT_AssemblyFile, "asm",
|
||||
"Emit an assembly ('.s') file"),
|
||||
clEnumValN(OFT_Null, "null",
|
||||
"Don't emit anything (for timing purposes)"),
|
||||
clEnumValN(OFT_ObjectFile, "obj",
|
||||
"Emit a native object ('.o') file"),
|
||||
clEnumValEnd));
|
||||
|
||||
static cl::list<std::string>
|
||||
IncludeDirs("I", cl::desc("Directory of include files"),
|
||||
cl::value_desc("directory"), cl::Prefix);
|
||||
|
||||
static cl::opt<std::string>
|
||||
ArchName("arch", cl::desc("Target arch to assemble for, "
|
||||
"see -version for available targets"));
|
||||
|
||||
static cl::opt<std::string>
|
||||
TripleName("triple", cl::desc("Target triple to assemble for, "
|
||||
"see -version for available targets"));
|
||||
|
||||
static cl::opt<std::string>
|
||||
MCPU("mcpu",
|
||||
cl::desc("Target a specific cpu type (-mcpu=help for details)"),
|
||||
cl::value_desc("cpu-name"),
|
||||
cl::init(""));
|
||||
|
||||
static cl::list<std::string>
|
||||
MAttrs("mattr",
|
||||
cl::CommaSeparated,
|
||||
cl::desc("Target specific attributes (-mattr=help for details)"),
|
||||
cl::value_desc("a1,+a2,-a3,..."));
|
||||
|
||||
static cl::opt<Reloc::Model>
|
||||
RelocModel("relocation-model",
|
||||
cl::desc("Choose relocation model"),
|
||||
cl::init(Reloc::Default),
|
||||
cl::values(
|
||||
clEnumValN(Reloc::Default, "default",
|
||||
"Target default relocation model"),
|
||||
clEnumValN(Reloc::Static, "static",
|
||||
"Non-relocatable code"),
|
||||
clEnumValN(Reloc::PIC_, "pic",
|
||||
"Fully relocatable, position independent code"),
|
||||
clEnumValN(Reloc::DynamicNoPIC, "dynamic-no-pic",
|
||||
"Relocatable external references, non-relocatable code"),
|
||||
clEnumValEnd));
|
||||
|
||||
static cl::opt<llvm::CodeModel::Model>
|
||||
CMModel("code-model",
|
||||
cl::desc("Choose code model"),
|
||||
cl::init(CodeModel::Default),
|
||||
cl::values(clEnumValN(CodeModel::Default, "default",
|
||||
"Target default code model"),
|
||||
clEnumValN(CodeModel::Small, "small",
|
||||
"Small code model"),
|
||||
clEnumValN(CodeModel::Kernel, "kernel",
|
||||
"Kernel code model"),
|
||||
clEnumValN(CodeModel::Medium, "medium",
|
||||
"Medium code model"),
|
||||
clEnumValN(CodeModel::Large, "large",
|
||||
"Large code model"),
|
||||
clEnumValEnd));
|
||||
|
||||
static cl::opt<bool>
|
||||
NoInitialTextSection("n", cl::desc("Don't assume assembly file starts "
|
||||
"in the text section"));
|
||||
|
||||
static cl::opt<bool>
|
||||
SaveTempLabels("L", cl::desc("Don't discard temporary labels"));
|
||||
|
||||
enum ActionType {
|
||||
AC_AsLex,
|
||||
AC_Assemble,
|
||||
AC_Disassemble,
|
||||
AC_EDisassemble
|
||||
};
|
||||
|
||||
static cl::opt<ActionType>
|
||||
Action(cl::desc("Action to perform:"),
|
||||
cl::init(AC_Assemble),
|
||||
cl::values(clEnumValN(AC_AsLex, "as-lex",
|
||||
"Lex tokens from a .s file"),
|
||||
clEnumValN(AC_Assemble, "assemble",
|
||||
"Assemble a .s file (default)"),
|
||||
clEnumValN(AC_Disassemble, "disassemble",
|
||||
"Disassemble strings of hex bytes"),
|
||||
clEnumValN(AC_EDisassemble, "edis",
|
||||
"Enhanced disassembly of strings of hex bytes"),
|
||||
clEnumValEnd));
|
||||
|
||||
static const Target *GetTarget(const char *ProgName) {
|
||||
// Figure out the target triple.
|
||||
if (TripleName.empty())
|
||||
TripleName = sys::getHostTriple();
|
||||
Triple TheTriple(Triple::normalize(TripleName));
|
||||
|
||||
const Target *TheTarget = 0;
|
||||
if (!ArchName.empty()) {
|
||||
for (TargetRegistry::iterator it = TargetRegistry::begin(),
|
||||
ie = TargetRegistry::end(); it != ie; ++it) {
|
||||
if (ArchName == it->getName()) {
|
||||
TheTarget = &*it;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!TheTarget) {
|
||||
errs() << ProgName << ": error: invalid target '" << ArchName << "'.\n";
|
||||
return 0;
|
||||
}
|
||||
|
||||
// Adjust the triple to match (if known), otherwise stick with the
|
||||
// module/host triple.
|
||||
Triple::ArchType Type = Triple::getArchTypeForLLVMName(ArchName);
|
||||
if (Type != Triple::UnknownArch)
|
||||
TheTriple.setArch(Type);
|
||||
} else {
|
||||
// Get the target specific parser.
|
||||
std::string Error;
|
||||
TheTarget = TargetRegistry::lookupTarget(TheTriple.getTriple(), Error);
|
||||
if (TheTarget == 0) {
|
||||
errs() << ProgName << ": error: unable to get target for '"
|
||||
<< TheTriple.getTriple()
|
||||
<< "', see --version and --triple.\n";
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
TripleName = TheTriple.getTriple();
|
||||
return TheTarget;
|
||||
}
|
||||
|
||||
static tool_output_file *GetOutputStream() {
|
||||
if (OutputFilename == "")
|
||||
OutputFilename = "-";
|
||||
|
||||
std::string Err;
|
||||
tool_output_file *Out = new tool_output_file(OutputFilename.c_str(), Err,
|
||||
raw_fd_ostream::F_Binary);
|
||||
if (!Err.empty()) {
|
||||
errs() << Err << '\n';
|
||||
delete Out;
|
||||
return 0;
|
||||
}
|
||||
|
||||
return Out;
|
||||
}
|
||||
|
||||
static int AsLexInput(const char *ProgName) {
|
||||
OwningPtr<MemoryBuffer> BufferPtr;
|
||||
if (error_code ec = MemoryBuffer::getFileOrSTDIN(InputFilename, BufferPtr)) {
|
||||
errs() << ProgName << ": " << ec.message() << '\n';
|
||||
return 1;
|
||||
}
|
||||
MemoryBuffer *Buffer = BufferPtr.take();
|
||||
|
||||
SourceMgr SrcMgr;
|
||||
|
||||
// Tell SrcMgr about this buffer, which is what TGParser will pick up.
|
||||
SrcMgr.AddNewSourceBuffer(Buffer, SMLoc());
|
||||
|
||||
// Record the location of the include directories so that the lexer can find
|
||||
// it later.
|
||||
SrcMgr.setIncludeDirs(IncludeDirs);
|
||||
|
||||
const Target *TheTarget = GetTarget(ProgName);
|
||||
if (!TheTarget)
|
||||
return 1;
|
||||
|
||||
llvm::OwningPtr<MCAsmInfo> MAI(TheTarget->createMCAsmInfo(TripleName));
|
||||
assert(MAI && "Unable to create target asm info!");
|
||||
|
||||
AsmLexer Lexer(*MAI);
|
||||
Lexer.setBuffer(SrcMgr.getMemoryBuffer(0));
|
||||
|
||||
OwningPtr<tool_output_file> Out(GetOutputStream());
|
||||
if (!Out)
|
||||
return 1;
|
||||
|
||||
bool Error = false;
|
||||
while (Lexer.Lex().isNot(AsmToken::Eof)) {
|
||||
AsmToken Tok = Lexer.getTok();
|
||||
|
||||
switch (Tok.getKind()) {
|
||||
default:
|
||||
SrcMgr.PrintMessage(Lexer.getLoc(), "unknown token", "warning");
|
||||
Error = true;
|
||||
break;
|
||||
case AsmToken::Error:
|
||||
Error = true; // error already printed.
|
||||
break;
|
||||
case AsmToken::Identifier:
|
||||
Out->os() << "identifier: " << Lexer.getTok().getString();
|
||||
break;
|
||||
case AsmToken::Integer:
|
||||
Out->os() << "int: " << Lexer.getTok().getString();
|
||||
break;
|
||||
case AsmToken::Real:
|
||||
Out->os() << "real: " << Lexer.getTok().getString();
|
||||
break;
|
||||
case AsmToken::Register:
|
||||
Out->os() << "register: " << Lexer.getTok().getRegVal();
|
||||
break;
|
||||
case AsmToken::String:
|
||||
Out->os() << "string: " << Lexer.getTok().getString();
|
||||
break;
|
||||
|
||||
case AsmToken::Amp: Out->os() << "Amp"; break;
|
||||
case AsmToken::AmpAmp: Out->os() << "AmpAmp"; break;
|
||||
case AsmToken::At: Out->os() << "At"; break;
|
||||
case AsmToken::Caret: Out->os() << "Caret"; break;
|
||||
case AsmToken::Colon: Out->os() << "Colon"; break;
|
||||
case AsmToken::Comma: Out->os() << "Comma"; break;
|
||||
case AsmToken::Dollar: Out->os() << "Dollar"; break;
|
||||
case AsmToken::Dot: Out->os() << "Dot"; break;
|
||||
case AsmToken::EndOfStatement: Out->os() << "EndOfStatement"; break;
|
||||
case AsmToken::Eof: Out->os() << "Eof"; break;
|
||||
case AsmToken::Equal: Out->os() << "Equal"; break;
|
||||
case AsmToken::EqualEqual: Out->os() << "EqualEqual"; break;
|
||||
case AsmToken::Exclaim: Out->os() << "Exclaim"; break;
|
||||
case AsmToken::ExclaimEqual: Out->os() << "ExclaimEqual"; break;
|
||||
case AsmToken::Greater: Out->os() << "Greater"; break;
|
||||
case AsmToken::GreaterEqual: Out->os() << "GreaterEqual"; break;
|
||||
case AsmToken::GreaterGreater: Out->os() << "GreaterGreater"; break;
|
||||
case AsmToken::Hash: Out->os() << "Hash"; break;
|
||||
case AsmToken::LBrac: Out->os() << "LBrac"; break;
|
||||
case AsmToken::LCurly: Out->os() << "LCurly"; break;
|
||||
case AsmToken::LParen: Out->os() << "LParen"; break;
|
||||
case AsmToken::Less: Out->os() << "Less"; break;
|
||||
case AsmToken::LessEqual: Out->os() << "LessEqual"; break;
|
||||
case AsmToken::LessGreater: Out->os() << "LessGreater"; break;
|
||||
case AsmToken::LessLess: Out->os() << "LessLess"; break;
|
||||
case AsmToken::Minus: Out->os() << "Minus"; break;
|
||||
case AsmToken::Percent: Out->os() << "Percent"; break;
|
||||
case AsmToken::Pipe: Out->os() << "Pipe"; break;
|
||||
case AsmToken::PipePipe: Out->os() << "PipePipe"; break;
|
||||
case AsmToken::Plus: Out->os() << "Plus"; break;
|
||||
case AsmToken::RBrac: Out->os() << "RBrac"; break;
|
||||
case AsmToken::RCurly: Out->os() << "RCurly"; break;
|
||||
case AsmToken::RParen: Out->os() << "RParen"; break;
|
||||
case AsmToken::Slash: Out->os() << "Slash"; break;
|
||||
case AsmToken::Star: Out->os() << "Star"; break;
|
||||
case AsmToken::Tilde: Out->os() << "Tilde"; break;
|
||||
}
|
||||
|
||||
// Print the token string.
|
||||
Out->os() << " (\"";
|
||||
Out->os().write_escaped(Tok.getString());
|
||||
Out->os() << "\")\n";
|
||||
}
|
||||
|
||||
// Keep output if no errors.
|
||||
if (Error == 0) Out->keep();
|
||||
|
||||
return Error;
|
||||
}
|
||||
|
||||
static int AssembleInput(const char *ProgName) {
|
||||
const Target *TheTarget = GetTarget(ProgName);
|
||||
if (!TheTarget)
|
||||
return 1;
|
||||
|
||||
OwningPtr<MemoryBuffer> BufferPtr;
|
||||
if (error_code ec = MemoryBuffer::getFileOrSTDIN(InputFilename, BufferPtr)) {
|
||||
errs() << ProgName << ": " << ec.message() << '\n';
|
||||
return 1;
|
||||
}
|
||||
MemoryBuffer *Buffer = BufferPtr.take();
|
||||
|
||||
SourceMgr SrcMgr;
|
||||
|
||||
// Tell SrcMgr about this buffer, which is what the parser will pick up.
|
||||
SrcMgr.AddNewSourceBuffer(Buffer, SMLoc());
|
||||
|
||||
// Record the location of the include directories so that the lexer can find
|
||||
// it later.
|
||||
SrcMgr.setIncludeDirs(IncludeDirs);
|
||||
|
||||
|
||||
llvm::OwningPtr<MCAsmInfo> MAI(TheTarget->createMCAsmInfo(TripleName));
|
||||
assert(MAI && "Unable to create target asm info!");
|
||||
|
||||
llvm::OwningPtr<MCRegisterInfo> MRI(TheTarget->createMCRegInfo(TripleName));
|
||||
assert(MRI && "Unable to create target register info!");
|
||||
|
||||
// FIXME: This is not pretty. MCContext has a ptr to MCObjectFileInfo and
|
||||
// MCObjectFileInfo needs a MCContext reference in order to initialize itself.
|
||||
OwningPtr<MCObjectFileInfo> MOFI(new MCObjectFileInfo());
|
||||
MCContext Ctx(*MAI, *MRI, MOFI.get());
|
||||
MOFI->InitMCObjectFileInfo(TripleName, RelocModel, CMModel, Ctx);
|
||||
|
||||
if (SaveTempLabels)
|
||||
Ctx.setAllowTemporaryLabels(false);
|
||||
|
||||
// Package up features to be passed to target/subtarget
|
||||
std::string FeaturesStr;
|
||||
if (MAttrs.size()) {
|
||||
SubtargetFeatures Features;
|
||||
for (unsigned i = 0; i != MAttrs.size(); ++i)
|
||||
Features.AddFeature(MAttrs[i]);
|
||||
FeaturesStr = Features.getString();
|
||||
}
|
||||
|
||||
OwningPtr<tool_output_file> Out(GetOutputStream());
|
||||
if (!Out)
|
||||
return 1;
|
||||
|
||||
formatted_raw_ostream FOS(Out->os());
|
||||
OwningPtr<MCStreamer> Str;
|
||||
|
||||
OwningPtr<MCInstrInfo> MCII(TheTarget->createMCInstrInfo());
|
||||
OwningPtr<MCSubtargetInfo>
|
||||
STI(TheTarget->createMCSubtargetInfo(TripleName, MCPU, FeaturesStr));
|
||||
|
||||
// FIXME: There is a bit of code duplication with addPassesToEmitFile.
|
||||
if (FileType == OFT_AssemblyFile) {
|
||||
MCInstPrinter *IP =
|
||||
TheTarget->createMCInstPrinter(OutputAsmVariant, *MAI, *STI);
|
||||
MCCodeEmitter *CE = 0;
|
||||
MCAsmBackend *MAB = 0;
|
||||
if (ShowEncoding) {
|
||||
CE = TheTarget->createMCCodeEmitter(*MCII, *STI, Ctx);
|
||||
MAB = TheTarget->createMCAsmBackend(TripleName);
|
||||
}
|
||||
Str.reset(TheTarget->createAsmStreamer(Ctx, FOS, /*asmverbose*/true,
|
||||
/*useLoc*/ true,
|
||||
/*useCFI*/ true, IP, CE, MAB,
|
||||
ShowInst));
|
||||
} else if (FileType == OFT_Null) {
|
||||
Str.reset(createNullStreamer(Ctx));
|
||||
} else {
|
||||
assert(FileType == OFT_ObjectFile && "Invalid file type!");
|
||||
MCCodeEmitter *CE = TheTarget->createMCCodeEmitter(*MCII, *STI, Ctx);
|
||||
MCAsmBackend *MAB = TheTarget->createMCAsmBackend(TripleName);
|
||||
Str.reset(TheTarget->createMCObjectStreamer(TripleName, Ctx, *MAB,
|
||||
FOS, CE, RelaxAll,
|
||||
NoExecStack));
|
||||
}
|
||||
|
||||
if (EnableLogging) {
|
||||
Str.reset(createLoggingStreamer(Str.take(), errs()));
|
||||
}
|
||||
|
||||
OwningPtr<MCAsmParser> Parser(createMCAsmParser(SrcMgr, Ctx,
|
||||
*Str.get(), *MAI));
|
||||
OwningPtr<MCTargetAsmParser> TAP(TheTarget->createMCAsmParser(*STI, *Parser));
|
||||
if (!TAP) {
|
||||
errs() << ProgName
|
||||
<< ": error: this target does not support assembly parsing.\n";
|
||||
return 1;
|
||||
}
|
||||
|
||||
Parser->setShowParsedOperands(ShowInstOperands);
|
||||
Parser->setTargetParser(*TAP.get());
|
||||
|
||||
int Res = Parser->Run(NoInitialTextSection);
|
||||
|
||||
// Keep output if no errors.
|
||||
if (Res == 0) Out->keep();
|
||||
|
||||
return Res;
|
||||
}
|
||||
|
||||
static int DisassembleInput(const char *ProgName, bool Enhanced) {
|
||||
const Target *TheTarget = GetTarget(ProgName);
|
||||
if (!TheTarget)
|
||||
return 0;
|
||||
|
||||
OwningPtr<MemoryBuffer> Buffer;
|
||||
if (error_code ec = MemoryBuffer::getFileOrSTDIN(InputFilename, Buffer)) {
|
||||
errs() << ProgName << ": " << ec.message() << '\n';
|
||||
return 1;
|
||||
}
|
||||
|
||||
OwningPtr<tool_output_file> Out(GetOutputStream());
|
||||
if (!Out)
|
||||
return 1;
|
||||
|
||||
int Res;
|
||||
if (Enhanced) {
|
||||
Res =
|
||||
Disassembler::disassembleEnhanced(TripleName, *Buffer.take(), Out->os());
|
||||
} else {
|
||||
// Package up features to be passed to target/subtarget
|
||||
std::string FeaturesStr;
|
||||
if (MAttrs.size()) {
|
||||
SubtargetFeatures Features;
|
||||
for (unsigned i = 0; i != MAttrs.size(); ++i)
|
||||
Features.AddFeature(MAttrs[i]);
|
||||
FeaturesStr = Features.getString();
|
||||
}
|
||||
|
||||
Res = Disassembler::disassemble(*TheTarget, TripleName, MCPU, FeaturesStr,
|
||||
*Buffer.take(), Out->os());
|
||||
}
|
||||
|
||||
// Keep output if no errors.
|
||||
if (Res == 0) Out->keep();
|
||||
|
||||
return Res;
|
||||
}
|
||||
|
||||
|
||||
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.
|
||||
|
||||
// Initialize targets and assembly printers/parsers.
|
||||
llvm::InitializeAllTargetInfos();
|
||||
llvm::InitializeAllTargetMCs();
|
||||
llvm::InitializeAllAsmParsers();
|
||||
llvm::InitializeAllDisassemblers();
|
||||
|
||||
cl::ParseCommandLineOptions(argc, argv, "llvm machine code playground\n");
|
||||
TripleName = Triple::normalize(TripleName);
|
||||
|
||||
switch (Action) {
|
||||
default:
|
||||
case AC_AsLex:
|
||||
return AsLexInput(argv[0]);
|
||||
case AC_Assemble:
|
||||
return AssembleInput(argv[0]);
|
||||
case AC_Disassemble:
|
||||
return DisassembleInput(argv[0], false);
|
||||
case AC_EDisassemble:
|
||||
return DisassembleInput(argv[0], true);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
5
contrib/llvm/tools/llvm-nm/CMakeLists.txt
Normal file
5
contrib/llvm/tools/llvm-nm/CMakeLists.txt
Normal file
@ -0,0 +1,5 @@
|
||||
set(LLVM_LINK_COMPONENTS archive bitreader object)
|
||||
|
||||
add_llvm_tool(llvm-nm
|
||||
llvm-nm.cpp
|
||||
)
|
17
contrib/llvm/tools/llvm-nm/Makefile
Normal file
17
contrib/llvm/tools/llvm-nm/Makefile
Normal file
@ -0,0 +1,17 @@
|
||||
##===- tools/llvm-nm/Makefile ------------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
LEVEL = ../..
|
||||
|
||||
TOOLNAME = llvm-nm
|
||||
LINK_COMPONENTS = archive bitreader object
|
||||
|
||||
# This tool has no plugins, optimize startup time.
|
||||
TOOL_NO_EXPORTS = 1
|
||||
|
||||
include $(LEVEL)/Makefile.common
|
392
contrib/llvm/tools/llvm-nm/llvm-nm.cpp
Normal file
392
contrib/llvm/tools/llvm-nm/llvm-nm.cpp
Normal file
@ -0,0 +1,392 @@
|
||||
//===-- llvm-nm.cpp - Symbol table 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 traditional Unix "nm",
|
||||
// that is, it prints out the names of symbols in a bitcode file,
|
||||
// along with some information about each symbol.
|
||||
//
|
||||
// This "nm" does not print symbols' addresses. It supports many of
|
||||
// the features of GNU "nm", including its different output formats.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/Bitcode/ReaderWriter.h"
|
||||
#include "llvm/Bitcode/Archive.h"
|
||||
#include "llvm/Object/Archive.h"
|
||||
#include "llvm/Object/ObjectFile.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/FileSystem.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/MemoryBuffer.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include "llvm/Support/Format.h"
|
||||
#include "llvm/Support/system_error.h"
|
||||
#include <algorithm>
|
||||
#include <cctype>
|
||||
#include <cerrno>
|
||||
#include <cstring>
|
||||
#include <vector>
|
||||
using namespace llvm;
|
||||
using namespace object;
|
||||
|
||||
namespace {
|
||||
enum OutputFormatTy { bsd, sysv, posix };
|
||||
cl::opt<OutputFormatTy>
|
||||
OutputFormat("format",
|
||||
cl::desc("Specify output format"),
|
||||
cl::values(clEnumVal(bsd, "BSD format"),
|
||||
clEnumVal(sysv, "System V format"),
|
||||
clEnumVal(posix, "POSIX.2 format"),
|
||||
clEnumValEnd), cl::init(bsd));
|
||||
cl::alias OutputFormat2("f", cl::desc("Alias for --format"),
|
||||
cl::aliasopt(OutputFormat));
|
||||
|
||||
cl::list<std::string>
|
||||
InputFilenames(cl::Positional, cl::desc("<input bitcode files>"),
|
||||
cl::ZeroOrMore);
|
||||
|
||||
cl::opt<bool> UndefinedOnly("undefined-only",
|
||||
cl::desc("Show only undefined symbols"));
|
||||
cl::alias UndefinedOnly2("u", cl::desc("Alias for --undefined-only"),
|
||||
cl::aliasopt(UndefinedOnly));
|
||||
|
||||
cl::opt<bool> DefinedOnly("defined-only",
|
||||
cl::desc("Show only defined symbols"));
|
||||
|
||||
cl::opt<bool> ExternalOnly("extern-only",
|
||||
cl::desc("Show only external symbols"));
|
||||
cl::alias ExternalOnly2("g", cl::desc("Alias for --extern-only"),
|
||||
cl::aliasopt(ExternalOnly));
|
||||
|
||||
cl::opt<bool> BSDFormat("B", cl::desc("Alias for --format=bsd"));
|
||||
cl::opt<bool> POSIXFormat("P", cl::desc("Alias for --format=posix"));
|
||||
|
||||
cl::opt<bool> PrintFileName("print-file-name",
|
||||
cl::desc("Precede each symbol with the object file it came from"));
|
||||
|
||||
cl::alias PrintFileNameA("A", cl::desc("Alias for --print-file-name"),
|
||||
cl::aliasopt(PrintFileName));
|
||||
cl::alias PrintFileNameo("o", cl::desc("Alias for --print-file-name"),
|
||||
cl::aliasopt(PrintFileName));
|
||||
|
||||
cl::opt<bool> DebugSyms("debug-syms",
|
||||
cl::desc("Show all symbols, even debugger only"));
|
||||
cl::alias DebugSymsa("a", cl::desc("Alias for --debug-syms"),
|
||||
cl::aliasopt(DebugSyms));
|
||||
|
||||
cl::opt<bool> NumericSort("numeric-sort",
|
||||
cl::desc("Sort symbols by address"));
|
||||
cl::alias NumericSortn("n", cl::desc("Alias for --numeric-sort"),
|
||||
cl::aliasopt(NumericSort));
|
||||
cl::alias NumericSortv("v", cl::desc("Alias for --numeric-sort"),
|
||||
cl::aliasopt(NumericSort));
|
||||
|
||||
cl::opt<bool> NoSort("no-sort",
|
||||
cl::desc("Show symbols in order encountered"));
|
||||
cl::alias NoSortp("p", cl::desc("Alias for --no-sort"),
|
||||
cl::aliasopt(NoSort));
|
||||
|
||||
cl::opt<bool> PrintSize("print-size",
|
||||
cl::desc("Show symbol size instead of address"));
|
||||
cl::alias PrintSizeS("S", cl::desc("Alias for --print-size"),
|
||||
cl::aliasopt(PrintSize));
|
||||
|
||||
cl::opt<bool> SizeSort("size-sort", cl::desc("Sort symbols by size"));
|
||||
|
||||
bool PrintAddress = true;
|
||||
|
||||
bool MultipleFiles = false;
|
||||
|
||||
std::string ToolName;
|
||||
}
|
||||
|
||||
namespace {
|
||||
struct NMSymbol {
|
||||
uint64_t Address;
|
||||
uint64_t Size;
|
||||
char TypeChar;
|
||||
StringRef Name;
|
||||
};
|
||||
|
||||
static bool CompareSymbolAddress(const NMSymbol &a, const NMSymbol &b) {
|
||||
if (a.Address < b.Address)
|
||||
return true;
|
||||
else if (a.Address == b.Address && a.Name < b.Name)
|
||||
return true;
|
||||
else
|
||||
return false;
|
||||
|
||||
}
|
||||
|
||||
static bool CompareSymbolSize(const NMSymbol &a, const NMSymbol &b) {
|
||||
if (a.Size < b.Size)
|
||||
return true;
|
||||
else if (a.Size == b.Size && a.Name < b.Name)
|
||||
return true;
|
||||
else
|
||||
return false;
|
||||
}
|
||||
|
||||
static bool CompareSymbolName(const NMSymbol &a, const NMSymbol &b) {
|
||||
return a.Name < b.Name;
|
||||
}
|
||||
|
||||
StringRef CurrentFilename;
|
||||
typedef std::vector<NMSymbol> SymbolListT;
|
||||
SymbolListT SymbolList;
|
||||
|
||||
bool error(error_code ec) {
|
||||
if (!ec) return false;
|
||||
|
||||
outs() << ToolName << ": error reading file: " << ec.message() << ".\n";
|
||||
outs().flush();
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
static void SortAndPrintSymbolList() {
|
||||
if (!NoSort) {
|
||||
if (NumericSort)
|
||||
std::sort(SymbolList.begin(), SymbolList.end(), CompareSymbolAddress);
|
||||
else if (SizeSort)
|
||||
std::sort(SymbolList.begin(), SymbolList.end(), CompareSymbolSize);
|
||||
else
|
||||
std::sort(SymbolList.begin(), SymbolList.end(), CompareSymbolName);
|
||||
}
|
||||
|
||||
if (OutputFormat == posix && MultipleFiles) {
|
||||
outs() << '\n' << CurrentFilename << ":\n";
|
||||
} else if (OutputFormat == bsd && MultipleFiles) {
|
||||
outs() << "\n" << CurrentFilename << ":\n";
|
||||
} else if (OutputFormat == sysv) {
|
||||
outs() << "\n\nSymbols from " << CurrentFilename << ":\n\n"
|
||||
<< "Name Value Class Type"
|
||||
<< " Size Line Section\n";
|
||||
}
|
||||
|
||||
for (SymbolListT::iterator i = SymbolList.begin(),
|
||||
e = SymbolList.end(); i != e; ++i) {
|
||||
if ((i->TypeChar != 'U') && UndefinedOnly)
|
||||
continue;
|
||||
if ((i->TypeChar == 'U') && DefinedOnly)
|
||||
continue;
|
||||
if (SizeSort && !PrintAddress && i->Size == UnknownAddressOrSize)
|
||||
continue;
|
||||
|
||||
char SymbolAddrStr[10] = "";
|
||||
char SymbolSizeStr[10] = "";
|
||||
|
||||
if (OutputFormat == sysv || i->Address == object::UnknownAddressOrSize)
|
||||
strcpy(SymbolAddrStr, " ");
|
||||
if (OutputFormat == sysv)
|
||||
strcpy(SymbolSizeStr, " ");
|
||||
|
||||
if (i->Address != object::UnknownAddressOrSize)
|
||||
format("%08"PRIx64, i->Address).print(SymbolAddrStr, sizeof(SymbolAddrStr));
|
||||
if (i->Size != object::UnknownAddressOrSize)
|
||||
format("%08"PRIx64, i->Size).print(SymbolSizeStr, sizeof(SymbolSizeStr));
|
||||
|
||||
if (OutputFormat == posix) {
|
||||
outs() << i->Name << " " << i->TypeChar << " "
|
||||
<< SymbolAddrStr << SymbolSizeStr << "\n";
|
||||
} else if (OutputFormat == bsd) {
|
||||
if (PrintAddress)
|
||||
outs() << SymbolAddrStr << ' ';
|
||||
if (PrintSize) {
|
||||
outs() << SymbolSizeStr;
|
||||
if (i->Size != object::UnknownAddressOrSize)
|
||||
outs() << ' ';
|
||||
}
|
||||
outs() << i->TypeChar << " " << i->Name << "\n";
|
||||
} else if (OutputFormat == sysv) {
|
||||
std::string PaddedName (i->Name);
|
||||
while (PaddedName.length () < 20)
|
||||
PaddedName += " ";
|
||||
outs() << PaddedName << "|" << SymbolAddrStr << "| "
|
||||
<< i->TypeChar
|
||||
<< " | |" << SymbolSizeStr << "| |\n";
|
||||
}
|
||||
}
|
||||
|
||||
SymbolList.clear();
|
||||
}
|
||||
|
||||
static char TypeCharForSymbol(GlobalValue &GV) {
|
||||
if (GV.isDeclaration()) return 'U';
|
||||
if (GV.hasLinkOnceLinkage()) return 'C';
|
||||
if (GV.hasCommonLinkage()) return 'C';
|
||||
if (GV.hasWeakLinkage()) return 'W';
|
||||
if (isa<Function>(GV) && GV.hasInternalLinkage()) return 't';
|
||||
if (isa<Function>(GV)) return 'T';
|
||||
if (isa<GlobalVariable>(GV) && GV.hasInternalLinkage()) return 'd';
|
||||
if (isa<GlobalVariable>(GV)) return 'D';
|
||||
if (const GlobalAlias *GA = dyn_cast<GlobalAlias>(&GV)) {
|
||||
const GlobalValue *AliasedGV = GA->getAliasedGlobal();
|
||||
if (isa<Function>(AliasedGV)) return 'T';
|
||||
if (isa<GlobalVariable>(AliasedGV)) return 'D';
|
||||
}
|
||||
return '?';
|
||||
}
|
||||
|
||||
static void DumpSymbolNameForGlobalValue(GlobalValue &GV) {
|
||||
// Private linkage and available_externally linkage don't exist in symtab.
|
||||
if (GV.hasPrivateLinkage() ||
|
||||
GV.hasLinkerPrivateLinkage() ||
|
||||
GV.hasLinkerPrivateWeakLinkage() ||
|
||||
GV.hasLinkerPrivateWeakDefAutoLinkage() ||
|
||||
GV.hasAvailableExternallyLinkage())
|
||||
return;
|
||||
char TypeChar = TypeCharForSymbol(GV);
|
||||
if (GV.hasLocalLinkage () && ExternalOnly)
|
||||
return;
|
||||
|
||||
NMSymbol s;
|
||||
s.Address = object::UnknownAddressOrSize;
|
||||
s.Size = object::UnknownAddressOrSize;
|
||||
s.TypeChar = TypeChar;
|
||||
s.Name = GV.getName();
|
||||
SymbolList.push_back(s);
|
||||
}
|
||||
|
||||
static void DumpSymbolNamesFromModule(Module *M) {
|
||||
CurrentFilename = M->getModuleIdentifier();
|
||||
std::for_each (M->begin(), M->end(), DumpSymbolNameForGlobalValue);
|
||||
std::for_each (M->global_begin(), M->global_end(),
|
||||
DumpSymbolNameForGlobalValue);
|
||||
std::for_each (M->alias_begin(), M->alias_end(),
|
||||
DumpSymbolNameForGlobalValue);
|
||||
|
||||
SortAndPrintSymbolList();
|
||||
}
|
||||
|
||||
static void DumpSymbolNamesFromObject(ObjectFile *obj) {
|
||||
error_code ec;
|
||||
for (symbol_iterator i = obj->begin_symbols(),
|
||||
e = obj->end_symbols();
|
||||
i != e; i.increment(ec)) {
|
||||
if (error(ec)) break;
|
||||
bool internal;
|
||||
if (error(i->isInternal(internal))) break;
|
||||
if (!DebugSyms && internal)
|
||||
continue;
|
||||
NMSymbol s;
|
||||
s.Size = object::UnknownAddressOrSize;
|
||||
s.Address = object::UnknownAddressOrSize;
|
||||
if (PrintSize || SizeSort) {
|
||||
if (error(i->getSize(s.Size))) break;
|
||||
}
|
||||
if (PrintAddress)
|
||||
if (error(i->getOffset(s.Address))) break;
|
||||
if (error(i->getNMTypeChar(s.TypeChar))) break;
|
||||
if (error(i->getName(s.Name))) break;
|
||||
SymbolList.push_back(s);
|
||||
}
|
||||
|
||||
CurrentFilename = obj->getFileName();
|
||||
SortAndPrintSymbolList();
|
||||
}
|
||||
|
||||
static void DumpSymbolNamesFromFile(std::string &Filename) {
|
||||
LLVMContext &Context = getGlobalContext();
|
||||
std::string ErrorMessage;
|
||||
sys::Path aPath(Filename);
|
||||
bool exists;
|
||||
if (sys::fs::exists(aPath.str(), exists) || !exists)
|
||||
errs() << ToolName << ": '" << Filename << "': " << "No such file\n";
|
||||
// Note: Currently we do not support reading an archive from stdin.
|
||||
if (Filename == "-" || aPath.isBitcodeFile()) {
|
||||
OwningPtr<MemoryBuffer> Buffer;
|
||||
if (error_code ec = MemoryBuffer::getFileOrSTDIN(Filename, Buffer))
|
||||
ErrorMessage = ec.message();
|
||||
Module *Result = 0;
|
||||
if (Buffer.get())
|
||||
Result = ParseBitcodeFile(Buffer.get(), Context, &ErrorMessage);
|
||||
|
||||
if (Result) {
|
||||
DumpSymbolNamesFromModule(Result);
|
||||
delete Result;
|
||||
} else
|
||||
errs() << ToolName << ": " << Filename << ": " << ErrorMessage << "\n";
|
||||
|
||||
} else if (aPath.isArchive()) {
|
||||
OwningPtr<Binary> arch;
|
||||
if (error_code ec = object::createBinary(aPath.str(), arch)) {
|
||||
errs() << ToolName << ": " << Filename << ": " << ec.message() << ".\n";
|
||||
return;
|
||||
}
|
||||
if (object::Archive *a = dyn_cast<object::Archive>(arch.get())) {
|
||||
for (object::Archive::child_iterator i = a->begin_children(),
|
||||
e = a->end_children(); i != e; ++i) {
|
||||
OwningPtr<Binary> child;
|
||||
if (error_code ec = i->getAsBinary(child)) {
|
||||
// Try opening it as a bitcode file.
|
||||
OwningPtr<MemoryBuffer> buff(i->getBuffer());
|
||||
Module *Result = 0;
|
||||
if (buff)
|
||||
Result = ParseBitcodeFile(buff.get(), Context, &ErrorMessage);
|
||||
|
||||
if (Result) {
|
||||
DumpSymbolNamesFromModule(Result);
|
||||
delete Result;
|
||||
}
|
||||
continue;
|
||||
}
|
||||
if (object::ObjectFile *o = dyn_cast<ObjectFile>(child.get())) {
|
||||
outs() << o->getFileName() << ":\n";
|
||||
DumpSymbolNamesFromObject(o);
|
||||
}
|
||||
}
|
||||
}
|
||||
} else if (aPath.isObjectFile()) {
|
||||
OwningPtr<Binary> obj;
|
||||
if (error_code ec = object::createBinary(aPath.str(), obj)) {
|
||||
errs() << ToolName << ": " << Filename << ": " << ec.message() << ".\n";
|
||||
return;
|
||||
}
|
||||
if (object::ObjectFile *o = dyn_cast<ObjectFile>(obj.get()))
|
||||
DumpSymbolNamesFromObject(o);
|
||||
} else {
|
||||
errs() << ToolName << ": " << Filename << ": "
|
||||
<< "unrecognizable file type\n";
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
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 symbol table dumper\n");
|
||||
|
||||
ToolName = argv[0];
|
||||
if (BSDFormat) OutputFormat = bsd;
|
||||
if (POSIXFormat) OutputFormat = posix;
|
||||
|
||||
// The relative order of these is important. If you pass --size-sort it should
|
||||
// only print out the size. However, if you pass -S --size-sort, it should
|
||||
// print out both the size and address.
|
||||
if (SizeSort && !PrintSize) PrintAddress = false;
|
||||
if (OutputFormat == sysv || SizeSort) PrintSize = true;
|
||||
|
||||
switch (InputFilenames.size()) {
|
||||
case 0: InputFilenames.push_back("-");
|
||||
case 1: break;
|
||||
default: MultipleFiles = true;
|
||||
}
|
||||
|
||||
std::for_each(InputFilenames.begin(), InputFilenames.end(),
|
||||
DumpSymbolNamesFromFile);
|
||||
return 0;
|
||||
}
|
14
contrib/llvm/tools/llvm-objdump/CMakeLists.txt
Normal file
14
contrib/llvm/tools/llvm-objdump/CMakeLists.txt
Normal file
@ -0,0 +1,14 @@
|
||||
set(LLVM_LINK_COMPONENTS
|
||||
${LLVM_TARGETS_TO_BUILD}
|
||||
DebugInfo
|
||||
MC
|
||||
MCParser
|
||||
MCDisassembler
|
||||
Object
|
||||
)
|
||||
|
||||
add_llvm_tool(llvm-objdump
|
||||
llvm-objdump.cpp
|
||||
MachODump.cpp
|
||||
MCFunction.cpp
|
||||
)
|
138
contrib/llvm/tools/llvm-objdump/MCFunction.cpp
Normal file
138
contrib/llvm/tools/llvm-objdump/MCFunction.cpp
Normal file
@ -0,0 +1,138 @@
|
||||
//===-- MCFunction.cpp ----------------------------------------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file defines the algorithm to break down a region of machine code
|
||||
// into basic blocks and try to reconstruct a CFG from it.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "MCFunction.h"
|
||||
#include "llvm/ADT/STLExtras.h"
|
||||
#include "llvm/MC/MCDisassembler.h"
|
||||
#include "llvm/MC/MCInst.h"
|
||||
#include "llvm/MC/MCInstPrinter.h"
|
||||
#include "llvm/MC/MCInstrAnalysis.h"
|
||||
#include "llvm/MC/MCInstrDesc.h"
|
||||
#include "llvm/MC/MCInstrInfo.h"
|
||||
#include "llvm/Support/MemoryObject.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/system_error.h"
|
||||
#include <set>
|
||||
using namespace llvm;
|
||||
|
||||
MCFunction
|
||||
MCFunction::createFunctionFromMC(StringRef Name, const MCDisassembler *DisAsm,
|
||||
const MemoryObject &Region, uint64_t Start,
|
||||
uint64_t End, const MCInstrAnalysis *Ana,
|
||||
raw_ostream &DebugOut,
|
||||
SmallVectorImpl<uint64_t> &Calls) {
|
||||
std::vector<MCDecodedInst> Instructions;
|
||||
std::set<uint64_t> Splits;
|
||||
Splits.insert(Start);
|
||||
uint64_t Size;
|
||||
|
||||
MCFunction f(Name);
|
||||
|
||||
{
|
||||
DenseSet<uint64_t> VisitedInsts;
|
||||
SmallVector<uint64_t, 16> WorkList;
|
||||
WorkList.push_back(Start);
|
||||
// Disassemble code and gather basic block split points.
|
||||
while (!WorkList.empty()) {
|
||||
uint64_t Index = WorkList.pop_back_val();
|
||||
if (VisitedInsts.find(Index) != VisitedInsts.end())
|
||||
continue; // Already visited this location.
|
||||
|
||||
for (;Index < End; Index += Size) {
|
||||
VisitedInsts.insert(Index);
|
||||
|
||||
MCInst Inst;
|
||||
if (DisAsm->getInstruction(Inst, Size, Region, Index, DebugOut, nulls())){
|
||||
Instructions.push_back(MCDecodedInst(Index, Size, Inst));
|
||||
if (Ana->isBranch(Inst)) {
|
||||
uint64_t targ = Ana->evaluateBranch(Inst, Index, Size);
|
||||
if (targ != -1ULL && targ == Index+Size)
|
||||
continue; // Skip nop jumps.
|
||||
|
||||
// If we could determine the branch target, make a note to start a
|
||||
// new basic block there and add the target to the worklist.
|
||||
if (targ != -1ULL) {
|
||||
Splits.insert(targ);
|
||||
WorkList.push_back(targ);
|
||||
WorkList.push_back(Index+Size);
|
||||
}
|
||||
Splits.insert(Index+Size);
|
||||
break;
|
||||
} else if (Ana->isReturn(Inst)) {
|
||||
// Return instruction. This basic block ends here.
|
||||
Splits.insert(Index+Size);
|
||||
break;
|
||||
} else if (Ana->isCall(Inst)) {
|
||||
uint64_t targ = Ana->evaluateBranch(Inst, Index, Size);
|
||||
// Add the call to the call list if the destination is known.
|
||||
if (targ != -1ULL && targ != Index+Size)
|
||||
Calls.push_back(targ);
|
||||
}
|
||||
} else {
|
||||
errs().write_hex(Index) << ": warning: invalid instruction encoding\n";
|
||||
if (Size == 0)
|
||||
Size = 1; // skip illegible bytes
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Make sure the instruction list is sorted.
|
||||
std::sort(Instructions.begin(), Instructions.end());
|
||||
|
||||
// Create basic blocks.
|
||||
unsigned ii = 0, ie = Instructions.size();
|
||||
for (std::set<uint64_t>::iterator spi = Splits.begin(),
|
||||
spe = llvm::prior(Splits.end()); spi != spe; ++spi) {
|
||||
MCBasicBlock BB;
|
||||
uint64_t BlockEnd = *llvm::next(spi);
|
||||
// Add instructions to the BB.
|
||||
for (; ii != ie; ++ii) {
|
||||
if (Instructions[ii].Address < *spi ||
|
||||
Instructions[ii].Address >= BlockEnd)
|
||||
break;
|
||||
BB.addInst(Instructions[ii]);
|
||||
}
|
||||
f.addBlock(*spi, BB);
|
||||
}
|
||||
|
||||
std::sort(f.Blocks.begin(), f.Blocks.end());
|
||||
|
||||
// Calculate successors of each block.
|
||||
for (MCFunction::iterator i = f.begin(), e = f.end(); i != e; ++i) {
|
||||
MCBasicBlock &BB = const_cast<MCBasicBlock&>(i->second);
|
||||
if (BB.getInsts().empty()) continue;
|
||||
const MCDecodedInst &Inst = BB.getInsts().back();
|
||||
|
||||
if (Ana->isBranch(Inst.Inst)) {
|
||||
uint64_t targ = Ana->evaluateBranch(Inst.Inst, Inst.Address, Inst.Size);
|
||||
if (targ == -1ULL) {
|
||||
// Indirect branch. Bail and add all blocks of the function as a
|
||||
// successor.
|
||||
for (MCFunction::iterator i = f.begin(), e = f.end(); i != e; ++i)
|
||||
BB.addSucc(i->first);
|
||||
} else if (targ != Inst.Address+Inst.Size)
|
||||
BB.addSucc(targ);
|
||||
// Conditional branches can also fall through to the next block.
|
||||
if (Ana->isConditionalBranch(Inst.Inst) && llvm::next(i) != e)
|
||||
BB.addSucc(llvm::next(i)->first);
|
||||
} else {
|
||||
// No branch. Fall through to the next block.
|
||||
if (!Ana->isReturn(Inst.Inst) && llvm::next(i) != e)
|
||||
BB.addSucc(llvm::next(i)->first);
|
||||
}
|
||||
}
|
||||
|
||||
return f;
|
||||
}
|
100
contrib/llvm/tools/llvm-objdump/MCFunction.h
Normal file
100
contrib/llvm/tools/llvm-objdump/MCFunction.h
Normal file
@ -0,0 +1,100 @@
|
||||
//===-- MCFunction.h ------------------------------------------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file defines the data structures to hold a CFG reconstructed from
|
||||
// machine code.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef LLVM_OBJECTDUMP_MCFUNCTION_H
|
||||
#define LLVM_OBJECTDUMP_MCFUNCTION_H
|
||||
|
||||
#include "llvm/ADT/ArrayRef.h"
|
||||
#include "llvm/ADT/DenseSet.h"
|
||||
#include "llvm/MC/MCInst.h"
|
||||
#include <map>
|
||||
|
||||
namespace llvm {
|
||||
|
||||
class MCDisassembler;
|
||||
class MCInstrAnalysis;
|
||||
class MemoryObject;
|
||||
class raw_ostream;
|
||||
|
||||
/// MCDecodedInst - Small container to hold an MCInst and associated info like
|
||||
/// address and size.
|
||||
struct MCDecodedInst {
|
||||
uint64_t Address;
|
||||
uint64_t Size;
|
||||
MCInst Inst;
|
||||
|
||||
MCDecodedInst() {}
|
||||
MCDecodedInst(uint64_t Address, uint64_t Size, MCInst Inst)
|
||||
: Address(Address), Size(Size), Inst(Inst) {}
|
||||
|
||||
bool operator<(const MCDecodedInst &RHS) const {
|
||||
return Address < RHS.Address;
|
||||
}
|
||||
};
|
||||
|
||||
/// MCBasicBlock - Consists of multiple MCDecodedInsts and a list of successing
|
||||
/// MCBasicBlocks.
|
||||
class MCBasicBlock {
|
||||
std::vector<MCDecodedInst> Insts;
|
||||
typedef DenseSet<uint64_t> SetTy;
|
||||
SetTy Succs;
|
||||
public:
|
||||
ArrayRef<MCDecodedInst> getInsts() const { return Insts; }
|
||||
|
||||
typedef SetTy::const_iterator succ_iterator;
|
||||
succ_iterator succ_begin() const { return Succs.begin(); }
|
||||
succ_iterator succ_end() const { return Succs.end(); }
|
||||
|
||||
bool contains(uint64_t Addr) const { return Succs.count(Addr); }
|
||||
|
||||
void addInst(const MCDecodedInst &Inst) { Insts.push_back(Inst); }
|
||||
void addSucc(uint64_t Addr) { Succs.insert(Addr); }
|
||||
|
||||
bool operator<(const MCBasicBlock &RHS) const {
|
||||
return Insts.size() < RHS.Insts.size();
|
||||
}
|
||||
};
|
||||
|
||||
/// MCFunction - Represents a named function in machine code, containing
|
||||
/// multiple MCBasicBlocks.
|
||||
class MCFunction {
|
||||
const StringRef Name;
|
||||
// Keep BBs sorted by address.
|
||||
typedef std::vector<std::pair<uint64_t, MCBasicBlock> > MapTy;
|
||||
MapTy Blocks;
|
||||
public:
|
||||
MCFunction(StringRef Name) : Name(Name) {}
|
||||
|
||||
// Create an MCFunction from a region of binary machine code.
|
||||
static MCFunction
|
||||
createFunctionFromMC(StringRef Name, const MCDisassembler *DisAsm,
|
||||
const MemoryObject &Region, uint64_t Start, uint64_t End,
|
||||
const MCInstrAnalysis *Ana, raw_ostream &DebugOut,
|
||||
SmallVectorImpl<uint64_t> &Calls);
|
||||
|
||||
typedef MapTy::const_iterator iterator;
|
||||
iterator begin() const { return Blocks.begin(); }
|
||||
iterator end() const { return Blocks.end(); }
|
||||
|
||||
StringRef getName() const { return Name; }
|
||||
|
||||
MCBasicBlock &addBlock(uint64_t Address, const MCBasicBlock &BB) {
|
||||
Blocks.push_back(std::make_pair(Address, BB));
|
||||
return Blocks.back().second;
|
||||
}
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
#endif
|
617
contrib/llvm/tools/llvm-objdump/MachODump.cpp
Normal file
617
contrib/llvm/tools/llvm-objdump/MachODump.cpp
Normal file
@ -0,0 +1,617 @@
|
||||
//===-- MachODump.cpp - Object file 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 file implements the MachO-specific dumper for llvm-objdump.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm-objdump.h"
|
||||
#include "MCFunction.h"
|
||||
#include "llvm/Support/MachO.h"
|
||||
#include "llvm/Object/MachOObject.h"
|
||||
#include "llvm/ADT/OwningPtr.h"
|
||||
#include "llvm/ADT/Triple.h"
|
||||
#include "llvm/ADT/STLExtras.h"
|
||||
#include "llvm/DebugInfo/DIContext.h"
|
||||
#include "llvm/MC/MCAsmInfo.h"
|
||||
#include "llvm/MC/MCDisassembler.h"
|
||||
#include "llvm/MC/MCInst.h"
|
||||
#include "llvm/MC/MCInstPrinter.h"
|
||||
#include "llvm/MC/MCInstrAnalysis.h"
|
||||
#include "llvm/MC/MCInstrDesc.h"
|
||||
#include "llvm/MC/MCInstrInfo.h"
|
||||
#include "llvm/MC/MCSubtargetInfo.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/Debug.h"
|
||||
#include "llvm/Support/Format.h"
|
||||
#include "llvm/Support/GraphWriter.h"
|
||||
#include "llvm/Support/MemoryBuffer.h"
|
||||
#include "llvm/Support/TargetRegistry.h"
|
||||
#include "llvm/Support/TargetSelect.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::opt<bool>
|
||||
CFG("cfg", cl::desc("Create a CFG for every symbol in the object file and"
|
||||
"write it to a graphviz file (MachO-only)"));
|
||||
|
||||
static cl::opt<bool>
|
||||
UseDbg("g", cl::desc("Print line information from debug info if available"));
|
||||
|
||||
static cl::opt<std::string>
|
||||
DSYMFile("dsym", cl::desc("Use .dSYM file for debug info"));
|
||||
|
||||
static const Target *GetTarget(const MachOObject *MachOObj) {
|
||||
// Figure out the target triple.
|
||||
llvm::Triple TT("unknown-unknown-unknown");
|
||||
switch (MachOObj->getHeader().CPUType) {
|
||||
case llvm::MachO::CPUTypeI386:
|
||||
TT.setArch(Triple::ArchType(Triple::x86));
|
||||
break;
|
||||
case llvm::MachO::CPUTypeX86_64:
|
||||
TT.setArch(Triple::ArchType(Triple::x86_64));
|
||||
break;
|
||||
case llvm::MachO::CPUTypeARM:
|
||||
TT.setArch(Triple::ArchType(Triple::arm));
|
||||
break;
|
||||
case llvm::MachO::CPUTypePowerPC:
|
||||
TT.setArch(Triple::ArchType(Triple::ppc));
|
||||
break;
|
||||
case llvm::MachO::CPUTypePowerPC64:
|
||||
TT.setArch(Triple::ArchType(Triple::ppc64));
|
||||
break;
|
||||
}
|
||||
|
||||
TripleName = TT.str();
|
||||
|
||||
// Get the target specific parser.
|
||||
std::string Error;
|
||||
const Target *TheTarget = TargetRegistry::lookupTarget(TripleName, Error);
|
||||
if (TheTarget)
|
||||
return TheTarget;
|
||||
|
||||
errs() << "llvm-objdump: error: unable to get target for '" << TripleName
|
||||
<< "', see --version and --triple.\n";
|
||||
return 0;
|
||||
}
|
||||
|
||||
struct Section {
|
||||
char Name[16];
|
||||
uint64_t Address;
|
||||
uint64_t Size;
|
||||
uint32_t Offset;
|
||||
uint32_t NumRelocs;
|
||||
uint64_t RelocTableOffset;
|
||||
};
|
||||
|
||||
struct Symbol {
|
||||
uint64_t Value;
|
||||
uint32_t StringIndex;
|
||||
uint8_t SectionIndex;
|
||||
bool operator<(const Symbol &RHS) const { return Value < RHS.Value; }
|
||||
};
|
||||
|
||||
template <typename T>
|
||||
static Section copySection(const T &Sect) {
|
||||
Section S;
|
||||
memcpy(S.Name, Sect->Name, 16);
|
||||
S.Address = Sect->Address;
|
||||
S.Size = Sect->Size;
|
||||
S.Offset = Sect->Offset;
|
||||
S.NumRelocs = Sect->NumRelocationTableEntries;
|
||||
S.RelocTableOffset = Sect->RelocationTableOffset;
|
||||
return S;
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
static Symbol copySymbol(const T &STE) {
|
||||
Symbol S;
|
||||
S.StringIndex = STE->StringIndex;
|
||||
S.SectionIndex = STE->SectionIndex;
|
||||
S.Value = STE->Value;
|
||||
return S;
|
||||
}
|
||||
|
||||
// Print additional information about an address, if available.
|
||||
static void DumpAddress(uint64_t Address, ArrayRef<Section> Sections,
|
||||
MachOObject *MachOObj, raw_ostream &OS) {
|
||||
for (unsigned i = 0; i != Sections.size(); ++i) {
|
||||
uint64_t addr = Address-Sections[i].Address;
|
||||
if (Sections[i].Address <= Address &&
|
||||
Sections[i].Address + Sections[i].Size > Address) {
|
||||
StringRef bytes = MachOObj->getData(Sections[i].Offset,
|
||||
Sections[i].Size);
|
||||
// Print constant strings.
|
||||
if (!strcmp(Sections[i].Name, "__cstring"))
|
||||
OS << '"' << bytes.substr(addr, bytes.find('\0', addr)) << '"';
|
||||
// Print constant CFStrings.
|
||||
if (!strcmp(Sections[i].Name, "__cfstring"))
|
||||
OS << "@\"" << bytes.substr(addr, bytes.find('\0', addr)) << '"';
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
typedef std::map<uint64_t, MCFunction*> FunctionMapTy;
|
||||
typedef SmallVector<MCFunction, 16> FunctionListTy;
|
||||
static void createMCFunctionAndSaveCalls(StringRef Name,
|
||||
const MCDisassembler *DisAsm,
|
||||
MemoryObject &Object, uint64_t Start,
|
||||
uint64_t End,
|
||||
MCInstrAnalysis *InstrAnalysis,
|
||||
uint64_t Address,
|
||||
raw_ostream &DebugOut,
|
||||
FunctionMapTy &FunctionMap,
|
||||
FunctionListTy &Functions) {
|
||||
SmallVector<uint64_t, 16> Calls;
|
||||
MCFunction f =
|
||||
MCFunction::createFunctionFromMC(Name, DisAsm, Object, Start, End,
|
||||
InstrAnalysis, DebugOut, Calls);
|
||||
Functions.push_back(f);
|
||||
FunctionMap[Address] = &Functions.back();
|
||||
|
||||
// Add the gathered callees to the map.
|
||||
for (unsigned i = 0, e = Calls.size(); i != e; ++i)
|
||||
FunctionMap.insert(std::make_pair(Calls[i], (MCFunction*)0));
|
||||
}
|
||||
|
||||
// Write a graphviz file for the CFG inside an MCFunction.
|
||||
static void emitDOTFile(const char *FileName, const MCFunction &f,
|
||||
MCInstPrinter *IP) {
|
||||
// Start a new dot file.
|
||||
std::string Error;
|
||||
raw_fd_ostream Out(FileName, Error);
|
||||
if (!Error.empty()) {
|
||||
errs() << "llvm-objdump: warning: " << Error << '\n';
|
||||
return;
|
||||
}
|
||||
|
||||
Out << "digraph " << f.getName() << " {\n";
|
||||
Out << "graph [ rankdir = \"LR\" ];\n";
|
||||
for (MCFunction::iterator i = f.begin(), e = f.end(); i != e; ++i) {
|
||||
bool hasPreds = false;
|
||||
// Only print blocks that have predecessors.
|
||||
// FIXME: Slow.
|
||||
for (MCFunction::iterator pi = f.begin(), pe = f.end(); pi != pe;
|
||||
++pi)
|
||||
if (pi->second.contains(i->first)) {
|
||||
hasPreds = true;
|
||||
break;
|
||||
}
|
||||
|
||||
if (!hasPreds && i != f.begin())
|
||||
continue;
|
||||
|
||||
Out << '"' << i->first << "\" [ label=\"<a>";
|
||||
// Print instructions.
|
||||
for (unsigned ii = 0, ie = i->second.getInsts().size(); ii != ie;
|
||||
++ii) {
|
||||
// Escape special chars and print the instruction in mnemonic form.
|
||||
std::string Str;
|
||||
raw_string_ostream OS(Str);
|
||||
IP->printInst(&i->second.getInsts()[ii].Inst, OS, "");
|
||||
Out << DOT::EscapeString(OS.str()) << '|';
|
||||
}
|
||||
Out << "<o>\" shape=\"record\" ];\n";
|
||||
|
||||
// Add edges.
|
||||
for (MCBasicBlock::succ_iterator si = i->second.succ_begin(),
|
||||
se = i->second.succ_end(); si != se; ++si)
|
||||
Out << i->first << ":o -> " << *si <<":a\n";
|
||||
}
|
||||
Out << "}\n";
|
||||
}
|
||||
|
||||
static void getSectionsAndSymbols(const macho::Header &Header,
|
||||
MachOObject *MachOObj,
|
||||
InMemoryStruct<macho::SymtabLoadCommand> *SymtabLC,
|
||||
std::vector<Section> &Sections,
|
||||
std::vector<Symbol> &Symbols,
|
||||
SmallVectorImpl<uint64_t> &FoundFns) {
|
||||
// Make a list of all symbols in the object file.
|
||||
for (unsigned i = 0; i != Header.NumLoadCommands; ++i) {
|
||||
const MachOObject::LoadCommandInfo &LCI = MachOObj->getLoadCommandInfo(i);
|
||||
if (LCI.Command.Type == macho::LCT_Segment) {
|
||||
InMemoryStruct<macho::SegmentLoadCommand> SegmentLC;
|
||||
MachOObj->ReadSegmentLoadCommand(LCI, SegmentLC);
|
||||
|
||||
// Store the sections in this segment.
|
||||
for (unsigned SectNum = 0; SectNum != SegmentLC->NumSections; ++SectNum) {
|
||||
InMemoryStruct<macho::Section> Sect;
|
||||
MachOObj->ReadSection(LCI, SectNum, Sect);
|
||||
Sections.push_back(copySection(Sect));
|
||||
|
||||
}
|
||||
} else if (LCI.Command.Type == macho::LCT_Segment64) {
|
||||
InMemoryStruct<macho::Segment64LoadCommand> Segment64LC;
|
||||
MachOObj->ReadSegment64LoadCommand(LCI, Segment64LC);
|
||||
|
||||
// Store the sections in this segment.
|
||||
for (unsigned SectNum = 0; SectNum != Segment64LC->NumSections;
|
||||
++SectNum) {
|
||||
InMemoryStruct<macho::Section64> Sect64;
|
||||
MachOObj->ReadSection64(LCI, SectNum, Sect64);
|
||||
Sections.push_back(copySection(Sect64));
|
||||
}
|
||||
} else if (LCI.Command.Type == macho::LCT_FunctionStarts) {
|
||||
// We found a function starts segment, parse the addresses for later
|
||||
// consumption.
|
||||
InMemoryStruct<macho::LinkeditDataLoadCommand> LLC;
|
||||
MachOObj->ReadLinkeditDataLoadCommand(LCI, LLC);
|
||||
|
||||
MachOObj->ReadULEB128s(LLC->DataOffset, FoundFns);
|
||||
}
|
||||
}
|
||||
// Store the symbols.
|
||||
if (SymtabLC) {
|
||||
for (unsigned i = 0; i != (*SymtabLC)->NumSymbolTableEntries; ++i) {
|
||||
if (MachOObj->is64Bit()) {
|
||||
InMemoryStruct<macho::Symbol64TableEntry> STE;
|
||||
MachOObj->ReadSymbol64TableEntry((*SymtabLC)->SymbolTableOffset, i,
|
||||
STE);
|
||||
Symbols.push_back(copySymbol(STE));
|
||||
} else {
|
||||
InMemoryStruct<macho::SymbolTableEntry> STE;
|
||||
MachOObj->ReadSymbolTableEntry((*SymtabLC)->SymbolTableOffset, i,
|
||||
STE);
|
||||
Symbols.push_back(copySymbol(STE));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void llvm::DisassembleInputMachO(StringRef Filename) {
|
||||
OwningPtr<MemoryBuffer> Buff;
|
||||
|
||||
if (error_code ec = MemoryBuffer::getFileOrSTDIN(Filename, Buff)) {
|
||||
errs() << "llvm-objdump: " << Filename << ": " << ec.message() << "\n";
|
||||
return;
|
||||
}
|
||||
|
||||
OwningPtr<MachOObject> MachOObj(MachOObject::LoadFromBuffer(Buff.take()));
|
||||
|
||||
const Target *TheTarget = GetTarget(MachOObj.get());
|
||||
if (!TheTarget) {
|
||||
// GetTarget prints out stuff.
|
||||
return;
|
||||
}
|
||||
OwningPtr<const MCInstrInfo> InstrInfo(TheTarget->createMCInstrInfo());
|
||||
OwningPtr<MCInstrAnalysis>
|
||||
InstrAnalysis(TheTarget->createMCInstrAnalysis(InstrInfo.get()));
|
||||
|
||||
// Set up disassembler.
|
||||
OwningPtr<const MCAsmInfo> AsmInfo(TheTarget->createMCAsmInfo(TripleName));
|
||||
OwningPtr<const MCSubtargetInfo>
|
||||
STI(TheTarget->createMCSubtargetInfo(TripleName, "", ""));
|
||||
OwningPtr<const MCDisassembler> DisAsm(TheTarget->createMCDisassembler(*STI));
|
||||
int AsmPrinterVariant = AsmInfo->getAssemblerDialect();
|
||||
OwningPtr<MCInstPrinter> IP(TheTarget->createMCInstPrinter(
|
||||
AsmPrinterVariant, *AsmInfo, *STI));
|
||||
|
||||
if (!InstrAnalysis || !AsmInfo || !STI || !DisAsm || !IP) {
|
||||
errs() << "error: couldn't initialize disassembler for target "
|
||||
<< TripleName << '\n';
|
||||
return;
|
||||
}
|
||||
|
||||
outs() << '\n' << Filename << ":\n\n";
|
||||
|
||||
const macho::Header &Header = MachOObj->getHeader();
|
||||
|
||||
const MachOObject::LoadCommandInfo *SymtabLCI = 0;
|
||||
// First, find the symbol table segment.
|
||||
for (unsigned i = 0; i != Header.NumLoadCommands; ++i) {
|
||||
const MachOObject::LoadCommandInfo &LCI = MachOObj->getLoadCommandInfo(i);
|
||||
if (LCI.Command.Type == macho::LCT_Symtab) {
|
||||
SymtabLCI = &LCI;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// Read and register the symbol table data.
|
||||
InMemoryStruct<macho::SymtabLoadCommand> SymtabLC;
|
||||
MachOObj->ReadSymtabLoadCommand(*SymtabLCI, SymtabLC);
|
||||
MachOObj->RegisterStringTable(*SymtabLC);
|
||||
|
||||
std::vector<Section> Sections;
|
||||
std::vector<Symbol> Symbols;
|
||||
SmallVector<uint64_t, 8> FoundFns;
|
||||
|
||||
getSectionsAndSymbols(Header, MachOObj.get(), &SymtabLC, Sections, Symbols,
|
||||
FoundFns);
|
||||
|
||||
// Make a copy of the unsorted symbol list. FIXME: duplication
|
||||
std::vector<Symbol> UnsortedSymbols(Symbols);
|
||||
// Sort the symbols by address, just in case they didn't come in that way.
|
||||
array_pod_sort(Symbols.begin(), Symbols.end());
|
||||
|
||||
#ifndef NDEBUG
|
||||
raw_ostream &DebugOut = DebugFlag ? dbgs() : nulls();
|
||||
#else
|
||||
raw_ostream &DebugOut = nulls();
|
||||
#endif
|
||||
|
||||
StringRef DebugAbbrevSection, DebugInfoSection, DebugArangesSection,
|
||||
DebugLineSection, DebugStrSection;
|
||||
OwningPtr<DIContext> diContext;
|
||||
OwningPtr<MachOObject> DSYMObj;
|
||||
MachOObject *DbgInfoObj = MachOObj.get();
|
||||
// Try to find debug info and set up the DIContext for it.
|
||||
if (UseDbg) {
|
||||
ArrayRef<Section> DebugSections = Sections;
|
||||
std::vector<Section> DSYMSections;
|
||||
|
||||
// A separate DSym file path was specified, parse it as a macho file,
|
||||
// get the sections and supply it to the section name parsing machinery.
|
||||
if (!DSYMFile.empty()) {
|
||||
OwningPtr<MemoryBuffer> Buf;
|
||||
if (error_code ec = MemoryBuffer::getFileOrSTDIN(DSYMFile.c_str(), Buf)) {
|
||||
errs() << "llvm-objdump: " << Filename << ": " << ec.message() << '\n';
|
||||
return;
|
||||
}
|
||||
DSYMObj.reset(MachOObject::LoadFromBuffer(Buf.take()));
|
||||
const macho::Header &Header = DSYMObj->getHeader();
|
||||
|
||||
std::vector<Symbol> Symbols;
|
||||
SmallVector<uint64_t, 8> FoundFns;
|
||||
getSectionsAndSymbols(Header, DSYMObj.get(), 0, DSYMSections, Symbols,
|
||||
FoundFns);
|
||||
DebugSections = DSYMSections;
|
||||
DbgInfoObj = DSYMObj.get();
|
||||
}
|
||||
|
||||
// Find the named debug info sections.
|
||||
for (unsigned SectIdx = 0; SectIdx != DebugSections.size(); SectIdx++) {
|
||||
if (!strcmp(DebugSections[SectIdx].Name, "__debug_abbrev"))
|
||||
DebugAbbrevSection = DbgInfoObj->getData(DebugSections[SectIdx].Offset,
|
||||
DebugSections[SectIdx].Size);
|
||||
else if (!strcmp(DebugSections[SectIdx].Name, "__debug_info"))
|
||||
DebugInfoSection = DbgInfoObj->getData(DebugSections[SectIdx].Offset,
|
||||
DebugSections[SectIdx].Size);
|
||||
else if (!strcmp(DebugSections[SectIdx].Name, "__debug_aranges"))
|
||||
DebugArangesSection = DbgInfoObj->getData(DebugSections[SectIdx].Offset,
|
||||
DebugSections[SectIdx].Size);
|
||||
else if (!strcmp(DebugSections[SectIdx].Name, "__debug_line"))
|
||||
DebugLineSection = DbgInfoObj->getData(DebugSections[SectIdx].Offset,
|
||||
DebugSections[SectIdx].Size);
|
||||
else if (!strcmp(DebugSections[SectIdx].Name, "__debug_str"))
|
||||
DebugStrSection = DbgInfoObj->getData(DebugSections[SectIdx].Offset,
|
||||
DebugSections[SectIdx].Size);
|
||||
}
|
||||
|
||||
// Setup the DIContext.
|
||||
diContext.reset(DIContext::getDWARFContext(DbgInfoObj->isLittleEndian(),
|
||||
DebugInfoSection,
|
||||
DebugAbbrevSection,
|
||||
DebugArangesSection,
|
||||
DebugLineSection,
|
||||
DebugStrSection));
|
||||
}
|
||||
|
||||
FunctionMapTy FunctionMap;
|
||||
FunctionListTy Functions;
|
||||
|
||||
for (unsigned SectIdx = 0; SectIdx != Sections.size(); SectIdx++) {
|
||||
if (strcmp(Sections[SectIdx].Name, "__text"))
|
||||
continue; // Skip non-text sections
|
||||
|
||||
// Insert the functions from the function starts segment into our map.
|
||||
uint64_t VMAddr = Sections[SectIdx].Address - Sections[SectIdx].Offset;
|
||||
for (unsigned i = 0, e = FoundFns.size(); i != e; ++i)
|
||||
FunctionMap.insert(std::make_pair(FoundFns[i]+VMAddr, (MCFunction*)0));
|
||||
|
||||
StringRef Bytes = MachOObj->getData(Sections[SectIdx].Offset,
|
||||
Sections[SectIdx].Size);
|
||||
StringRefMemoryObject memoryObject(Bytes);
|
||||
bool symbolTableWorked = false;
|
||||
|
||||
// Parse relocations.
|
||||
std::vector<std::pair<uint64_t, uint32_t> > Relocs;
|
||||
for (unsigned j = 0; j != Sections[SectIdx].NumRelocs; ++j) {
|
||||
InMemoryStruct<macho::RelocationEntry> RE;
|
||||
MachOObj->ReadRelocationEntry(Sections[SectIdx].RelocTableOffset, j, RE);
|
||||
Relocs.push_back(std::make_pair(RE->Word0, RE->Word1 & 0xffffff));
|
||||
}
|
||||
array_pod_sort(Relocs.begin(), Relocs.end());
|
||||
|
||||
// Disassemble symbol by symbol.
|
||||
for (unsigned SymIdx = 0; SymIdx != Symbols.size(); SymIdx++) {
|
||||
// Make sure the symbol is defined in this section.
|
||||
if ((unsigned)Symbols[SymIdx].SectionIndex - 1 != SectIdx)
|
||||
continue;
|
||||
|
||||
// Start at the address of the symbol relative to the section's address.
|
||||
uint64_t Start = Symbols[SymIdx].Value - Sections[SectIdx].Address;
|
||||
// Stop disassembling either at the beginning of the next symbol or at
|
||||
// the end of the section.
|
||||
uint64_t End = (SymIdx+1 == Symbols.size() ||
|
||||
Symbols[SymIdx].SectionIndex != Symbols[SymIdx+1].SectionIndex) ?
|
||||
Sections[SectIdx].Size :
|
||||
Symbols[SymIdx+1].Value - Sections[SectIdx].Address;
|
||||
uint64_t Size;
|
||||
|
||||
if (Start >= End)
|
||||
continue;
|
||||
|
||||
symbolTableWorked = true;
|
||||
|
||||
if (!CFG) {
|
||||
// Normal disassembly, print addresses, bytes and mnemonic form.
|
||||
outs() << MachOObj->getStringAtIndex(Symbols[SymIdx].StringIndex)
|
||||
<< ":\n";
|
||||
DILineInfo lastLine;
|
||||
for (uint64_t Index = Start; Index < End; Index += Size) {
|
||||
MCInst Inst;
|
||||
|
||||
if (DisAsm->getInstruction(Inst, Size, memoryObject, Index,
|
||||
DebugOut, nulls())) {
|
||||
outs() << format("%8llx:\t", Sections[SectIdx].Address + Index);
|
||||
DumpBytes(StringRef(Bytes.data() + Index, Size));
|
||||
IP->printInst(&Inst, outs(), "");
|
||||
|
||||
// Print debug info.
|
||||
if (diContext) {
|
||||
DILineInfo dli =
|
||||
diContext->getLineInfoForAddress(Sections[SectIdx].Address +
|
||||
Index);
|
||||
// Print valid line info if it changed.
|
||||
if (dli != lastLine && dli.getLine() != 0)
|
||||
outs() << "\t## " << dli.getFileName() << ':'
|
||||
<< dli.getLine() << ':' << dli.getColumn();
|
||||
lastLine = dli;
|
||||
}
|
||||
outs() << "\n";
|
||||
} else {
|
||||
errs() << "llvm-objdump: warning: invalid instruction encoding\n";
|
||||
if (Size == 0)
|
||||
Size = 1; // skip illegible bytes
|
||||
}
|
||||
}
|
||||
} else {
|
||||
// Create CFG and use it for disassembly.
|
||||
createMCFunctionAndSaveCalls(
|
||||
MachOObj->getStringAtIndex(Symbols[SymIdx].StringIndex),
|
||||
DisAsm.get(), memoryObject, Start, End, InstrAnalysis.get(),
|
||||
Start, DebugOut, FunctionMap, Functions);
|
||||
}
|
||||
}
|
||||
|
||||
if (CFG) {
|
||||
if (!symbolTableWorked) {
|
||||
// Reading the symbol table didn't work, create a big __TEXT symbol.
|
||||
createMCFunctionAndSaveCalls("__TEXT", DisAsm.get(), memoryObject,
|
||||
0, Sections[SectIdx].Size,
|
||||
InstrAnalysis.get(),
|
||||
Sections[SectIdx].Offset, DebugOut,
|
||||
FunctionMap, Functions);
|
||||
}
|
||||
for (std::map<uint64_t, MCFunction*>::iterator mi = FunctionMap.begin(),
|
||||
me = FunctionMap.end(); mi != me; ++mi)
|
||||
if (mi->second == 0) {
|
||||
// Create functions for the remaining callees we have gathered,
|
||||
// but we didn't find a name for them.
|
||||
SmallVector<uint64_t, 16> Calls;
|
||||
MCFunction f =
|
||||
MCFunction::createFunctionFromMC("unknown", DisAsm.get(),
|
||||
memoryObject, mi->first,
|
||||
Sections[SectIdx].Size,
|
||||
InstrAnalysis.get(), DebugOut,
|
||||
Calls);
|
||||
Functions.push_back(f);
|
||||
mi->second = &Functions.back();
|
||||
for (unsigned i = 0, e = Calls.size(); i != e; ++i) {
|
||||
std::pair<uint64_t, MCFunction*> p(Calls[i], (MCFunction*)0);
|
||||
if (FunctionMap.insert(p).second)
|
||||
mi = FunctionMap.begin();
|
||||
}
|
||||
}
|
||||
|
||||
DenseSet<uint64_t> PrintedBlocks;
|
||||
for (unsigned ffi = 0, ffe = Functions.size(); ffi != ffe; ++ffi) {
|
||||
MCFunction &f = Functions[ffi];
|
||||
for (MCFunction::iterator fi = f.begin(), fe = f.end(); fi != fe; ++fi){
|
||||
if (!PrintedBlocks.insert(fi->first).second)
|
||||
continue; // We already printed this block.
|
||||
|
||||
// We assume a block has predecessors when it's the first block after
|
||||
// a symbol.
|
||||
bool hasPreds = FunctionMap.find(fi->first) != FunctionMap.end();
|
||||
|
||||
// See if this block has predecessors.
|
||||
// FIXME: Slow.
|
||||
for (MCFunction::iterator pi = f.begin(), pe = f.end(); pi != pe;
|
||||
++pi)
|
||||
if (pi->second.contains(fi->first)) {
|
||||
hasPreds = true;
|
||||
break;
|
||||
}
|
||||
|
||||
// No predecessors, this is a data block. Print as .byte directives.
|
||||
if (!hasPreds) {
|
||||
uint64_t End = llvm::next(fi) == fe ? Sections[SectIdx].Size :
|
||||
llvm::next(fi)->first;
|
||||
outs() << "# " << End-fi->first << " bytes of data:\n";
|
||||
for (unsigned pos = fi->first; pos != End; ++pos) {
|
||||
outs() << format("%8x:\t", Sections[SectIdx].Address + pos);
|
||||
DumpBytes(StringRef(Bytes.data() + pos, 1));
|
||||
outs() << format("\t.byte 0x%02x\n", (uint8_t)Bytes[pos]);
|
||||
}
|
||||
continue;
|
||||
}
|
||||
|
||||
if (fi->second.contains(fi->first)) // Print a header for simple loops
|
||||
outs() << "# Loop begin:\n";
|
||||
|
||||
DILineInfo lastLine;
|
||||
// Walk over the instructions and print them.
|
||||
for (unsigned ii = 0, ie = fi->second.getInsts().size(); ii != ie;
|
||||
++ii) {
|
||||
const MCDecodedInst &Inst = fi->second.getInsts()[ii];
|
||||
|
||||
// If there's a symbol at this address, print its name.
|
||||
if (FunctionMap.find(Sections[SectIdx].Address + Inst.Address) !=
|
||||
FunctionMap.end())
|
||||
outs() << FunctionMap[Sections[SectIdx].Address + Inst.Address]->
|
||||
getName() << ":\n";
|
||||
|
||||
outs() << format("%8llx:\t", Sections[SectIdx].Address +
|
||||
Inst.Address);
|
||||
DumpBytes(StringRef(Bytes.data() + Inst.Address, Inst.Size));
|
||||
|
||||
if (fi->second.contains(fi->first)) // Indent simple loops.
|
||||
outs() << '\t';
|
||||
|
||||
IP->printInst(&Inst.Inst, outs(), "");
|
||||
|
||||
// Look for relocations inside this instructions, if there is one
|
||||
// print its target and additional information if available.
|
||||
for (unsigned j = 0; j != Relocs.size(); ++j)
|
||||
if (Relocs[j].first >= Sections[SectIdx].Address + Inst.Address &&
|
||||
Relocs[j].first < Sections[SectIdx].Address + Inst.Address +
|
||||
Inst.Size) {
|
||||
outs() << "\t# "
|
||||
<< MachOObj->getStringAtIndex(
|
||||
UnsortedSymbols[Relocs[j].second].StringIndex)
|
||||
<< ' ';
|
||||
DumpAddress(UnsortedSymbols[Relocs[j].second].Value, Sections,
|
||||
MachOObj.get(), outs());
|
||||
}
|
||||
|
||||
// If this instructions contains an address, see if we can evaluate
|
||||
// it and print additional information.
|
||||
uint64_t targ = InstrAnalysis->evaluateBranch(Inst.Inst,
|
||||
Inst.Address,
|
||||
Inst.Size);
|
||||
if (targ != -1ULL)
|
||||
DumpAddress(targ, Sections, MachOObj.get(), outs());
|
||||
|
||||
// Print debug info.
|
||||
if (diContext) {
|
||||
DILineInfo dli =
|
||||
diContext->getLineInfoForAddress(Sections[SectIdx].Address +
|
||||
Inst.Address);
|
||||
// Print valid line info if it changed.
|
||||
if (dli != lastLine && dli.getLine() != 0)
|
||||
outs() << "\t## " << dli.getFileName() << ':'
|
||||
<< dli.getLine() << ':' << dli.getColumn();
|
||||
lastLine = dli;
|
||||
}
|
||||
|
||||
outs() << '\n';
|
||||
}
|
||||
}
|
||||
|
||||
emitDOTFile((f.getName().str() + ".dot").c_str(), f, IP.get());
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
18
contrib/llvm/tools/llvm-objdump/Makefile
Normal file
18
contrib/llvm/tools/llvm-objdump/Makefile
Normal file
@ -0,0 +1,18 @@
|
||||
##===- tools/llvm-objdump/Makefile -------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
LEVEL = ../..
|
||||
|
||||
TOOLNAME = llvm-objdump
|
||||
LINK_COMPONENTS = $(TARGETS_TO_BUILD) DebugInfo MC MCParser MCDisassembler \
|
||||
Object
|
||||
|
||||
# This tool has no plugins, optimize startup time.
|
||||
TOOL_NO_EXPORTS = 1
|
||||
|
||||
include $(LEVEL)/Makefile.common
|
459
contrib/llvm/tools/llvm-objdump/llvm-objdump.cpp
Normal file
459
contrib/llvm/tools/llvm-objdump/llvm-objdump.cpp
Normal file
@ -0,0 +1,459 @@
|
||||
//===-- llvm-objdump.cpp - Object file 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 binutils "objdump", that is, it
|
||||
// dumps out a plethora of information about an object file depending on the
|
||||
// flags.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm-objdump.h"
|
||||
#include "MCFunction.h"
|
||||
#include "llvm/Object/Archive.h"
|
||||
#include "llvm/Object/ObjectFile.h"
|
||||
#include "llvm/ADT/OwningPtr.h"
|
||||
#include "llvm/ADT/Triple.h"
|
||||
#include "llvm/ADT/STLExtras.h"
|
||||
#include "llvm/MC/MCAsmInfo.h"
|
||||
#include "llvm/MC/MCDisassembler.h"
|
||||
#include "llvm/MC/MCInst.h"
|
||||
#include "llvm/MC/MCInstPrinter.h"
|
||||
#include "llvm/MC/MCSubtargetInfo.h"
|
||||
#include "llvm/Support/Casting.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/Debug.h"
|
||||
#include "llvm/Support/FileSystem.h"
|
||||
#include "llvm/Support/Format.h"
|
||||
#include "llvm/Support/GraphWriter.h"
|
||||
#include "llvm/Support/Host.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/SourceMgr.h"
|
||||
#include "llvm/Support/TargetRegistry.h"
|
||||
#include "llvm/Support/TargetSelect.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<bool>
|
||||
Disassemble("disassemble",
|
||||
cl::desc("Display assembler mnemonics for the machine instructions"));
|
||||
static cl::alias
|
||||
Disassembled("d", cl::desc("Alias for --disassemble"),
|
||||
cl::aliasopt(Disassemble));
|
||||
|
||||
static cl::opt<bool>
|
||||
Relocations("r", cl::desc("Display the relocation entries in the file"));
|
||||
|
||||
static cl::opt<bool>
|
||||
MachO("macho", cl::desc("Use MachO specific object file parser"));
|
||||
static cl::alias
|
||||
MachOm("m", cl::desc("Alias for --macho"), cl::aliasopt(MachO));
|
||||
|
||||
cl::opt<std::string>
|
||||
llvm::TripleName("triple", cl::desc("Target triple to disassemble for, "
|
||||
"see -version for available targets"));
|
||||
|
||||
cl::opt<std::string>
|
||||
llvm::ArchName("arch", cl::desc("Target arch to disassemble for, "
|
||||
"see -version for available targets"));
|
||||
|
||||
static cl::opt<bool>
|
||||
SectionHeaders("section-headers", cl::desc("Display summaries of the headers "
|
||||
"for each section."));
|
||||
static cl::alias
|
||||
SectionHeadersShort("headers", cl::desc("Alias for --section-headers"),
|
||||
cl::aliasopt(SectionHeaders));
|
||||
static cl::alias
|
||||
SectionHeadersShorter("h", cl::desc("Alias for --section-headers"),
|
||||
cl::aliasopt(SectionHeaders));
|
||||
|
||||
static StringRef ToolName;
|
||||
|
||||
static bool error(error_code ec) {
|
||||
if (!ec) return false;
|
||||
|
||||
outs() << ToolName << ": error reading file: " << ec.message() << ".\n";
|
||||
outs().flush();
|
||||
return true;
|
||||
}
|
||||
|
||||
static const Target *GetTarget(const ObjectFile *Obj = NULL) {
|
||||
// Figure out the target triple.
|
||||
llvm::Triple TT("unknown-unknown-unknown");
|
||||
if (TripleName.empty()) {
|
||||
if (Obj)
|
||||
TT.setArch(Triple::ArchType(Obj->getArch()));
|
||||
} else
|
||||
TT.setTriple(Triple::normalize(TripleName));
|
||||
|
||||
if (!ArchName.empty())
|
||||
TT.setArchName(ArchName);
|
||||
|
||||
TripleName = TT.str();
|
||||
|
||||
// Get the target specific parser.
|
||||
std::string Error;
|
||||
const Target *TheTarget = TargetRegistry::lookupTarget(TripleName, Error);
|
||||
if (TheTarget)
|
||||
return TheTarget;
|
||||
|
||||
errs() << ToolName << ": error: unable to get target for '" << TripleName
|
||||
<< "', see --version and --triple.\n";
|
||||
return 0;
|
||||
}
|
||||
|
||||
void llvm::DumpBytes(StringRef bytes) {
|
||||
static const char hex_rep[] = "0123456789abcdef";
|
||||
// FIXME: The real way to do this is to figure out the longest instruction
|
||||
// and align to that size before printing. I'll fix this when I get
|
||||
// around to outputting relocations.
|
||||
// 15 is the longest x86 instruction
|
||||
// 3 is for the hex rep of a byte + a space.
|
||||
// 1 is for the null terminator.
|
||||
enum { OutputSize = (15 * 3) + 1 };
|
||||
char output[OutputSize];
|
||||
|
||||
assert(bytes.size() <= 15
|
||||
&& "DumpBytes only supports instructions of up to 15 bytes");
|
||||
memset(output, ' ', sizeof(output));
|
||||
unsigned index = 0;
|
||||
for (StringRef::iterator i = bytes.begin(),
|
||||
e = bytes.end(); i != e; ++i) {
|
||||
output[index] = hex_rep[(*i & 0xF0) >> 4];
|
||||
output[index + 1] = hex_rep[*i & 0xF];
|
||||
index += 3;
|
||||
}
|
||||
|
||||
output[sizeof(output) - 1] = 0;
|
||||
outs() << output;
|
||||
}
|
||||
|
||||
static bool RelocAddressLess(RelocationRef a, RelocationRef b) {
|
||||
uint64_t a_addr, b_addr;
|
||||
if (error(a.getAddress(a_addr))) return false;
|
||||
if (error(b.getAddress(b_addr))) return false;
|
||||
return a_addr < b_addr;
|
||||
}
|
||||
|
||||
static void DisassembleObject(const ObjectFile *Obj, bool InlineRelocs) {
|
||||
const Target *TheTarget = GetTarget(Obj);
|
||||
if (!TheTarget) {
|
||||
// GetTarget prints out stuff.
|
||||
return;
|
||||
}
|
||||
|
||||
outs() << '\n';
|
||||
outs() << Obj->getFileName()
|
||||
<< ":\tfile format " << Obj->getFileFormatName() << "\n\n";
|
||||
|
||||
error_code ec;
|
||||
for (section_iterator i = Obj->begin_sections(),
|
||||
e = Obj->end_sections();
|
||||
i != e; i.increment(ec)) {
|
||||
if (error(ec)) break;
|
||||
bool text;
|
||||
if (error(i->isText(text))) break;
|
||||
if (!text) continue;
|
||||
|
||||
uint64_t SectionAddr;
|
||||
if (error(i->getAddress(SectionAddr))) break;
|
||||
|
||||
// Make a list of all the symbols in this section.
|
||||
std::vector<std::pair<uint64_t, StringRef> > Symbols;
|
||||
for (symbol_iterator si = Obj->begin_symbols(),
|
||||
se = Obj->end_symbols();
|
||||
si != se; si.increment(ec)) {
|
||||
bool contains;
|
||||
if (!error(i->containsSymbol(*si, contains)) && contains) {
|
||||
uint64_t Address;
|
||||
if (error(si->getOffset(Address))) break;
|
||||
StringRef Name;
|
||||
if (error(si->getName(Name))) break;
|
||||
Symbols.push_back(std::make_pair(Address, Name));
|
||||
}
|
||||
}
|
||||
|
||||
// Sort the symbols by address, just in case they didn't come in that way.
|
||||
array_pod_sort(Symbols.begin(), Symbols.end());
|
||||
|
||||
// Make a list of all the relocations for this section.
|
||||
std::vector<RelocationRef> Rels;
|
||||
if (InlineRelocs) {
|
||||
for (relocation_iterator ri = i->begin_relocations(),
|
||||
re = i->end_relocations();
|
||||
ri != re; ri.increment(ec)) {
|
||||
if (error(ec)) break;
|
||||
Rels.push_back(*ri);
|
||||
}
|
||||
}
|
||||
|
||||
// Sort relocations by address.
|
||||
std::sort(Rels.begin(), Rels.end(), RelocAddressLess);
|
||||
|
||||
StringRef name;
|
||||
if (error(i->getName(name))) break;
|
||||
outs() << "Disassembly of section " << name << ':';
|
||||
|
||||
// If the section has no symbols just insert a dummy one and disassemble
|
||||
// the whole section.
|
||||
if (Symbols.empty())
|
||||
Symbols.push_back(std::make_pair(0, name));
|
||||
|
||||
// Set up disassembler.
|
||||
OwningPtr<const MCAsmInfo> AsmInfo(TheTarget->createMCAsmInfo(TripleName));
|
||||
|
||||
if (!AsmInfo) {
|
||||
errs() << "error: no assembly info for target " << TripleName << "\n";
|
||||
return;
|
||||
}
|
||||
|
||||
OwningPtr<const MCSubtargetInfo> STI(
|
||||
TheTarget->createMCSubtargetInfo(TripleName, "", ""));
|
||||
|
||||
if (!STI) {
|
||||
errs() << "error: no subtarget info for target " << TripleName << "\n";
|
||||
return;
|
||||
}
|
||||
|
||||
OwningPtr<const MCDisassembler> DisAsm(
|
||||
TheTarget->createMCDisassembler(*STI));
|
||||
if (!DisAsm) {
|
||||
errs() << "error: no disassembler for target " << TripleName << "\n";
|
||||
return;
|
||||
}
|
||||
|
||||
int AsmPrinterVariant = AsmInfo->getAssemblerDialect();
|
||||
OwningPtr<MCInstPrinter> IP(TheTarget->createMCInstPrinter(
|
||||
AsmPrinterVariant, *AsmInfo, *STI));
|
||||
if (!IP) {
|
||||
errs() << "error: no instruction printer for target " << TripleName
|
||||
<< '\n';
|
||||
return;
|
||||
}
|
||||
|
||||
StringRef Bytes;
|
||||
if (error(i->getContents(Bytes))) break;
|
||||
StringRefMemoryObject memoryObject(Bytes);
|
||||
uint64_t Size;
|
||||
uint64_t Index;
|
||||
uint64_t SectSize;
|
||||
if (error(i->getSize(SectSize))) break;
|
||||
|
||||
std::vector<RelocationRef>::const_iterator rel_cur = Rels.begin();
|
||||
std::vector<RelocationRef>::const_iterator rel_end = Rels.end();
|
||||
// Disassemble symbol by symbol.
|
||||
for (unsigned si = 0, se = Symbols.size(); si != se; ++si) {
|
||||
uint64_t Start = Symbols[si].first;
|
||||
uint64_t End;
|
||||
// The end is either the size of the section or the beginning of the next
|
||||
// symbol.
|
||||
if (si == se - 1)
|
||||
End = SectSize;
|
||||
// Make sure this symbol takes up space.
|
||||
else if (Symbols[si + 1].first != Start)
|
||||
End = Symbols[si + 1].first - 1;
|
||||
else
|
||||
// This symbol has the same address as the next symbol. Skip it.
|
||||
continue;
|
||||
|
||||
outs() << '\n' << Symbols[si].second << ":\n";
|
||||
|
||||
#ifndef NDEBUG
|
||||
raw_ostream &DebugOut = DebugFlag ? dbgs() : nulls();
|
||||
#else
|
||||
raw_ostream &DebugOut = nulls();
|
||||
#endif
|
||||
|
||||
for (Index = Start; Index < End; Index += Size) {
|
||||
MCInst Inst;
|
||||
|
||||
if (DisAsm->getInstruction(Inst, Size, memoryObject, Index,
|
||||
DebugOut, nulls())) {
|
||||
outs() << format("%8"PRIx64":\t", SectionAddr + Index);
|
||||
DumpBytes(StringRef(Bytes.data() + Index, Size));
|
||||
IP->printInst(&Inst, outs(), "");
|
||||
outs() << "\n";
|
||||
} else {
|
||||
errs() << ToolName << ": warning: invalid instruction encoding\n";
|
||||
if (Size == 0)
|
||||
Size = 1; // skip illegible bytes
|
||||
}
|
||||
|
||||
// Print relocation for instruction.
|
||||
while (rel_cur != rel_end) {
|
||||
uint64_t addr;
|
||||
SmallString<16> name;
|
||||
SmallString<32> val;
|
||||
if (error(rel_cur->getAddress(addr))) goto skip_print_rel;
|
||||
// Stop when rel_cur's address is past the current instruction.
|
||||
if (addr > Index + Size) break;
|
||||
if (error(rel_cur->getTypeName(name))) goto skip_print_rel;
|
||||
if (error(rel_cur->getValueString(val))) goto skip_print_rel;
|
||||
|
||||
outs() << format("\t\t\t%8"PRIx64": ", SectionAddr + addr) << name << "\t"
|
||||
<< val << "\n";
|
||||
|
||||
skip_print_rel:
|
||||
++rel_cur;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void PrintRelocations(const ObjectFile *o) {
|
||||
error_code ec;
|
||||
for (section_iterator si = o->begin_sections(), se = o->end_sections();
|
||||
si != se; si.increment(ec)){
|
||||
if (error(ec)) return;
|
||||
if (si->begin_relocations() == si->end_relocations())
|
||||
continue;
|
||||
StringRef secname;
|
||||
if (error(si->getName(secname))) continue;
|
||||
outs() << "RELOCATION RECORDS FOR [" << secname << "]:\n";
|
||||
for (relocation_iterator ri = si->begin_relocations(),
|
||||
re = si->end_relocations();
|
||||
ri != re; ri.increment(ec)) {
|
||||
if (error(ec)) return;
|
||||
|
||||
uint64_t address;
|
||||
SmallString<32> relocname;
|
||||
SmallString<32> valuestr;
|
||||
if (error(ri->getTypeName(relocname))) continue;
|
||||
if (error(ri->getAddress(address))) continue;
|
||||
if (error(ri->getValueString(valuestr))) continue;
|
||||
outs() << address << " " << relocname << " " << valuestr << "\n";
|
||||
}
|
||||
outs() << "\n";
|
||||
}
|
||||
}
|
||||
|
||||
static void PrintSectionHeaders(const ObjectFile *o) {
|
||||
outs() << "Sections:\n"
|
||||
"Idx Name Size Address Type\n";
|
||||
error_code ec;
|
||||
unsigned i = 0;
|
||||
for (section_iterator si = o->begin_sections(), se = o->end_sections();
|
||||
si != se; si.increment(ec)) {
|
||||
if (error(ec)) return;
|
||||
StringRef Name;
|
||||
if (error(si->getName(Name))) return;
|
||||
uint64_t Address;
|
||||
if (error(si->getAddress(Address))) return;
|
||||
uint64_t Size;
|
||||
if (error(si->getSize(Size))) return;
|
||||
bool Text, Data, BSS;
|
||||
if (error(si->isText(Text))) return;
|
||||
if (error(si->isData(Data))) return;
|
||||
if (error(si->isBSS(BSS))) return;
|
||||
std::string Type = (std::string(Text ? "TEXT " : "") +
|
||||
(Data ? "DATA " : "") + (BSS ? "BSS" : ""));
|
||||
outs() << format("%3d %-13s %09"PRIx64" %017"PRIx64" %s\n", i, Name.str().c_str(), Size,
|
||||
Address, Type.c_str());
|
||||
++i;
|
||||
}
|
||||
}
|
||||
|
||||
static void DumpObject(const ObjectFile *o) {
|
||||
if (Disassemble)
|
||||
DisassembleObject(o, Relocations);
|
||||
if (Relocations && !Disassemble)
|
||||
PrintRelocations(o);
|
||||
if (SectionHeaders)
|
||||
PrintSectionHeaders(o);
|
||||
}
|
||||
|
||||
/// @brief Dump each object file in \a a;
|
||||
static void DumpArchive(const Archive *a) {
|
||||
for (Archive::child_iterator i = a->begin_children(),
|
||||
e = a->end_children(); i != e; ++i) {
|
||||
OwningPtr<Binary> child;
|
||||
if (error_code ec = i->getAsBinary(child)) {
|
||||
errs() << ToolName << ": '" << a->getFileName() << "': " << ec.message()
|
||||
<< ".\n";
|
||||
continue;
|
||||
}
|
||||
if (ObjectFile *o = dyn_cast<ObjectFile>(child.get()))
|
||||
DumpObject(o);
|
||||
else
|
||||
errs() << ToolName << ": '" << a->getFileName() << "': "
|
||||
<< "Unrecognized file type.\n";
|
||||
}
|
||||
}
|
||||
|
||||
/// @brief Open file and figure out how to dump it.
|
||||
static void DumpInput(StringRef file) {
|
||||
// If file isn't stdin, check that it exists.
|
||||
if (file != "-" && !sys::fs::exists(file)) {
|
||||
errs() << ToolName << ": '" << file << "': " << "No such file\n";
|
||||
return;
|
||||
}
|
||||
|
||||
if (MachO && Disassemble) {
|
||||
DisassembleInputMachO(file);
|
||||
return;
|
||||
}
|
||||
|
||||
// Attempt to open the binary.
|
||||
OwningPtr<Binary> binary;
|
||||
if (error_code ec = createBinary(file, binary)) {
|
||||
errs() << ToolName << ": '" << file << "': " << ec.message() << ".\n";
|
||||
return;
|
||||
}
|
||||
|
||||
if (Archive *a = dyn_cast<Archive>(binary.get())) {
|
||||
DumpArchive(a);
|
||||
} else if (ObjectFile *o = dyn_cast<ObjectFile>(binary.get())) {
|
||||
DumpObject(o);
|
||||
} else {
|
||||
errs() << ToolName << ": '" << file << "': " << "Unrecognized file type.\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.
|
||||
|
||||
// Initialize targets and assembly printers/parsers.
|
||||
llvm::InitializeAllTargetInfos();
|
||||
llvm::InitializeAllTargetMCs();
|
||||
llvm::InitializeAllAsmParsers();
|
||||
llvm::InitializeAllDisassemblers();
|
||||
|
||||
cl::ParseCommandLineOptions(argc, argv, "llvm object file dumper\n");
|
||||
TripleName = Triple::normalize(TripleName);
|
||||
|
||||
ToolName = argv[0];
|
||||
|
||||
// Defaults to a.out if no filenames specified.
|
||||
if (InputFilenames.size() == 0)
|
||||
InputFilenames.push_back("a.out");
|
||||
|
||||
if (!Disassemble && !Relocations && !SectionHeaders) {
|
||||
cl::PrintHelpMessage();
|
||||
return 2;
|
||||
}
|
||||
|
||||
std::for_each(InputFilenames.begin(), InputFilenames.end(),
|
||||
DumpInput);
|
||||
|
||||
return 0;
|
||||
}
|
46
contrib/llvm/tools/llvm-objdump/llvm-objdump.h
Normal file
46
contrib/llvm/tools/llvm-objdump/llvm-objdump.h
Normal file
@ -0,0 +1,46 @@
|
||||
//===-- llvm-objdump.h ----------------------------------------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef LLVM_OBJDUMP_H
|
||||
#define LLVM_OBJDUMP_H
|
||||
|
||||
#include "llvm/ADT/StringRef.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/DataTypes.h"
|
||||
#include "llvm/Support/MemoryObject.h"
|
||||
|
||||
namespace llvm {
|
||||
|
||||
extern cl::opt<std::string> TripleName;
|
||||
extern cl::opt<std::string> ArchName;
|
||||
|
||||
// Various helper functions.
|
||||
void DumpBytes(StringRef bytes);
|
||||
void DisassembleInputMachO(StringRef Filename);
|
||||
|
||||
class StringRefMemoryObject : public MemoryObject {
|
||||
private:
|
||||
StringRef Bytes;
|
||||
public:
|
||||
StringRefMemoryObject(StringRef bytes) : Bytes(bytes) {}
|
||||
|
||||
uint64_t getBase() const { return 0; }
|
||||
uint64_t getExtent() const { return Bytes.size(); }
|
||||
|
||||
int readByte(uint64_t Addr, uint8_t *Byte) const {
|
||||
if (Addr >= getExtent())
|
||||
return -1;
|
||||
*Byte = Bytes[Addr];
|
||||
return 0;
|
||||
}
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
#endif
|
5
contrib/llvm/tools/llvm-prof/CMakeLists.txt
Normal file
5
contrib/llvm/tools/llvm-prof/CMakeLists.txt
Normal file
@ -0,0 +1,5 @@
|
||||
set(LLVM_LINK_COMPONENTS bitreader analysis)
|
||||
|
||||
add_llvm_tool(llvm-prof
|
||||
llvm-prof.cpp
|
||||
)
|
17
contrib/llvm/tools/llvm-prof/Makefile
Normal file
17
contrib/llvm/tools/llvm-prof/Makefile
Normal file
@ -0,0 +1,17 @@
|
||||
##===- tools/llvm-prof/Makefile ----------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
LEVEL = ../..
|
||||
|
||||
TOOLNAME = llvm-prof
|
||||
LINK_COMPONENTS = bitreader analysis
|
||||
|
||||
# This tool has no plugins, optimize startup time.
|
||||
TOOL_NO_EXPORTS = 1
|
||||
|
||||
include $(LEVEL)/Makefile.common
|
293
contrib/llvm/tools/llvm-prof/llvm-prof.cpp
Normal file
293
contrib/llvm/tools/llvm-prof/llvm-prof.cpp
Normal file
@ -0,0 +1,293 @@
|
||||
//===- llvm-prof.cpp - Read in and process llvmprof.out data files --------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This tools is meant for use with the various LLVM profiling instrumentation
|
||||
// passes. It reads in the data file produced by executing an instrumented
|
||||
// program, and outputs a nice report.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/InstrTypes.h"
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/PassManager.h"
|
||||
#include "llvm/Assembly/AssemblyAnnotationWriter.h"
|
||||
#include "llvm/Analysis/ProfileInfo.h"
|
||||
#include "llvm/Analysis/ProfileInfoLoader.h"
|
||||
#include "llvm/Analysis/Passes.h"
|
||||
#include "llvm/Bitcode/ReaderWriter.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/FormattedStream.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/MemoryBuffer.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/Format.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include "llvm/Support/system_error.h"
|
||||
#include <algorithm>
|
||||
#include <iomanip>
|
||||
#include <map>
|
||||
#include <set>
|
||||
|
||||
using namespace llvm;
|
||||
|
||||
namespace {
|
||||
cl::opt<std::string>
|
||||
BitcodeFile(cl::Positional, cl::desc("<program bitcode file>"),
|
||||
cl::Required);
|
||||
|
||||
cl::opt<std::string>
|
||||
ProfileDataFile(cl::Positional, cl::desc("<llvmprof.out file>"),
|
||||
cl::Optional, cl::init("llvmprof.out"));
|
||||
|
||||
cl::opt<bool>
|
||||
PrintAnnotatedLLVM("annotated-llvm",
|
||||
cl::desc("Print LLVM code with frequency annotations"));
|
||||
cl::alias PrintAnnotated2("A", cl::desc("Alias for --annotated-llvm"),
|
||||
cl::aliasopt(PrintAnnotatedLLVM));
|
||||
cl::opt<bool>
|
||||
PrintAllCode("print-all-code",
|
||||
cl::desc("Print annotated code for the entire program"));
|
||||
}
|
||||
|
||||
// PairSecondSort - A sorting predicate to sort by the second element of a pair.
|
||||
template<class T>
|
||||
struct PairSecondSortReverse
|
||||
: public std::binary_function<std::pair<T, double>,
|
||||
std::pair<T, double>, bool> {
|
||||
bool operator()(const std::pair<T, double> &LHS,
|
||||
const std::pair<T, double> &RHS) const {
|
||||
return LHS.second > RHS.second;
|
||||
}
|
||||
};
|
||||
|
||||
static double ignoreMissing(double w) {
|
||||
if (w == ProfileInfo::MissingValue) return 0;
|
||||
return w;
|
||||
}
|
||||
|
||||
namespace {
|
||||
class ProfileAnnotator : public AssemblyAnnotationWriter {
|
||||
ProfileInfo &PI;
|
||||
public:
|
||||
ProfileAnnotator(ProfileInfo &pi) : PI(pi) {}
|
||||
|
||||
virtual void emitFunctionAnnot(const Function *F,
|
||||
formatted_raw_ostream &OS) {
|
||||
double w = PI.getExecutionCount(F);
|
||||
if (w != ProfileInfo::MissingValue) {
|
||||
OS << ";;; %" << F->getName() << " called "<<(unsigned)w
|
||||
<<" times.\n;;;\n";
|
||||
}
|
||||
}
|
||||
virtual void emitBasicBlockStartAnnot(const BasicBlock *BB,
|
||||
formatted_raw_ostream &OS) {
|
||||
double w = PI.getExecutionCount(BB);
|
||||
if (w != ProfileInfo::MissingValue) {
|
||||
if (w != 0) {
|
||||
OS << "\t;;; Basic block executed " << (unsigned)w << " times.\n";
|
||||
} else {
|
||||
OS << "\t;;; Never executed!\n";
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
virtual void emitBasicBlockEndAnnot(const BasicBlock *BB,
|
||||
formatted_raw_ostream &OS) {
|
||||
// Figure out how many times each successor executed.
|
||||
std::vector<std::pair<ProfileInfo::Edge, double> > SuccCounts;
|
||||
|
||||
const TerminatorInst *TI = BB->getTerminator();
|
||||
for (unsigned s = 0, e = TI->getNumSuccessors(); s != e; ++s) {
|
||||
BasicBlock* Succ = TI->getSuccessor(s);
|
||||
double w = ignoreMissing(PI.getEdgeWeight(std::make_pair(BB, Succ)));
|
||||
if (w != 0)
|
||||
SuccCounts.push_back(std::make_pair(std::make_pair(BB, Succ), w));
|
||||
}
|
||||
if (!SuccCounts.empty()) {
|
||||
OS << "\t;;; Out-edge counts:";
|
||||
for (unsigned i = 0, e = SuccCounts.size(); i != e; ++i)
|
||||
OS << " [" << (SuccCounts[i]).second << " -> "
|
||||
<< (SuccCounts[i]).first.second->getName() << "]";
|
||||
OS << "\n";
|
||||
}
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
namespace {
|
||||
/// ProfileInfoPrinterPass - Helper pass to dump the profile information for
|
||||
/// a module.
|
||||
//
|
||||
// FIXME: This should move elsewhere.
|
||||
class ProfileInfoPrinterPass : public ModulePass {
|
||||
ProfileInfoLoader &PIL;
|
||||
public:
|
||||
static char ID; // Class identification, replacement for typeinfo.
|
||||
explicit ProfileInfoPrinterPass(ProfileInfoLoader &_PIL)
|
||||
: ModulePass(ID), PIL(_PIL) {}
|
||||
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.setPreservesAll();
|
||||
AU.addRequired<ProfileInfo>();
|
||||
}
|
||||
|
||||
bool runOnModule(Module &M);
|
||||
};
|
||||
}
|
||||
|
||||
char ProfileInfoPrinterPass::ID = 0;
|
||||
|
||||
bool ProfileInfoPrinterPass::runOnModule(Module &M) {
|
||||
ProfileInfo &PI = getAnalysis<ProfileInfo>();
|
||||
std::map<const Function *, unsigned> FuncFreqs;
|
||||
std::map<const BasicBlock*, unsigned> BlockFreqs;
|
||||
std::map<ProfileInfo::Edge, unsigned> EdgeFreqs;
|
||||
|
||||
// Output a report. Eventually, there will be multiple reports selectable on
|
||||
// the command line, for now, just keep things simple.
|
||||
|
||||
// Emit the most frequent function table...
|
||||
std::vector<std::pair<Function*, double> > FunctionCounts;
|
||||
std::vector<std::pair<BasicBlock*, double> > Counts;
|
||||
for (Module::iterator FI = M.begin(), FE = M.end(); FI != FE; ++FI) {
|
||||
if (FI->isDeclaration()) continue;
|
||||
double w = ignoreMissing(PI.getExecutionCount(FI));
|
||||
FunctionCounts.push_back(std::make_pair(FI, w));
|
||||
for (Function::iterator BB = FI->begin(), BBE = FI->end();
|
||||
BB != BBE; ++BB) {
|
||||
double w = ignoreMissing(PI.getExecutionCount(BB));
|
||||
Counts.push_back(std::make_pair(BB, w));
|
||||
}
|
||||
}
|
||||
|
||||
// Sort by the frequency, backwards.
|
||||
sort(FunctionCounts.begin(), FunctionCounts.end(),
|
||||
PairSecondSortReverse<Function*>());
|
||||
|
||||
double TotalExecutions = 0;
|
||||
for (unsigned i = 0, e = FunctionCounts.size(); i != e; ++i)
|
||||
TotalExecutions += FunctionCounts[i].second;
|
||||
|
||||
outs() << "===" << std::string(73, '-') << "===\n"
|
||||
<< "LLVM profiling output for execution";
|
||||
if (PIL.getNumExecutions() != 1) outs() << "s";
|
||||
outs() << ":\n";
|
||||
|
||||
for (unsigned i = 0, e = PIL.getNumExecutions(); i != e; ++i) {
|
||||
outs() << " ";
|
||||
if (e != 1) outs() << i+1 << ". ";
|
||||
outs() << PIL.getExecution(i) << "\n";
|
||||
}
|
||||
|
||||
outs() << "\n===" << std::string(73, '-') << "===\n";
|
||||
outs() << "Function execution frequencies:\n\n";
|
||||
|
||||
// Print out the function frequencies...
|
||||
outs() << " ## Frequency\n";
|
||||
for (unsigned i = 0, e = FunctionCounts.size(); i != e; ++i) {
|
||||
if (FunctionCounts[i].second == 0) {
|
||||
outs() << "\n NOTE: " << e-i << " function"
|
||||
<< (e-i-1 ? "s were" : " was") << " never executed!\n";
|
||||
break;
|
||||
}
|
||||
|
||||
outs() << format("%3d", i+1) << ". "
|
||||
<< format("%5.2g", FunctionCounts[i].second) << "/"
|
||||
<< format("%g", TotalExecutions) << " "
|
||||
<< FunctionCounts[i].first->getNameStr() << "\n";
|
||||
}
|
||||
|
||||
std::set<Function*> FunctionsToPrint;
|
||||
|
||||
TotalExecutions = 0;
|
||||
for (unsigned i = 0, e = Counts.size(); i != e; ++i)
|
||||
TotalExecutions += Counts[i].second;
|
||||
|
||||
// Sort by the frequency, backwards.
|
||||
sort(Counts.begin(), Counts.end(),
|
||||
PairSecondSortReverse<BasicBlock*>());
|
||||
|
||||
outs() << "\n===" << std::string(73, '-') << "===\n";
|
||||
outs() << "Top 20 most frequently executed basic blocks:\n\n";
|
||||
|
||||
// Print out the function frequencies...
|
||||
outs() <<" ## %% \tFrequency\n";
|
||||
unsigned BlocksToPrint = Counts.size();
|
||||
if (BlocksToPrint > 20) BlocksToPrint = 20;
|
||||
for (unsigned i = 0; i != BlocksToPrint; ++i) {
|
||||
if (Counts[i].second == 0) break;
|
||||
Function *F = Counts[i].first->getParent();
|
||||
outs() << format("%3d", i+1) << ". "
|
||||
<< format("%5g", Counts[i].second/(double)TotalExecutions*100) << "% "
|
||||
<< format("%5.0f", Counts[i].second) << "/"
|
||||
<< format("%g", TotalExecutions) << "\t"
|
||||
<< F->getNameStr() << "() - "
|
||||
<< Counts[i].first->getNameStr() << "\n";
|
||||
FunctionsToPrint.insert(F);
|
||||
}
|
||||
|
||||
if (PrintAnnotatedLLVM || PrintAllCode) {
|
||||
outs() << "\n===" << std::string(73, '-') << "===\n";
|
||||
outs() << "Annotated LLVM code for the module:\n\n";
|
||||
|
||||
ProfileAnnotator PA(PI);
|
||||
|
||||
if (FunctionsToPrint.empty() || PrintAllCode)
|
||||
M.print(outs(), &PA);
|
||||
else
|
||||
// Print just a subset of the functions.
|
||||
for (std::set<Function*>::iterator I = FunctionsToPrint.begin(),
|
||||
E = FunctionsToPrint.end(); I != E; ++I)
|
||||
(*I)->print(outs(), &PA);
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
// Print a stack trace if we signal out.
|
||||
sys::PrintStackTraceOnErrorSignal();
|
||||
PrettyStackTraceProgram X(argc, argv);
|
||||
|
||||
LLVMContext &Context = getGlobalContext();
|
||||
llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
|
||||
|
||||
cl::ParseCommandLineOptions(argc, argv, "llvm profile dump decoder\n");
|
||||
|
||||
// Read in the bitcode file...
|
||||
std::string ErrorMessage;
|
||||
OwningPtr<MemoryBuffer> Buffer;
|
||||
error_code ec;
|
||||
Module *M = 0;
|
||||
if (!(ec = MemoryBuffer::getFileOrSTDIN(BitcodeFile, Buffer))) {
|
||||
M = ParseBitcodeFile(Buffer.get(), Context, &ErrorMessage);
|
||||
} else
|
||||
ErrorMessage = ec.message();
|
||||
if (M == 0) {
|
||||
errs() << argv[0] << ": " << BitcodeFile << ": "
|
||||
<< ErrorMessage << "\n";
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Read the profiling information. This is redundant since we load it again
|
||||
// using the standard profile info provider pass, but for now this gives us
|
||||
// access to additional information not exposed via the ProfileInfo
|
||||
// interface.
|
||||
ProfileInfoLoader PIL(argv[0], ProfileDataFile, *M);
|
||||
|
||||
// Run the printer pass.
|
||||
PassManager PassMgr;
|
||||
PassMgr.add(createProfileLoaderPass(ProfileDataFile));
|
||||
PassMgr.add(new ProfileInfoPrinterPass(PIL));
|
||||
PassMgr.run(*M);
|
||||
|
||||
return 0;
|
||||
}
|
6
contrib/llvm/tools/llvm-ranlib/CMakeLists.txt
Normal file
6
contrib/llvm/tools/llvm-ranlib/CMakeLists.txt
Normal file
@ -0,0 +1,6 @@
|
||||
set(LLVM_LINK_COMPONENTS archive)
|
||||
set(LLVM_REQUIRES_EH 1)
|
||||
|
||||
add_llvm_tool(llvm-ranlib
|
||||
llvm-ranlib.cpp
|
||||
)
|
18
contrib/llvm/tools/llvm-ranlib/Makefile
Normal file
18
contrib/llvm/tools/llvm-ranlib/Makefile
Normal file
@ -0,0 +1,18 @@
|
||||
##===- tools/llvm-ranlib/Makefile --------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
|
||||
LEVEL = ../..
|
||||
TOOLNAME = llvm-ranlib
|
||||
LINK_COMPONENTS = archive
|
||||
REQUIRES_EH := 1
|
||||
|
||||
# This tool has no plugins, optimize startup time.
|
||||
TOOL_NO_EXPORTS = 1
|
||||
|
||||
include $(LEVEL)/Makefile.common
|
101
contrib/llvm/tools/llvm-ranlib/llvm-ranlib.cpp
Normal file
101
contrib/llvm/tools/llvm-ranlib/llvm-ranlib.cpp
Normal file
@ -0,0 +1,101 @@
|
||||
//===-- llvm-ranlib.cpp - LLVM archive index generator --------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// Adds or updates an index (symbol table) for an LLVM archive file.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/Bitcode/Archive.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/FileSystem.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/Format.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include <memory>
|
||||
using namespace llvm;
|
||||
|
||||
// llvm-ar operation code and modifier flags
|
||||
static cl::opt<std::string>
|
||||
ArchiveName(cl::Positional, cl::Optional, cl::desc("<archive-file>"));
|
||||
|
||||
static cl::opt<bool>
|
||||
Verbose("verbose",cl::Optional,cl::init(false),
|
||||
cl::desc("Print the symbol table"));
|
||||
|
||||
// printSymbolTable - print out the archive's symbol table.
|
||||
void printSymbolTable(Archive* TheArchive) {
|
||||
outs() << "\nArchive Symbol Table:\n";
|
||||
const Archive::SymTabType& symtab = TheArchive->getSymbolTable();
|
||||
for (Archive::SymTabType::const_iterator I=symtab.begin(), E=symtab.end();
|
||||
I != E; ++I ) {
|
||||
unsigned offset = TheArchive->getFirstFileOffset() + I->second;
|
||||
outs() << " " << format("%9u", offset) << "\t" << I->first <<"\n";
|
||||
}
|
||||
}
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
// Print a stack trace if we signal out.
|
||||
llvm::sys::PrintStackTraceOnErrorSignal();
|
||||
llvm::PrettyStackTraceProgram X(argc, argv);
|
||||
|
||||
LLVMContext &Context = getGlobalContext();
|
||||
llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
|
||||
|
||||
// Have the command line options parsed and handle things
|
||||
// like --help and --version.
|
||||
cl::ParseCommandLineOptions(argc, argv,
|
||||
"LLVM Archive Index Generator (llvm-ranlib)\n\n"
|
||||
" This program adds or updates an index of bitcode symbols\n"
|
||||
" to an LLVM archive file."
|
||||
);
|
||||
|
||||
int exitCode = 0;
|
||||
|
||||
// Make sure we don't exit with "unhandled exception".
|
||||
try {
|
||||
|
||||
// Check the path name of the archive
|
||||
sys::Path ArchivePath;
|
||||
if (!ArchivePath.set(ArchiveName))
|
||||
throw std::string("Archive name invalid: ") + ArchiveName;
|
||||
|
||||
// Make sure it exists, we don't create empty archives
|
||||
bool Exists;
|
||||
if (llvm::sys::fs::exists(ArchivePath.str(), Exists) || !Exists)
|
||||
throw std::string("Archive file does not exist");
|
||||
|
||||
std::string err_msg;
|
||||
std::auto_ptr<Archive>
|
||||
AutoArchive(Archive::OpenAndLoad(ArchivePath, Context, &err_msg));
|
||||
Archive* TheArchive = AutoArchive.get();
|
||||
if (!TheArchive)
|
||||
throw err_msg;
|
||||
|
||||
if (TheArchive->writeToDisk(true, false, false, &err_msg ))
|
||||
throw err_msg;
|
||||
|
||||
if (Verbose)
|
||||
printSymbolTable(TheArchive);
|
||||
|
||||
} catch (const char* msg) {
|
||||
errs() << argv[0] << ": " << msg << "\n\n";
|
||||
exitCode = 1;
|
||||
} catch (const std::string& msg) {
|
||||
errs() << argv[0] << ": " << msg << "\n";
|
||||
exitCode = 2;
|
||||
} catch (...) {
|
||||
errs() << argv[0] << ": An unexpected unknown exception occurred.\n";
|
||||
exitCode = 3;
|
||||
}
|
||||
return exitCode;
|
||||
}
|
5
contrib/llvm/tools/llvm-rtdyld/CMakeLists.txt
Normal file
5
contrib/llvm/tools/llvm-rtdyld/CMakeLists.txt
Normal file
@ -0,0 +1,5 @@
|
||||
set(LLVM_LINK_COMPONENTS ${LLVM_TARGETS_TO_BUILD} support MC object RuntimeDyld JIT)
|
||||
|
||||
add_llvm_tool(llvm-rtdyld
|
||||
llvm-rtdyld.cpp
|
||||
)
|
23
contrib/llvm/tools/llvm-rtdyld/Makefile
Normal file
23
contrib/llvm/tools/llvm-rtdyld/Makefile
Normal file
@ -0,0 +1,23 @@
|
||||
##===- tools/llvm-rtdyld/Makefile --------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
|
||||
LEVEL = ../..
|
||||
TOOLNAME = llvm-rtdyld
|
||||
|
||||
# This tool has no plugins, optimize startup time.
|
||||
TOOL_NO_EXPORTS = 1
|
||||
|
||||
# Include this here so we can get the configuration of the targets
|
||||
# that have been configured for construction. We have to do this
|
||||
# early so we can set up LINK_COMPONENTS before including Makefile.rules
|
||||
include $(LEVEL)/Makefile.config
|
||||
|
||||
LINK_COMPONENTS := $(TARGETS_TO_BUILD) support MC object RuntimeDyld JIT
|
||||
|
||||
include $(LLVM_SRC_ROOT)/Makefile.rules
|
151
contrib/llvm/tools/llvm-rtdyld/llvm-rtdyld.cpp
Normal file
151
contrib/llvm/tools/llvm-rtdyld/llvm-rtdyld.cpp
Normal file
@ -0,0 +1,151 @@
|
||||
//===-- llvm-rtdyld.cpp - MCJIT Testing Tool ------------------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This is a testing tool for use with the MC-JIT LLVM components.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/ADT/StringMap.h"
|
||||
#include "llvm/ADT/OwningPtr.h"
|
||||
#include "llvm/ExecutionEngine/RuntimeDyld.h"
|
||||
#include "llvm/Object/MachOObject.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/Memory.h"
|
||||
#include "llvm/Support/MemoryBuffer.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/system_error.h"
|
||||
using namespace llvm;
|
||||
using namespace llvm::object;
|
||||
|
||||
static cl::list<std::string>
|
||||
InputFileList(cl::Positional, cl::ZeroOrMore,
|
||||
cl::desc("<input file>"));
|
||||
|
||||
enum ActionType {
|
||||
AC_Execute
|
||||
};
|
||||
|
||||
static cl::opt<ActionType>
|
||||
Action(cl::desc("Action to perform:"),
|
||||
cl::init(AC_Execute),
|
||||
cl::values(clEnumValN(AC_Execute, "execute",
|
||||
"Load, link, and execute the inputs."),
|
||||
clEnumValEnd));
|
||||
|
||||
static cl::opt<std::string>
|
||||
EntryPoint("entry",
|
||||
cl::desc("Function to call as entry point."),
|
||||
cl::init("_main"));
|
||||
|
||||
/* *** */
|
||||
|
||||
// A trivial memory manager that doesn't do anything fancy, just uses the
|
||||
// support library allocation routines directly.
|
||||
class TrivialMemoryManager : public RTDyldMemoryManager {
|
||||
public:
|
||||
SmallVector<sys::MemoryBlock, 16> FunctionMemory;
|
||||
|
||||
uint8_t *startFunctionBody(const char *Name, uintptr_t &Size);
|
||||
void endFunctionBody(const char *Name, uint8_t *FunctionStart,
|
||||
uint8_t *FunctionEnd);
|
||||
};
|
||||
|
||||
uint8_t *TrivialMemoryManager::startFunctionBody(const char *Name,
|
||||
uintptr_t &Size) {
|
||||
return (uint8_t*)sys::Memory::AllocateRWX(Size, 0, 0).base();
|
||||
}
|
||||
|
||||
void TrivialMemoryManager::endFunctionBody(const char *Name,
|
||||
uint8_t *FunctionStart,
|
||||
uint8_t *FunctionEnd) {
|
||||
uintptr_t Size = FunctionEnd - FunctionStart + 1;
|
||||
FunctionMemory.push_back(sys::MemoryBlock(FunctionStart, Size));
|
||||
}
|
||||
|
||||
static const char *ProgramName;
|
||||
|
||||
static void Message(const char *Type, const Twine &Msg) {
|
||||
errs() << ProgramName << ": " << Type << ": " << Msg << "\n";
|
||||
}
|
||||
|
||||
static int Error(const Twine &Msg) {
|
||||
Message("error", Msg);
|
||||
return 1;
|
||||
}
|
||||
|
||||
/* *** */
|
||||
|
||||
static int executeInput() {
|
||||
// Instantiate a dynamic linker.
|
||||
TrivialMemoryManager *MemMgr = new TrivialMemoryManager;
|
||||
RuntimeDyld Dyld(MemMgr);
|
||||
|
||||
// If we don't have any input files, read from stdin.
|
||||
if (!InputFileList.size())
|
||||
InputFileList.push_back("-");
|
||||
for(unsigned i = 0, e = InputFileList.size(); i != e; ++i) {
|
||||
// Load the input memory buffer.
|
||||
OwningPtr<MemoryBuffer> InputBuffer;
|
||||
if (error_code ec = MemoryBuffer::getFileOrSTDIN(InputFileList[i],
|
||||
InputBuffer))
|
||||
return Error("unable to read input: '" + ec.message() + "'");
|
||||
|
||||
// Load the object file into it.
|
||||
if (Dyld.loadObject(InputBuffer.take())) {
|
||||
return Error(Dyld.getErrorString());
|
||||
}
|
||||
}
|
||||
|
||||
// Resolve all the relocations we can.
|
||||
Dyld.resolveRelocations();
|
||||
|
||||
// FIXME: Error out if there are unresolved relocations.
|
||||
|
||||
// Get the address of the entry point (_main by default).
|
||||
void *MainAddress = Dyld.getSymbolAddress(EntryPoint);
|
||||
if (MainAddress == 0)
|
||||
return Error("no definition for '" + EntryPoint + "'");
|
||||
|
||||
// Invalidate the instruction cache for each loaded function.
|
||||
for (unsigned i = 0, e = MemMgr->FunctionMemory.size(); i != e; ++i) {
|
||||
sys::MemoryBlock &Data = MemMgr->FunctionMemory[i];
|
||||
// Make sure the memory is executable.
|
||||
std::string ErrorStr;
|
||||
sys::Memory::InvalidateInstructionCache(Data.base(), Data.size());
|
||||
if (!sys::Memory::setExecutable(Data, &ErrorStr))
|
||||
return Error("unable to mark function executable: '" + ErrorStr + "'");
|
||||
}
|
||||
|
||||
// Dispatch to _main().
|
||||
errs() << "loaded '" << EntryPoint << "' at: " << (void*)MainAddress << "\n";
|
||||
|
||||
int (*Main)(int, const char**) =
|
||||
(int(*)(int,const char**)) uintptr_t(MainAddress);
|
||||
const char **Argv = new const char*[2];
|
||||
// Use the name of the first input object module as argv[0] for the target.
|
||||
Argv[0] = InputFileList[0].c_str();
|
||||
Argv[1] = 0;
|
||||
return Main(1, Argv);
|
||||
}
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
ProgramName = argv[0];
|
||||
llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
|
||||
|
||||
cl::ParseCommandLineOptions(argc, argv, "llvm MC-JIT tool\n");
|
||||
|
||||
switch (Action) {
|
||||
default:
|
||||
case AC_Execute:
|
||||
return executeInput();
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
3
contrib/llvm/tools/llvm-stub/CMakeLists.txt
Normal file
3
contrib/llvm/tools/llvm-stub/CMakeLists.txt
Normal file
@ -0,0 +1,3 @@
|
||||
add_llvm_tool(llvm-stub
|
||||
llvm-stub.c
|
||||
)
|
13
contrib/llvm/tools/llvm-stub/Makefile
Normal file
13
contrib/llvm/tools/llvm-stub/Makefile
Normal file
@ -0,0 +1,13 @@
|
||||
##===- tools/llvm-stub/Makefile ----------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
|
||||
LEVEL = ../..
|
||||
TOOLNAME = llvm-stub
|
||||
include $(LEVEL)/Makefile.common
|
||||
|
77
contrib/llvm/tools/llvm-stub/llvm-stub.c
Normal file
77
contrib/llvm/tools/llvm-stub/llvm-stub.c
Normal file
@ -0,0 +1,77 @@
|
||||
/*===- llvm-stub.c - Stub executable to run llvm bitcode files ------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This tool is used by the gccld program to enable transparent execution of
|
||||
// bitcode files by the user. Specifically, gccld outputs two files when asked
|
||||
// to compile a <program> file:
|
||||
// 1. It outputs the LLVM bitcode file to <program>.bc
|
||||
// 2. It outputs a stub executable that runs lli on <program>.bc
|
||||
//
|
||||
// This allows the end user to just say ./<program> and have the JIT executed
|
||||
// automatically. On unix, the stub executable emitted is actually a bourne
|
||||
// shell script that does the forwarding. Windows does not like #!/bin/sh
|
||||
// programs in .exe files, so we make it an actual program, defined here.
|
||||
//
|
||||
//===----------------------------------------------------------------------===*/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
|
||||
#include "llvm/Config/config.h"
|
||||
|
||||
#if defined(HAVE_UNISTD_H) && !defined(_MSC_VER)
|
||||
#include <unistd.h>
|
||||
#endif
|
||||
|
||||
#ifdef _WIN32
|
||||
#include <process.h>
|
||||
#include <io.h>
|
||||
#endif
|
||||
|
||||
int main(int argc, char** argv) {
|
||||
const char *Interp = getenv("LLVMINTERP");
|
||||
const char **Args;
|
||||
if (Interp == 0) Interp = "lli";
|
||||
|
||||
/* Set up the command line options to pass to the JIT. */
|
||||
Args = (const char**)malloc(sizeof(char*) * (argc+2));
|
||||
/* argv[0] is the JIT */
|
||||
Args[0] = Interp;
|
||||
|
||||
#ifdef LLVM_ON_WIN32
|
||||
{
|
||||
int len = strlen(argv[0]);
|
||||
if (len < 4 || strcmp(argv[0] + len - 4, ".exe") != 0) {
|
||||
/* .exe suffix is stripped off of argv[0] if the executable was run on the
|
||||
* command line without one. Put it back on.
|
||||
*/
|
||||
argv[0] = strcat(strcpy((char*)malloc(len + 5), argv[0]), ".exe");
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
/* argv[1] is argv[0] + ".bc". */
|
||||
Args[1] = strcat(strcpy((char*)malloc(strlen(argv[0])+4), argv[0]), ".bc");
|
||||
|
||||
/* The rest of the args are as before. */
|
||||
memcpy((char **)Args+2, argv+1, sizeof(char*)*argc);
|
||||
|
||||
/* Run the JIT. */
|
||||
#if !defined(_WIN32) || defined(__MINGW64__)
|
||||
execvp(Interp, (char **)Args); /* POSIX execvp takes a char *const[]. */
|
||||
#else
|
||||
execvp(Interp, Args); /* windows execvp takes a const char *const *. */
|
||||
#endif
|
||||
/* if _execv returns, the JIT could not be started. */
|
||||
fprintf(stderr, "Could not execute the LLVM JIT. Either add 'lli' to your"
|
||||
" path, or set the\ninterpreter you want to use in the LLVMINTERP "
|
||||
"environment variable.\n");
|
||||
return 1;
|
||||
}
|
5
contrib/llvm/tools/macho-dump/CMakeLists.txt
Normal file
5
contrib/llvm/tools/macho-dump/CMakeLists.txt
Normal file
@ -0,0 +1,5 @@
|
||||
set(LLVM_LINK_COMPONENTS ${LLVM_TARGETS_TO_BUILD} support object)
|
||||
|
||||
add_llvm_tool(macho-dump
|
||||
macho-dump.cpp
|
||||
)
|
23
contrib/llvm/tools/macho-dump/Makefile
Normal file
23
contrib/llvm/tools/macho-dump/Makefile
Normal file
@ -0,0 +1,23 @@
|
||||
##===- tools/macho-dump/Makefile ---------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
|
||||
LEVEL = ../..
|
||||
TOOLNAME = macho-dump
|
||||
|
||||
# This tool has no plugins, optimize startup time.
|
||||
TOOL_NO_EXPORTS = 1
|
||||
|
||||
# Include this here so we can get the configuration of the targets
|
||||
# that have been configured for construction. We have to do this
|
||||
# early so we can set up LINK_COMPONENTS before including Makefile.rules
|
||||
include $(LEVEL)/Makefile.config
|
||||
|
||||
LINK_COMPONENTS := support object
|
||||
|
||||
include $(LLVM_SRC_ROOT)/Makefile.rules
|
400
contrib/llvm/tools/macho-dump/macho-dump.cpp
Normal file
400
contrib/llvm/tools/macho-dump/macho-dump.cpp
Normal file
@ -0,0 +1,400 @@
|
||||
//===-- macho-dump.cpp - Mach Object Dumping Tool -------------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This is a testing tool for use with the MC/Mach-O LLVM components.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/Object/MachOObject.h"
|
||||
#include "llvm/ADT/StringExtras.h"
|
||||
#include "llvm/ADT/Twine.h"
|
||||
#include "llvm/Support/CommandLine.h"
|
||||
#include "llvm/Support/Format.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/MemoryBuffer.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/Support/system_error.h"
|
||||
using namespace llvm;
|
||||
using namespace llvm::object;
|
||||
|
||||
static cl::opt<std::string>
|
||||
InputFile(cl::Positional, cl::desc("<input file>"), cl::init("-"));
|
||||
|
||||
static cl::opt<bool>
|
||||
ShowSectionData("dump-section-data", cl::desc("Dump the contents of sections"),
|
||||
cl::init(false));
|
||||
|
||||
///
|
||||
|
||||
static const char *ProgramName;
|
||||
|
||||
static void Message(const char *Type, const Twine &Msg) {
|
||||
errs() << ProgramName << ": " << Type << ": " << Msg << "\n";
|
||||
}
|
||||
|
||||
static int Error(const Twine &Msg) {
|
||||
Message("error", Msg);
|
||||
return 1;
|
||||
}
|
||||
|
||||
static void Warning(const Twine &Msg) {
|
||||
Message("warning", Msg);
|
||||
}
|
||||
|
||||
///
|
||||
|
||||
static void DumpSegmentCommandData(StringRef Name,
|
||||
uint64_t VMAddr, uint64_t VMSize,
|
||||
uint64_t FileOffset, uint64_t FileSize,
|
||||
uint32_t MaxProt, uint32_t InitProt,
|
||||
uint32_t NumSections, uint32_t Flags) {
|
||||
outs() << " ('segment_name', '";
|
||||
outs().write_escaped(Name, /*UseHexEscapes=*/true) << "')\n";
|
||||
outs() << " ('vm_addr', " << VMAddr << ")\n";
|
||||
outs() << " ('vm_size', " << VMSize << ")\n";
|
||||
outs() << " ('file_offset', " << FileOffset << ")\n";
|
||||
outs() << " ('file_size', " << FileSize << ")\n";
|
||||
outs() << " ('maxprot', " << MaxProt << ")\n";
|
||||
outs() << " ('initprot', " << InitProt << ")\n";
|
||||
outs() << " ('num_sections', " << NumSections << ")\n";
|
||||
outs() << " ('flags', " << Flags << ")\n";
|
||||
}
|
||||
|
||||
static int DumpSectionData(MachOObject &Obj, unsigned Index, StringRef Name,
|
||||
StringRef SegmentName, uint64_t Address,
|
||||
uint64_t Size, uint32_t Offset,
|
||||
uint32_t Align, uint32_t RelocationTableOffset,
|
||||
uint32_t NumRelocationTableEntries,
|
||||
uint32_t Flags, uint32_t Reserved1,
|
||||
uint32_t Reserved2, uint64_t Reserved3 = ~0ULL) {
|
||||
outs() << " # Section " << Index << "\n";
|
||||
outs() << " (('section_name', '";
|
||||
outs().write_escaped(Name, /*UseHexEscapes=*/true) << "')\n";
|
||||
outs() << " ('segment_name', '";
|
||||
outs().write_escaped(SegmentName, /*UseHexEscapes=*/true) << "')\n";
|
||||
outs() << " ('address', " << Address << ")\n";
|
||||
outs() << " ('size', " << Size << ")\n";
|
||||
outs() << " ('offset', " << Offset << ")\n";
|
||||
outs() << " ('alignment', " << Align << ")\n";
|
||||
outs() << " ('reloc_offset', " << RelocationTableOffset << ")\n";
|
||||
outs() << " ('num_reloc', " << NumRelocationTableEntries << ")\n";
|
||||
outs() << " ('flags', " << format("0x%x", Flags) << ")\n";
|
||||
outs() << " ('reserved1', " << Reserved1 << ")\n";
|
||||
outs() << " ('reserved2', " << Reserved2 << ")\n";
|
||||
if (Reserved3 != ~0ULL)
|
||||
outs() << " ('reserved3', " << Reserved3 << ")\n";
|
||||
outs() << " ),\n";
|
||||
|
||||
// Dump the relocation entries.
|
||||
int Res = 0;
|
||||
outs() << " ('_relocations', [\n";
|
||||
for (unsigned i = 0; i != NumRelocationTableEntries; ++i) {
|
||||
InMemoryStruct<macho::RelocationEntry> RE;
|
||||
Obj.ReadRelocationEntry(RelocationTableOffset, i, RE);
|
||||
if (!RE) {
|
||||
Res = Error("unable to read relocation table entry '" + Twine(i) + "'");
|
||||
break;
|
||||
}
|
||||
|
||||
outs() << " # Relocation " << i << "\n";
|
||||
outs() << " (('word-0', " << format("0x%x", RE->Word0) << "),\n";
|
||||
outs() << " ('word-1', " << format("0x%x", RE->Word1) << ")),\n";
|
||||
}
|
||||
outs() << " ])\n";
|
||||
|
||||
// Dump the section data, if requested.
|
||||
if (ShowSectionData) {
|
||||
outs() << " ('_section_data', '";
|
||||
StringRef Data = Obj.getData(Offset, Size);
|
||||
for (unsigned i = 0; i != Data.size(); ++i) {
|
||||
if (i && (i % 4) == 0)
|
||||
outs() << ' ';
|
||||
outs() << hexdigit((Data[i] >> 4) & 0xF, /*LowerCase=*/true);
|
||||
outs() << hexdigit((Data[i] >> 0) & 0xF, /*LowerCase=*/true);
|
||||
}
|
||||
outs() << "')\n";
|
||||
}
|
||||
|
||||
return Res;
|
||||
}
|
||||
|
||||
static int DumpSegmentCommand(MachOObject &Obj,
|
||||
const MachOObject::LoadCommandInfo &LCI) {
|
||||
InMemoryStruct<macho::SegmentLoadCommand> SLC;
|
||||
Obj.ReadSegmentLoadCommand(LCI, SLC);
|
||||
if (!SLC)
|
||||
return Error("unable to read segment load command");
|
||||
|
||||
DumpSegmentCommandData(StringRef(SLC->Name, 16), SLC->VMAddress,
|
||||
SLC->VMSize, SLC->FileOffset, SLC->FileSize,
|
||||
SLC->MaxVMProtection, SLC->InitialVMProtection,
|
||||
SLC->NumSections, SLC->Flags);
|
||||
|
||||
// Dump the sections.
|
||||
int Res = 0;
|
||||
outs() << " ('sections', [\n";
|
||||
for (unsigned i = 0; i != SLC->NumSections; ++i) {
|
||||
InMemoryStruct<macho::Section> Sect;
|
||||
Obj.ReadSection(LCI, i, Sect);
|
||||
if (!SLC) {
|
||||
Res = Error("unable to read section '" + Twine(i) + "'");
|
||||
break;
|
||||
}
|
||||
|
||||
if ((Res = DumpSectionData(Obj, i, StringRef(Sect->Name, 16),
|
||||
StringRef(Sect->SegmentName, 16), Sect->Address,
|
||||
Sect->Size, Sect->Offset, Sect->Align,
|
||||
Sect->RelocationTableOffset,
|
||||
Sect->NumRelocationTableEntries, Sect->Flags,
|
||||
Sect->Reserved1, Sect->Reserved2)))
|
||||
break;
|
||||
}
|
||||
outs() << " ])\n";
|
||||
|
||||
return Res;
|
||||
}
|
||||
|
||||
static int DumpSegment64Command(MachOObject &Obj,
|
||||
const MachOObject::LoadCommandInfo &LCI) {
|
||||
InMemoryStruct<macho::Segment64LoadCommand> SLC;
|
||||
Obj.ReadSegment64LoadCommand(LCI, SLC);
|
||||
if (!SLC)
|
||||
return Error("unable to read segment load command");
|
||||
|
||||
DumpSegmentCommandData(StringRef(SLC->Name, 16), SLC->VMAddress,
|
||||
SLC->VMSize, SLC->FileOffset, SLC->FileSize,
|
||||
SLC->MaxVMProtection, SLC->InitialVMProtection,
|
||||
SLC->NumSections, SLC->Flags);
|
||||
|
||||
// Dump the sections.
|
||||
int Res = 0;
|
||||
outs() << " ('sections', [\n";
|
||||
for (unsigned i = 0; i != SLC->NumSections; ++i) {
|
||||
InMemoryStruct<macho::Section64> Sect;
|
||||
Obj.ReadSection64(LCI, i, Sect);
|
||||
if (!SLC) {
|
||||
Res = Error("unable to read section '" + Twine(i) + "'");
|
||||
break;
|
||||
}
|
||||
|
||||
if ((Res = DumpSectionData(Obj, i, StringRef(Sect->Name, 16),
|
||||
StringRef(Sect->SegmentName, 16), Sect->Address,
|
||||
Sect->Size, Sect->Offset, Sect->Align,
|
||||
Sect->RelocationTableOffset,
|
||||
Sect->NumRelocationTableEntries, Sect->Flags,
|
||||
Sect->Reserved1, Sect->Reserved2,
|
||||
Sect->Reserved3)))
|
||||
break;
|
||||
}
|
||||
outs() << " ])\n";
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void DumpSymbolTableEntryData(MachOObject &Obj,
|
||||
unsigned Index, uint32_t StringIndex,
|
||||
uint8_t Type, uint8_t SectionIndex,
|
||||
uint16_t Flags, uint64_t Value) {
|
||||
outs() << " # Symbol " << Index << "\n";
|
||||
outs() << " (('n_strx', " << StringIndex << ")\n";
|
||||
outs() << " ('n_type', " << format("0x%x", Type) << ")\n";
|
||||
outs() << " ('n_sect', " << uint32_t(SectionIndex) << ")\n";
|
||||
outs() << " ('n_desc', " << Flags << ")\n";
|
||||
outs() << " ('n_value', " << Value << ")\n";
|
||||
outs() << " ('_string', '" << Obj.getStringAtIndex(StringIndex) << "')\n";
|
||||
outs() << " ),\n";
|
||||
}
|
||||
|
||||
static int DumpSymtabCommand(MachOObject &Obj,
|
||||
const MachOObject::LoadCommandInfo &LCI) {
|
||||
InMemoryStruct<macho::SymtabLoadCommand> SLC;
|
||||
Obj.ReadSymtabLoadCommand(LCI, SLC);
|
||||
if (!SLC)
|
||||
return Error("unable to read segment load command");
|
||||
|
||||
outs() << " ('symoff', " << SLC->SymbolTableOffset << ")\n";
|
||||
outs() << " ('nsyms', " << SLC->NumSymbolTableEntries << ")\n";
|
||||
outs() << " ('stroff', " << SLC->StringTableOffset << ")\n";
|
||||
outs() << " ('strsize', " << SLC->StringTableSize << ")\n";
|
||||
|
||||
// Cache the string table data.
|
||||
Obj.RegisterStringTable(*SLC);
|
||||
|
||||
// Dump the string data.
|
||||
outs() << " ('_string_data', '";
|
||||
outs().write_escaped(Obj.getStringTableData(),
|
||||
/*UseHexEscapes=*/true) << "')\n";
|
||||
|
||||
// Dump the symbol table.
|
||||
int Res = 0;
|
||||
outs() << " ('_symbols', [\n";
|
||||
for (unsigned i = 0; i != SLC->NumSymbolTableEntries; ++i) {
|
||||
if (Obj.is64Bit()) {
|
||||
InMemoryStruct<macho::Symbol64TableEntry> STE;
|
||||
Obj.ReadSymbol64TableEntry(SLC->SymbolTableOffset, i, STE);
|
||||
if (!STE) {
|
||||
Res = Error("unable to read symbol: '" + Twine(i) + "'");
|
||||
break;
|
||||
}
|
||||
|
||||
DumpSymbolTableEntryData(Obj, i, STE->StringIndex, STE->Type,
|
||||
STE->SectionIndex, STE->Flags, STE->Value);
|
||||
} else {
|
||||
InMemoryStruct<macho::SymbolTableEntry> STE;
|
||||
Obj.ReadSymbolTableEntry(SLC->SymbolTableOffset, i, STE);
|
||||
if (!SLC) {
|
||||
Res = Error("unable to read symbol: '" + Twine(i) + "'");
|
||||
break;
|
||||
}
|
||||
|
||||
DumpSymbolTableEntryData(Obj, i, STE->StringIndex, STE->Type,
|
||||
STE->SectionIndex, STE->Flags, STE->Value);
|
||||
}
|
||||
}
|
||||
outs() << " ])\n";
|
||||
|
||||
return Res;
|
||||
}
|
||||
|
||||
static int DumpDysymtabCommand(MachOObject &Obj,
|
||||
const MachOObject::LoadCommandInfo &LCI) {
|
||||
InMemoryStruct<macho::DysymtabLoadCommand> DLC;
|
||||
Obj.ReadDysymtabLoadCommand(LCI, DLC);
|
||||
if (!DLC)
|
||||
return Error("unable to read segment load command");
|
||||
|
||||
outs() << " ('ilocalsym', " << DLC->LocalSymbolsIndex << ")\n";
|
||||
outs() << " ('nlocalsym', " << DLC->NumLocalSymbols << ")\n";
|
||||
outs() << " ('iextdefsym', " << DLC->ExternalSymbolsIndex << ")\n";
|
||||
outs() << " ('nextdefsym', " << DLC->NumExternalSymbols << ")\n";
|
||||
outs() << " ('iundefsym', " << DLC->UndefinedSymbolsIndex << ")\n";
|
||||
outs() << " ('nundefsym', " << DLC->NumUndefinedSymbols << ")\n";
|
||||
outs() << " ('tocoff', " << DLC->TOCOffset << ")\n";
|
||||
outs() << " ('ntoc', " << DLC->NumTOCEntries << ")\n";
|
||||
outs() << " ('modtaboff', " << DLC->ModuleTableOffset << ")\n";
|
||||
outs() << " ('nmodtab', " << DLC->NumModuleTableEntries << ")\n";
|
||||
outs() << " ('extrefsymoff', " << DLC->ReferenceSymbolTableOffset << ")\n";
|
||||
outs() << " ('nextrefsyms', "
|
||||
<< DLC->NumReferencedSymbolTableEntries << ")\n";
|
||||
outs() << " ('indirectsymoff', " << DLC->IndirectSymbolTableOffset << ")\n";
|
||||
outs() << " ('nindirectsyms', "
|
||||
<< DLC->NumIndirectSymbolTableEntries << ")\n";
|
||||
outs() << " ('extreloff', " << DLC->ExternalRelocationTableOffset << ")\n";
|
||||
outs() << " ('nextrel', " << DLC->NumExternalRelocationTableEntries << ")\n";
|
||||
outs() << " ('locreloff', " << DLC->LocalRelocationTableOffset << ")\n";
|
||||
outs() << " ('nlocrel', " << DLC->NumLocalRelocationTableEntries << ")\n";
|
||||
|
||||
// Dump the indirect symbol table.
|
||||
int Res = 0;
|
||||
outs() << " ('_indirect_symbols', [\n";
|
||||
for (unsigned i = 0; i != DLC->NumIndirectSymbolTableEntries; ++i) {
|
||||
InMemoryStruct<macho::IndirectSymbolTableEntry> ISTE;
|
||||
Obj.ReadIndirectSymbolTableEntry(*DLC, i, ISTE);
|
||||
if (!ISTE) {
|
||||
Res = Error("unable to read segment load command");
|
||||
break;
|
||||
}
|
||||
|
||||
outs() << " # Indirect Symbol " << i << "\n";
|
||||
outs() << " (('symbol_index', "
|
||||
<< format("0x%x", ISTE->Index) << "),),\n";
|
||||
}
|
||||
outs() << " ])\n";
|
||||
|
||||
return Res;
|
||||
}
|
||||
|
||||
static int DumpLinkeditDataCommand(MachOObject &Obj,
|
||||
const MachOObject::LoadCommandInfo &LCI) {
|
||||
InMemoryStruct<macho::LinkeditDataLoadCommand> LLC;
|
||||
Obj.ReadLinkeditDataLoadCommand(LCI, LLC);
|
||||
if (!LLC)
|
||||
return Error("unable to read segment load command");
|
||||
|
||||
outs() << " ('dataoff', " << LLC->DataOffset << ")\n"
|
||||
<< " ('datasize', " << LLC->DataSize << ")\n"
|
||||
<< " ('_addresses', [\n";
|
||||
|
||||
SmallVector<uint64_t, 8> Addresses;
|
||||
Obj.ReadULEB128s(LLC->DataOffset, Addresses);
|
||||
for (unsigned i = 0, e = Addresses.size(); i != e; ++i)
|
||||
outs() << " # Address " << i << '\n'
|
||||
<< " ('address', " << format("0x%x", Addresses[i]) << "),\n";
|
||||
|
||||
outs() << " ])\n";
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
static int DumpLoadCommand(MachOObject &Obj, unsigned Index) {
|
||||
const MachOObject::LoadCommandInfo &LCI = Obj.getLoadCommandInfo(Index);
|
||||
int Res = 0;
|
||||
|
||||
outs() << " # Load Command " << Index << "\n"
|
||||
<< " (('command', " << LCI.Command.Type << ")\n"
|
||||
<< " ('size', " << LCI.Command.Size << ")\n";
|
||||
switch (LCI.Command.Type) {
|
||||
case macho::LCT_Segment:
|
||||
Res = DumpSegmentCommand(Obj, LCI);
|
||||
break;
|
||||
case macho::LCT_Segment64:
|
||||
Res = DumpSegment64Command(Obj, LCI);
|
||||
break;
|
||||
case macho::LCT_Symtab:
|
||||
Res = DumpSymtabCommand(Obj, LCI);
|
||||
break;
|
||||
case macho::LCT_Dysymtab:
|
||||
Res = DumpDysymtabCommand(Obj, LCI);
|
||||
break;
|
||||
case macho::LCT_CodeSignature:
|
||||
case macho::LCT_SegmentSplitInfo:
|
||||
case macho::LCT_FunctionStarts:
|
||||
Res = DumpLinkeditDataCommand(Obj, LCI);
|
||||
break;
|
||||
default:
|
||||
Warning("unknown load command: " + Twine(LCI.Command.Type));
|
||||
break;
|
||||
}
|
||||
outs() << " ),\n";
|
||||
|
||||
return Res;
|
||||
}
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
ProgramName = argv[0];
|
||||
llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
|
||||
|
||||
cl::ParseCommandLineOptions(argc, argv, "llvm Mach-O dumping tool\n");
|
||||
|
||||
// Load the input file.
|
||||
std::string ErrorStr;
|
||||
OwningPtr<MemoryBuffer> InputBuffer;
|
||||
if (error_code ec = MemoryBuffer::getFileOrSTDIN(InputFile, InputBuffer))
|
||||
return Error("unable to read input: '" + ec.message() + "'");
|
||||
|
||||
// Construct the Mach-O wrapper object.
|
||||
OwningPtr<MachOObject> InputObject(
|
||||
MachOObject::LoadFromBuffer(InputBuffer.take(), &ErrorStr));
|
||||
if (!InputObject)
|
||||
return Error("unable to load object: '" + ErrorStr + "'");
|
||||
|
||||
// Print the header
|
||||
InputObject->printHeader(outs());
|
||||
|
||||
// Print the load commands.
|
||||
int Res = 0;
|
||||
outs() << "('load_commands', [\n";
|
||||
for (unsigned i = 0; i != InputObject->getHeader().NumLoadCommands; ++i)
|
||||
if ((Res = DumpLoadCommand(*InputObject, i)))
|
||||
break;
|
||||
outs() << "])\n";
|
||||
|
||||
return Res;
|
||||
}
|
94
contrib/llvm/tools/opt/AnalysisWrappers.cpp
Normal file
94
contrib/llvm/tools/opt/AnalysisWrappers.cpp
Normal file
@ -0,0 +1,94 @@
|
||||
//===- AnalysisWrappers.cpp - Wrappers around non-pass analyses -----------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file defines pass wrappers around LLVM analyses that don't make sense to
|
||||
// be passes. It provides a nice standard pass interface to these classes so
|
||||
// that they can be printed out by analyze.
|
||||
//
|
||||
// These classes are separated out of analyze.cpp so that it is more clear which
|
||||
// code is the integral part of the analyze tool, and which part of the code is
|
||||
// just making it so more passes are available.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/Pass.h"
|
||||
#include "llvm/Support/CallSite.h"
|
||||
#include "llvm/Analysis/CallGraph.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
using namespace llvm;
|
||||
|
||||
namespace {
|
||||
/// ExternalFunctionsPassedConstants - This pass prints out call sites to
|
||||
/// external functions that are called with constant arguments. This can be
|
||||
/// useful when looking for standard library functions we should constant fold
|
||||
/// or handle in alias analyses.
|
||||
struct ExternalFunctionsPassedConstants : public ModulePass {
|
||||
static char ID; // Pass ID, replacement for typeid
|
||||
ExternalFunctionsPassedConstants() : ModulePass(ID) {}
|
||||
virtual bool runOnModule(Module &M) {
|
||||
for (Module::iterator I = M.begin(), E = M.end(); I != E; ++I) {
|
||||
if (!I->isDeclaration()) continue;
|
||||
|
||||
bool PrintedFn = false;
|
||||
for (Value::use_iterator UI = I->use_begin(), E = I->use_end();
|
||||
UI != E; ++UI) {
|
||||
Instruction *User = dyn_cast<Instruction>(*UI);
|
||||
if (!User) continue;
|
||||
|
||||
CallSite CS(cast<Value>(User));
|
||||
if (!CS) continue;
|
||||
|
||||
for (CallSite::arg_iterator AI = CS.arg_begin(),
|
||||
E = CS.arg_end(); AI != E; ++AI) {
|
||||
if (!isa<Constant>(*AI)) continue;
|
||||
|
||||
if (!PrintedFn) {
|
||||
errs() << "Function '" << I->getName() << "':\n";
|
||||
PrintedFn = true;
|
||||
}
|
||||
errs() << *User;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.setPreservesAll();
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
char ExternalFunctionsPassedConstants::ID = 0;
|
||||
static RegisterPass<ExternalFunctionsPassedConstants>
|
||||
P1("print-externalfnconstants",
|
||||
"Print external fn callsites passed constants");
|
||||
|
||||
namespace {
|
||||
struct CallGraphPrinter : public ModulePass {
|
||||
static char ID; // Pass ID, replacement for typeid
|
||||
CallGraphPrinter() : ModulePass(ID) {}
|
||||
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.setPreservesAll();
|
||||
AU.addRequiredTransitive<CallGraph>();
|
||||
}
|
||||
virtual bool runOnModule(Module &M) {
|
||||
getAnalysis<CallGraph>().print(errs(), &M);
|
||||
return false;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
char CallGraphPrinter::ID = 0;
|
||||
static RegisterPass<CallGraphPrinter>
|
||||
P2("print-callgraph", "Print a call graph");
|
8
contrib/llvm/tools/opt/CMakeLists.txt
Normal file
8
contrib/llvm/tools/opt/CMakeLists.txt
Normal file
@ -0,0 +1,8 @@
|
||||
set(LLVM_LINK_COMPONENTS bitreader asmparser bitwriter instrumentation scalaropts ipo)
|
||||
|
||||
add_llvm_tool(opt
|
||||
AnalysisWrappers.cpp
|
||||
GraphPrinters.cpp
|
||||
PrintSCC.cpp
|
||||
opt.cpp
|
||||
)
|
118
contrib/llvm/tools/opt/GraphPrinters.cpp
Normal file
118
contrib/llvm/tools/opt/GraphPrinters.cpp
Normal file
@ -0,0 +1,118 @@
|
||||
//===- GraphPrinters.cpp - DOT printers for various graph types -----------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file defines several printers for various different types of graphs used
|
||||
// by the LLVM infrastructure. It uses the generic graph interface to convert
|
||||
// the graph into a .dot graph. These graphs can then be processed with the
|
||||
// "dot" tool to convert them to postscript or some other suitable format.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/Support/GraphWriter.h"
|
||||
#include "llvm/Pass.h"
|
||||
#include "llvm/Value.h"
|
||||
#include "llvm/Analysis/CallGraph.h"
|
||||
#include "llvm/Analysis/Dominators.h"
|
||||
#include "llvm/Support/ToolOutputFile.h"
|
||||
using namespace llvm;
|
||||
|
||||
template<typename GraphType>
|
||||
static void WriteGraphToFile(raw_ostream &O, const std::string &GraphName,
|
||||
const GraphType >) {
|
||||
std::string Filename = GraphName + ".dot";
|
||||
O << "Writing '" << Filename << "'...";
|
||||
std::string ErrInfo;
|
||||
tool_output_file F(Filename.c_str(), ErrInfo);
|
||||
|
||||
if (ErrInfo.empty()) {
|
||||
WriteGraph(F.os(), GT);
|
||||
F.os().close();
|
||||
if (!F.os().has_error()) {
|
||||
O << "\n";
|
||||
F.keep();
|
||||
return;
|
||||
}
|
||||
}
|
||||
O << " error opening file for writing!\n";
|
||||
F.os().clear_error();
|
||||
}
|
||||
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
// Call Graph Printer
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
namespace llvm {
|
||||
template<>
|
||||
struct DOTGraphTraits<CallGraph*> : public DefaultDOTGraphTraits {
|
||||
|
||||
DOTGraphTraits (bool isSimple=false) : DefaultDOTGraphTraits(isSimple) {}
|
||||
|
||||
static std::string getGraphName(CallGraph *F) {
|
||||
return "Call Graph";
|
||||
}
|
||||
|
||||
static std::string getNodeLabel(CallGraphNode *Node, CallGraph *Graph) {
|
||||
if (Node->getFunction())
|
||||
return ((Value*)Node->getFunction())->getName();
|
||||
return "external node";
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
|
||||
namespace {
|
||||
struct CallGraphPrinter : public ModulePass {
|
||||
static char ID; // Pass ID, replacement for typeid
|
||||
CallGraphPrinter() : ModulePass(ID) {}
|
||||
|
||||
virtual bool runOnModule(Module &M) {
|
||||
WriteGraphToFile(llvm::errs(), "callgraph", &getAnalysis<CallGraph>());
|
||||
return false;
|
||||
}
|
||||
|
||||
void print(raw_ostream &OS, const llvm::Module*) const {}
|
||||
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.addRequired<CallGraph>();
|
||||
AU.setPreservesAll();
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
char CallGraphPrinter::ID = 0;
|
||||
static RegisterPass<CallGraphPrinter> P2("dot-callgraph",
|
||||
"Print Call Graph to 'dot' file");
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
// DomInfoPrinter Pass
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
namespace {
|
||||
class DomInfoPrinter : public FunctionPass {
|
||||
public:
|
||||
static char ID; // Pass identification, replacement for typeid
|
||||
DomInfoPrinter() : FunctionPass(ID) {}
|
||||
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.setPreservesAll();
|
||||
AU.addRequired<DominatorTree>();
|
||||
|
||||
}
|
||||
|
||||
virtual bool runOnFunction(Function &F) {
|
||||
getAnalysis<DominatorTree>().dump();
|
||||
return false;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
char DomInfoPrinter::ID = 0;
|
||||
static RegisterPass<DomInfoPrinter>
|
||||
DIP("print-dom-info", "Dominator Info Printer", true, true);
|
14
contrib/llvm/tools/opt/Makefile
Normal file
14
contrib/llvm/tools/opt/Makefile
Normal file
@ -0,0 +1,14 @@
|
||||
##===- tools/opt/Makefile ----------------------------------*- Makefile -*-===##
|
||||
#
|
||||
# The LLVM Compiler Infrastructure
|
||||
#
|
||||
# This file is distributed under the University of Illinois Open Source
|
||||
# License. See LICENSE.TXT for details.
|
||||
#
|
||||
##===----------------------------------------------------------------------===##
|
||||
LEVEL = ../..
|
||||
TOOLNAME = opt
|
||||
|
||||
LINK_COMPONENTS := bitreader bitwriter asmparser instrumentation scalaropts ipo
|
||||
|
||||
include $(LEVEL)/Makefile.common
|
112
contrib/llvm/tools/opt/PrintSCC.cpp
Normal file
112
contrib/llvm/tools/opt/PrintSCC.cpp
Normal file
@ -0,0 +1,112 @@
|
||||
//===- PrintSCC.cpp - Enumerate SCCs in some key graphs -------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This file provides passes to print out SCCs in a CFG or a CallGraph.
|
||||
// Normally, you would not use these passes; instead, you would use the
|
||||
// scc_iterator directly to enumerate SCCs and process them in some way. These
|
||||
// passes serve three purposes:
|
||||
//
|
||||
// (1) As a reference for how to use the scc_iterator.
|
||||
// (2) To print out the SCCs for a CFG or a CallGraph:
|
||||
// analyze -print-cfg-sccs to print the SCCs in each CFG of a module.
|
||||
// analyze -print-cfg-sccs -stats to print the #SCCs and the maximum SCC size.
|
||||
// analyze -print-cfg-sccs -debug > /dev/null to watch the algorithm in action.
|
||||
//
|
||||
// and similarly:
|
||||
// analyze -print-callgraph-sccs [-stats] [-debug] to print SCCs in the CallGraph
|
||||
//
|
||||
// (3) To test the scc_iterator.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/Pass.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/Analysis/CallGraph.h"
|
||||
#include "llvm/Support/CFG.h"
|
||||
#include "llvm/Support/raw_ostream.h"
|
||||
#include "llvm/ADT/SCCIterator.h"
|
||||
using namespace llvm;
|
||||
|
||||
namespace {
|
||||
struct CFGSCC : public FunctionPass {
|
||||
static char ID; // Pass identification, replacement for typeid
|
||||
CFGSCC() : FunctionPass(ID) {}
|
||||
bool runOnFunction(Function& func);
|
||||
|
||||
void print(raw_ostream &O, const Module* = 0) const { }
|
||||
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.setPreservesAll();
|
||||
}
|
||||
};
|
||||
|
||||
struct CallGraphSCC : public ModulePass {
|
||||
static char ID; // Pass identification, replacement for typeid
|
||||
CallGraphSCC() : ModulePass(ID) {}
|
||||
|
||||
// run - Print out SCCs in the call graph for the specified module.
|
||||
bool runOnModule(Module &M);
|
||||
|
||||
void print(raw_ostream &O, const Module* = 0) const { }
|
||||
|
||||
// getAnalysisUsage - This pass requires the CallGraph.
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.setPreservesAll();
|
||||
AU.addRequired<CallGraph>();
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
char CFGSCC::ID = 0;
|
||||
static RegisterPass<CFGSCC>
|
||||
Y("print-cfg-sccs", "Print SCCs of each function CFG");
|
||||
|
||||
char CallGraphSCC::ID = 0;
|
||||
static RegisterPass<CallGraphSCC>
|
||||
Z("print-callgraph-sccs", "Print SCCs of the Call Graph");
|
||||
|
||||
bool CFGSCC::runOnFunction(Function &F) {
|
||||
unsigned sccNum = 0;
|
||||
errs() << "SCCs for Function " << F.getName() << " in PostOrder:";
|
||||
for (scc_iterator<Function*> SCCI = scc_begin(&F),
|
||||
E = scc_end(&F); SCCI != E; ++SCCI) {
|
||||
std::vector<BasicBlock*> &nextSCC = *SCCI;
|
||||
errs() << "\nSCC #" << ++sccNum << " : ";
|
||||
for (std::vector<BasicBlock*>::const_iterator I = nextSCC.begin(),
|
||||
E = nextSCC.end(); I != E; ++I)
|
||||
errs() << (*I)->getName() << ", ";
|
||||
if (nextSCC.size() == 1 && SCCI.hasLoop())
|
||||
errs() << " (Has self-loop).";
|
||||
}
|
||||
errs() << "\n";
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
// run - Print out SCCs in the call graph for the specified module.
|
||||
bool CallGraphSCC::runOnModule(Module &M) {
|
||||
CallGraphNode* rootNode = getAnalysis<CallGraph>().getRoot();
|
||||
unsigned sccNum = 0;
|
||||
errs() << "SCCs for the program in PostOrder:";
|
||||
for (scc_iterator<CallGraphNode*> SCCI = scc_begin(rootNode),
|
||||
E = scc_end(rootNode); SCCI != E; ++SCCI) {
|
||||
const std::vector<CallGraphNode*> &nextSCC = *SCCI;
|
||||
errs() << "\nSCC #" << ++sccNum << " : ";
|
||||
for (std::vector<CallGraphNode*>::const_iterator I = nextSCC.begin(),
|
||||
E = nextSCC.end(); I != E; ++I)
|
||||
errs() << ((*I)->getFunction() ? (*I)->getFunction()->getNameStr()
|
||||
: std::string("external node")) << ", ";
|
||||
if (nextSCC.size() == 1 && SCCI.hasLoop())
|
||||
errs() << " (Has self-loop).";
|
||||
}
|
||||
errs() << "\n";
|
||||
|
||||
return true;
|
||||
}
|
713
contrib/llvm/tools/opt/opt.cpp
Normal file
713
contrib/llvm/tools/opt/opt.cpp
Normal file
@ -0,0 +1,713 @@
|
||||
//===- opt.cpp - The LLVM Modular Optimizer -------------------------------===//
|
||||
//
|
||||
// The LLVM Compiler Infrastructure
|
||||
//
|
||||
// This file is distributed under the University of Illinois Open Source
|
||||
// License. See LICENSE.TXT for details.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// Optimizations may be specified an arbitrary number of times on the command
|
||||
// line, They are run in the order specified.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/LLVMContext.h"
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/PassManager.h"
|
||||
#include "llvm/CallGraphSCCPass.h"
|
||||
#include "llvm/Bitcode/ReaderWriter.h"
|
||||
#include "llvm/Assembly/PrintModulePass.h"
|
||||
#include "llvm/Analysis/DebugInfo.h"
|
||||
#include "llvm/Analysis/Verifier.h"
|
||||
#include "llvm/Analysis/LoopPass.h"
|
||||
#include "llvm/Analysis/RegionPass.h"
|
||||
#include "llvm/Analysis/CallGraph.h"
|
||||
#include "llvm/Target/TargetData.h"
|
||||
#include "llvm/Target/TargetLibraryInfo.h"
|
||||
#include "llvm/Target/TargetMachine.h"
|
||||
#include "llvm/ADT/StringSet.h"
|
||||
#include "llvm/ADT/Triple.h"
|
||||
#include "llvm/Support/PassNameParser.h"
|
||||
#include "llvm/Support/Signals.h"
|
||||
#include "llvm/Support/Debug.h"
|
||||
#include "llvm/Support/IRReader.h"
|
||||
#include "llvm/Support/ManagedStatic.h"
|
||||
#include "llvm/Support/PluginLoader.h"
|
||||
#include "llvm/Support/PrettyStackTrace.h"
|
||||
#include "llvm/Support/SystemUtils.h"
|
||||
#include "llvm/Support/ToolOutputFile.h"
|
||||
#include "llvm/LinkAllPasses.h"
|
||||
#include "llvm/LinkAllVMCore.h"
|
||||
#include "llvm/Transforms/IPO/PassManagerBuilder.h"
|
||||
#include <memory>
|
||||
#include <algorithm>
|
||||
using namespace llvm;
|
||||
|
||||
// The OptimizationList is automatically populated with registered Passes by the
|
||||
// PassNameParser.
|
||||
//
|
||||
static cl::list<const PassInfo*, bool, PassNameParser>
|
||||
PassList(cl::desc("Optimizations available:"));
|
||||
|
||||
// Other command line options...
|
||||
//
|
||||
static cl::opt<std::string>
|
||||
InputFilename(cl::Positional, cl::desc("<input bitcode file>"),
|
||||
cl::init("-"), cl::value_desc("filename"));
|
||||
|
||||
static cl::opt<std::string>
|
||||
OutputFilename("o", cl::desc("Override output filename"),
|
||||
cl::value_desc("filename"));
|
||||
|
||||
static cl::opt<bool>
|
||||
Force("f", cl::desc("Enable binary output on terminals"));
|
||||
|
||||
static cl::opt<bool>
|
||||
PrintEachXForm("p", cl::desc("Print module after each transformation"));
|
||||
|
||||
static cl::opt<bool>
|
||||
NoOutput("disable-output",
|
||||
cl::desc("Do not write result bitcode file"), cl::Hidden);
|
||||
|
||||
static cl::opt<bool>
|
||||
OutputAssembly("S", cl::desc("Write output as LLVM assembly"));
|
||||
|
||||
static cl::opt<bool>
|
||||
NoVerify("disable-verify", cl::desc("Do not verify result module"), cl::Hidden);
|
||||
|
||||
static cl::opt<bool>
|
||||
VerifyEach("verify-each", cl::desc("Verify after each transform"));
|
||||
|
||||
static cl::opt<bool>
|
||||
StripDebug("strip-debug",
|
||||
cl::desc("Strip debugger symbol info from translation unit"));
|
||||
|
||||
static cl::opt<bool>
|
||||
DisableInline("disable-inlining", cl::desc("Do not run the inliner pass"));
|
||||
|
||||
static cl::opt<bool>
|
||||
DisableOptimizations("disable-opt",
|
||||
cl::desc("Do not run any optimization passes"));
|
||||
|
||||
static cl::opt<bool>
|
||||
DisableInternalize("disable-internalize",
|
||||
cl::desc("Do not mark all symbols as internal"));
|
||||
|
||||
static cl::opt<bool>
|
||||
StandardCompileOpts("std-compile-opts",
|
||||
cl::desc("Include the standard compile time optimizations"));
|
||||
|
||||
static cl::opt<bool>
|
||||
StandardLinkOpts("std-link-opts",
|
||||
cl::desc("Include the standard link time optimizations"));
|
||||
|
||||
static cl::opt<bool>
|
||||
OptLevelO1("O1",
|
||||
cl::desc("Optimization level 1. Similar to llvm-gcc -O1"));
|
||||
|
||||
static cl::opt<bool>
|
||||
OptLevelO2("O2",
|
||||
cl::desc("Optimization level 2. Similar to llvm-gcc -O2"));
|
||||
|
||||
static cl::opt<bool>
|
||||
OptLevelO3("O3",
|
||||
cl::desc("Optimization level 3. Similar to llvm-gcc -O3"));
|
||||
|
||||
static cl::opt<bool>
|
||||
UnitAtATime("funit-at-a-time",
|
||||
cl::desc("Enable IPO. This is same as llvm-gcc's -funit-at-a-time"),
|
||||
cl::init(true));
|
||||
|
||||
static cl::opt<bool>
|
||||
DisableSimplifyLibCalls("disable-simplify-libcalls",
|
||||
cl::desc("Disable simplify-libcalls"));
|
||||
|
||||
static cl::opt<bool>
|
||||
Quiet("q", cl::desc("Obsolete option"), cl::Hidden);
|
||||
|
||||
static cl::alias
|
||||
QuietA("quiet", cl::desc("Alias for -q"), cl::aliasopt(Quiet));
|
||||
|
||||
static cl::opt<bool>
|
||||
AnalyzeOnly("analyze", cl::desc("Only perform analysis, no optimization"));
|
||||
|
||||
static cl::opt<bool>
|
||||
PrintBreakpoints("print-breakpoints-for-testing",
|
||||
cl::desc("Print select breakpoints location for testing"));
|
||||
|
||||
static cl::opt<std::string>
|
||||
DefaultDataLayout("default-data-layout",
|
||||
cl::desc("data layout string to use if not specified by module"),
|
||||
cl::value_desc("layout-string"), cl::init(""));
|
||||
|
||||
// ---------- Define Printers for module and function passes ------------
|
||||
namespace {
|
||||
|
||||
struct CallGraphSCCPassPrinter : public CallGraphSCCPass {
|
||||
static char ID;
|
||||
const PassInfo *PassToPrint;
|
||||
raw_ostream &Out;
|
||||
std::string PassName;
|
||||
|
||||
CallGraphSCCPassPrinter(const PassInfo *PI, raw_ostream &out) :
|
||||
CallGraphSCCPass(ID), PassToPrint(PI), Out(out) {
|
||||
std::string PassToPrintName = PassToPrint->getPassName();
|
||||
PassName = "CallGraphSCCPass Printer: " + PassToPrintName;
|
||||
}
|
||||
|
||||
virtual bool runOnSCC(CallGraphSCC &SCC) {
|
||||
if (!Quiet)
|
||||
Out << "Printing analysis '" << PassToPrint->getPassName() << "':\n";
|
||||
|
||||
// Get and print pass...
|
||||
for (CallGraphSCC::iterator I = SCC.begin(), E = SCC.end(); I != E; ++I) {
|
||||
Function *F = (*I)->getFunction();
|
||||
if (F)
|
||||
getAnalysisID<Pass>(PassToPrint->getTypeInfo()).print(Out,
|
||||
F->getParent());
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
virtual const char *getPassName() const { return PassName.c_str(); }
|
||||
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.addRequiredID(PassToPrint->getTypeInfo());
|
||||
AU.setPreservesAll();
|
||||
}
|
||||
};
|
||||
|
||||
char CallGraphSCCPassPrinter::ID = 0;
|
||||
|
||||
struct ModulePassPrinter : public ModulePass {
|
||||
static char ID;
|
||||
const PassInfo *PassToPrint;
|
||||
raw_ostream &Out;
|
||||
std::string PassName;
|
||||
|
||||
ModulePassPrinter(const PassInfo *PI, raw_ostream &out)
|
||||
: ModulePass(ID), PassToPrint(PI), Out(out) {
|
||||
std::string PassToPrintName = PassToPrint->getPassName();
|
||||
PassName = "ModulePass Printer: " + PassToPrintName;
|
||||
}
|
||||
|
||||
virtual bool runOnModule(Module &M) {
|
||||
if (!Quiet)
|
||||
Out << "Printing analysis '" << PassToPrint->getPassName() << "':\n";
|
||||
|
||||
// Get and print pass...
|
||||
getAnalysisID<Pass>(PassToPrint->getTypeInfo()).print(Out, &M);
|
||||
return false;
|
||||
}
|
||||
|
||||
virtual const char *getPassName() const { return PassName.c_str(); }
|
||||
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.addRequiredID(PassToPrint->getTypeInfo());
|
||||
AU.setPreservesAll();
|
||||
}
|
||||
};
|
||||
|
||||
char ModulePassPrinter::ID = 0;
|
||||
struct FunctionPassPrinter : public FunctionPass {
|
||||
const PassInfo *PassToPrint;
|
||||
raw_ostream &Out;
|
||||
static char ID;
|
||||
std::string PassName;
|
||||
|
||||
FunctionPassPrinter(const PassInfo *PI, raw_ostream &out)
|
||||
: FunctionPass(ID), PassToPrint(PI), Out(out) {
|
||||
std::string PassToPrintName = PassToPrint->getPassName();
|
||||
PassName = "FunctionPass Printer: " + PassToPrintName;
|
||||
}
|
||||
|
||||
virtual bool runOnFunction(Function &F) {
|
||||
if (!Quiet)
|
||||
Out << "Printing analysis '" << PassToPrint->getPassName()
|
||||
<< "' for function '" << F.getName() << "':\n";
|
||||
|
||||
// Get and print pass...
|
||||
getAnalysisID<Pass>(PassToPrint->getTypeInfo()).print(Out,
|
||||
F.getParent());
|
||||
return false;
|
||||
}
|
||||
|
||||
virtual const char *getPassName() const { return PassName.c_str(); }
|
||||
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.addRequiredID(PassToPrint->getTypeInfo());
|
||||
AU.setPreservesAll();
|
||||
}
|
||||
};
|
||||
|
||||
char FunctionPassPrinter::ID = 0;
|
||||
|
||||
struct LoopPassPrinter : public LoopPass {
|
||||
static char ID;
|
||||
const PassInfo *PassToPrint;
|
||||
raw_ostream &Out;
|
||||
std::string PassName;
|
||||
|
||||
LoopPassPrinter(const PassInfo *PI, raw_ostream &out) :
|
||||
LoopPass(ID), PassToPrint(PI), Out(out) {
|
||||
std::string PassToPrintName = PassToPrint->getPassName();
|
||||
PassName = "LoopPass Printer: " + PassToPrintName;
|
||||
}
|
||||
|
||||
|
||||
virtual bool runOnLoop(Loop *L, LPPassManager &LPM) {
|
||||
if (!Quiet)
|
||||
Out << "Printing analysis '" << PassToPrint->getPassName() << "':\n";
|
||||
|
||||
// Get and print pass...
|
||||
getAnalysisID<Pass>(PassToPrint->getTypeInfo()).print(Out,
|
||||
L->getHeader()->getParent()->getParent());
|
||||
return false;
|
||||
}
|
||||
|
||||
virtual const char *getPassName() const { return PassName.c_str(); }
|
||||
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.addRequiredID(PassToPrint->getTypeInfo());
|
||||
AU.setPreservesAll();
|
||||
}
|
||||
};
|
||||
|
||||
char LoopPassPrinter::ID = 0;
|
||||
|
||||
struct RegionPassPrinter : public RegionPass {
|
||||
static char ID;
|
||||
const PassInfo *PassToPrint;
|
||||
raw_ostream &Out;
|
||||
std::string PassName;
|
||||
|
||||
RegionPassPrinter(const PassInfo *PI, raw_ostream &out) : RegionPass(ID),
|
||||
PassToPrint(PI), Out(out) {
|
||||
std::string PassToPrintName = PassToPrint->getPassName();
|
||||
PassName = "RegionPass Printer: " + PassToPrintName;
|
||||
}
|
||||
|
||||
virtual bool runOnRegion(Region *R, RGPassManager &RGM) {
|
||||
if (!Quiet) {
|
||||
Out << "Printing analysis '" << PassToPrint->getPassName() << "' for "
|
||||
<< "region: '" << R->getNameStr() << "' in function '"
|
||||
<< R->getEntry()->getParent()->getNameStr() << "':\n";
|
||||
}
|
||||
// Get and print pass...
|
||||
getAnalysisID<Pass>(PassToPrint->getTypeInfo()).print(Out,
|
||||
R->getEntry()->getParent()->getParent());
|
||||
return false;
|
||||
}
|
||||
|
||||
virtual const char *getPassName() const { return PassName.c_str(); }
|
||||
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.addRequiredID(PassToPrint->getTypeInfo());
|
||||
AU.setPreservesAll();
|
||||
}
|
||||
};
|
||||
|
||||
char RegionPassPrinter::ID = 0;
|
||||
|
||||
struct BasicBlockPassPrinter : public BasicBlockPass {
|
||||
const PassInfo *PassToPrint;
|
||||
raw_ostream &Out;
|
||||
static char ID;
|
||||
std::string PassName;
|
||||
|
||||
BasicBlockPassPrinter(const PassInfo *PI, raw_ostream &out)
|
||||
: BasicBlockPass(ID), PassToPrint(PI), Out(out) {
|
||||
std::string PassToPrintName = PassToPrint->getPassName();
|
||||
PassName = "BasicBlockPass Printer: " + PassToPrintName;
|
||||
}
|
||||
|
||||
virtual bool runOnBasicBlock(BasicBlock &BB) {
|
||||
if (!Quiet)
|
||||
Out << "Printing Analysis info for BasicBlock '" << BB.getName()
|
||||
<< "': Pass " << PassToPrint->getPassName() << ":\n";
|
||||
|
||||
// Get and print pass...
|
||||
getAnalysisID<Pass>(PassToPrint->getTypeInfo()).print(Out,
|
||||
BB.getParent()->getParent());
|
||||
return false;
|
||||
}
|
||||
|
||||
virtual const char *getPassName() const { return PassName.c_str(); }
|
||||
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.addRequiredID(PassToPrint->getTypeInfo());
|
||||
AU.setPreservesAll();
|
||||
}
|
||||
};
|
||||
|
||||
char BasicBlockPassPrinter::ID = 0;
|
||||
|
||||
struct BreakpointPrinter : public ModulePass {
|
||||
raw_ostream &Out;
|
||||
static char ID;
|
||||
|
||||
BreakpointPrinter(raw_ostream &out)
|
||||
: ModulePass(ID), Out(out) {
|
||||
}
|
||||
|
||||
void getContextName(DIDescriptor Context, std::string &N) {
|
||||
if (Context.isNameSpace()) {
|
||||
DINameSpace NS(Context);
|
||||
if (!NS.getName().empty()) {
|
||||
getContextName(NS.getContext(), N);
|
||||
N = N + NS.getName().str() + "::";
|
||||
}
|
||||
} else if (Context.isType()) {
|
||||
DIType TY(Context);
|
||||
if (!TY.getName().empty()) {
|
||||
getContextName(TY.getContext(), N);
|
||||
N = N + TY.getName().str() + "::";
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
virtual bool runOnModule(Module &M) {
|
||||
StringSet<> Processed;
|
||||
if (NamedMDNode *NMD = M.getNamedMetadata("llvm.dbg.sp"))
|
||||
for (unsigned i = 0, e = NMD->getNumOperands(); i != e; ++i) {
|
||||
std::string Name;
|
||||
DISubprogram SP(NMD->getOperand(i));
|
||||
if (SP.Verify())
|
||||
getContextName(SP.getContext(), Name);
|
||||
Name = Name + SP.getDisplayName().str();
|
||||
if (!Name.empty() && Processed.insert(Name)) {
|
||||
Out << Name << "\n";
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.setPreservesAll();
|
||||
}
|
||||
};
|
||||
|
||||
} // anonymous namespace
|
||||
|
||||
char BreakpointPrinter::ID = 0;
|
||||
|
||||
static inline void addPass(PassManagerBase &PM, Pass *P) {
|
||||
// Add the pass to the pass manager...
|
||||
PM.add(P);
|
||||
|
||||
// If we are verifying all of the intermediate steps, add the verifier...
|
||||
if (VerifyEach) PM.add(createVerifierPass());
|
||||
}
|
||||
|
||||
/// AddOptimizationPasses - This routine adds optimization passes
|
||||
/// based on selected optimization level, OptLevel. This routine
|
||||
/// duplicates llvm-gcc behaviour.
|
||||
///
|
||||
/// OptLevel - Optimization Level
|
||||
static void AddOptimizationPasses(PassManagerBase &MPM,FunctionPassManager &FPM,
|
||||
unsigned OptLevel) {
|
||||
PassManagerBuilder Builder;
|
||||
Builder.OptLevel = OptLevel;
|
||||
|
||||
if (DisableInline) {
|
||||
// No inlining pass
|
||||
} else if (OptLevel > 1) {
|
||||
unsigned Threshold = 225;
|
||||
if (OptLevel > 2)
|
||||
Threshold = 275;
|
||||
Builder.Inliner = createFunctionInliningPass(Threshold);
|
||||
} else {
|
||||
Builder.Inliner = createAlwaysInlinerPass();
|
||||
}
|
||||
Builder.DisableUnitAtATime = !UnitAtATime;
|
||||
Builder.DisableUnrollLoops = OptLevel == 0;
|
||||
Builder.DisableSimplifyLibCalls = DisableSimplifyLibCalls;
|
||||
|
||||
Builder.populateFunctionPassManager(FPM);
|
||||
Builder.populateModulePassManager(MPM);
|
||||
}
|
||||
|
||||
static void AddStandardCompilePasses(PassManagerBase &PM) {
|
||||
PM.add(createVerifierPass()); // Verify that input is correct
|
||||
|
||||
// If the -strip-debug command line option was specified, do it.
|
||||
if (StripDebug)
|
||||
addPass(PM, createStripSymbolsPass(true));
|
||||
|
||||
if (DisableOptimizations) return;
|
||||
|
||||
// -std-compile-opts adds the same module passes as -O3.
|
||||
PassManagerBuilder Builder;
|
||||
if (!DisableInline)
|
||||
Builder.Inliner = createFunctionInliningPass();
|
||||
Builder.OptLevel = 3;
|
||||
Builder.DisableSimplifyLibCalls = DisableSimplifyLibCalls;
|
||||
Builder.populateModulePassManager(PM);
|
||||
}
|
||||
|
||||
static void AddStandardLinkPasses(PassManagerBase &PM) {
|
||||
PM.add(createVerifierPass()); // Verify that input is correct
|
||||
|
||||
// If the -strip-debug command line option was specified, do it.
|
||||
if (StripDebug)
|
||||
addPass(PM, createStripSymbolsPass(true));
|
||||
|
||||
if (DisableOptimizations) return;
|
||||
|
||||
PassManagerBuilder Builder;
|
||||
Builder.populateLTOPassManager(PM, /*Internalize=*/ !DisableInternalize,
|
||||
/*RunInliner=*/ !DisableInline);
|
||||
}
|
||||
|
||||
|
||||
//===----------------------------------------------------------------------===//
|
||||
// main for opt
|
||||
//
|
||||
int main(int argc, char **argv) {
|
||||
sys::PrintStackTraceOnErrorSignal();
|
||||
llvm::PrettyStackTraceProgram X(argc, argv);
|
||||
|
||||
// Enable debug stream buffering.
|
||||
EnableDebugBuffering = true;
|
||||
|
||||
llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
|
||||
LLVMContext &Context = getGlobalContext();
|
||||
|
||||
// Initialize passes
|
||||
PassRegistry &Registry = *PassRegistry::getPassRegistry();
|
||||
initializeCore(Registry);
|
||||
initializeScalarOpts(Registry);
|
||||
initializeIPO(Registry);
|
||||
initializeAnalysis(Registry);
|
||||
initializeIPA(Registry);
|
||||
initializeTransformUtils(Registry);
|
||||
initializeInstCombine(Registry);
|
||||
initializeInstrumentation(Registry);
|
||||
initializeTarget(Registry);
|
||||
|
||||
cl::ParseCommandLineOptions(argc, argv,
|
||||
"llvm .bc -> .bc modular optimizer and analysis printer\n");
|
||||
|
||||
if (AnalyzeOnly && NoOutput) {
|
||||
errs() << argv[0] << ": analyze mode conflicts with no-output mode.\n";
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Allocate a full target machine description only if necessary.
|
||||
// FIXME: The choice of target should be controllable on the command line.
|
||||
std::auto_ptr<TargetMachine> target;
|
||||
|
||||
SMDiagnostic Err;
|
||||
|
||||
// Load the input module...
|
||||
std::auto_ptr<Module> M;
|
||||
M.reset(ParseIRFile(InputFilename, Err, Context));
|
||||
|
||||
if (M.get() == 0) {
|
||||
Err.Print(argv[0], errs());
|
||||
return 1;
|
||||
}
|
||||
|
||||
// Figure out what stream we are supposed to write to...
|
||||
OwningPtr<tool_output_file> Out;
|
||||
if (NoOutput) {
|
||||
if (!OutputFilename.empty())
|
||||
errs() << "WARNING: The -o (output filename) option is ignored when\n"
|
||||
"the --disable-output option is used.\n";
|
||||
} else {
|
||||
// Default to standard output.
|
||||
if (OutputFilename.empty())
|
||||
OutputFilename = "-";
|
||||
|
||||
std::string ErrorInfo;
|
||||
Out.reset(new tool_output_file(OutputFilename.c_str(), ErrorInfo,
|
||||
raw_fd_ostream::F_Binary));
|
||||
if (!ErrorInfo.empty()) {
|
||||
errs() << ErrorInfo << '\n';
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
|
||||
// If the output is set to be emitted to standard out, and standard out is a
|
||||
// console, print out a warning message and refuse to do it. We don't
|
||||
// impress anyone by spewing tons of binary goo to a terminal.
|
||||
if (!Force && !NoOutput && !AnalyzeOnly && !OutputAssembly)
|
||||
if (CheckBitcodeOutputToConsole(Out->os(), !Quiet))
|
||||
NoOutput = true;
|
||||
|
||||
// Create a PassManager to hold and optimize the collection of passes we are
|
||||
// about to build.
|
||||
//
|
||||
PassManager Passes;
|
||||
|
||||
// Add an appropriate TargetLibraryInfo pass for the module's triple.
|
||||
TargetLibraryInfo *TLI = new TargetLibraryInfo(Triple(M->getTargetTriple()));
|
||||
|
||||
// The -disable-simplify-libcalls flag actually disables all builtin optzns.
|
||||
if (DisableSimplifyLibCalls)
|
||||
TLI->disableAllFunctions();
|
||||
Passes.add(TLI);
|
||||
|
||||
// Add an appropriate TargetData instance for this module.
|
||||
TargetData *TD = 0;
|
||||
const std::string &ModuleDataLayout = M.get()->getDataLayout();
|
||||
if (!ModuleDataLayout.empty())
|
||||
TD = new TargetData(ModuleDataLayout);
|
||||
else if (!DefaultDataLayout.empty())
|
||||
TD = new TargetData(DefaultDataLayout);
|
||||
|
||||
if (TD)
|
||||
Passes.add(TD);
|
||||
|
||||
OwningPtr<FunctionPassManager> FPasses;
|
||||
if (OptLevelO1 || OptLevelO2 || OptLevelO3) {
|
||||
FPasses.reset(new FunctionPassManager(M.get()));
|
||||
if (TD)
|
||||
FPasses->add(new TargetData(*TD));
|
||||
}
|
||||
|
||||
if (PrintBreakpoints) {
|
||||
// Default to standard output.
|
||||
if (!Out) {
|
||||
if (OutputFilename.empty())
|
||||
OutputFilename = "-";
|
||||
|
||||
std::string ErrorInfo;
|
||||
Out.reset(new tool_output_file(OutputFilename.c_str(), ErrorInfo,
|
||||
raw_fd_ostream::F_Binary));
|
||||
if (!ErrorInfo.empty()) {
|
||||
errs() << ErrorInfo << '\n';
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
Passes.add(new BreakpointPrinter(Out->os()));
|
||||
NoOutput = true;
|
||||
}
|
||||
|
||||
// If the -strip-debug command line option was specified, add it. If
|
||||
// -std-compile-opts was also specified, it will handle StripDebug.
|
||||
if (StripDebug && !StandardCompileOpts)
|
||||
addPass(Passes, createStripSymbolsPass(true));
|
||||
|
||||
// Create a new optimization pass for each one specified on the command line
|
||||
for (unsigned i = 0; i < PassList.size(); ++i) {
|
||||
// Check to see if -std-compile-opts was specified before this option. If
|
||||
// so, handle it.
|
||||
if (StandardCompileOpts &&
|
||||
StandardCompileOpts.getPosition() < PassList.getPosition(i)) {
|
||||
AddStandardCompilePasses(Passes);
|
||||
StandardCompileOpts = false;
|
||||
}
|
||||
|
||||
if (StandardLinkOpts &&
|
||||
StandardLinkOpts.getPosition() < PassList.getPosition(i)) {
|
||||
AddStandardLinkPasses(Passes);
|
||||
StandardLinkOpts = false;
|
||||
}
|
||||
|
||||
if (OptLevelO1 && OptLevelO1.getPosition() < PassList.getPosition(i)) {
|
||||
AddOptimizationPasses(Passes, *FPasses, 1);
|
||||
OptLevelO1 = false;
|
||||
}
|
||||
|
||||
if (OptLevelO2 && OptLevelO2.getPosition() < PassList.getPosition(i)) {
|
||||
AddOptimizationPasses(Passes, *FPasses, 2);
|
||||
OptLevelO2 = false;
|
||||
}
|
||||
|
||||
if (OptLevelO3 && OptLevelO3.getPosition() < PassList.getPosition(i)) {
|
||||
AddOptimizationPasses(Passes, *FPasses, 3);
|
||||
OptLevelO3 = false;
|
||||
}
|
||||
|
||||
const PassInfo *PassInf = PassList[i];
|
||||
Pass *P = 0;
|
||||
if (PassInf->getNormalCtor())
|
||||
P = PassInf->getNormalCtor()();
|
||||
else
|
||||
errs() << argv[0] << ": cannot create pass: "
|
||||
<< PassInf->getPassName() << "\n";
|
||||
if (P) {
|
||||
PassKind Kind = P->getPassKind();
|
||||
addPass(Passes, P);
|
||||
|
||||
if (AnalyzeOnly) {
|
||||
switch (Kind) {
|
||||
case PT_BasicBlock:
|
||||
Passes.add(new BasicBlockPassPrinter(PassInf, Out->os()));
|
||||
break;
|
||||
case PT_Region:
|
||||
Passes.add(new RegionPassPrinter(PassInf, Out->os()));
|
||||
break;
|
||||
case PT_Loop:
|
||||
Passes.add(new LoopPassPrinter(PassInf, Out->os()));
|
||||
break;
|
||||
case PT_Function:
|
||||
Passes.add(new FunctionPassPrinter(PassInf, Out->os()));
|
||||
break;
|
||||
case PT_CallGraphSCC:
|
||||
Passes.add(new CallGraphSCCPassPrinter(PassInf, Out->os()));
|
||||
break;
|
||||
default:
|
||||
Passes.add(new ModulePassPrinter(PassInf, Out->os()));
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (PrintEachXForm)
|
||||
Passes.add(createPrintModulePass(&errs()));
|
||||
}
|
||||
|
||||
// If -std-compile-opts was specified at the end of the pass list, add them.
|
||||
if (StandardCompileOpts) {
|
||||
AddStandardCompilePasses(Passes);
|
||||
StandardCompileOpts = false;
|
||||
}
|
||||
|
||||
if (StandardLinkOpts) {
|
||||
AddStandardLinkPasses(Passes);
|
||||
StandardLinkOpts = false;
|
||||
}
|
||||
|
||||
if (OptLevelO1)
|
||||
AddOptimizationPasses(Passes, *FPasses, 1);
|
||||
|
||||
if (OptLevelO2)
|
||||
AddOptimizationPasses(Passes, *FPasses, 2);
|
||||
|
||||
if (OptLevelO3)
|
||||
AddOptimizationPasses(Passes, *FPasses, 3);
|
||||
|
||||
if (OptLevelO1 || OptLevelO2 || OptLevelO3) {
|
||||
FPasses->doInitialization();
|
||||
for (Module::iterator F = M->begin(), E = M->end(); F != E; ++F)
|
||||
FPasses->run(*F);
|
||||
FPasses->doFinalization();
|
||||
}
|
||||
|
||||
// Check that the module is well formed on completion of optimization
|
||||
if (!NoVerify && !VerifyEach)
|
||||
Passes.add(createVerifierPass());
|
||||
|
||||
// Write bitcode or assembly to the output as the last step...
|
||||
if (!NoOutput && !AnalyzeOnly) {
|
||||
if (OutputAssembly)
|
||||
Passes.add(createPrintModulePass(&Out->os()));
|
||||
else
|
||||
Passes.add(createBitcodeWriterPass(Out->os()));
|
||||
}
|
||||
|
||||
// Before executing passes, print the final values of the LLVM options.
|
||||
cl::PrintOptionValues();
|
||||
|
||||
// Now that we have all of the passes ready, run them.
|
||||
Passes.run(*M.get());
|
||||
|
||||
// Declare success.
|
||||
if (!NoOutput || PrintBreakpoints)
|
||||
Out->keep();
|
||||
|
||||
return 0;
|
||||
}
|
@ -1,5 +1,7 @@
|
||||
# $FreeBSD$
|
||||
|
||||
.include <bsd.own.mk>
|
||||
|
||||
.if !make(install)
|
||||
SUBDIR= libclanganalysis \
|
||||
libclangarcmigrate \
|
||||
@ -60,6 +62,19 @@ SUBDIR= libclanganalysis \
|
||||
libllvmx86info \
|
||||
libllvmx86instprinter \
|
||||
libllvmx86utils
|
||||
|
||||
.if ${MK_CLANG_EXTRAS} != "no"
|
||||
SUBDIR+=libllvmarchive \
|
||||
libllvmdebuginfo \
|
||||
libllvmexecutionengine \
|
||||
libllvminterpreter \
|
||||
libllvmjit \
|
||||
libllvmlinker \
|
||||
libllvmmcdisassembler \
|
||||
libllvmmcjit \
|
||||
libllvmobject \
|
||||
libllvmruntimedyld
|
||||
.endif
|
||||
.endif
|
||||
|
||||
SUBDIR+= include
|
||||
|
@ -1,5 +1,7 @@
|
||||
# $FreeBSD$
|
||||
|
||||
.include <bsd.own.mk>
|
||||
|
||||
LIB= llvmanalysis
|
||||
|
||||
SRCDIR= lib/Analysis
|
||||
@ -55,6 +57,14 @@ SRCS= AliasAnalysis.cpp \
|
||||
TypeBasedAliasAnalysis.cpp \
|
||||
ValueTracking.cpp
|
||||
|
||||
.if ${MK_CLANG_EXTRAS} != "no"
|
||||
SRCS+= BlockFrequencyInfo.cpp \
|
||||
LibCallSemantics.cpp \
|
||||
PathNumbering.cpp \
|
||||
PathProfileInfo.cpp \
|
||||
PathProfileVerifier.cpp
|
||||
.endif
|
||||
|
||||
TGHDRS= Intrinsics
|
||||
|
||||
.include "../clang.lib.mk"
|
||||
|
10
lib/clang/libllvmarchive/Makefile
Normal file
10
lib/clang/libllvmarchive/Makefile
Normal file
@ -0,0 +1,10 @@
|
||||
# $FreeBSD$
|
||||
|
||||
LIB= llvmarchive
|
||||
|
||||
SRCDIR= lib/Archive
|
||||
SRCS= Archive.cpp \
|
||||
ArchiveReader.cpp \
|
||||
ArchiveWriter.cpp
|
||||
|
||||
.include "../clang.lib.mk"
|
17
lib/clang/libllvmdebuginfo/Makefile
Normal file
17
lib/clang/libllvmdebuginfo/Makefile
Normal file
@ -0,0 +1,17 @@
|
||||
# $FreeBSD$
|
||||
|
||||
LIB= llvmdebuginfo
|
||||
|
||||
SRCDIR= lib/DebugInfo
|
||||
SRCS= DIContext.cpp \
|
||||
DWARFAbbreviationDeclaration.cpp \
|
||||
DWARFCompileUnit.cpp \
|
||||
DWARFContext.cpp \
|
||||
DWARFDebugAbbrev.cpp \
|
||||
DWARFDebugArangeSet.cpp \
|
||||
DWARFDebugAranges.cpp \
|
||||
DWARFDebugInfoEntry.cpp \
|
||||
DWARFDebugLine.cpp \
|
||||
DWARFFormValue.cpp
|
||||
|
||||
.include "../clang.lib.mk"
|
10
lib/clang/libllvmexecutionengine/Makefile
Normal file
10
lib/clang/libllvmexecutionengine/Makefile
Normal file
@ -0,0 +1,10 @@
|
||||
# $FreeBSD$
|
||||
|
||||
LIB= llvmexecutionengine
|
||||
|
||||
SRCDIR= lib/ExecutionEngine
|
||||
SRCS= ExecutionEngine.cpp \
|
||||
ExecutionEngineBindings.cpp \
|
||||
TargetSelect.cpp
|
||||
|
||||
.include "../clang.lib.mk"
|
12
lib/clang/libllvminterpreter/Makefile
Normal file
12
lib/clang/libllvminterpreter/Makefile
Normal file
@ -0,0 +1,12 @@
|
||||
# $FreeBSD$
|
||||
|
||||
LIB= llvminterpreter
|
||||
|
||||
SRCDIR= lib/ExecutionEngine/Interpreter
|
||||
SRCS= Execution.cpp \
|
||||
ExternalFunctions.cpp \
|
||||
Interpreter.cpp \
|
||||
|
||||
TGHDRS= Intrinsics
|
||||
|
||||
.include "../clang.lib.mk"
|
@ -1,5 +1,7 @@
|
||||
# $FreeBSD$
|
||||
|
||||
.include <bsd.own.mk>
|
||||
|
||||
LIB= llvmipa
|
||||
|
||||
SRCDIR= lib/Analysis/IPA
|
||||
@ -8,6 +10,10 @@ SRCS= CallGraph.cpp \
|
||||
FindUsedTypes.cpp \
|
||||
GlobalsModRef.cpp
|
||||
|
||||
.if ${MK_CLANG_EXTRAS} != "no"
|
||||
SRCS+= IPA.cpp
|
||||
.endif
|
||||
|
||||
TGHDRS= Intrinsics
|
||||
|
||||
.include "../clang.lib.mk"
|
||||
|
@ -1,5 +1,7 @@
|
||||
# $FreeBSD$
|
||||
|
||||
.include <bsd.own.mk>
|
||||
|
||||
LIB= llvmipo
|
||||
|
||||
SRCDIR= lib/Transforms/IPO
|
||||
@ -23,6 +25,10 @@ SRCS= ArgumentPromotion.cpp \
|
||||
StripDeadPrototypes.cpp \
|
||||
StripSymbols.cpp
|
||||
|
||||
.if ${MK_CLANG_EXTRAS} != "no"
|
||||
SRCS+= IPO.cpp
|
||||
.endif
|
||||
|
||||
TGHDRS= Intrinsics
|
||||
|
||||
.include "../clang.lib.mk"
|
||||
|
14
lib/clang/libllvmjit/Makefile
Normal file
14
lib/clang/libllvmjit/Makefile
Normal file
@ -0,0 +1,14 @@
|
||||
# $FreeBSD$
|
||||
|
||||
LIB= llvmjit
|
||||
|
||||
SRCDIR= lib/ExecutionEngine/JIT
|
||||
SRCS= Intercept.cpp \
|
||||
JIT.cpp \
|
||||
JITDebugRegisterer.cpp \
|
||||
JITDwarfEmitter.cpp \
|
||||
JITEmitter.cpp \
|
||||
JITMemoryManager.cpp \
|
||||
OProfileJITEventListener.cpp
|
||||
|
||||
.include "../clang.lib.mk"
|
11
lib/clang/libllvmlinker/Makefile
Normal file
11
lib/clang/libllvmlinker/Makefile
Normal file
@ -0,0 +1,11 @@
|
||||
# $FreeBSD$
|
||||
|
||||
LIB= llvmlinker
|
||||
|
||||
SRCDIR= lib/Linker
|
||||
SRCS= LinkArchives.cpp \
|
||||
LinkItems.cpp \
|
||||
LinkModules.cpp \
|
||||
Linker.cpp
|
||||
|
||||
.include "../clang.lib.mk"
|
@ -1,5 +1,7 @@
|
||||
# $FreeBSD$
|
||||
|
||||
.include <bsd.own.mk>
|
||||
|
||||
LIB= llvmmc
|
||||
|
||||
SRCDIR= lib/MC
|
||||
@ -43,4 +45,8 @@ SRCS= ELFObjectWriter.cpp \
|
||||
WinCOFFObjectWriter.cpp \
|
||||
WinCOFFStreamer.cpp
|
||||
|
||||
.if ${MK_CLANG_EXTRAS} != "no"
|
||||
SRCS+= MCDisassembler.cpp
|
||||
.endif
|
||||
|
||||
.include "../clang.lib.mk"
|
||||
|
12
lib/clang/libllvmmcdisassembler/Makefile
Normal file
12
lib/clang/libllvmmcdisassembler/Makefile
Normal file
@ -0,0 +1,12 @@
|
||||
# $FreeBSD$
|
||||
|
||||
LIB= llvmmcdisassembler
|
||||
|
||||
SRCDIR= lib/MC/MCDisassembler
|
||||
SRCS= Disassembler.cpp \
|
||||
EDDisassembler.cpp \
|
||||
EDInst.cpp \
|
||||
EDOperand.cpp \
|
||||
EDToken.cpp
|
||||
|
||||
.include "../clang.lib.mk"
|
9
lib/clang/libllvmmcjit/Makefile
Normal file
9
lib/clang/libllvmmcjit/Makefile
Normal file
@ -0,0 +1,9 @@
|
||||
# $FreeBSD$
|
||||
|
||||
LIB= llvmmcjit
|
||||
|
||||
SRCDIR= lib/ExecutionEngine/MCJIT
|
||||
SRCS= Intercept.cpp \
|
||||
MCJIT.cpp
|
||||
|
||||
.include "../clang.lib.mk"
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user