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