//===- 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"); |