1e60b36cfSFlorian Hahn //===-- VPlanTransforms.cpp - Utility VPlan to VPlan transforms -----------===//
2e60b36cfSFlorian Hahn //
3e60b36cfSFlorian Hahn // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4e60b36cfSFlorian Hahn // See https://llvm.org/LICENSE.txt for license information.
5e60b36cfSFlorian Hahn // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6e60b36cfSFlorian Hahn //
7e60b36cfSFlorian Hahn //===----------------------------------------------------------------------===//
8e60b36cfSFlorian Hahn ///
9e60b36cfSFlorian Hahn /// \file
10e60b36cfSFlorian Hahn /// This file implements a set of utility VPlan to VPlan transformations.
11e60b36cfSFlorian Hahn ///
12e60b36cfSFlorian Hahn //===----------------------------------------------------------------------===//
13e60b36cfSFlorian Hahn 
14e60b36cfSFlorian Hahn #include "VPlanTransforms.h"
15e60b36cfSFlorian Hahn #include "llvm/ADT/PostOrderIterator.h"
16e60b36cfSFlorian Hahn 
17e60b36cfSFlorian Hahn using namespace llvm;
18e60b36cfSFlorian Hahn 
19e60b36cfSFlorian Hahn void VPlanTransforms::VPInstructionsToVPRecipes(
20e60b36cfSFlorian Hahn     Loop *OrigLoop, VPlanPtr &Plan,
21d0d38df0SDavid Green     LoopVectorizationLegality::InductionList &Inductions,
220de8aeaeSMauri Mustonen     SmallPtrSetImpl<Instruction *> &DeadInstructions, ScalarEvolution &SE) {
23e60b36cfSFlorian Hahn 
24e60b36cfSFlorian Hahn   auto *TopRegion = cast<VPRegionBlock>(Plan->getEntry());
25e60b36cfSFlorian Hahn   ReversePostOrderTraversal<VPBlockBase *> RPOT(TopRegion->getEntry());
26e60b36cfSFlorian Hahn 
27e60b36cfSFlorian Hahn   for (VPBlockBase *Base : RPOT) {
28e60b36cfSFlorian Hahn     // Do not widen instructions in pre-header and exit blocks.
29e60b36cfSFlorian Hahn     if (Base->getNumPredecessors() == 0 || Base->getNumSuccessors() == 0)
30e60b36cfSFlorian Hahn       continue;
31e60b36cfSFlorian Hahn 
32e60b36cfSFlorian Hahn     VPBasicBlock *VPBB = Base->getEntryBasicBlock();
33e60b36cfSFlorian Hahn     // Introduce each ingredient into VPlan.
34e60b36cfSFlorian Hahn     for (auto I = VPBB->begin(), E = VPBB->end(); I != E;) {
35e60b36cfSFlorian Hahn       VPRecipeBase *Ingredient = &*I++;
36a0e1313cSFlorian Hahn       VPValue *VPV = Ingredient->getVPSingleValue();
3715a74b64SFlorian Hahn       Instruction *Inst = cast<Instruction>(VPV->getUnderlyingValue());
38e60b36cfSFlorian Hahn       if (DeadInstructions.count(Inst)) {
3976afbf60SFlorian Hahn         VPValue DummyValue;
4015a74b64SFlorian Hahn         VPV->replaceAllUsesWith(&DummyValue);
41e60b36cfSFlorian Hahn         Ingredient->eraseFromParent();
42e60b36cfSFlorian Hahn         continue;
43e60b36cfSFlorian Hahn       }
44e60b36cfSFlorian Hahn 
45e60b36cfSFlorian Hahn       VPRecipeBase *NewRecipe = nullptr;
4615a74b64SFlorian Hahn       if (auto *VPPhi = dyn_cast<VPWidenPHIRecipe>(Ingredient)) {
4715a74b64SFlorian Hahn         auto *Phi = cast<PHINode>(VPPhi->getUnderlyingValue());
48d0d38df0SDavid Green         InductionDescriptor II = Inductions.lookup(Phi);
49e60b36cfSFlorian Hahn         if (II.getKind() == InductionDescriptor::IK_IntInduction ||
50e60b36cfSFlorian Hahn             II.getKind() == InductionDescriptor::IK_FpInduction) {
51816dba48SFlorian Hahn           VPValue *Start = Plan->getOrAddVPValue(II.getStartValue());
52daaa0e35SFlorian Hahn           NewRecipe = new VPWidenIntOrFpInductionRecipe(Phi, Start, nullptr);
5315a74b64SFlorian Hahn         } else {
5415a74b64SFlorian Hahn           Plan->addVPValue(Phi, VPPhi);
5515a74b64SFlorian Hahn           continue;
5615a74b64SFlorian Hahn         }
5715a74b64SFlorian Hahn       } else {
5815a74b64SFlorian Hahn         assert(isa<VPInstruction>(Ingredient) &&
5915a74b64SFlorian Hahn                "only VPInstructions expected here");
6015a74b64SFlorian Hahn         assert(!isa<PHINode>(Inst) && "phis should be handled above");
6115a74b64SFlorian Hahn         // Create VPWidenMemoryInstructionRecipe for loads and stores.
6215a74b64SFlorian Hahn         if (LoadInst *Load = dyn_cast<LoadInst>(Inst)) {
6315a74b64SFlorian Hahn           NewRecipe = new VPWidenMemoryInstructionRecipe(
6415a74b64SFlorian Hahn               *Load, Plan->getOrAddVPValue(getLoadStorePointerOperand(Inst)),
6515a74b64SFlorian Hahn               nullptr /*Mask*/);
6615a74b64SFlorian Hahn         } else if (StoreInst *Store = dyn_cast<StoreInst>(Inst)) {
6715a74b64SFlorian Hahn           NewRecipe = new VPWidenMemoryInstructionRecipe(
6815a74b64SFlorian Hahn               *Store, Plan->getOrAddVPValue(getLoadStorePointerOperand(Inst)),
6915a74b64SFlorian Hahn               Plan->getOrAddVPValue(Store->getValueOperand()),
7015a74b64SFlorian Hahn               nullptr /*Mask*/);
71e60b36cfSFlorian Hahn         } else if (GetElementPtrInst *GEP = dyn_cast<GetElementPtrInst>(Inst)) {
72c0cdba72SFlorian Hahn           NewRecipe = new VPWidenGEPRecipe(
73c0cdba72SFlorian Hahn               GEP, Plan->mapToVPValues(GEP->operands()), OrigLoop);
74494b5ba3SMauri Mustonen         } else if (CallInst *CI = dyn_cast<CallInst>(Inst)) {
75494b5ba3SMauri Mustonen           NewRecipe = new VPWidenCallRecipe(
76494b5ba3SMauri Mustonen               *CI, Plan->mapToVPValues(CI->arg_operands()));
770de8aeaeSMauri Mustonen         } else if (SelectInst *SI = dyn_cast<SelectInst>(Inst)) {
780de8aeaeSMauri Mustonen           bool InvariantCond =
790de8aeaeSMauri Mustonen               SE.isLoopInvariant(SE.getSCEV(SI->getOperand(0)), OrigLoop);
800de8aeaeSMauri Mustonen           NewRecipe = new VPWidenSelectRecipe(
810de8aeaeSMauri Mustonen               *SI, Plan->mapToVPValues(SI->operands()), InvariantCond);
8215a74b64SFlorian Hahn         } else {
83e8937985SFlorian Hahn           NewRecipe =
84e8937985SFlorian Hahn               new VPWidenRecipe(*Inst, Plan->mapToVPValues(Inst->operands()));
8515a74b64SFlorian Hahn         }
8615a74b64SFlorian Hahn       }
87e60b36cfSFlorian Hahn 
88e60b36cfSFlorian Hahn       NewRecipe->insertBefore(Ingredient);
8976afbf60SFlorian Hahn       if (NewRecipe->getNumDefinedValues() == 1)
90a0e1313cSFlorian Hahn         VPV->replaceAllUsesWith(NewRecipe->getVPSingleValue());
9176afbf60SFlorian Hahn       else
9276afbf60SFlorian Hahn         assert(NewRecipe->getNumDefinedValues() == 0 &&
9376afbf60SFlorian Hahn                "Only recpies with zero or one defined values expected");
94e60b36cfSFlorian Hahn       Ingredient->eraseFromParent();
9554a14c26SFlorian Hahn       Plan->removeVPValueFor(Inst);
9654a14c26SFlorian Hahn       for (auto *Def : NewRecipe->definedValues()) {
9754a14c26SFlorian Hahn         Plan->addVPValue(Inst, Def);
9854a14c26SFlorian Hahn       }
99e60b36cfSFlorian Hahn     }
100e60b36cfSFlorian Hahn   }
101e60b36cfSFlorian Hahn }
102*65d3dd7cSFlorian Hahn 
103*65d3dd7cSFlorian Hahn bool VPlanTransforms::sinkScalarOperands(VPlan &Plan) {
104*65d3dd7cSFlorian Hahn   auto Iter = depth_first(
105*65d3dd7cSFlorian Hahn       VPBlockRecursiveTraversalWrapper<VPBlockBase *>(Plan.getEntry()));
106*65d3dd7cSFlorian Hahn   bool Changed = false;
107*65d3dd7cSFlorian Hahn   // First, collect the operands of all predicated replicate recipes as seeds
108*65d3dd7cSFlorian Hahn   // for sinking.
109*65d3dd7cSFlorian Hahn   SetVector<VPValue *> WorkList;
110*65d3dd7cSFlorian Hahn   for (VPBasicBlock *VPBB : VPBlockUtils::blocksOnly<VPBasicBlock>(Iter)) {
111*65d3dd7cSFlorian Hahn     for (auto &Recipe : *VPBB) {
112*65d3dd7cSFlorian Hahn       auto *RepR = dyn_cast<VPReplicateRecipe>(&Recipe);
113*65d3dd7cSFlorian Hahn       if (!RepR || !RepR->isPredicated())
114*65d3dd7cSFlorian Hahn         continue;
115*65d3dd7cSFlorian Hahn       WorkList.insert(RepR->op_begin(), RepR->op_end());
116*65d3dd7cSFlorian Hahn     }
117*65d3dd7cSFlorian Hahn   }
118*65d3dd7cSFlorian Hahn 
119*65d3dd7cSFlorian Hahn   // Try to sink each replicate recipe in the worklist.
120*65d3dd7cSFlorian Hahn   while (!WorkList.empty()) {
121*65d3dd7cSFlorian Hahn     auto *C = WorkList.pop_back_val();
122*65d3dd7cSFlorian Hahn     auto *SinkCandidate = dyn_cast_or_null<VPReplicateRecipe>(C->Def);
123*65d3dd7cSFlorian Hahn     if (!SinkCandidate)
124*65d3dd7cSFlorian Hahn       continue;
125*65d3dd7cSFlorian Hahn 
126*65d3dd7cSFlorian Hahn     // All users of SinkCandidate must be in the same block in order to perform
127*65d3dd7cSFlorian Hahn     // sinking. Therefore the destination block for sinking must match the block
128*65d3dd7cSFlorian Hahn     // containing the first user.
129*65d3dd7cSFlorian Hahn     auto *FirstUser = dyn_cast<VPRecipeBase>(*SinkCandidate->user_begin());
130*65d3dd7cSFlorian Hahn     if (!FirstUser)
131*65d3dd7cSFlorian Hahn       continue;
132*65d3dd7cSFlorian Hahn     VPBasicBlock *SinkTo = FirstUser->getParent();
133*65d3dd7cSFlorian Hahn     if (SinkCandidate->getParent() == SinkTo ||
134*65d3dd7cSFlorian Hahn         SinkCandidate->mayHaveSideEffects() ||
135*65d3dd7cSFlorian Hahn         SinkCandidate->mayReadOrWriteMemory())
136*65d3dd7cSFlorian Hahn       continue;
137*65d3dd7cSFlorian Hahn 
138*65d3dd7cSFlorian Hahn     // All recipe users of the sink candidate must be in the same block SinkTo.
139*65d3dd7cSFlorian Hahn     if (any_of(SinkCandidate->users(), [SinkTo](VPUser *U) {
140*65d3dd7cSFlorian Hahn           auto *UI = dyn_cast<VPRecipeBase>(U);
141*65d3dd7cSFlorian Hahn           return !UI || UI->getParent() != SinkTo;
142*65d3dd7cSFlorian Hahn         }))
143*65d3dd7cSFlorian Hahn       continue;
144*65d3dd7cSFlorian Hahn 
145*65d3dd7cSFlorian Hahn     SinkCandidate->moveBefore(*SinkTo, SinkTo->getFirstNonPhi());
146*65d3dd7cSFlorian Hahn     WorkList.insert(SinkCandidate->op_begin(), SinkCandidate->op_end());
147*65d3dd7cSFlorian Hahn     Changed = true;
148*65d3dd7cSFlorian Hahn   }
149*65d3dd7cSFlorian Hahn   return Changed;
150*65d3dd7cSFlorian Hahn }
151