diff --git a/llvm/lib/Transforms/Scalar/GuardWidening.cpp b/llvm/lib/Transforms/Scalar/GuardWidening.cpp --- a/llvm/lib/Transforms/Scalar/GuardWidening.cpp +++ b/llvm/lib/Transforms/Scalar/GuardWidening.cpp @@ -593,10 +593,35 @@ Instruction *InsertPtAtDef = getFreezeInsertPt(Orig, DT); if (!InsertPtAtDef) return new FreezeInst(Orig, "gw.freeze", InsertPt); + if (isa(Orig) || isa(Orig)) + return new FreezeInst(Orig, "gw.freeze", InsertPtAtDef); + SmallSet Visited; SmallVector Worklist; SmallSet DropPoisonFlags; SmallVector NeedFreeze; + DenseMap CacheOfFreezes; + + // A bit overloaded data structures. Visited contains constant/GV + // if we already met it. In this case CacheOfFreezes has a freeze if it is + // required. + auto handleConstantOrGlobal = [&](Use &U) { + Value *Def = U.get(); + if (!isa(Def) && !isa(Def)) + return false; + + if (Visited.insert(Def).second) { + if (isGuaranteedNotToBePoison(Def, nullptr, InsertPt, &DT)) + return true; + CacheOfFreezes[Def] = new FreezeInst(Def, Def->getName() + ".gw.fr", + getFreezeInsertPt(Def, DT)); + } + + if (CacheOfFreezes.count(Def)) + U.set(CacheOfFreezes[Def]); + return true; + }; + Worklist.push_back(Orig); while (!Worklist.empty()) { Value *V = Worklist.pop_back_val(); @@ -621,29 +646,22 @@ continue; } DropPoisonFlags.insert(I); - append_range(Worklist, I->operands()); + for (Use &U : I->operands()) + if (!handleConstantOrGlobal(U)) + Worklist.push_back(U.get()); } for (Instruction *I : DropPoisonFlags) I->dropPoisonGeneratingFlagsAndMetadata(); Value *Result = Orig; - SmallPtrSet NeedFreezeSet(NeedFreeze.begin(), NeedFreeze.end()); for (Value *V : NeedFreeze) { auto *FreezeInsertPt = getFreezeInsertPt(V, DT); FreezeInst *FI = new FreezeInst(V, V->getName() + ".gw.fr", FreezeInsertPt); ++FreezeAdded; if (V == Orig) Result = FI; - if (isa(V) || isa(V)) - V->replaceUsesWithIf( - FI, [&](const Use & U)->bool { return U.getUser() != FI; }); - else - // if it is a constant or global, just make a change only in instructions - // we visited and which are not marked as NeedFreeze itself. - V->replaceUsesWithIf(FI, [&](const Use & U)->bool { - return U.getUser() != FI && Visited.contains(U) && - !NeedFreezeSet.count(U); - }); + V->replaceUsesWithIf( + FI, [&](const Use & U)->bool { return U.getUser() != FI; }); } return Result;