forked from OSchip/llvm-project
Fix infinite loop in the BU algorithm. Unfortunately this dies a serious
death when handling moderately sized SCC's, but what can you do llvm-svn: 4689
This commit is contained in:
parent
773da868a0
commit
cca3599c0b
|
@ -46,27 +46,170 @@ void BUDataStructures::releaseMemory() {
|
||||||
GlobalsGraph = 0;
|
GlobalsGraph = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Return true if a graph was inlined
|
||||||
|
// Can not modify the part of the AuxCallList < FirstResolvableCall.
|
||||||
|
//
|
||||||
|
bool BUDataStructures::ResolveFunctionCalls(DSGraph &G,
|
||||||
|
unsigned &FirstResolvableCall,
|
||||||
|
std::map<Function*, DSCallSite> &InProcess,
|
||||||
|
unsigned Indent) {
|
||||||
|
std::vector<DSCallSite> &FCs = G.getAuxFunctionCalls();
|
||||||
|
bool Changed = false;
|
||||||
|
|
||||||
|
// Loop while there are call sites that we can resolve!
|
||||||
|
while (FirstResolvableCall != FCs.size()) {
|
||||||
|
DSCallSite Call = FCs[FirstResolvableCall];
|
||||||
|
|
||||||
|
// If the function list is incomplete...
|
||||||
|
if (Call.getCallee().getNode()->NodeType & DSNode::Incomplete) {
|
||||||
|
// If incomplete, we cannot resolve it, so leave it at the beginning of
|
||||||
|
// the call list with the other unresolvable calls...
|
||||||
|
++FirstResolvableCall;
|
||||||
|
} else {
|
||||||
|
// Start inlining all of the functions we can... some may not be
|
||||||
|
// inlinable if they are external...
|
||||||
|
//
|
||||||
|
const std::vector<GlobalValue*> &Callees =
|
||||||
|
Call.getCallee().getNode()->getGlobals();
|
||||||
|
|
||||||
|
bool hasExternalTarget = false;
|
||||||
|
|
||||||
|
// Loop over the functions, inlining whatever we can...
|
||||||
|
for (unsigned c = 0, e = Callees.size(); c != e; ++c) {
|
||||||
|
// Must be a function type, so this cast should succeed unless something
|
||||||
|
// really wierd is happening.
|
||||||
|
Function &FI = cast<Function>(*Callees[c]);
|
||||||
|
|
||||||
|
if (FI.getName() == "printf" || FI.getName() == "sscanf" ||
|
||||||
|
FI.getName() == "fprintf" || FI.getName() == "open" ||
|
||||||
|
FI.getName() == "sprintf" || FI.getName() == "fputs") {
|
||||||
|
// Ignore
|
||||||
|
} else if (FI.isExternal()) {
|
||||||
|
// If the function is external, then we cannot resolve this call site!
|
||||||
|
hasExternalTarget = true;
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
std::map<Function*, DSCallSite>::iterator I =
|
||||||
|
InProcess.lower_bound(&FI);
|
||||||
|
|
||||||
|
if (I != InProcess.end() && I->first == &FI) { // Recursion detected?
|
||||||
|
// Merge two call sites to eliminate recursion...
|
||||||
|
Call.mergeWith(I->second);
|
||||||
|
|
||||||
|
DEBUG(std::cerr << std::string(Indent*2, ' ')
|
||||||
|
<< "* Recursion detected for function " << FI.getName()<<"\n");
|
||||||
|
} else {
|
||||||
|
DEBUG(std::cerr << std::string(Indent*2, ' ')
|
||||||
|
<< "Inlining: " << FI.getName() << "\n");
|
||||||
|
|
||||||
|
// Get the data structure graph for the called function, closing it
|
||||||
|
// if possible...
|
||||||
|
//
|
||||||
|
DSGraph &GI = calculateGraph(FI, Indent+1); // Graph to inline
|
||||||
|
|
||||||
|
DEBUG(std::cerr << std::string(Indent*2, ' ')
|
||||||
|
<< "Got graph for: " << FI.getName() << "["
|
||||||
|
<< GI.getGraphSize() << "+"
|
||||||
|
<< GI.getAuxFunctionCalls().size() << "] "
|
||||||
|
<< " in: " << G.getFunction().getName() << "["
|
||||||
|
<< G.getGraphSize() << "+"
|
||||||
|
<< G.getAuxFunctionCalls().size() << "]\n");
|
||||||
|
|
||||||
|
// Keep track of how many call sites are added by the inlining...
|
||||||
|
unsigned NumCalls = FCs.size();
|
||||||
|
|
||||||
|
// Resolve the arguments and return value
|
||||||
|
G.mergeInGraph(Call, GI, DSGraph::StripAllocaBit |
|
||||||
|
DSGraph::DontCloneCallNodes);
|
||||||
|
|
||||||
|
// Added a call site?
|
||||||
|
if (FCs.size() != NumCalls) {
|
||||||
|
// Otherwise we need to inline the graph. Temporarily add the
|
||||||
|
// current function to the InProcess map to be able to handle
|
||||||
|
// recursion successfully.
|
||||||
|
//
|
||||||
|
I = InProcess.insert(I, std::make_pair(&FI, Call));
|
||||||
|
|
||||||
|
// ResolveFunctionCalls - Resolve the function calls that just got
|
||||||
|
// inlined...
|
||||||
|
//
|
||||||
|
Changed |= ResolveFunctionCalls(G, NumCalls, InProcess, Indent+1);
|
||||||
|
|
||||||
|
// Now that we are done processing the inlined graph, remove our
|
||||||
|
// cycle detector record...
|
||||||
|
//
|
||||||
|
//InProcess.erase(I);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hasExternalTarget) {
|
||||||
|
// If we cannot resolve this call site...
|
||||||
|
++FirstResolvableCall;
|
||||||
|
} else {
|
||||||
|
Changed = true;
|
||||||
|
FCs.erase(FCs.begin()+FirstResolvableCall);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Changed;
|
||||||
|
}
|
||||||
|
|
||||||
DSGraph &BUDataStructures::calculateGraph(Function &F, unsigned Indent) {
|
DSGraph &BUDataStructures::calculateGraph(Function &F, unsigned Indent) {
|
||||||
// Make sure this graph has not already been calculated, or that we don't get
|
// Make sure this graph has not already been calculated, or 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];
|
DSGraph *&GraphPtr = DSInfo[&F];
|
||||||
if (Graph) return *Graph;
|
if (GraphPtr) return *GraphPtr;
|
||||||
|
|
||||||
// Copy the local version into DSInfo...
|
// Copy the local version into DSInfo...
|
||||||
Graph = new DSGraph(getAnalysis<LocalDataStructures>().getDSGraph(F));
|
GraphPtr = new DSGraph(getAnalysis<LocalDataStructures>().getDSGraph(F));
|
||||||
Graph->setGlobalsGraph(GlobalsGraph);
|
DSGraph &Graph = *GraphPtr;
|
||||||
Graph->setPrintAuxCalls();
|
|
||||||
|
Graph.setGlobalsGraph(GlobalsGraph);
|
||||||
|
Graph.setPrintAuxCalls();
|
||||||
|
|
||||||
// Start resolving calls...
|
// Start resolving calls...
|
||||||
std::vector<DSCallSite> &FCs = Graph->getAuxFunctionCalls();
|
std::vector<DSCallSite> &FCs = Graph.getAuxFunctionCalls();
|
||||||
|
|
||||||
// Start with a copy of the original call sites...
|
// Start with a copy of the original call sites...
|
||||||
FCs = Graph->getFunctionCalls();
|
FCs = Graph.getFunctionCalls();
|
||||||
|
|
||||||
DEBUG(std::cerr << std::string(Indent*4, ' ')
|
DEBUG(std::cerr << std::string(Indent*2, ' ')
|
||||||
<< "[BU] Calculating graph for: " << F.getName() << "\n");
|
<< "[BU] Calculating graph for: " << F.getName() << "\n");
|
||||||
|
|
||||||
|
bool Changed;
|
||||||
|
while (1) {
|
||||||
|
unsigned FirstResolvableCall = 0;
|
||||||
|
std::map<Function *, DSCallSite> InProcess;
|
||||||
|
|
||||||
|
// Insert a call site for self to handle self recursion...
|
||||||
|
std::vector<DSNodeHandle> Args;
|
||||||
|
Args.reserve(F.asize());
|
||||||
|
for (Function::aiterator I = F.abegin(), E = F.aend(); I != E; ++I)
|
||||||
|
if (isPointerType(I->getType()))
|
||||||
|
Args.push_back(Graph.getNodeForValue(I));
|
||||||
|
|
||||||
|
InProcess.insert(std::make_pair(&F,
|
||||||
|
DSCallSite(*(CallInst*)0, Graph.getRetNode(),(DSNode*)0,Args)));
|
||||||
|
|
||||||
|
Changed = ResolveFunctionCalls(Graph, FirstResolvableCall, InProcess,
|
||||||
|
Indent);
|
||||||
|
|
||||||
|
if (Changed) {
|
||||||
|
Graph.maskIncompleteMarkers();
|
||||||
|
Graph.markIncompleteNodes();
|
||||||
|
Graph.removeDeadNodes();
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#if 0
|
||||||
bool Inlined;
|
bool Inlined;
|
||||||
do {
|
do {
|
||||||
Inlined = false;
|
Inlined = false;
|
||||||
|
@ -93,17 +236,17 @@ DSGraph &BUDataStructures::calculateGraph(Function &F, unsigned Indent) {
|
||||||
if (&FI == &F) {
|
if (&FI == &F) {
|
||||||
// Self recursion... simply link up the formal arguments with the
|
// Self recursion... simply link up the formal arguments with the
|
||||||
// actual arguments...
|
// actual arguments...
|
||||||
DEBUG(std::cerr << std::string(Indent*4, ' ')
|
DEBUG(std::cerr << std::string(Indent*2, ' ')
|
||||||
<< "[BU] Self Inlining: " << F.getName() << "\n");
|
<< "[BU] Self Inlining: " << F.getName() << "\n");
|
||||||
|
|
||||||
// Handle self recursion by resolving the arguments and return value
|
// Handle self recursion by resolving the arguments and return value
|
||||||
Graph->mergeInGraph(Call, *Graph, DSGraph::StripAllocaBit);
|
Graph.mergeInGraph(Call, Graph, DSGraph::StripAllocaBit);
|
||||||
|
|
||||||
// Erase the entry in the callees vector
|
// Erase the entry in the callees vector
|
||||||
Callees.erase(Callees.begin()+c--);
|
Callees.erase(Callees.begin()+c--);
|
||||||
|
|
||||||
} else if (!FI.isExternal()) {
|
} else if (!FI.isExternal()) {
|
||||||
DEBUG(std::cerr << std::string(Indent*4, ' ')
|
DEBUG(std::cerr << std::string(Indent*2, ' ')
|
||||||
<< "[BU] In " << F.getName() << " inlining: "
|
<< "[BU] In " << F.getName() << " inlining: "
|
||||||
<< FI.getName() << "\n");
|
<< FI.getName() << "\n");
|
||||||
|
|
||||||
|
@ -113,13 +256,13 @@ DSGraph &BUDataStructures::calculateGraph(Function &F, unsigned Indent) {
|
||||||
//
|
//
|
||||||
DSGraph &GI = calculateGraph(FI, Indent+1); // Graph to inline
|
DSGraph &GI = calculateGraph(FI, Indent+1); // Graph to inline
|
||||||
|
|
||||||
DEBUG(std::cerr << std::string(Indent*4, ' ')
|
DEBUG(std::cerr << std::string(Indent*2, ' ')
|
||||||
<< "[BU] Got graph for " << FI.getName()
|
<< "[BU] Got graph for " << FI.getName()
|
||||||
<< " in: " << F.getName() << "[" << GI.getGraphSize() << "+"
|
<< " in: " << F.getName() << "[" << GI.getGraphSize() << "+"
|
||||||
<< GI.getAuxFunctionCalls().size() << "]\n");
|
<< GI.getAuxFunctionCalls().size() << "]\n");
|
||||||
|
|
||||||
// Handle self recursion by resolving the arguments and return value
|
// Handle self recursion by resolving the arguments and return value
|
||||||
Graph->mergeInGraph(Call, GI, DSGraph::StripAllocaBit |
|
Graph.mergeInGraph(Call, GI, DSGraph::StripAllocaBit |
|
||||||
DSGraph::DontCloneCallNodes);
|
DSGraph::DontCloneCallNodes);
|
||||||
|
|
||||||
// Erase the entry in the Callees vector
|
// Erase the entry in the Callees vector
|
||||||
|
@ -150,23 +293,26 @@ DSGraph &BUDataStructures::calculateGraph(Function &F, unsigned Indent) {
|
||||||
Inlined = true;
|
Inlined = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#if 0
|
||||||
// If we just inlined a function that had call nodes, chances are that
|
// If we just inlined a function that had call nodes, chances are that
|
||||||
// the call nodes are redundant with ones we already have. Eliminate
|
// the call nodes are redundant with ones we already have. Eliminate
|
||||||
// those call nodes now.
|
// those call nodes now.
|
||||||
//
|
//
|
||||||
if (FCs.size() > OldNumCalls)
|
if (FCs.size() >= OldNumCalls)
|
||||||
Graph->removeTriviallyDeadNodes();
|
Graph.removeTriviallyDeadNodes();
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
if (FCs.size() > 200) {
|
if (FCs.size() > 200) {
|
||||||
std::cerr << "Aborted inlining fn: '" << F.getName() << "'!"
|
std::cerr << "Aborted inlining fn: '" << F.getName() << "'!"
|
||||||
<< std::endl;
|
<< std::endl;
|
||||||
Graph->maskIncompleteMarkers();
|
Graph.maskIncompleteMarkers();
|
||||||
Graph->markIncompleteNodes();
|
Graph.markIncompleteNodes();
|
||||||
Graph->removeDeadNodes();
|
Graph.removeDeadNodes();
|
||||||
Graph->writeGraphToFile(std::cerr, "crap."+F.getName());
|
Graph.writeGraphToFile(std::cerr, "crap."+F.getName());
|
||||||
exit(1);
|
exit(1);
|
||||||
return *Graph;
|
return Graph;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -174,19 +320,20 @@ DSGraph &BUDataStructures::calculateGraph(Function &F, unsigned Indent) {
|
||||||
// Recompute the Incomplete markers. If there are any function calls left
|
// Recompute the Incomplete markers. If there are any function calls left
|
||||||
// now that are complete, we must loop!
|
// now that are complete, we must loop!
|
||||||
if (Inlined) {
|
if (Inlined) {
|
||||||
Graph->maskIncompleteMarkers();
|
Graph.maskIncompleteMarkers();
|
||||||
Graph->markIncompleteNodes();
|
Graph.markIncompleteNodes();
|
||||||
Graph->removeDeadNodes();
|
Graph.removeDeadNodes();
|
||||||
}
|
}
|
||||||
|
|
||||||
} while (Inlined && !FCs.empty());
|
} while (Inlined && !FCs.empty());
|
||||||
|
#endif
|
||||||
|
|
||||||
DEBUG(std::cerr << std::string(Indent*4, ' ')
|
DEBUG(std::cerr << std::string(Indent*2, ' ')
|
||||||
<< "[BU] Done inlining: " << F.getName() << " ["
|
<< "[BU] Done inlining: " << F.getName() << " ["
|
||||||
<< Graph->getGraphSize() << "+" << Graph->getAuxFunctionCalls().size()
|
<< Graph.getGraphSize() << "+" << Graph.getAuxFunctionCalls().size()
|
||||||
<< "]\n");
|
<< "]\n");
|
||||||
|
|
||||||
//Graph->writeGraphToFile(std::cerr, "bu_" + F.getName());
|
Graph.writeGraphToFile(std::cerr, "bu_" + F.getName());
|
||||||
|
|
||||||
return *Graph;
|
return Graph;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue