xref: /llvm-project/llvm/lib/Transforms/Utils/AssumeBundleBuilder.cpp (revision da100de0a68fb3c17bf0856f432465c0721c841f)
1 //===- AssumeBundleBuilder.cpp - tools to preserve informations -*- C++ -*-===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 #include "llvm/Transforms/Utils/AssumeBundleBuilder.h"
10 #include "llvm/ADT/DepthFirstIterator.h"
11 #include "llvm/ADT/MapVector.h"
12 #include "llvm/Analysis/AssumeBundleQueries.h"
13 #include "llvm/Analysis/AssumptionCache.h"
14 #include "llvm/Analysis/ValueTracking.h"
15 #include "llvm/IR/Dominators.h"
16 #include "llvm/IR/Function.h"
17 #include "llvm/IR/InstIterator.h"
18 #include "llvm/IR/IntrinsicInst.h"
19 #include "llvm/IR/Module.h"
20 #include "llvm/InitializePasses.h"
21 #include "llvm/Support/CommandLine.h"
22 #include "llvm/Transforms/Utils/Local.h"
23 
24 using namespace llvm;
25 
26 cl::opt<bool> ShouldPreserveAllAttributes(
27     "assume-preserve-all", cl::init(false), cl::Hidden,
28     cl::desc("enable preservation of all attrbitues. even those that are "
29              "unlikely to be usefull"));
30 
31 cl::opt<bool> EnableKnowledgeRetention(
32     "enable-knowledge-retention", cl::init(false), cl::Hidden,
33     cl::desc(
34         "enable preservation of attributes throughout code transformation"));
35 
36 namespace {
37 
38 bool isUsefullToPreserve(Attribute::AttrKind Kind) {
39   switch (Kind) {
40     case Attribute::NonNull:
41     case Attribute::Alignment:
42     case Attribute::Dereferenceable:
43     case Attribute::DereferenceableOrNull:
44     case Attribute::Cold:
45       return true;
46     default:
47       return false;
48   }
49 }
50 
51 /// This class contain all knowledge that have been gather while building an
52 /// llvm.assume and the function to manipulate it.
53 struct AssumeBuilderState {
54   Module *M;
55 
56   using MapKey = std::pair<Value *, Attribute::AttrKind>;
57   SmallMapVector<MapKey, unsigned, 8> AssumedKnowledgeMap;
58   Instruction *InsertBeforeInstruction = nullptr;
59   AssumptionCache* AC = nullptr;
60   DominatorTree* DT = nullptr;
61 
62   AssumeBuilderState(Module *M, Instruction *I = nullptr,
63                      AssumptionCache *AC = nullptr, DominatorTree *DT = nullptr)
64       : M(M), InsertBeforeInstruction(I), AC(AC), DT(DT) {}
65 
66   bool tryToPreserveWithoutAddingAssume(RetainedKnowledge RK) {
67     if (!InsertBeforeInstruction || !AC || !RK.WasOn)
68       return false;
69     bool HasBeenPreserved = false;
70     Use* ToUpdate = nullptr;
71     getKnowledgeForValue(
72         RK.WasOn, {RK.AttrKind}, AC,
73         [&](RetainedKnowledge RKOther, Instruction *Assume,
74             const CallInst::BundleOpInfo *Bundle) {
75           if (!isValidAssumeForContext(Assume, InsertBeforeInstruction, DT))
76             return false;
77           if (RKOther.ArgValue >= RK.ArgValue) {
78             HasBeenPreserved = true;
79             return true;
80           } else if (isValidAssumeForContext(InsertBeforeInstruction, Assume,
81                                              DT)) {
82             HasBeenPreserved = true;
83             IntrinsicInst *Intr = cast<IntrinsicInst>(Assume);
84             ToUpdate = &Intr->op_begin()[Bundle->Begin + ABA_Argument];
85             return true;
86           }
87           return false;
88         });
89     if (ToUpdate)
90       ToUpdate->set(
91           ConstantInt::get(Type::getInt64Ty(M->getContext()), RK.ArgValue));
92     return HasBeenPreserved;
93   }
94 
95   void addKnowledge(RetainedKnowledge RK) {
96     if (RK.AttrKind == Attribute::None || tryToPreserveWithoutAddingAssume(RK))
97       return;
98     MapKey Key{RK.WasOn, RK.AttrKind};
99     auto Lookup = AssumedKnowledgeMap.find(Key);
100     if (Lookup == AssumedKnowledgeMap.end()) {
101       AssumedKnowledgeMap[Key] = RK.ArgValue;
102       return;
103     }
104     assert(((Lookup->second == 0 && RK.ArgValue == 0) ||
105             (Lookup->second != 0 && RK.ArgValue != 0)) &&
106            "inconsistent argument value");
107 
108     /// This is only desirable because for all attributes taking an argument
109     /// higher is better.
110     Lookup->second = std::max(Lookup->second, RK.ArgValue);
111   }
112 
113   void addAttribute(Attribute Attr, Value *WasOn) {
114     if (Attr.isTypeAttribute() || Attr.isStringAttribute() ||
115         (!ShouldPreserveAllAttributes &&
116          !isUsefullToPreserve(Attr.getKindAsEnum())))
117       return;
118     unsigned AttrArg = 0;
119     if (Attr.isIntAttribute())
120       AttrArg = Attr.getValueAsInt();
121     addKnowledge({Attr.getKindAsEnum(), AttrArg, WasOn});
122   }
123 
124   void addCall(const CallBase *Call) {
125     auto addAttrList = [&](AttributeList AttrList) {
126       for (unsigned Idx = AttributeList::FirstArgIndex;
127            Idx < AttrList.getNumAttrSets(); Idx++)
128         for (Attribute Attr : AttrList.getAttributes(Idx))
129           addAttribute(Attr, Call->getArgOperand(Idx - 1));
130       for (Attribute Attr : AttrList.getFnAttributes())
131         addAttribute(Attr, nullptr);
132     };
133     addAttrList(Call->getAttributes());
134     if (Function *Fn = Call->getCalledFunction())
135       addAttrList(Fn->getAttributes());
136   }
137 
138   IntrinsicInst *build() {
139     if (AssumedKnowledgeMap.empty())
140       return nullptr;
141     Function *FnAssume = Intrinsic::getDeclaration(M, Intrinsic::assume);
142     LLVMContext &C = M->getContext();
143     SmallVector<OperandBundleDef, 8> OpBundle;
144     for (auto &MapElem : AssumedKnowledgeMap) {
145       SmallVector<Value *, 2> Args;
146       if (MapElem.first.first)
147         Args.push_back(MapElem.first.first);
148 
149       /// This is only valid because for all attribute that currently exist a
150       /// value of 0 is useless. and should not be preserved.
151       if (MapElem.second)
152         Args.push_back(ConstantInt::get(Type::getInt64Ty(M->getContext()),
153                                         MapElem.second));
154       OpBundle.push_back(OperandBundleDefT<Value *>(
155           std::string(Attribute::getNameFromAttrKind(MapElem.first.second)),
156           Args));
157     }
158     return cast<IntrinsicInst>(CallInst::Create(
159         FnAssume, ArrayRef<Value *>({ConstantInt::getTrue(C)}), OpBundle));
160   }
161 
162   void addAccessedPtr(Instruction *MemInst, Value *Pointer, Type *AccType,
163                       MaybeAlign MA) {
164     unsigned DerefSize = MemInst->getModule()
165                              ->getDataLayout()
166                              .getTypeStoreSize(AccType)
167                              .getKnownMinSize();
168     if (DerefSize != 0) {
169       addKnowledge({Attribute::Dereferenceable, DerefSize, Pointer});
170       if (!NullPointerIsDefined(MemInst->getFunction(),
171                                 Pointer->getType()->getPointerAddressSpace()))
172         addKnowledge({Attribute::NonNull, 0u, Pointer});
173     }
174     if (MA.valueOrOne() > 1)
175       addKnowledge(
176           {Attribute::Alignment, unsigned(MA.valueOrOne().value()), Pointer});
177   }
178 
179   void addInstruction(Instruction *I) {
180     if (auto *Call = dyn_cast<CallBase>(I))
181       return addCall(Call);
182     if (auto *Load = dyn_cast<LoadInst>(I))
183       return addAccessedPtr(I, Load->getPointerOperand(), Load->getType(),
184                             Load->getAlign());
185     if (auto *Store = dyn_cast<StoreInst>(I))
186       return addAccessedPtr(I, Store->getPointerOperand(),
187                             Store->getValueOperand()->getType(),
188                             Store->getAlign());
189     // TODO: Add support for the other Instructions.
190     // TODO: Maybe we should look around and merge with other llvm.assume.
191   }
192 };
193 
194 } // namespace
195 
196 IntrinsicInst *llvm::buildAssumeFromInst(Instruction *I) {
197   if (!EnableKnowledgeRetention)
198     return nullptr;
199   AssumeBuilderState Builder(I->getModule());
200   Builder.addInstruction(I);
201   return Builder.build();
202 }
203 
204 void llvm::salvageKnowledge(Instruction *I, AssumptionCache *AC, DominatorTree* DT) {
205   if (!EnableKnowledgeRetention)
206     return;
207   AssumeBuilderState Builder(I->getModule(), I, AC, DT);
208   Builder.addInstruction(I);
209   if (IntrinsicInst *Intr = Builder.build()) {
210     Intr->insertBefore(I);
211     if (AC)
212       AC->registerAssumption(Intr);
213   }
214 }
215 
216 namespace {
217 
218 struct AssumeSimplify {
219   Function &F;
220   AssumptionCache &AC;
221   DominatorTree *DT;
222   LLVMContext &C;
223   SmallDenseSet<IntrinsicInst *> CleanupToDo;
224   StringMapEntry<uint32_t> *IgnoreTag;
225   SmallDenseMap<BasicBlock *, SmallVector<IntrinsicInst *, 4>, 8> BBToAssume;
226   bool MadeChange = false;
227 
228   AssumeSimplify(Function &F, AssumptionCache &AC, DominatorTree *DT,
229                  LLVMContext &C)
230       : F(F), AC(AC), DT(DT), C(C),
231         IgnoreTag(C.getOrInsertBundleTag(IgnoreBundleTag)) {}
232 
233   void buildMapping(bool FilterBooleanArgument) {
234     BBToAssume.clear();
235     for (Value *V : AC.assumptions()) {
236       if (!V)
237         continue;
238       IntrinsicInst *Assume = cast<IntrinsicInst>(V);
239       if (FilterBooleanArgument) {
240         auto *Arg = dyn_cast<ConstantInt>(Assume->getOperand(0));
241         if (!Arg || Arg->isZero())
242           continue;
243       }
244       BBToAssume[Assume->getParent()].push_back(Assume);
245     }
246 
247     for (auto &Elem : BBToAssume) {
248       llvm::sort(Elem.second,
249                  [](const IntrinsicInst *LHS, const IntrinsicInst *RHS) {
250                    return LHS->comesBefore(RHS);
251                  });
252     }
253   }
254 
255   /// Remove all asumes in CleanupToDo if there boolean argument is true and
256   /// ForceCleanup is set or the assume doesn't hold valuable knowledge.
257   void RunCleanup(bool ForceCleanup) {
258     for (IntrinsicInst *Assume : CleanupToDo) {
259       auto *Arg = dyn_cast<ConstantInt>(Assume->getOperand(0));
260       if (!Arg || Arg->isZero() ||
261           (!ForceCleanup && !isAssumeWithEmptyBundle(*Assume)))
262         continue;
263       MadeChange = true;
264       Assume->eraseFromParent();
265     }
266     CleanupToDo.clear();
267   }
268 
269   /// Remove knowledge stored in assume when it is already know by an attribute
270   /// or an other assume. This can when valid update an existing knowledge in an
271   /// attribute or an other assume.
272   void dropRedundantKnowledge() {
273     struct MapValue {
274       IntrinsicInst *Assume;
275       unsigned ArgValue;
276       CallInst::BundleOpInfo *BOI;
277     };
278     buildMapping(false);
279     SmallDenseMap<std::pair<Value *, Attribute::AttrKind>,
280                   SmallVector<MapValue, 2>, 16>
281         Knowledge;
282     for (BasicBlock *BB : depth_first(&F))
283       for (Value *V : BBToAssume[BB]) {
284         if (!V)
285           continue;
286         IntrinsicInst *Assume = cast<IntrinsicInst>(V);
287         for (CallInst::BundleOpInfo &BOI : Assume->bundle_op_infos()) {
288           auto RemoveFromAssume = [&]() {
289             CleanupToDo.insert(Assume);
290             if (BOI.Begin != BOI.End) {
291               Use *U = &Assume->op_begin()[BOI.Begin + ABA_WasOn];
292               U->set(UndefValue::get(U->get()->getType()));
293             }
294             BOI.Tag = IgnoreTag;
295           };
296           if (BOI.Tag == IgnoreTag) {
297             CleanupToDo.insert(Assume);
298             continue;
299           }
300           RetainedKnowledge RK = getKnowledgeFromBundle(*Assume, BOI);
301           if (auto *Arg = dyn_cast_or_null<Argument>(RK.WasOn)) {
302             bool HasSameKindAttr = Arg->hasAttribute(RK.AttrKind);
303             if (HasSameKindAttr)
304               if (!Attribute::doesAttrKindHaveArgument(RK.AttrKind) ||
305                   Arg->getAttribute(RK.AttrKind).getValueAsInt() >=
306                       RK.ArgValue) {
307                 RemoveFromAssume();
308                 continue;
309               }
310             if (isValidAssumeForContext(
311                     Assume, &*F.getEntryBlock().getFirstInsertionPt()) ||
312                 Assume == &*F.getEntryBlock().getFirstInsertionPt()) {
313               if (HasSameKindAttr)
314                 Arg->removeAttr(RK.AttrKind);
315               Arg->addAttr(Attribute::get(C, RK.AttrKind, RK.ArgValue));
316               MadeChange = true;
317               RemoveFromAssume();
318               continue;
319             }
320           }
321           auto &Lookup = Knowledge[{RK.WasOn, RK.AttrKind}];
322           for (MapValue &Elem : Lookup) {
323             if (!isValidAssumeForContext(Elem.Assume, Assume, DT))
324               continue;
325             if (Elem.ArgValue >= RK.ArgValue) {
326               RemoveFromAssume();
327               continue;
328             } else if (isValidAssumeForContext(Assume, Elem.Assume, DT)) {
329               Elem.Assume->op_begin()[Elem.BOI->Begin + ABA_Argument].set(
330                   ConstantInt::get(Type::getInt64Ty(C), RK.ArgValue));
331               MadeChange = true;
332               RemoveFromAssume();
333               continue;
334             }
335           }
336           Lookup.push_back({Assume, RK.ArgValue, &BOI});
337         }
338       }
339   }
340 
341   using MergeIterator = SmallVectorImpl<IntrinsicInst *>::iterator;
342 
343   /// Merge all Assumes from Begin to End in and insert the resulting assume as
344   /// high as possible in the basicblock.
345   void mergeRange(BasicBlock *BB, MergeIterator Begin, MergeIterator End) {
346     if (Begin == End || std::next(Begin) == End)
347       return;
348     /// Provide no additional information so that AssumeBuilderState doesn't
349     /// try to do any punning since it already has been done better.
350     AssumeBuilderState Builder(F.getParent());
351 
352     /// For now it is initialized to the best value it could have
353     Instruction *InsertPt = BB->getFirstNonPHI();
354     if (isa<LandingPadInst>(InsertPt))
355       InsertPt = InsertPt->getNextNode();
356     for (IntrinsicInst *I : make_range(Begin, End)) {
357       CleanupToDo.insert(I);
358       for (CallInst::BundleOpInfo &BOI : I->bundle_op_infos()) {
359         RetainedKnowledge RK = getKnowledgeFromBundle(*I, BOI);
360         if (!RK)
361           continue;
362         Builder.addKnowledge(RK);
363         if (auto *I = dyn_cast_or_null<Instruction>(RK.WasOn))
364           if (I->getParent() == InsertPt->getParent() &&
365               (InsertPt->comesBefore(I) || InsertPt == I))
366             InsertPt = I->getNextNode();
367       }
368     }
369 
370     /// Adjust InsertPt if it is before Begin, since mergeAssumes only
371     /// guarantees we can place the resulting assume between Begin and End.
372     if (InsertPt->comesBefore(*Begin))
373       for (auto It = (*Begin)->getIterator(), E = InsertPt->getIterator();
374            It != E; --It)
375         if (!isGuaranteedToTransferExecutionToSuccessor(&*It)) {
376           InsertPt = It->getNextNode();
377           break;
378         }
379     IntrinsicInst *MergedAssume = Builder.build();
380     if (!MergedAssume)
381       return;
382     MadeChange = true;
383     MergedAssume->insertBefore(InsertPt);
384     AC.registerAssumption(MergedAssume);
385   }
386 
387   /// Merge assume when they are in the same BasicBlock and for all instruction
388   /// between them isGuaranteedToTransferExecutionToSuccessor returns true.
389   void mergeAssumes() {
390     buildMapping(true);
391 
392     SmallVector<MergeIterator, 4> SplitPoints;
393     for (auto &Elem : BBToAssume) {
394       SmallVectorImpl<IntrinsicInst *> &AssumesInBB = Elem.second;
395       if (AssumesInBB.size() < 2)
396         continue;
397       /// AssumesInBB is already sorted by order in the block.
398 
399       BasicBlock::iterator It = AssumesInBB.front()->getIterator();
400       BasicBlock::iterator E = AssumesInBB.back()->getIterator();
401       SplitPoints.push_back(AssumesInBB.begin());
402       MergeIterator LastSplit = AssumesInBB.begin();
403       for (; It != E; ++It)
404         if (!isGuaranteedToTransferExecutionToSuccessor(&*It)) {
405           for (; (*LastSplit)->comesBefore(&*It); ++LastSplit)
406             ;
407           if (SplitPoints.back() != LastSplit)
408             SplitPoints.push_back(LastSplit);
409         }
410       SplitPoints.push_back(AssumesInBB.end());
411       for (auto SplitIt = SplitPoints.begin();
412            SplitIt != std::prev(SplitPoints.end()); SplitIt++) {
413         mergeRange(Elem.first, *SplitIt, *(SplitIt + 1));
414       }
415       SplitPoints.clear();
416     }
417   }
418 };
419 
420 bool simplifyAssumes(Function &F, AssumptionCache *AC, DominatorTree *DT) {
421   AssumeSimplify AS(F, *AC, DT, F.getContext());
422 
423   /// Remove knowledge that is already known by a dominating other assume or an
424   /// attribute.
425   AS.dropRedundantKnowledge();
426 
427   /// Remove assume that are empty.
428   AS.RunCleanup(false);
429 
430   /// Merge assume in the same basicblock when possible.
431   AS.mergeAssumes();
432 
433   /// Remove assume that were merged.
434   AS.RunCleanup(true);
435   return AS.MadeChange;
436 }
437 
438 } // namespace
439 
440 PreservedAnalyses AssumeSimplifyPass::run(Function &F,
441                                           FunctionAnalysisManager &AM) {
442   if (!EnableKnowledgeRetention)
443     return PreservedAnalyses::all();
444   simplifyAssumes(F, &AM.getResult<AssumptionAnalysis>(F),
445                   AM.getCachedResult<DominatorTreeAnalysis>(F));
446   return PreservedAnalyses::all();
447 }
448 
449 class AssumeSimplifyPassLegacyPass : public FunctionPass {
450 public:
451   static char ID;
452 
453   AssumeSimplifyPassLegacyPass() : FunctionPass(ID) {
454     initializeAssumeSimplifyPassLegacyPassPass(
455         *PassRegistry::getPassRegistry());
456   }
457   bool runOnFunction(Function &F) override {
458     if (skipFunction(F) || !EnableKnowledgeRetention)
459       return false;
460     DominatorTreeWrapperPass *DT =
461         getAnalysisIfAvailable<DominatorTreeWrapperPass>();
462     AssumptionCache &AC =
463         getAnalysis<AssumptionCacheTracker>().getAssumptionCache(F);
464     return simplifyAssumes(F, &AC, DT ? &DT->getDomTree() : nullptr);
465   }
466 
467   void getAnalysisUsage(AnalysisUsage &AU) const override {
468     AU.setPreservesAll();
469   }
470 };
471 
472 char AssumeSimplifyPassLegacyPass::ID = 0;
473 
474 INITIALIZE_PASS_BEGIN(AssumeSimplifyPassLegacyPass, "assume-simplify",
475                       "Assume Simplify", false, false)
476 INITIALIZE_PASS_DEPENDENCY(DominatorTreeWrapperPass)
477 INITIALIZE_PASS_DEPENDENCY(AssumptionCacheTracker)
478 INITIALIZE_PASS_END(AssumeSimplifyPassLegacyPass, "assume-simplify",
479                     "Assume Simplify", false, false)
480 
481 FunctionPass *llvm::createAssumeSimplifyPass() {
482   return new AssumeSimplifyPassLegacyPass();
483 }
484 
485 PreservedAnalyses AssumeBuilderPass::run(Function &F,
486                                          FunctionAnalysisManager &AM) {
487   AssumptionCache* AC = AM.getCachedResult<AssumptionAnalysis>(F);
488   DominatorTree* DT = AM.getCachedResult<DominatorTreeAnalysis>(F);
489   for (Instruction &I : instructions(F))
490     salvageKnowledge(&I, AC, DT);
491   return PreservedAnalyses::all();
492 }
493