mirror of
https://github.com/RPCS3/llvm.git
synced 2025-01-30 23:33:57 +00:00
Since there was interest on the mailing list, this is a utility pass that
uses DSA to make find targets of calls. It provides a very convinient interface to DSA results to do things with indirect calls, such as write a devirtualizer (which I have and may commit one of these days). git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@28545 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
6d1727cf14
commit
632cd52162
54
include/llvm/Analysis/CallTargets.h
Normal file
54
include/llvm/Analysis/CallTargets.h
Normal file
@ -0,0 +1,54 @@
|
||||
//=- llvm/Analysis/CallTargets.h - Resolve Indirect Call Targets --*- C++ -*-=//
|
||||
//
|
||||
// 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.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This pass uses DSA to map targets of all calls, and reports on if it
|
||||
// thinks it knows all targets of a given call.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#ifndef LLVM_ANALYSIS_CALLTARGETS_H
|
||||
#define LLVM_ANALYSIS_CALLTARGETS_H
|
||||
|
||||
#include "llvm/Pass.h"
|
||||
#include "llvm/Support/CallSite.h"
|
||||
|
||||
#include <set>
|
||||
#include <list>
|
||||
|
||||
namespace llvm {
|
||||
|
||||
class CallTargetFinder : public ModulePass {
|
||||
std::map<CallSite, std::vector<Function*> > IndMap;
|
||||
std::set<CallSite> CompleteSites;
|
||||
std::list<CallSite> AllSites;
|
||||
|
||||
void findIndTargets(Module &M);
|
||||
public:
|
||||
virtual bool runOnModule(Module &M);
|
||||
|
||||
virtual void getAnalysisUsage(AnalysisUsage &AU) const;
|
||||
|
||||
virtual void print(std::ostream &O, const Module *M) const;
|
||||
|
||||
// Given a CallSite, get an iterator of callees
|
||||
std::vector<Function*>::iterator begin(CallSite cs);
|
||||
std::vector<Function*>::iterator end(CallSite cs);
|
||||
|
||||
// Iterate over CallSites in program
|
||||
std::list<CallSite>::iterator cs_begin();
|
||||
std::list<CallSite>::iterator cs_end();
|
||||
|
||||
// Do we think we have complete knowledge of this site?
|
||||
// That is, do we think there are no missing callees
|
||||
bool isComplete(CallSite cs) const;
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
#endif
|
@ -22,6 +22,7 @@
|
||||
#include "llvm/Analysis/Passes.h"
|
||||
#include "llvm/Analysis/ScalarEvolution.h"
|
||||
#include "llvm/Analysis/DataStructure/DataStructure.h"
|
||||
#include "llvm/Analysis/CallTargets.h"
|
||||
#include "llvm/Function.h"
|
||||
#include <cstdlib>
|
||||
|
||||
@ -50,6 +51,7 @@ namespace {
|
||||
(void)new llvm::PostDominatorSet();
|
||||
(void)new llvm::FindUsedTypes();
|
||||
(void)new llvm::ScalarEvolution();
|
||||
(void)new llvm::CallTargetFinder();
|
||||
((llvm::Function*)0)->viewCFGOnly();
|
||||
llvm::AliasSetTracker X(*(llvm::AliasAnalysis*)0);
|
||||
X.add((llvm::Value*)0, 0); // for -print-alias-sets
|
||||
|
125
lib/Analysis/IPA/CallTargets.cpp
Normal file
125
lib/Analysis/IPA/CallTargets.cpp
Normal file
@ -0,0 +1,125 @@
|
||||
//=- lib/Analysis/IPA/CallTargets.cpp - Resolve Call Targets --*- C++ -*-=====//
|
||||
//
|
||||
// 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.
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
//
|
||||
// This pass uses DSA to map targets of all calls, and reports on if it
|
||||
// thinks it knows all targets of a given call.
|
||||
//
|
||||
// Loop over all callsites, and lookup the DSNode for that site. Pull the
|
||||
// Functions from the node as callees.
|
||||
// This is essentially a utility pass to simplify later passes that only depend
|
||||
// on call sites and callees to operate (such as a devirtualizer).
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#include "llvm/Module.h"
|
||||
#include "llvm/Instructions.h"
|
||||
#include "llvm/Analysis/DataStructure/DataStructure.h"
|
||||
#include "llvm/Analysis/DataStructure/DSGraph.h"
|
||||
#include "llvm/Analysis/CallTargets.h"
|
||||
#include "llvm/ADT/Statistic.h"
|
||||
#include <iostream>
|
||||
#include "llvm/Constants.h"
|
||||
|
||||
using namespace llvm;
|
||||
|
||||
namespace {
|
||||
Statistic<> DirCall("calltarget", "Number of direct calls");
|
||||
Statistic<> IndCall("calltarget", "Number of indirect calls");
|
||||
Statistic<> CompleteInd("calltarget", "Number of complete indirect calls");
|
||||
Statistic<> CompleteEmpty("calltarget", "Number of complete empty calls");
|
||||
|
||||
RegisterAnalysis<CallTargetFinder> X("calltarget", "Find Call Targets (uses DSA)");
|
||||
}
|
||||
|
||||
void CallTargetFinder::findIndTargets(Module &M)
|
||||
{
|
||||
TDDataStructures* T = &getAnalysis<TDDataStructures>();
|
||||
for (Module::iterator I = M.begin(), E = M.end(); I != E; ++I)
|
||||
if (!I->isExternal())
|
||||
for (Function::iterator F = I->begin(), FE = I->end(); F != FE; ++F)
|
||||
for (BasicBlock::iterator B = F->begin(), BE = F->end(); B != BE; ++B)
|
||||
if (isa<CallInst>(B) || isa<InvokeInst>(B)) {
|
||||
CallSite cs = CallSite::get(B);
|
||||
AllSites.push_back(cs);
|
||||
if (!cs.getCalledFunction()) {
|
||||
IndCall++;
|
||||
DSNode* N = T->getDSGraph(*cs.getCaller())
|
||||
.getNodeForValue(cs.getCalledValue()).getNode();
|
||||
N->addFullFunctionList(IndMap[cs]);
|
||||
if (N->isComplete() && IndMap[cs].size()) {
|
||||
CompleteSites.insert(cs);
|
||||
++CompleteInd;
|
||||
}
|
||||
if (N->isComplete() && !IndMap[cs].size()) {
|
||||
++CompleteEmpty;
|
||||
std::cerr << "Call site empty: '" << cs.getInstruction()->getName()
|
||||
<< "' In '" << cs.getInstruction()->getParent()->getParent()->getName()
|
||||
<< "'\n";
|
||||
}
|
||||
} else {
|
||||
++DirCall;
|
||||
IndMap[cs].push_back(cs.getCalledFunction());
|
||||
CompleteSites.insert(cs);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void CallTargetFinder::print(std::ostream &O, const Module *M) const
|
||||
{
|
||||
return;
|
||||
O << "[* = incomplete] CS: func list\n";
|
||||
for (std::map<CallSite, std::vector<Function*> >::const_iterator ii = IndMap.begin(),
|
||||
ee = IndMap.end(); ii != ee; ++ii) {
|
||||
if (!ii->first.getCalledFunction()) { //only print indirect
|
||||
if (!isComplete(ii->first)) {
|
||||
O << "* ";
|
||||
CallSite cs = ii->first;
|
||||
cs.getInstruction()->dump();
|
||||
O << cs.getInstruction()->getParent()->getParent()->getName() << " "
|
||||
<< cs.getInstruction()->getName() << " ";
|
||||
}
|
||||
O << ii->first.getInstruction() << ":";
|
||||
for (std::vector<Function*>::const_iterator i = ii->second.begin(),
|
||||
e = ii->second.end(); i != e; ++i) {
|
||||
O << " " << (*i)->getName();
|
||||
}
|
||||
O << "\n";
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
bool CallTargetFinder::runOnModule(Module &M) {
|
||||
findIndTargets(M);
|
||||
return false;
|
||||
}
|
||||
|
||||
void CallTargetFinder::getAnalysisUsage(AnalysisUsage &AU) const {
|
||||
AU.setPreservesAll();
|
||||
AU.addRequired<TDDataStructures>();
|
||||
}
|
||||
|
||||
std::vector<Function*>::iterator CallTargetFinder::begin(CallSite cs) {
|
||||
return IndMap[cs].begin();
|
||||
}
|
||||
|
||||
std::vector<Function*>::iterator CallTargetFinder::end(CallSite cs) {
|
||||
return IndMap[cs].end();
|
||||
}
|
||||
|
||||
bool CallTargetFinder::isComplete(CallSite cs) const {
|
||||
return CompleteSites.find(cs) != CompleteSites.end();
|
||||
}
|
||||
|
||||
std::list<CallSite>::iterator CallTargetFinder::cs_begin() {
|
||||
return AllSites.begin();
|
||||
}
|
||||
|
||||
std::list<CallSite>::iterator CallTargetFinder::cs_end() {
|
||||
return AllSites.end();
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user