mirror of
				https://github.com/c64scene-ar/llvm-6502.git
				synced 2025-10-25 10:27:04 +00:00 
			
		
		
		
	Reimplement TD pass completely it now works
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@4635 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
		| @@ -12,7 +12,6 @@ | |||||||
| #include "llvm/Module.h" | #include "llvm/Module.h" | ||||||
| #include "llvm/DerivedTypes.h" | #include "llvm/DerivedTypes.h" | ||||||
| #include "Support/Statistic.h" | #include "Support/Statistic.h" | ||||||
| #include <set> |  | ||||||
|  |  | ||||||
| static RegisterAnalysis<TDDataStructures> | static RegisterAnalysis<TDDataStructures> | ||||||
| Y("tddatastructure", "Top-down Data Structure Analysis Closure"); | Y("tddatastructure", "Top-down Data Structure Analysis Closure"); | ||||||
| @@ -21,7 +20,6 @@ Y("tddatastructure", "Top-down Data Structure Analysis Closure"); | |||||||
| // our memory... here... | // our memory... here... | ||||||
| // | // | ||||||
| void TDDataStructures::releaseMemory() { | void TDDataStructures::releaseMemory() { | ||||||
|   BUMaps.clear(); |  | ||||||
|   for (std::map<const Function*, DSGraph*>::iterator I = DSInfo.begin(), |   for (std::map<const Function*, DSGraph*>::iterator I = DSInfo.begin(), | ||||||
|          E = DSInfo.end(); I != E; ++I) |          E = DSInfo.end(); I != E; ++I) | ||||||
|     delete I->second; |     delete I->second; | ||||||
| @@ -37,21 +35,14 @@ void TDDataStructures::releaseMemory() { | |||||||
| bool TDDataStructures::run(Module &M) { | bool TDDataStructures::run(Module &M) { | ||||||
|   BUDataStructures &BU = getAnalysis<BUDataStructures>(); |   BUDataStructures &BU = getAnalysis<BUDataStructures>(); | ||||||
|  |  | ||||||
|   // Calculate the CallSitesForFunction mapping from the BU info... |   // Calculate top-down from main... | ||||||
|   for (Module::iterator I = M.begin(), E = M.end(); I != E; ++I) |   if (Function *F = M.getMainFunction()) | ||||||
|     if (!I->isExternal()) |     calculateGraph(*F); | ||||||
|       if (const std::vector<DSCallSite> *CS = BU.getCallSites(*I)) |  | ||||||
|         for (unsigned i = 0, e = CS->size(); i != e; ++i) |  | ||||||
|           if (Function *F = (*CS)[i].getResolvingCaller()) |  | ||||||
|             CallSitesForFunction[F].push_back(&(*CS)[i]); |  | ||||||
|  |  | ||||||
|   // Next calculate the graphs for each function... |   // Next calculate the graphs for each function unreachable function... | ||||||
|   for (Module::reverse_iterator I = M.rbegin(), E = M.rend(); I != E; ++I) |   for (Module::reverse_iterator I = M.rbegin(), E = M.rend(); I != E; ++I) | ||||||
|     if (!I->isExternal()) |     if (!I->isExternal()) | ||||||
|       calculateGraph(*I); |       calculateGraph(*I); | ||||||
|  |  | ||||||
|   // Destroy the temporary mapping... |  | ||||||
|   CallSitesForFunction.clear(); |  | ||||||
|   return false; |   return false; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -81,116 +72,131 @@ void TDDataStructures::ResolveCallSite(DSGraph &Graph, | |||||||
|     Graph.getRetNode().mergeWith(CallSite.getRetVal()); |     Graph.getRetNode().mergeWith(CallSite.getRetVal()); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | DSGraph &TDDataStructures::getOrCreateDSGraph(Function &F) { | ||||||
|  |   DSGraph *&G = DSInfo[&F]; | ||||||
|  |   if (G == 0) { // Not created yet?  Clone BU graph... | ||||||
|  |     G = new DSGraph(getAnalysis<BUDataStructures>().getDSGraph(F)); | ||||||
|  |     G->getAuxFunctionCalls().clear(); | ||||||
|  |   } | ||||||
|  |   return *G; | ||||||
|  | } | ||||||
|  |  | ||||||
| DSGraph &TDDataStructures::calculateGraph(Function &F) { | void TDDataStructures::calculateGraph(Function &F) { | ||||||
|   // Make sure this graph has not already been calculated, or that we don't get |   // Make sure this graph has not already been calculated, and that we don't get | ||||||
|   // into an infinite loop with mutually recursive functions. |   // into an infinite loop with mutually recursive functions. | ||||||
|   // |   // | ||||||
|   DSGraph *&Graph = DSInfo[&F]; |   if (GraphDone.count(&F)) return; | ||||||
|   if (Graph) return *Graph; |   GraphDone.insert(&F); | ||||||
|  |  | ||||||
|   BUDataStructures &BU = getAnalysis<BUDataStructures>(); |   // Get the current functions graph... | ||||||
|   DSGraph &BUGraph = BU.getDSGraph(F); |   DSGraph &Graph = getOrCreateDSGraph(F); | ||||||
|    |  | ||||||
|   // Copy the BU graph, keeping a mapping from the BUGraph to the current Graph |  | ||||||
|   std::map<const DSNode*, DSNodeHandle> BUNodeMap; |  | ||||||
|   Graph = new DSGraph(BUGraph, BUNodeMap); |  | ||||||
|  |  | ||||||
|   // We only need the BUMap entries for the nodes that are used in call sites. |   const std::vector<DSCallSite> &CallSites = Graph.getFunctionCalls(); | ||||||
|   // Calculate which nodes are needed. | #if 0 | ||||||
|   std::set<const DSNode*> NeededNodes; |   if (CallSites.empty()) { | ||||||
|   std::map<const Function*, std::vector<const DSCallSite*> >::iterator CSFFI |     DEBUG(std::cerr << "  [TD] No callees for: " << F.getName() << "\n"); | ||||||
|     = CallSitesForFunction.find(&F); |     return;  // If no call sites, the graph is the same as the BU graph! | ||||||
|   if (CSFFI == CallSitesForFunction.end()) { |  | ||||||
|     BUNodeMap.clear();  // No nodes are neccesary |  | ||||||
|   } else { |  | ||||||
|     std::vector<const DSCallSite*> &CSV = CSFFI->second; |  | ||||||
|     for (unsigned i = 0, e = CSV.size(); i != e; ++i) { |  | ||||||
|       NeededNodes.insert(CSV[i]->getRetVal().getNode()); |  | ||||||
|       for (unsigned j = 0, je = CSV[i]->getNumPtrArgs(); j != je; ++j) |  | ||||||
|         NeededNodes.insert(CSV[i]->getPtrArg(j).getNode()); |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|  | #endif | ||||||
|   // Loop through te BUNodeMap, keeping only the nodes that are "Needed" |   // Loop over all of the call sites, building a multi-map from Callees to | ||||||
|   for (std::map<const DSNode*, DSNodeHandle>::iterator I = BUNodeMap.begin(); |   // DSCallSite*'s.  With this map we can then loop over each callee, cloning | ||||||
|        I != BUNodeMap.end(); ) |   // this graph once into it, then resolving arguments. | ||||||
|     if (NeededNodes.count(I->first) && I->first)  // Keep needed nodes... |  | ||||||
|       ++I; |  | ||||||
|     else { |  | ||||||
|       std::map<const DSNode*, DSNodeHandle>::iterator J = I++; |  | ||||||
|       BUNodeMap.erase(J); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|   NeededNodes.clear();  // We are done with this temporary data structure |  | ||||||
|  |  | ||||||
|   // Convert the mapping from a node-to-node map into a node-to-nodehandle map |  | ||||||
|   BUNodeMapTy &BUMap = BUMaps[&F]; |  | ||||||
|   BUMap.insert(BUNodeMap.begin(), BUNodeMap.end()); |  | ||||||
|   BUNodeMap.clear();   // We are done with the temporary map. |  | ||||||
|  |  | ||||||
|   const std::vector<DSCallSite> *CallSitesP = BU.getCallSites(F); |  | ||||||
|   if (CallSitesP == 0) { |  | ||||||
|     DEBUG(std::cerr << "  [TD] No callers for: " << F.getName() << "\n"); |  | ||||||
|     return *Graph;  // If no call sites, the graph is the same as the BU graph! |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   // Loop over all call sites of this function, merging each one into this |  | ||||||
|   // graph. |  | ||||||
|   // |   // | ||||||
|   DEBUG(std::cerr << "  [TD] Inlining " << CallSitesP->size() |   std::multimap<Function*, const DSCallSite*> CalleeSites; | ||||||
|                   << " callers for: " << F.getName() << "\n"); |   for (unsigned i = 0, e = CallSites.size(); i != e; ++i) { | ||||||
|   const std::vector<DSCallSite> &CallSites = *CallSitesP; |     const DSCallSite &CS = CallSites[i]; | ||||||
|   for (unsigned c = 0, ce = CallSites.size(); c != ce; ++c) { |     const std::vector<GlobalValue*> Callees = | ||||||
|     const DSCallSite &CallSite = CallSites[c]; |       CS.getCallee().getNode()->getGlobals(); | ||||||
|     Function &Caller = *CallSite.getResolvingCaller(); |  | ||||||
|     assert(&Caller && !Caller.isExternal() && |  | ||||||
|            "Externals function cannot 'call'!"); |  | ||||||
|      |  | ||||||
|     DEBUG(std::cerr << "\t [TD] Inlining caller #" << c << " '" |  | ||||||
|           << Caller.getName() << "' into callee: " << F.getName() << "\n"); |  | ||||||
|      |  | ||||||
|     // Self recursion is not tracked in BU pass... |  | ||||||
|     assert(&Caller != &F && "This cannot happen!\n"); |  | ||||||
|  |  | ||||||
|     // Recursively compute the graph for the Caller.  It should be fully |     // Loop over all of the functions that this call may invoke... | ||||||
|     // resolved except if there is mutual recursion... |     for (unsigned c = 0, e = Callees.size(); c != e; ++c) | ||||||
|     // |       if (Function *F = dyn_cast<Function>(Callees[c]))  // If this is a fn... | ||||||
|     DSGraph &CG = calculateGraph(Caller);  // Graph to inline |         if (!F->isExternal())                            // If it's not external | ||||||
|      |           CalleeSites.insert(std::make_pair(F, &CS));    // Keep track of it! | ||||||
|     DEBUG(std::cerr << "\t\t[TD] Got graph for " << Caller.getName() |  | ||||||
|           << " in: " << F.getName() << "\n"); |  | ||||||
|      |  | ||||||
|     // Translate call site from having links into the BU graph |  | ||||||
|     DSCallSite CallSiteInCG(CallSite, BUMaps[&Caller]); |  | ||||||
|  |  | ||||||
|     // These two maps keep track of where scalars in the old graph _used_ |  | ||||||
|     // to point to, and of new nodes matching nodes of the old graph. |  | ||||||
|     std::map<Value*, DSNodeHandle> OldValMap; |  | ||||||
|     std::map<const DSNode*, DSNodeHandle> OldNodeMap; |  | ||||||
|  |  | ||||||
|     // FIXME: Eventually use DSGraph::mergeInGraph here... |  | ||||||
|     // Graph->mergeInGraph(CallSiteInCG, CG, false); |  | ||||||
|      |  | ||||||
|     // Clone the Caller's graph into the current graph, keeping |  | ||||||
|     // track of where scalars in the old graph _used_ to point... |  | ||||||
|     // Do this here because it only needs to happens once for each Caller! |  | ||||||
|     // Strip scalars but not allocas since they are alive in callee. |  | ||||||
|     //  |  | ||||||
|     DSNodeHandle RetVal = Graph->cloneInto(CG, OldValMap, OldNodeMap, |  | ||||||
|                                            DSGraph::KeepAllocaBit); |  | ||||||
|     ResolveCallSite(*Graph, DSCallSite(CallSiteInCG, OldNodeMap)); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Recompute the Incomplete markers and eliminate unreachable nodes. |   // Now that we have information about all of the callees, propogate the | ||||||
|   Graph->maskIncompleteMarkers(); |   // current graph into the callees. | ||||||
|   Graph->markIncompleteNodes(/*markFormals*/ !F.hasInternalLinkage() |   // | ||||||
|  |   DEBUG(std::cerr << "  [TD] Inlining '" << F.getName() << "' into " | ||||||
|  |                   << CalleeSites.size() << " callees.\n"); | ||||||
|  |  | ||||||
|  |   // Loop over all the callees... | ||||||
|  |   for (std::multimap<Function*, const DSCallSite*>::iterator | ||||||
|  |          I = CalleeSites.begin(), E = CalleeSites.end(); I != E; ) | ||||||
|  |     if (I->first == &F) {  // Bottom-up pass takes care of self loops! | ||||||
|  |       ++I; | ||||||
|  |     } else { | ||||||
|  |       // For each callee... | ||||||
|  |       Function *Callee = I->first; | ||||||
|  |       DSGraph &CG = getOrCreateDSGraph(*Callee);  // Get the callee's graph... | ||||||
|  |        | ||||||
|  |       DEBUG(std::cerr << "\t [TD] Inlining into callee '" << Callee->getName() | ||||||
|  |             << "'\n"); | ||||||
|  |        | ||||||
|  |       // Clone our current graph into the callee... | ||||||
|  |       std::map<Value*, DSNodeHandle> OldValMap; | ||||||
|  |       std::map<const DSNode*, DSNodeHandle> OldNodeMap; | ||||||
|  |       CG.cloneInto(Graph, OldValMap, OldNodeMap, | ||||||
|  |                    DSGraph::KeepAllocaBit | DSGraph::DontCloneCallNodes); | ||||||
|  |       OldValMap.clear();  // We don't care about the ValMap | ||||||
|  |        | ||||||
|  |       // Loop over all of the invocation sites of the callee, resolving | ||||||
|  |       // arguments to our graph.  This loop may iterate multiple times if the | ||||||
|  |       // current function calls this callee multiple times with different | ||||||
|  |       // signatures. | ||||||
|  |       // | ||||||
|  |       for (; I != E && I->first == Callee; ++I) { | ||||||
|  |         // Map call site into callee graph | ||||||
|  |         DSCallSite NewCS(*I->second, OldNodeMap); | ||||||
|  |          | ||||||
|  |         // Resolve the return values... | ||||||
|  |         NewCS.getRetVal().mergeWith(CG.getRetNode()); | ||||||
|  |          | ||||||
|  |         // Resolve all of the arguments... | ||||||
|  |         Function::aiterator AI = Callee->abegin(); | ||||||
|  |         for (unsigned i = 0, e = NewCS.getNumPtrArgs(); i != e; ++i, ++AI) { | ||||||
|  |           // Advance the argument iterator to the first pointer argument... | ||||||
|  |           while (!DS::isPointerType(AI->getType())) { | ||||||
|  |             ++AI; | ||||||
|  | #ifndef NDEBUG | ||||||
|  |             if (AI == Callee->aend()) | ||||||
|  |               std::cerr << "Bad call to Function: " << Callee->getName()<< "\n"; | ||||||
|  | #endif | ||||||
|  |             assert(AI != Callee->aend() && | ||||||
|  |                    "# Args provided is not # Args required!"); | ||||||
|  |           } | ||||||
|  |            | ||||||
|  |           // Add the link from the argument scalar to the provided value | ||||||
|  |           DSNodeHandle &NH = CG.getNodeForValue(AI); | ||||||
|  |           assert(NH.getNode() && "Pointer argument without scalarmap entry?"); | ||||||
|  |           NH.mergeWith(NewCS.getPtrArg(i)); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // Done with the nodemap... | ||||||
|  |       OldNodeMap.clear(); | ||||||
|  |  | ||||||
|  |       // Recompute the Incomplete markers and eliminate unreachable nodes. | ||||||
|  |       CG.maskIncompleteMarkers(); | ||||||
|  |       CG.markIncompleteNodes(/*markFormals*/ !F.hasInternalLinkage() | ||||||
|                              /*&& FIXME: NEED TO CHECK IF ALL CALLERS FOUND!*/); |                              /*&& FIXME: NEED TO CHECK IF ALL CALLERS FOUND!*/); | ||||||
|   Graph->removeDeadNodes(/*KeepAllGlobals*/ false, /*KeepCalls*/ false); |       CG.removeTriviallyDeadNodes(false); | ||||||
|  |       CG.removeDeadNodes(false, true) ;///*KeepAllGlobals*/ false, true); | ||||||
|  |       ///*KeepCalls*/ false); | ||||||
|  |     } | ||||||
|  |  | ||||||
|   DEBUG(std::cerr << "  [TD] Done inlining callers for: " << F.getName() << " [" |   DEBUG(std::cerr << "  [TD] Done inlining into callees for: " << F.getName() | ||||||
|         << Graph->getGraphSize() << "+" << Graph->getFunctionCalls().size() |         << " [" << Graph.getGraphSize() << "+" | ||||||
|         << "]\n"); |         << Graph.getFunctionCalls().size() << "]\n"); | ||||||
|  |  | ||||||
|   return *Graph; |    | ||||||
|  |   // Loop over all the callees... making sure they are all resolved now... | ||||||
|  |   Function *LastFunc = 0; | ||||||
|  |   for (std::multimap<Function*, const DSCallSite*>::iterator | ||||||
|  |          I = CalleeSites.begin(), E = CalleeSites.end(); I != E; ++I) | ||||||
|  |     if (I->first != LastFunc) {  // Only visit each callee once... | ||||||
|  |       LastFunc = I->first; | ||||||
|  |       calculateGraph(*I->first); | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user