2002-07-18 00:12:30 +00:00
|
|
|
//===- BottomUpClosure.cpp - Compute the bottom up interprocedure closure -===//
|
|
|
|
//
|
|
|
|
// This file implements the BUDataStructures class, which represents the
|
|
|
|
// Bottom-Up Interprocedural closure of the data structure graph over the
|
|
|
|
// program. This is useful for applications like pool allocation, but **not**
|
|
|
|
// applications like pointer analysis.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "llvm/Analysis/DataStructure.h"
|
|
|
|
#include "llvm/Module.h"
|
|
|
|
#include "llvm/DerivedTypes.h"
|
|
|
|
#include "Support/StatisticReporter.h"
|
2002-07-30 22:05:22 +00:00
|
|
|
#include <set>
|
2002-07-18 00:12:30 +00:00
|
|
|
using std::map;
|
|
|
|
|
2002-07-26 21:12:44 +00:00
|
|
|
static RegisterAnalysis<BUDataStructures>
|
2002-07-30 22:05:22 +00:00
|
|
|
X("budatastructure", "Bottom-up Data Structure Analysis Closure");
|
2002-07-27 01:12:15 +00:00
|
|
|
AnalysisID BUDataStructures::ID = X;
|
2002-07-18 00:12:30 +00:00
|
|
|
|
|
|
|
// releaseMemory - If the pass pipeline is done with this pass, we can release
|
|
|
|
// our memory... here...
|
|
|
|
//
|
|
|
|
void BUDataStructures::releaseMemory() {
|
2002-07-30 22:05:22 +00:00
|
|
|
for (map<const Function*, DSGraph*>::iterator I = DSInfo.begin(),
|
2002-07-18 00:12:30 +00:00
|
|
|
E = DSInfo.end(); I != E; ++I)
|
|
|
|
delete I->second;
|
|
|
|
|
|
|
|
// Empty map so next time memory is released, data structures are not
|
|
|
|
// re-deleted.
|
|
|
|
DSInfo.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
// run - Calculate the bottom up data structure graphs for each function in the
|
|
|
|
// program.
|
|
|
|
//
|
|
|
|
bool BUDataStructures::run(Module &M) {
|
|
|
|
// Simply calculate the graphs for each function...
|
|
|
|
for (Module::iterator I = M.begin(), E = M.end(); I != E; ++I)
|
|
|
|
if (!I->isExternal())
|
|
|
|
calculateGraph(*I);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ResolveArguments - Resolve the formal and actual arguments for a function
|
|
|
|
// call.
|
|
|
|
//
|
|
|
|
static void ResolveArguments(std::vector<DSNodeHandle> &Call, Function &F,
|
|
|
|
map<Value*, DSNodeHandle> &ValueMap) {
|
|
|
|
// Resolve all of the function arguments...
|
|
|
|
Function::aiterator AI = F.abegin();
|
|
|
|
for (unsigned i = 2, e = Call.size(); i != e; ++i) {
|
|
|
|
// Advance the argument iterator to the first pointer argument...
|
|
|
|
while (!isa<PointerType>(AI->getType())) ++AI;
|
|
|
|
|
|
|
|
// Add the link from the argument scalar to the provided value
|
|
|
|
DSNode *NN = ValueMap[AI];
|
|
|
|
NN->addEdgeTo(Call[i]);
|
|
|
|
++AI;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-07-30 22:05:22 +00:00
|
|
|
// MergeGlobalNodes - Merge all existing global nodes with globals
|
|
|
|
// inlined from the callee or with globals from the GlobalsGraph.
|
2002-07-18 00:12:30 +00:00
|
|
|
//
|
2002-07-30 22:05:22 +00:00
|
|
|
static void MergeGlobalNodes(DSGraph& Graph,
|
2002-07-18 00:12:30 +00:00
|
|
|
map<Value*, DSNodeHandle> &OldValMap) {
|
2002-07-30 22:05:22 +00:00
|
|
|
map<Value*, DSNodeHandle> &ValMap = Graph.getValueMap();
|
|
|
|
for (map<Value*, DSNodeHandle>::iterator I = ValMap.begin(), E = ValMap.end();
|
|
|
|
I != E; ++I)
|
|
|
|
if (GlobalValue* GV = dyn_cast<GlobalValue>(I->first)) {
|
|
|
|
map<Value*, DSNodeHandle>:: iterator NHI = OldValMap.find(GV);
|
|
|
|
if (NHI != OldValMap.end()) // was it inlined from the callee?
|
|
|
|
I->second->mergeWith(NHI->second);
|
|
|
|
else // get it from the GlobalsGraph
|
|
|
|
I->second->mergeWith(Graph.cloneGlobalInto(GV));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add unused inlined global nodes into the value map
|
2002-07-18 00:12:30 +00:00
|
|
|
for (map<Value*, DSNodeHandle>::iterator I = OldValMap.begin(),
|
|
|
|
E = OldValMap.end(); I != E; ++I)
|
|
|
|
if (isa<GlobalValue>(I->first)) {
|
2002-07-30 22:05:22 +00:00
|
|
|
DSNodeHandle &NH = ValMap[I->first]; // If global is not in ValMap...
|
|
|
|
if (NH == 0)
|
|
|
|
NH = I->second; // Add the one just inlined.
|
2002-07-18 00:12:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
DSGraph &BUDataStructures::calculateGraph(Function &F) {
|
|
|
|
// Make sure this graph has not already been calculated, or that we don't get
|
|
|
|
// into an infinite loop with mutually recursive functions.
|
|
|
|
//
|
|
|
|
DSGraph *&Graph = DSInfo[&F];
|
|
|
|
if (Graph) return *Graph;
|
|
|
|
|
|
|
|
// Copy the local version into DSInfo...
|
|
|
|
Graph = new DSGraph(getAnalysis<LocalDataStructures>().getDSGraph(F));
|
|
|
|
|
2002-07-30 22:05:22 +00:00
|
|
|
// Populate the GlobalsGraph with globals from this one.
|
|
|
|
Graph->GlobalsGraph->cloneGlobals(*Graph, /*cloneCalls*/ false);
|
|
|
|
|
2002-07-18 16:13:52 +00:00
|
|
|
// Save a copy of the original call nodes for the top-down pass
|
|
|
|
Graph->saveOrigFunctionCalls();
|
2002-07-30 22:05:22 +00:00
|
|
|
|
2002-07-18 00:12:30 +00:00
|
|
|
// Start resolving calls...
|
|
|
|
std::vector<std::vector<DSNodeHandle> > &FCs = Graph->getFunctionCalls();
|
|
|
|
|
2002-07-30 22:05:22 +00:00
|
|
|
DEBUG(std::cerr << " [BU] Inlining: " << F.getName() << "\n");
|
|
|
|
|
|
|
|
// Add F to the PendingCallers list of each direct callee for use in the
|
|
|
|
// top-down pass so we don't have to compute this again. We don't want
|
|
|
|
// to do it for indirect callees inlined later, so remember which calls
|
|
|
|
// are in the original FCs set.
|
|
|
|
std::set<const DSNode*> directCallees;
|
|
|
|
for (unsigned i = 0; i < FCs.size(); ++i)
|
|
|
|
directCallees.insert(FCs[i][1]); // ptr to function node
|
2002-07-18 00:12:30 +00:00
|
|
|
|
|
|
|
bool Inlined;
|
|
|
|
do {
|
|
|
|
Inlined = false;
|
2002-07-30 22:05:22 +00:00
|
|
|
|
2002-07-18 00:12:30 +00:00
|
|
|
for (unsigned i = 0; i != FCs.size(); ++i) {
|
|
|
|
// Copy the call, because inlining graphs may invalidate the FCs vector.
|
|
|
|
std::vector<DSNodeHandle> Call = FCs[i];
|
|
|
|
|
|
|
|
// If the function list is not incomplete...
|
|
|
|
if ((Call[1]->NodeType & DSNode::Incomplete) == 0) {
|
|
|
|
// Start inlining all of the functions we can... some may not be
|
|
|
|
// inlinable if they are external...
|
|
|
|
//
|
2002-07-30 22:05:22 +00:00
|
|
|
std::vector<GlobalValue*> Callees(Call[1]->getGlobals());
|
2002-07-18 00:12:30 +00:00
|
|
|
|
|
|
|
// Loop over the functions, inlining whatever we can...
|
2002-07-30 22:05:22 +00:00
|
|
|
for (unsigned c = 0; c != Callees.size(); ++c) {
|
2002-07-18 00:12:30 +00:00
|
|
|
// Must be a function type, so this cast MUST succeed.
|
2002-07-30 22:05:22 +00:00
|
|
|
Function &FI = cast<Function>(*Callees[c]);
|
2002-07-18 00:12:30 +00:00
|
|
|
if (&FI == &F) {
|
|
|
|
// Self recursion... simply link up the formal arguments with the
|
|
|
|
// actual arguments...
|
2002-07-30 22:05:22 +00:00
|
|
|
|
|
|
|
DEBUG(std::cerr << "\t[BU] Self Inlining: " << F.getName() << "\n");
|
2002-07-18 00:12:30 +00:00
|
|
|
|
|
|
|
if (Call[0]) // Handle the return value if present...
|
|
|
|
Graph->RetNode->mergeWith(Call[0]);
|
|
|
|
|
|
|
|
// Resolve the arguments in the call to the actual values...
|
|
|
|
ResolveArguments(Call, F, Graph->getValueMap());
|
|
|
|
|
2002-07-30 22:05:22 +00:00
|
|
|
// Erase the entry in the callees vector
|
|
|
|
Callees.erase(Callees.begin()+c--);
|
2002-07-18 00:12:30 +00:00
|
|
|
} else if (!FI.isExternal()) {
|
2002-07-30 22:05:22 +00:00
|
|
|
DEBUG(std::cerr << "\t[BU] In " << F.getName() << " inlining: "
|
2002-07-18 00:12:30 +00:00
|
|
|
<< FI.getName() << "\n");
|
2002-07-18 16:13:52 +00:00
|
|
|
|
2002-07-18 00:12:30 +00:00
|
|
|
// Get the data structure graph for the called function, closing it
|
|
|
|
// if possible (which is only impossible in the case of mutual
|
|
|
|
// recursion...
|
|
|
|
//
|
|
|
|
DSGraph &GI = calculateGraph(FI); // Graph to inline
|
|
|
|
|
2002-07-30 22:05:22 +00:00
|
|
|
DEBUG(std::cerr << "\t\t[BU] Got graph for " << FI.getName()
|
|
|
|
<< " in: " << F.getName() << "\n");
|
2002-07-18 00:12:30 +00:00
|
|
|
|
2002-07-18 16:13:52 +00:00
|
|
|
// Clone the callee's graph into the current graph, keeping
|
2002-07-30 22:05:22 +00:00
|
|
|
// track of where scalars in the old graph _used_ to point,
|
|
|
|
// and of the new nodes matching nodes of the old graph.
|
2002-07-18 16:13:52 +00:00
|
|
|
std::map<Value*, DSNodeHandle> OldValMap;
|
2002-07-30 22:05:22 +00:00
|
|
|
std::map<const DSNode*, DSNode*> OldNodeMap;
|
2002-07-18 00:12:30 +00:00
|
|
|
|
|
|
|
// The clone call may invalidate any of the vectors in the data
|
2002-07-30 22:05:22 +00:00
|
|
|
// structure graph. Strip locals and don't copy the list of callers
|
|
|
|
DSNode *RetVal = Graph->cloneInto(GI, OldValMap, OldNodeMap,
|
|
|
|
/*StripScalars*/ true,
|
|
|
|
/*StripAllocas*/ true,
|
|
|
|
/*CopyCallers*/ false,
|
|
|
|
/*CopyOrigCalls*/ false);
|
2002-07-18 00:12:30 +00:00
|
|
|
|
|
|
|
ResolveArguments(Call, FI, OldValMap);
|
|
|
|
|
2002-07-18 01:58:24 +00:00
|
|
|
if (Call[0]) // Handle the return value if present
|
|
|
|
RetVal->mergeWith(Call[0]);
|
2002-07-30 22:05:22 +00:00
|
|
|
|
2002-07-18 00:12:30 +00:00
|
|
|
// Merge global value nodes in the inlined graph with the global
|
|
|
|
// value nodes in the current graph if there are duplicates.
|
|
|
|
//
|
2002-07-30 22:05:22 +00:00
|
|
|
MergeGlobalNodes(*Graph, OldValMap);
|
|
|
|
|
|
|
|
// If this was an original call, add F to the PendingCallers list
|
|
|
|
if (directCallees.find(Call[1]) != directCallees.end())
|
|
|
|
GI.addCaller(F);
|
|
|
|
|
|
|
|
// Erase the entry in the Callees vector
|
|
|
|
Callees.erase(Callees.begin()+c--);
|
2002-07-18 00:12:30 +00:00
|
|
|
|
2002-07-19 18:11:43 +00:00
|
|
|
} else if (FI.getName() == "printf" || FI.getName() == "sscanf" ||
|
|
|
|
FI.getName() == "fprintf" || FI.getName() == "open" ||
|
|
|
|
FI.getName() == "sprintf") {
|
|
|
|
|
2002-07-18 00:12:30 +00:00
|
|
|
// Erase the entry in the globals vector
|
2002-07-30 22:05:22 +00:00
|
|
|
Callees.erase(Callees.begin()+c--);
|
2002-07-18 00:12:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-07-30 22:05:22 +00:00
|
|
|
if (Callees.empty()) { // Inlined all of the function calls?
|
2002-07-18 00:12:30 +00:00
|
|
|
// Erase the call if it is resolvable...
|
|
|
|
FCs.erase(FCs.begin()+i--); // Don't skip a the next call...
|
|
|
|
Inlined = true;
|
2002-07-30 22:05:22 +00:00
|
|
|
} else if (Callees.size() != Call[1]->getGlobals().size()) {
|
2002-07-18 00:12:30 +00:00
|
|
|
// Was able to inline SOME, but not all of the functions. Construct a
|
|
|
|
// new global node here.
|
|
|
|
//
|
|
|
|
assert(0 && "Unimpl!");
|
|
|
|
Inlined = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Recompute the Incomplete markers. If there are any function calls left
|
|
|
|
// now that are complete, we must loop!
|
|
|
|
if (Inlined) {
|
|
|
|
Graph->maskIncompleteMarkers();
|
|
|
|
Graph->markIncompleteNodes();
|
2002-07-30 22:05:22 +00:00
|
|
|
Graph->removeDeadNodes(/*KeepAllGlobals*/ false, /*KeepCalls*/ true);
|
2002-07-18 00:12:30 +00:00
|
|
|
}
|
|
|
|
} while (Inlined && !FCs.empty());
|
|
|
|
|
2002-07-30 22:05:22 +00:00
|
|
|
// Copy any unresolved call nodes into the Globals graph and
|
|
|
|
// filter out unresolved call nodes inlined from the callee.
|
|
|
|
if (!FCs.empty())
|
|
|
|
Graph->GlobalsGraph->cloneCalls(*Graph);
|
|
|
|
|
|
|
|
Graph->maskIncompleteMarkers();
|
|
|
|
Graph->markIncompleteNodes();
|
|
|
|
Graph->removeDeadNodes(/*KeepAllGlobals*/ false, /*KeepCalls*/ false);
|
|
|
|
|
2002-07-31 19:32:12 +00:00
|
|
|
DEBUG(std::cerr << " [BU] Done inlining: " << F.getName() << "\n");
|
2002-07-30 22:05:22 +00:00
|
|
|
|
2002-07-18 00:12:30 +00:00
|
|
|
return *Graph;
|
|
|
|
}
|