From a75dd4dc56946195b4fd4e5cbc359059477a62db Mon Sep 17 00:00:00 2001 From: Owen Anderson Date: Tue, 12 Jun 2007 00:50:47 +0000 Subject: [PATCH] Fix a few more bugs, including an instance of walking in reverse topological rather than topological order. This fixes a testcase extracted from llvm-test. llvm-svn: 37550 --- llvm/lib/Transforms/Scalar/GVNPRE.cpp | 80 ++++++++++++++++----------- 1 file changed, 49 insertions(+), 31 deletions(-) diff --git a/llvm/lib/Transforms/Scalar/GVNPRE.cpp b/llvm/lib/Transforms/Scalar/GVNPRE.cpp index c43603e6ddc6..e11f214b1342 100644 --- a/llvm/lib/Transforms/Scalar/GVNPRE.cpp +++ b/llvm/lib/Transforms/Scalar/GVNPRE.cpp @@ -43,14 +43,21 @@ struct ExprLT { if (BinaryOperator* rightBO = dyn_cast(right)) return cmpBinaryOperator(leftBO, rightBO); else - return left < right; + if (isa(right)) { + return false; + } else { + return true; + } } else if (CmpInst* leftCmp = dyn_cast(left)) { if (CmpInst* rightCmp = dyn_cast(right)) return cmpComparison(leftCmp, rightCmp); else - return left < right; + return true; } else { - return left < right; + if (isa(right) || isa(right)) + return false; + else + return left < right; } } @@ -92,7 +99,7 @@ namespace { typedef std::map ValueTable; ValueTable VN; std::set MS; - std::set createdExpressions; + std::vector createdExpressions; virtual void getAnalysisUsage(AnalysisUsage &AU) const { AU.setPreservesCFG(); @@ -109,9 +116,9 @@ namespace { Value* find_leader(std::set& vals, Value* v); Value* phi_translate(std::set& set, - Value* V, BasicBlock* pred); - void phi_translate_set(std::set& anticIn, BasicBlock* B, - std::set& out); + Value* V, BasicBlock* pred, BasicBlock* succ); + void phi_translate_set(std::set& anticIn, BasicBlock* pred, + BasicBlock* succ, std::set& out); void topo_sort(std::set& set, std::vector& vec); @@ -162,7 +169,7 @@ Value* GVNPRE::find_leader(std::set& vals, Value* v) { } Value* GVNPRE::phi_translate(std::set& set, - Value* V, BasicBlock* pred) { + Value* V, BasicBlock* pred, BasicBlock* succ) { if (V == 0) return 0; @@ -170,7 +177,7 @@ Value* GVNPRE::phi_translate(std::set& set, Value* newOp1 = isa(BO->getOperand(0)) ? phi_translate(set, find_leader(set, BO->getOperand(0)), - pred) + pred, succ) : BO->getOperand(0); if (newOp1 == 0) return 0; @@ -178,7 +185,7 @@ Value* GVNPRE::phi_translate(std::set& set, Value* newOp2 = isa(BO->getOperand(1)) ? phi_translate(set, find_leader(set, BO->getOperand(1)), - pred) + pred, succ) : BO->getOperand(1); if (newOp2 == 0) return 0; @@ -192,7 +199,7 @@ Value* GVNPRE::phi_translate(std::set& set, nextValueNumber++; if (!find_leader(set, newVal)) { DOUT << "Creating value: " << std::hex << newVal << std::dec << "\n"; - createdExpressions.insert(newVal); + createdExpressions.push_back(newVal); return newVal; } else { ValueTable::iterator I = VN.find(newVal); @@ -208,13 +215,13 @@ Value* GVNPRE::phi_translate(std::set& set, } } } else if (PHINode* P = dyn_cast(V)) { - if (P->getParent() == pred->getTerminator()->getSuccessor(0)) + if (P->getParent() == succ) return P->getIncomingValueForBlock(pred); } else if (CmpInst* C = dyn_cast(V)) { Value* newOp1 = isa(C->getOperand(0)) ? phi_translate(set, find_leader(set, C->getOperand(0)), - pred) + pred, succ) : C->getOperand(0); if (newOp1 == 0) return 0; @@ -222,7 +229,7 @@ Value* GVNPRE::phi_translate(std::set& set, Value* newOp2 = isa(C->getOperand(1)) ? phi_translate(set, find_leader(set, C->getOperand(1)), - pred) + pred, succ) : C->getOperand(1); if (newOp2 == 0) return 0; @@ -237,7 +244,7 @@ Value* GVNPRE::phi_translate(std::set& set, nextValueNumber++; if (!find_leader(set, newVal)) { DOUT << "Creating value: " << std::hex << newVal << std::dec << "\n"; - createdExpressions.insert(newVal); + createdExpressions.push_back(newVal); return newVal; } else { ValueTable::iterator I = VN.find(newVal); @@ -257,11 +264,12 @@ Value* GVNPRE::phi_translate(std::set& set, return V; } -void GVNPRE::phi_translate_set(std::set& anticIn, BasicBlock* B, - std::set& out) { +void GVNPRE::phi_translate_set(std::set& anticIn, + BasicBlock* pred, BasicBlock* succ, + std::set& out) { for (std::set::iterator I = anticIn.begin(), E = anticIn.end(); I != E; ++I) { - Value* V = phi_translate(anticIn, *I, B); + Value* V = phi_translate(anticIn, *I, pred, succ); if (V != 0) out.insert(V); } @@ -544,10 +552,12 @@ bool GVNPRE::runOnFunction(Function &F) { if (BB->getTerminator()->getNumSuccessors() == 1) { if (visited.find(BB->getTerminator()->getSuccessor(0)) == visited.end()) - phi_translate_set(MS, BB, anticOut); + phi_translate_set(MS, BB, BB->getTerminator()->getSuccessor(0), + anticOut); else - phi_translate_set( - anticipatedIn[BB->getTerminator()->getSuccessor(0)], BB, anticOut); + phi_translate_set(anticipatedIn[BB->getTerminator()->getSuccessor(0)], + BB, BB->getTerminator()->getSuccessor(0), + anticOut); } else if (BB->getTerminator()->getNumSuccessors() > 1) { BasicBlock* first = BB->getTerminator()->getSuccessor(0); anticOut.insert(anticipatedIn[first].begin(), @@ -673,9 +683,8 @@ bool GVNPRE::runOnFunction(Function &F) { dump_unique(anticIn); DOUT << "\n"; - while (!workList.empty()) { - Value* e = workList.back(); - workList.pop_back(); + for (unsigned i = 0; i < workList.size(); ++i) { + Value* e = workList[i]; if (isa(e) || isa(e)) { if (find_leader(availableOut[DI->getIDom()->getBlock()], e) != 0) @@ -687,7 +696,7 @@ bool GVNPRE::runOnFunction(Function &F) { for (pred_iterator PI = pred_begin(BB), PE = pred_end(BB); PI != PE; ++PI) { - Value *e2 = phi_translate(anticIn, e, *PI); + Value *e2 = phi_translate(anticIn, e, *PI, BB); Value *e3 = find_leader(availableOut[*PI], e2); if (e3 == 0) { @@ -720,13 +729,21 @@ bool GVNPRE::runOnFunction(Function &F) { Value* s1 = 0; if (isa(U->getOperand(0))) - s1 = find_leader(availableOut[*PI], U->getOperand(0)); + s1 = find_leader(availableOut[*PI], + phi_translate(availableOut[*PI], + U->getOperand(0), + *PI, BB) + ); else s1 = U->getOperand(0); Value* s2 = 0; if (isa(U->getOperand(1))) - s2 = find_leader(availableOut[*PI], U->getOperand(1)); + s2 = find_leader(availableOut[*PI], + phi_translate(availableOut[*PI], + U->getOperand(1), + *PI, BB) + ); else s2 = U->getOperand(1); @@ -842,7 +859,6 @@ bool GVNPRE::runOnFunction(Function &F) { while (!replace.empty()) { std::pair rep = replace.back(); replace.pop_back(); - rep.first->replaceAllUsesWith(rep.second); } @@ -851,9 +867,11 @@ bool GVNPRE::runOnFunction(Function &F) { (*I)->eraseFromParent(); // Phase 4: Cleanup - for (std::set::iterator I = createdExpressions.begin(), - E = createdExpressions.end(); I != E; ++I) { - delete *I; + while (!createdExpressions.empty()) { + Instruction* I = createdExpressions.back(); + createdExpressions.pop_back(); + + delete I; } return false;