mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-10 02:36:06 +00:00
fd90dd5d55
Shrinkify LLVM's footprint by removing the analyze tool and moving its functionality into the opt tool. THis eliminates one of the largest tools from LLVM and doesn't make opt much bigger because it already included most of the analysis passes. To get the old analyze functionality pass the -analyze option to opt. Note that the integeration here is dead simple. The "main" of analyze was just copied to opt and invoked if the -analyze option was given. There may be opportunities for further integration such as removing the distinction between transform passes and analysis passes. To use the analysis functionality, if you previously did this: analyze $FNAME -domset -disable-verify you would now do this: opt -analyze $FNAME -domset -disable-verify Pretty simple. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@29762 91177308-0d34-0410-b5e6-96231b3b80d8
323 lines
10 KiB
C++
323 lines
10 KiB
C++
//===- opt.cpp - The LLVM Modular Optimizer -------------------------------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file was developed by the LLVM research group and 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/Module.h"
|
|
#include "llvm/Assembly/Parser.h"
|
|
#include "llvm/PassManager.h"
|
|
#include "llvm/Bytecode/Reader.h"
|
|
#include "llvm/Bytecode/WriteBytecodePass.h"
|
|
#include "llvm/Assembly/PrintModulePass.h"
|
|
#include "llvm/Analysis/Verifier.h"
|
|
#include "llvm/Target/TargetData.h"
|
|
#include "llvm/Target/TargetMachine.h"
|
|
#include "llvm/Support/PassNameParser.h"
|
|
#include "llvm/System/Signals.h"
|
|
#include "llvm/Support/PluginLoader.h"
|
|
#include "llvm/Support/SystemUtils.h"
|
|
#include "llvm/Support/Timer.h"
|
|
#include "llvm/Analysis/LinkAllAnalyses.h"
|
|
#include "llvm/Transforms/LinkAllPasses.h"
|
|
#include "llvm/LinkAllVMCore.h"
|
|
#include <fstream>
|
|
#include <memory>
|
|
#include <algorithm>
|
|
|
|
using namespace llvm;
|
|
|
|
// The OptimizationList is automatically populated with registered Passes by the
|
|
// PassNameParser.
|
|
//
|
|
static cl::list<const PassInfo*, bool,
|
|
FilteredPassNameParser<PassInfo::Optimization> >
|
|
OptimizationList(cl::desc("Optimizations available:"));
|
|
|
|
|
|
// Other command line options...
|
|
//
|
|
static cl::opt<std::string>
|
|
InputFilename(cl::Positional, cl::desc("<input bytecode file>"),
|
|
cl::init("-"), cl::value_desc("filename"));
|
|
|
|
static cl::opt<std::string>
|
|
OutputFilename("o", cl::desc("Override output filename"),
|
|
cl::value_desc("filename"), cl::init("-"));
|
|
|
|
static cl::opt<bool>
|
|
Force("f", cl::desc("Overwrite output files"));
|
|
|
|
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 bytecode file"), cl::Hidden);
|
|
|
|
static cl::opt<bool>
|
|
NoVerify("disable-verify", cl::desc("Do not verify result module"), cl::Hidden);
|
|
|
|
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"));
|
|
|
|
// The AnalysesList is automatically populated with registered Passes by the
|
|
// PassNameParser.
|
|
static
|
|
cl::list<const PassInfo*, bool, FilteredPassNameParser<PassInfo::Analysis> >
|
|
AnalysesList(cl::desc("Analyses available:"));
|
|
|
|
static Timer BytecodeLoadTimer("Bytecode Loader");
|
|
|
|
// ---------- Define Printers for module and function passes ------------
|
|
namespace {
|
|
|
|
struct ModulePassPrinter : public ModulePass {
|
|
const PassInfo *PassToPrint;
|
|
ModulePassPrinter(const PassInfo *PI) : PassToPrint(PI) {}
|
|
|
|
virtual bool runOnModule(Module &M) {
|
|
if (!Quiet) {
|
|
std::cout << "Printing analysis '" << PassToPrint->getPassName()
|
|
<< "':\n";
|
|
getAnalysisID<Pass>(PassToPrint).print(std::cout, &M);
|
|
}
|
|
|
|
// Get and print pass...
|
|
return false;
|
|
}
|
|
|
|
virtual const char *getPassName() const { return "'Pass' Printer"; }
|
|
|
|
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
|
AU.addRequiredID(PassToPrint);
|
|
AU.setPreservesAll();
|
|
}
|
|
};
|
|
|
|
struct FunctionPassPrinter : public FunctionPass {
|
|
const PassInfo *PassToPrint;
|
|
FunctionPassPrinter(const PassInfo *PI) : PassToPrint(PI) {}
|
|
|
|
virtual bool runOnFunction(Function &F) {
|
|
if (!Quiet) {
|
|
std::cout << "Printing analysis '" << PassToPrint->getPassName()
|
|
<< "' for function '" << F.getName() << "':\n";
|
|
}
|
|
// Get and print pass...
|
|
getAnalysisID<Pass>(PassToPrint).print(std::cout, F.getParent());
|
|
return false;
|
|
}
|
|
|
|
virtual const char *getPassName() const { return "FunctionPass Printer"; }
|
|
|
|
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
|
AU.addRequiredID(PassToPrint);
|
|
AU.setPreservesAll();
|
|
}
|
|
};
|
|
|
|
struct BasicBlockPassPrinter : public BasicBlockPass {
|
|
const PassInfo *PassToPrint;
|
|
BasicBlockPassPrinter(const PassInfo *PI) : PassToPrint(PI) {}
|
|
|
|
virtual bool runOnBasicBlock(BasicBlock &BB) {
|
|
if (!Quiet) {
|
|
std::cout << "Printing Analysis info for BasicBlock '" << BB.getName()
|
|
<< "': Pass " << PassToPrint->getPassName() << ":\n";
|
|
}
|
|
|
|
// Get and print pass...
|
|
getAnalysisID<Pass>(PassToPrint).print(
|
|
std::cout, BB.getParent()->getParent());
|
|
return false;
|
|
}
|
|
|
|
virtual const char *getPassName() const { return "BasicBlockPass Printer"; }
|
|
|
|
virtual void getAnalysisUsage(AnalysisUsage &AU) const {
|
|
AU.addRequiredID(PassToPrint);
|
|
AU.setPreservesAll();
|
|
}
|
|
};
|
|
|
|
} // anonymous namespace
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// main for opt
|
|
//
|
|
int main(int argc, char **argv) {
|
|
try {
|
|
cl::ParseCommandLineOptions(argc, argv,
|
|
" llvm .bc -> .bc modular optimizer and analysis printer \n");
|
|
sys::PrintStackTraceOnErrorSignal();
|
|
|
|
if (AnalyzeOnly) {
|
|
Module *CurMod = 0;
|
|
try {
|
|
#if 0
|
|
TimeRegion RegionTimer(BytecodeLoadTimer);
|
|
#endif
|
|
CurMod = ParseBytecodeFile(InputFilename);
|
|
if (!CurMod && !(CurMod = ParseAssemblyFile(InputFilename))){
|
|
std::cerr << argv[0] << ": input file didn't read correctly.\n";
|
|
return 1;
|
|
}
|
|
} catch (const ParseException &E) {
|
|
std::cerr << argv[0] << ": " << E.getMessage() << "\n";
|
|
return 1;
|
|
}
|
|
|
|
// Create a PassManager to hold and optimize the collection of passes we
|
|
// are about to build...
|
|
PassManager Passes;
|
|
|
|
// Add an appropriate TargetData instance for this module...
|
|
Passes.add(new TargetData(CurMod));
|
|
|
|
// Make sure the input LLVM is well formed.
|
|
if (!NoVerify)
|
|
Passes.add(createVerifierPass());
|
|
|
|
// Create a new optimization pass for each one specified on the
|
|
// command line
|
|
for (unsigned i = 0; i < AnalysesList.size(); ++i) {
|
|
const PassInfo *Analysis = AnalysesList[i];
|
|
|
|
if (Analysis->getNormalCtor()) {
|
|
Pass *P = Analysis->getNormalCtor()();
|
|
Passes.add(P);
|
|
|
|
if (BasicBlockPass *BBP = dynamic_cast<BasicBlockPass*>(P))
|
|
Passes.add(new BasicBlockPassPrinter(Analysis));
|
|
else if (FunctionPass *FP = dynamic_cast<FunctionPass*>(P))
|
|
Passes.add(new FunctionPassPrinter(Analysis));
|
|
else
|
|
Passes.add(new ModulePassPrinter(Analysis));
|
|
|
|
} else
|
|
std::cerr << argv[0] << ": cannot create pass: "
|
|
<< Analysis->getPassName() << "\n";
|
|
}
|
|
|
|
Passes.run(*CurMod);
|
|
|
|
delete CurMod;
|
|
return 0;
|
|
}
|
|
|
|
// 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;
|
|
|
|
TargetMachine* TM = NULL;
|
|
std::string ErrorMessage;
|
|
|
|
// Load the input module...
|
|
std::auto_ptr<Module> M(ParseBytecodeFile(InputFilename, &ErrorMessage));
|
|
if (M.get() == 0) {
|
|
std::cerr << argv[0] << ": ";
|
|
if (ErrorMessage.size())
|
|
std::cerr << ErrorMessage << "\n";
|
|
else
|
|
std::cerr << "bytecode didn't read correctly.\n";
|
|
return 1;
|
|
}
|
|
|
|
// Figure out what stream we are supposed to write to...
|
|
// FIXME: cout is not binary!
|
|
std::ostream *Out = &std::cout; // Default to printing to stdout...
|
|
if (OutputFilename != "-") {
|
|
if (!Force && std::ifstream(OutputFilename.c_str())) {
|
|
// If force is not specified, make sure not to overwrite a file!
|
|
std::cerr << argv[0] << ": error opening '" << OutputFilename
|
|
<< "': file exists!\n"
|
|
<< "Use -f command line argument to force output\n";
|
|
return 1;
|
|
}
|
|
std::ios::openmode io_mode = std::ios::out | std::ios::trunc |
|
|
std::ios::binary;
|
|
Out = new std::ofstream(OutputFilename.c_str(), io_mode);
|
|
|
|
if (!Out->good()) {
|
|
std::cerr << argv[0] << ": error opening " << OutputFilename << "!\n";
|
|
return 1;
|
|
}
|
|
|
|
// Make sure that the Output file gets unlinked from the disk if we get a
|
|
// SIGINT
|
|
sys::RemoveFileOnSignal(sys::Path(OutputFilename));
|
|
}
|
|
|
|
// 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 && CheckBytecodeOutputToConsole(Out,!Quiet)) {
|
|
NoOutput = true;
|
|
}
|
|
|
|
// Create a PassManager to hold and optimize the collection of passes we are
|
|
// about to build...
|
|
//
|
|
PassManager Passes;
|
|
|
|
// Add an appropriate TargetData instance for this module...
|
|
Passes.add(new TargetData(M.get()));
|
|
|
|
// Create a new optimization pass for each one specified on the command line
|
|
for (unsigned i = 0; i < OptimizationList.size(); ++i) {
|
|
const PassInfo *Opt = OptimizationList[i];
|
|
|
|
if (Opt->getNormalCtor())
|
|
Passes.add(Opt->getNormalCtor()());
|
|
else if (Opt->getTargetCtor()) {
|
|
#if 0
|
|
if (target.get() == NULL)
|
|
target.reset(allocateSparcTargetMachine()); // FIXME: target option
|
|
#endif
|
|
assert(target.get() && "Could not allocate target machine!");
|
|
Passes.add(Opt->getTargetCtor()(*target.get()));
|
|
} else
|
|
std::cerr << argv[0] << ": cannot create pass: " << Opt->getPassName()
|
|
<< "\n";
|
|
|
|
if (PrintEachXForm)
|
|
Passes.add(new PrintModulePass(&std::cerr));
|
|
}
|
|
|
|
// Check that the module is well formed on completion of optimization
|
|
if (!NoVerify)
|
|
Passes.add(createVerifierPass());
|
|
|
|
// Write bytecode out to disk or cout as the last step...
|
|
if (!NoOutput)
|
|
Passes.add(new WriteBytecodePass(Out, Out != &std::cout));
|
|
|
|
// Now that we have all of the passes ready, run them.
|
|
Passes.run(*M.get());
|
|
|
|
return 0;
|
|
|
|
} catch (const std::string& msg) {
|
|
std::cerr << argv[0] << ": " << msg << "\n";
|
|
} catch (...) {
|
|
std::cerr << argv[0] << ": Unexpected unknown exception occurred.\n";
|
|
}
|
|
return 1;
|
|
}
|