You've already forked linux-packaging-mono
acceptance-tests
data
debian
docs
external
Newtonsoft.Json
api-doc-tools
api-snapshot
aspnetwebstack
binary-reference-assemblies
bockbuild
boringssl
cecil
cecil-legacy
corefx
corert
helix-binaries
ikdasm
ikvm
illinker-test-assets
linker
llvm
bindings
cmake
docs
examples
include
lib
projects
resources
runtimes
scripts
test
tools
bugpoint
BugDriver.cpp
BugDriver.h
CMakeLists.txt
CrashDebugger.cpp
ExecutionDriver.cpp
ExtractFunction.cpp
FindBugs.cpp
LLVMBuild.txt
ListReducer.h
Miscompilation.cpp
OptimizerDriver.cpp
ToolRunner.cpp
ToolRunner.h
bugpoint.cpp
bugpoint-passes
dsymutil
gold
llc
lli
llvm-ar
llvm-as
llvm-as-fuzzer
llvm-bcanalyzer
llvm-c-test
llvm-cat
llvm-cfi-verify
llvm-config
llvm-cov
llvm-cvtres
llvm-cxxdump
llvm-cxxfilt
llvm-demangle-fuzzer
llvm-diff
llvm-dis
llvm-dwarfdump
llvm-dwp
llvm-extract
llvm-go
llvm-isel-fuzzer
llvm-jitlistener
llvm-link
llvm-lto
llvm-lto2
llvm-mc
llvm-mc-assemble-fuzzer
llvm-mc-disassemble-fuzzer
llvm-mcmarkup
llvm-modextract
llvm-mt
llvm-nm
llvm-objcopy
llvm-objdump
llvm-opt-fuzzer
llvm-opt-report
llvm-pdbutil
llvm-profdata
llvm-rc
llvm-readobj
llvm-rtdyld
llvm-shlib
llvm-size
llvm-special-case-list-fuzzer
llvm-split
llvm-stress
llvm-strings
llvm-symbolizer
llvm-xray
lto
msbuild
obj2yaml
opt
opt-viewer
sancov
sanstats
verify-uselistorder
xcode-toolchain
yaml2obj
CMakeLists.txt
LLVMBuild.txt
unittests
utils
.arcconfig
.clang-format
.clang-tidy
.gitattributes
.gitignore
CMakeLists.txt
CODE_OWNERS.TXT
CREDITS.TXT
LICENSE.TXT
LLVMBuild.txt
README.txt
RELEASE_TESTERS.TXT
configure
llvm.spec.in
nuget-buildtasks
nunit-lite
roslyn-binaries
rx
xunit-binaries
how-to-bump-roslyn-binaries.md
ikvm-native
libgc
llvm
m4
man
mcs
mk
mono
msvc
netcore
po
runtime
samples
scripts
support
tools
COPYING.LIB
LICENSE
Makefile.am
Makefile.in
NEWS
README.md
acinclude.m4
aclocal.m4
autogen.sh
code_of_conduct.md
compile
config.guess
config.h.in
config.rpath
config.sub
configure.REMOVED.git-id
configure.ac.REMOVED.git-id
depcomp
install-sh
ltmain.sh.REMOVED.git-id
missing
mkinstalldirs
mono-uninstalled.pc.in
test-driver
winconfig.h
288 lines
9.5 KiB
C++
288 lines
9.5 KiB
C++
![]() |
//===- 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/Bitcode/BitcodeWriter.h"
|
||
|
#include "llvm/IR/DataLayout.h"
|
||
|
#include "llvm/IR/Module.h"
|
||
|
#include "llvm/Support/CommandLine.h"
|
||
|
#include "llvm/Support/Debug.h"
|
||
|
#include "llvm/Support/FileUtilities.h"
|
||
|
#include "llvm/Support/Path.h"
|
||
|
#include "llvm/Support/Program.h"
|
||
|
#include "llvm/Support/ToolOutputFile.h"
|
||
|
|
||
|
#define DONT_GET_PLUGIN_LOADER_OPTION
|
||
|
#include "llvm/Support/PluginLoader.h"
|
||
|
|
||
|
|
||
|
using namespace llvm;
|
||
|
|
||
|
#define DEBUG_TYPE "bugpoint"
|
||
|
|
||
|
namespace llvm {
|
||
|
extern cl::opt<std::string> OutputPrefix;
|
||
|
}
|
||
|
|
||
|
static cl::opt<bool> PreserveBitcodeUseListOrder(
|
||
|
"preserve-bc-uselistorder",
|
||
|
cl::desc("Preserve use-list order when writing LLVM bitcode."),
|
||
|
cl::init(true), cl::Hidden);
|
||
|
|
||
|
// ChildOutput - This option captures the name of the child output file that
|
||
|
// is set up by the parent bugpoint process
|
||
|
static cl::opt<std::string> ChildOutput("child-output", cl::ReallyHidden);
|
||
|
static cl::opt<std::string>
|
||
|
OptCmd("opt-command", cl::init(""),
|
||
|
cl::desc("Path to opt. (default: search path "
|
||
|
"for 'opt'.)"));
|
||
|
|
||
|
/// writeProgramToFile - This writes the current "Program" to the named bitcode
|
||
|
/// file. If an error occurs, true is returned.
|
||
|
///
|
||
|
static bool writeProgramToFileAux(ToolOutputFile &Out, const Module *M) {
|
||
|
WriteBitcodeToFile(M, Out.os(), PreserveBitcodeUseListOrder);
|
||
|
Out.os().close();
|
||
|
if (!Out.os().has_error()) {
|
||
|
Out.keep();
|
||
|
return false;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool BugDriver::writeProgramToFile(const std::string &Filename, int FD,
|
||
|
const Module *M) const {
|
||
|
ToolOutputFile Out(Filename, FD);
|
||
|
return writeProgramToFileAux(Out, M);
|
||
|
}
|
||
|
|
||
|
bool BugDriver::writeProgramToFile(int FD, const Module *M) const {
|
||
|
raw_fd_ostream OS(FD, /*shouldClose*/ false);
|
||
|
WriteBitcodeToFile(M, OS, PreserveBitcodeUseListOrder);
|
||
|
OS.flush();
|
||
|
if (!OS.has_error())
|
||
|
return false;
|
||
|
OS.clear_error();
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool BugDriver::writeProgramToFile(const std::string &Filename,
|
||
|
const Module *M) const {
|
||
|
std::error_code EC;
|
||
|
ToolOutputFile Out(Filename, EC, sys::fs::F_None);
|
||
|
if (!EC)
|
||
|
return writeProgramToFileAux(Out, M);
|
||
|
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;
|
||
|
for (unsigned i = 0, e = PluginLoader::getNumPlugins(); i != e; ++i) {
|
||
|
outs() << " -load " << PluginLoader::getPlugin(i);
|
||
|
}
|
||
|
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();
|
||
|
SmallString<128> UniqueFilename;
|
||
|
std::error_code EC = sys::fs::createUniqueFile(
|
||
|
OutputPrefix + "-output-%%%%%%%.bc", UniqueFilename);
|
||
|
if (EC) {
|
||
|
errs() << getToolName()
|
||
|
<< ": Error making unique filename: " << EC.message() << "\n";
|
||
|
return 1;
|
||
|
}
|
||
|
OutputFilename = UniqueFilename.str();
|
||
|
|
||
|
// set up the input file name
|
||
|
Expected<sys::fs::TempFile> Temp =
|
||
|
sys::fs::TempFile::create(OutputPrefix + "-input-%%%%%%%.bc");
|
||
|
if (!Temp) {
|
||
|
errs() << getToolName()
|
||
|
<< ": Error making unique filename: " << toString(Temp.takeError())
|
||
|
<< "\n";
|
||
|
return 1;
|
||
|
}
|
||
|
DiscardTemp Discard{*Temp};
|
||
|
raw_fd_ostream OS(Temp->FD, /*shouldClose*/ false);
|
||
|
|
||
|
WriteBitcodeToFile(Program, OS, PreserveBitcodeUseListOrder);
|
||
|
OS.flush();
|
||
|
if (OS.has_error()) {
|
||
|
errs() << "Error writing bitcode file: " << Temp->TmpName << "\n";
|
||
|
OS.clear_error();
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
std::string tool = OptCmd;
|
||
|
if (OptCmd.empty()) {
|
||
|
if (ErrorOr<std::string> Path = sys::findProgramByName("opt"))
|
||
|
tool = *Path;
|
||
|
else
|
||
|
errs() << Path.getError().message() << "\n";
|
||
|
}
|
||
|
if (tool.empty()) {
|
||
|
errs() << "Cannot find `opt' in PATH!\n";
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
std::string Prog;
|
||
|
if (UseValgrind) {
|
||
|
if (ErrorOr<std::string> Path = sys::findProgramByName("valgrind"))
|
||
|
Prog = *Path;
|
||
|
else
|
||
|
errs() << Path.getError().message() << "\n";
|
||
|
} else
|
||
|
Prog = tool;
|
||
|
if (Prog.empty()) {
|
||
|
errs() << "Cannot find `valgrind' in PATH!\n";
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
// setup the child process' arguments
|
||
|
SmallVector<const char *, 8> Args;
|
||
|
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(tool.c_str());
|
||
|
|
||
|
for (unsigned i = 0, e = OptArgs.size(); i != e; ++i)
|
||
|
Args.push_back(OptArgs[i].c_str());
|
||
|
Args.push_back("-disable-symbolication");
|
||
|
Args.push_back("-o");
|
||
|
Args.push_back(OutputFilename.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(Temp->TmpName.c_str());
|
||
|
for (unsigned i = 0; i < NumExtraArgs; ++i)
|
||
|
Args.push_back(*ExtraArgs);
|
||
|
Args.push_back(nullptr);
|
||
|
|
||
|
DEBUG(errs() << "\nAbout to run:\t";
|
||
|
for (unsigned i = 0, e = Args.size() - 1; i != e; ++i) errs()
|
||
|
<< " " << Args[i];
|
||
|
errs() << "\n";);
|
||
|
|
||
|
Optional<StringRef> Redirects[3] = {None, None, None};
|
||
|
// Redirect stdout and stderr to nowhere if SilencePasses is given.
|
||
|
if (SilencePasses) {
|
||
|
Redirects[1] = "";
|
||
|
Redirects[2] = "";
|
||
|
}
|
||
|
|
||
|
std::string ErrMsg;
|
||
|
int result = sys::ExecuteAndWait(Prog, Args.data(), nullptr, Redirects,
|
||
|
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::fs::remove(OutputFilename);
|
||
|
|
||
|
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;
|
||
|
}
|
||
|
|
||
|
std::unique_ptr<Module>
|
||
|
BugDriver::runPassesOn(Module *M, const std::vector<std::string> &Passes,
|
||
|
unsigned NumExtraArgs, const char *const *ExtraArgs) {
|
||
|
std::string BitcodeResult;
|
||
|
if (runPasses(M, Passes, BitcodeResult, false /*delete*/, true /*quiet*/,
|
||
|
NumExtraArgs, ExtraArgs)) {
|
||
|
return nullptr;
|
||
|
}
|
||
|
|
||
|
std::unique_ptr<Module> Ret = parseInputFile(BitcodeResult, Context);
|
||
|
if (!Ret) {
|
||
|
errs() << getToolName() << ": Error reading bitcode file '" << BitcodeResult
|
||
|
<< "'!\n";
|
||
|
exit(1);
|
||
|
}
|
||
|
sys::fs::remove(BitcodeResult);
|
||
|
return Ret;
|
||
|
}
|