blob: 02a9e64408caed2e1abce6e701d57faaf4d144a2 [file] [log] [blame]
Chris Lattner55c10582002-10-03 20:38:41 +00001//===- BottomUpClosure.cpp - Compute bottom-up interprocedural closure ----===//
Chris Lattner0d9bab82002-07-18 00:12:30 +00002//
3// This file implements the BUDataStructures class, which represents the
4// Bottom-Up Interprocedural closure of the data structure graph over the
5// program. This is useful for applications like pool allocation, but **not**
Chris Lattner55c10582002-10-03 20:38:41 +00006// applications like alias analysis.
Chris Lattner0d9bab82002-07-18 00:12:30 +00007//
8//===----------------------------------------------------------------------===//
9
10#include "llvm/Analysis/DataStructure.h"
Chris Lattner55c10582002-10-03 20:38:41 +000011#include "llvm/Analysis/DSGraph.h"
Chris Lattner0d9bab82002-07-18 00:12:30 +000012#include "llvm/Module.h"
Chris Lattnerfccd06f2002-10-01 22:33:50 +000013#include "Support/Statistic.h"
Chris Lattner0d9bab82002-07-18 00:12:30 +000014using std::map;
15
Chris Lattner1e435162002-07-26 21:12:44 +000016static RegisterAnalysis<BUDataStructures>
Vikram S. Adve355e2ca2002-07-30 22:05:22 +000017X("budatastructure", "Bottom-up Data Structure Analysis Closure");
Chris Lattner0d9bab82002-07-18 00:12:30 +000018
Chris Lattnerb1060432002-11-07 05:20:53 +000019using namespace DS;
Chris Lattner55c10582002-10-03 20:38:41 +000020
21
Chris Lattner0d9bab82002-07-18 00:12:30 +000022// releaseMemory - If the pass pipeline is done with this pass, we can release
23// our memory... here...
24//
25void BUDataStructures::releaseMemory() {
Chris Lattner613692c2002-10-17 04:24:08 +000026 // Delete all call site information
27 CallSites.clear();
28
Vikram S. Adve355e2ca2002-07-30 22:05:22 +000029 for (map<const Function*, DSGraph*>::iterator I = DSInfo.begin(),
Chris Lattner0d9bab82002-07-18 00:12:30 +000030 E = DSInfo.end(); I != E; ++I)
31 delete I->second;
32
33 // Empty map so next time memory is released, data structures are not
34 // re-deleted.
35 DSInfo.clear();
36}
37
38// run - Calculate the bottom up data structure graphs for each function in the
39// program.
40//
41bool BUDataStructures::run(Module &M) {
42 // Simply calculate the graphs for each function...
43 for (Module::iterator I = M.begin(), E = M.end(); I != E; ++I)
44 if (!I->isExternal())
45 calculateGraph(*I);
46 return false;
47}
48
Chris Lattner0d9bab82002-07-18 00:12:30 +000049// ResolveArguments - Resolve the formal and actual arguments for a function
50// call.
51//
Vikram S. Adve42fd1692002-10-20 18:07:37 +000052static void ResolveArguments(DSCallSite &Call, Function &F,
Chris Lattnerc875f022002-11-03 21:27:48 +000053 map<Value*, DSNodeHandle> &ScalarMap) {
Chris Lattner0d9bab82002-07-18 00:12:30 +000054 // Resolve all of the function arguments...
55 Function::aiterator AI = F.abegin();
Chris Lattner92673292002-11-02 00:13:20 +000056 for (unsigned i = 0, e = Call.getNumPtrArgs(); i != e; ++i, ++AI) {
Chris Lattner0d9bab82002-07-18 00:12:30 +000057 // Advance the argument iterator to the first pointer argument...
Chris Lattner048912b2002-11-04 02:29:15 +000058 while (!isPointerType(AI->getType())) {
59 ++AI;
60#ifndef NDEBUG
61 if (AI == F.aend())
62 std::cerr << "Bad call to Function: " << F.getName() << "\n";
63#endif
64 assert(AI != F.aend() && "# Args provided is not # Args required!");
65 }
Chris Lattner0d9bab82002-07-18 00:12:30 +000066
67 // Add the link from the argument scalar to the provided value
Chris Lattnerc875f022002-11-03 21:27:48 +000068 ScalarMap[AI].mergeWith(Call.getPtrArg(i));
Chris Lattner0d9bab82002-07-18 00:12:30 +000069 }
70}
71
Chris Lattner0d9bab82002-07-18 00:12:30 +000072DSGraph &BUDataStructures::calculateGraph(Function &F) {
73 // Make sure this graph has not already been calculated, or that we don't get
74 // into an infinite loop with mutually recursive functions.
75 //
76 DSGraph *&Graph = DSInfo[&F];
77 if (Graph) return *Graph;
78
79 // Copy the local version into DSInfo...
80 Graph = new DSGraph(getAnalysis<LocalDataStructures>().getDSGraph(F));
81
Chris Lattner55c10582002-10-03 20:38:41 +000082#if 0
Vikram S. Adve355e2ca2002-07-30 22:05:22 +000083 // Populate the GlobalsGraph with globals from this one.
84 Graph->GlobalsGraph->cloneGlobals(*Graph, /*cloneCalls*/ false);
Chris Lattner55c10582002-10-03 20:38:41 +000085#endif
Vikram S. Adve355e2ca2002-07-30 22:05:22 +000086
Chris Lattner0d9bab82002-07-18 00:12:30 +000087 // Start resolving calls...
Vikram S. Adve42fd1692002-10-20 18:07:37 +000088 std::vector<DSCallSite> &FCs = Graph->getFunctionCalls();
Chris Lattner0d9bab82002-07-18 00:12:30 +000089
Vikram S. Adve355e2ca2002-07-30 22:05:22 +000090 DEBUG(std::cerr << " [BU] Inlining: " << F.getName() << "\n");
91
Chris Lattner0d9bab82002-07-18 00:12:30 +000092 bool Inlined;
93 do {
94 Inlined = false;
Vikram S. Adve355e2ca2002-07-30 22:05:22 +000095
Chris Lattner0d9bab82002-07-18 00:12:30 +000096 for (unsigned i = 0; i != FCs.size(); ++i) {
97 // Copy the call, because inlining graphs may invalidate the FCs vector.
Vikram S. Adve42fd1692002-10-20 18:07:37 +000098 DSCallSite Call = FCs[i];
Chris Lattner0d9bab82002-07-18 00:12:30 +000099
Chris Lattner55c10582002-10-03 20:38:41 +0000100 // If the function list is complete...
Chris Lattner0969c502002-10-21 02:08:03 +0000101 if ((Call.getCallee().getNode()->NodeType & DSNode::Incomplete)==0) {
Chris Lattner0d9bab82002-07-18 00:12:30 +0000102 // Start inlining all of the functions we can... some may not be
103 // inlinable if they are external...
104 //
Chris Lattner7836d602002-10-20 22:11:17 +0000105 std::vector<GlobalValue*> Callees =
Chris Lattner0969c502002-10-21 02:08:03 +0000106 Call.getCallee().getNode()->getGlobals();
Chris Lattner0d9bab82002-07-18 00:12:30 +0000107
108 // Loop over the functions, inlining whatever we can...
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000109 for (unsigned c = 0; c != Callees.size(); ++c) {
Chris Lattner0d9bab82002-07-18 00:12:30 +0000110 // Must be a function type, so this cast MUST succeed.
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000111 Function &FI = cast<Function>(*Callees[c]);
Chris Lattner613692c2002-10-17 04:24:08 +0000112
Chris Lattner0d9bab82002-07-18 00:12:30 +0000113 if (&FI == &F) {
114 // Self recursion... simply link up the formal arguments with the
115 // actual arguments...
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000116 DEBUG(std::cerr << "\t[BU] Self Inlining: " << F.getName() << "\n");
Chris Lattner0d9bab82002-07-18 00:12:30 +0000117
Chris Lattner7836d602002-10-20 22:11:17 +0000118 // Handle the return value if present...
Chris Lattner92673292002-11-02 00:13:20 +0000119 Graph->getRetNode().mergeWith(Call.getRetVal());
Chris Lattner0d9bab82002-07-18 00:12:30 +0000120
121 // Resolve the arguments in the call to the actual values...
Chris Lattnerc875f022002-11-03 21:27:48 +0000122 ResolveArguments(Call, F, Graph->getScalarMap());
Chris Lattner0d9bab82002-07-18 00:12:30 +0000123
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000124 // Erase the entry in the callees vector
125 Callees.erase(Callees.begin()+c--);
Chris Lattner613692c2002-10-17 04:24:08 +0000126
Chris Lattner0d9bab82002-07-18 00:12:30 +0000127 } else if (!FI.isExternal()) {
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000128 DEBUG(std::cerr << "\t[BU] In " << F.getName() << " inlining: "
Chris Lattner0d9bab82002-07-18 00:12:30 +0000129 << FI.getName() << "\n");
Vikram S. Advec44e9bf2002-07-18 16:13:52 +0000130
Chris Lattner0d9bab82002-07-18 00:12:30 +0000131 // Get the data structure graph for the called function, closing it
132 // if possible (which is only impossible in the case of mutual
133 // recursion...
134 //
135 DSGraph &GI = calculateGraph(FI); // Graph to inline
136
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000137 DEBUG(std::cerr << "\t\t[BU] Got graph for " << FI.getName()
138 << " in: " << F.getName() << "\n");
Chris Lattner0d9bab82002-07-18 00:12:30 +0000139
Vikram S. Adve26b98262002-10-20 21:41:02 +0000140 // Record that the original DSCallSite was a call site of FI.
141 // This may or may not have been known when the DSCallSite was
142 // originally created.
Chris Lattner198be222002-10-21 19:47:18 +0000143 std::vector<DSCallSite> &CallSitesForFunc = CallSites[&FI];
144 CallSitesForFunc.push_back(Call);
145 CallSitesForFunc.back().setResolvingCaller(&F);
Chris Lattner9faf18d2002-10-22 15:58:46 +0000146 CallSitesForFunc.back().setCallee(0);
Chris Lattner7a0b5bb2002-11-02 00:26:32 +0000147
Vikram S. Advec44e9bf2002-07-18 16:13:52 +0000148 // Clone the callee's graph into the current graph, keeping
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000149 // track of where scalars in the old graph _used_ to point,
150 // and of the new nodes matching nodes of the old graph.
Chris Lattner55c10582002-10-03 20:38:41 +0000151 map<Value*, DSNodeHandle> OldValMap;
152 map<const DSNode*, DSNode*> OldNodeMap;
Chris Lattner0d9bab82002-07-18 00:12:30 +0000153
154 // The clone call may invalidate any of the vectors in the data
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000155 // structure graph. Strip locals and don't copy the list of callers
Chris Lattner55c10582002-10-03 20:38:41 +0000156 DSNodeHandle RetVal = Graph->cloneInto(GI, OldValMap, OldNodeMap,
Chris Lattnere4ae3042002-10-21 19:50:29 +0000157 /*StripAllocas*/ true);
Chris Lattner0d9bab82002-07-18 00:12:30 +0000158
Chris Lattner55c10582002-10-03 20:38:41 +0000159 // Resolve the arguments in the call to the actual values...
Chris Lattner0d9bab82002-07-18 00:12:30 +0000160 ResolveArguments(Call, FI, OldValMap);
161
Chris Lattner92673292002-11-02 00:13:20 +0000162 // Handle the return value if present...
163 RetVal.mergeWith(Call.getRetVal());
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000164
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000165 // Erase the entry in the Callees vector
166 Callees.erase(Callees.begin()+c--);
167
Chris Lattner9eee58d2002-07-19 18:11:43 +0000168 } else if (FI.getName() == "printf" || FI.getName() == "sscanf" ||
169 FI.getName() == "fprintf" || FI.getName() == "open" ||
170 FI.getName() == "sprintf") {
Chris Lattner92673292002-11-02 00:13:20 +0000171 // FIXME: These special cases (eg printf) should go away when we can
172 // define functions that take a variable number of arguments.
Chris Lattner9eee58d2002-07-19 18:11:43 +0000173
Chris Lattner92673292002-11-02 00:13:20 +0000174 // FIXME: at the very least, this should update mod/ref info
Chris Lattner9eee58d2002-07-19 18:11:43 +0000175 // Erase the entry in the globals vector
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000176 Callees.erase(Callees.begin()+c--);
Chris Lattner0d9bab82002-07-18 00:12:30 +0000177 }
178 }
179
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000180 if (Callees.empty()) { // Inlined all of the function calls?
Chris Lattner0d9bab82002-07-18 00:12:30 +0000181 // Erase the call if it is resolvable...
182 FCs.erase(FCs.begin()+i--); // Don't skip a the next call...
183 Inlined = true;
Chris Lattner0969c502002-10-21 02:08:03 +0000184 } else if (Callees.size() !=
185 Call.getCallee().getNode()->getGlobals().size()) {
Chris Lattner0d9bab82002-07-18 00:12:30 +0000186 // Was able to inline SOME, but not all of the functions. Construct a
187 // new global node here.
188 //
189 assert(0 && "Unimpl!");
190 Inlined = true;
191 }
192 }
193 }
194
195 // Recompute the Incomplete markers. If there are any function calls left
196 // now that are complete, we must loop!
197 if (Inlined) {
198 Graph->maskIncompleteMarkers();
199 Graph->markIncompleteNodes();
Chris Lattner55c10582002-10-03 20:38:41 +0000200 Graph->removeDeadNodes(/*KeepAllGlobals*/ true, /*KeepCalls*/ true);
Chris Lattner0d9bab82002-07-18 00:12:30 +0000201 }
202 } while (Inlined && !FCs.empty());
203
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000204 Graph->maskIncompleteMarkers();
205 Graph->markIncompleteNodes();
Chris Lattnera00397e2002-10-03 21:55:28 +0000206 Graph->removeTriviallyDeadNodes(false);
Chris Lattner55c10582002-10-03 20:38:41 +0000207 Graph->removeDeadNodes(/*KeepAllGlobals*/ true, /*KeepCalls*/ true);
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000208
Chris Lattner221c9792002-08-07 21:41:11 +0000209 DEBUG(std::cerr << " [BU] Done inlining: " << F.getName() << " ["
210 << Graph->getGraphSize() << "+" << Graph->getFunctionCalls().size()
211 << "]\n");
Vikram S. Adve355e2ca2002-07-30 22:05:22 +0000212
Chris Lattner0d9bab82002-07-18 00:12:30 +0000213 return *Graph;
214}