1480093f4SDimitry Andric //===------ PPCLoopInstrFormPrep.cpp - Loop Instr Form Prep Pass ----------===// 2480093f4SDimitry Andric // 3480093f4SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4480093f4SDimitry Andric // See https://llvm.org/LICENSE.txt for license information. 5480093f4SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6480093f4SDimitry Andric // 7480093f4SDimitry Andric //===----------------------------------------------------------------------===// 8480093f4SDimitry Andric // 9480093f4SDimitry Andric // This file implements a pass to prepare loops for ppc preferred addressing 10480093f4SDimitry Andric // modes, leveraging different instruction form. (eg: DS/DQ form, D/DS form with 11480093f4SDimitry Andric // update) 12480093f4SDimitry Andric // Additional PHIs are created for loop induction variables used by load/store 13480093f4SDimitry Andric // instructions so that preferred addressing modes can be used. 14480093f4SDimitry Andric // 15480093f4SDimitry Andric // 1: DS/DQ form preparation, prepare the load/store instructions so that they 16480093f4SDimitry Andric // can satisfy the DS/DQ form displacement requirements. 17480093f4SDimitry Andric // Generically, this means transforming loops like this: 18480093f4SDimitry Andric // for (int i = 0; i < n; ++i) { 19480093f4SDimitry Andric // unsigned long x1 = *(unsigned long *)(p + i + 5); 20480093f4SDimitry Andric // unsigned long x2 = *(unsigned long *)(p + i + 9); 21480093f4SDimitry Andric // } 22480093f4SDimitry Andric // 23480093f4SDimitry Andric // to look like this: 24480093f4SDimitry Andric // 25480093f4SDimitry Andric // unsigned NewP = p + 5; 26480093f4SDimitry Andric // for (int i = 0; i < n; ++i) { 27480093f4SDimitry Andric // unsigned long x1 = *(unsigned long *)(i + NewP); 28480093f4SDimitry Andric // unsigned long x2 = *(unsigned long *)(i + NewP + 4); 29480093f4SDimitry Andric // } 30480093f4SDimitry Andric // 31480093f4SDimitry Andric // 2: D/DS form with update preparation, prepare the load/store instructions so 32480093f4SDimitry Andric // that we can use update form to do pre-increment. 33480093f4SDimitry Andric // Generically, this means transforming loops like this: 34480093f4SDimitry Andric // for (int i = 0; i < n; ++i) 35480093f4SDimitry Andric // array[i] = c; 36480093f4SDimitry Andric // 37480093f4SDimitry Andric // to look like this: 38480093f4SDimitry Andric // 39480093f4SDimitry Andric // T *p = array[-1]; 40480093f4SDimitry Andric // for (int i = 0; i < n; ++i) 41480093f4SDimitry Andric // *++p = c; 42480093f4SDimitry Andric //===----------------------------------------------------------------------===// 43480093f4SDimitry Andric 44480093f4SDimitry Andric #include "PPC.h" 45480093f4SDimitry Andric #include "PPCSubtarget.h" 46480093f4SDimitry Andric #include "PPCTargetMachine.h" 47480093f4SDimitry Andric #include "llvm/ADT/DepthFirstIterator.h" 48480093f4SDimitry Andric #include "llvm/ADT/SmallPtrSet.h" 49480093f4SDimitry Andric #include "llvm/ADT/SmallSet.h" 50480093f4SDimitry Andric #include "llvm/ADT/SmallVector.h" 51480093f4SDimitry Andric #include "llvm/ADT/Statistic.h" 52480093f4SDimitry Andric #include "llvm/Analysis/LoopInfo.h" 53480093f4SDimitry Andric #include "llvm/Analysis/ScalarEvolution.h" 54480093f4SDimitry Andric #include "llvm/Analysis/ScalarEvolutionExpressions.h" 55480093f4SDimitry Andric #include "llvm/IR/BasicBlock.h" 56480093f4SDimitry Andric #include "llvm/IR/CFG.h" 57480093f4SDimitry Andric #include "llvm/IR/Dominators.h" 58480093f4SDimitry Andric #include "llvm/IR/Instruction.h" 59480093f4SDimitry Andric #include "llvm/IR/Instructions.h" 60480093f4SDimitry Andric #include "llvm/IR/IntrinsicInst.h" 61e8d8bef9SDimitry Andric #include "llvm/IR/IntrinsicsPowerPC.h" 62480093f4SDimitry Andric #include "llvm/IR/Module.h" 63480093f4SDimitry Andric #include "llvm/IR/Type.h" 64480093f4SDimitry Andric #include "llvm/IR/Value.h" 65480093f4SDimitry Andric #include "llvm/InitializePasses.h" 66480093f4SDimitry Andric #include "llvm/Pass.h" 67480093f4SDimitry Andric #include "llvm/Support/Casting.h" 68480093f4SDimitry Andric #include "llvm/Support/CommandLine.h" 69480093f4SDimitry Andric #include "llvm/Support/Debug.h" 70480093f4SDimitry Andric #include "llvm/Transforms/Scalar.h" 71480093f4SDimitry Andric #include "llvm/Transforms/Utils.h" 72480093f4SDimitry Andric #include "llvm/Transforms/Utils/BasicBlockUtils.h" 73480093f4SDimitry Andric #include "llvm/Transforms/Utils/Local.h" 74480093f4SDimitry Andric #include "llvm/Transforms/Utils/LoopUtils.h" 755ffd83dbSDimitry Andric #include "llvm/Transforms/Utils/ScalarEvolutionExpander.h" 76480093f4SDimitry Andric #include <cassert> 77480093f4SDimitry Andric #include <iterator> 78480093f4SDimitry Andric #include <utility> 79480093f4SDimitry Andric 80*fe6060f1SDimitry Andric #define DEBUG_TYPE "ppc-loop-instr-form-prep" 81*fe6060f1SDimitry Andric 82480093f4SDimitry Andric using namespace llvm; 83480093f4SDimitry Andric 84480093f4SDimitry Andric static cl::opt<unsigned> MaxVarsPrep("ppc-formprep-max-vars", 85e8d8bef9SDimitry Andric cl::Hidden, cl::init(24), 86480093f4SDimitry Andric cl::desc("Potential common base number threshold per function for PPC loop " 87480093f4SDimitry Andric "prep")); 88480093f4SDimitry Andric 89480093f4SDimitry Andric static cl::opt<bool> PreferUpdateForm("ppc-formprep-prefer-update", 90480093f4SDimitry Andric cl::init(true), cl::Hidden, 91480093f4SDimitry Andric cl::desc("prefer update form when ds form is also a update form")); 92480093f4SDimitry Andric 93480093f4SDimitry Andric // Sum of following 3 per loop thresholds for all loops can not be larger 94480093f4SDimitry Andric // than MaxVarsPrep. 95e8d8bef9SDimitry Andric // now the thresholds for each kind prep are exterimental values on Power9. 96480093f4SDimitry Andric static cl::opt<unsigned> MaxVarsUpdateForm("ppc-preinc-prep-max-vars", 97480093f4SDimitry Andric cl::Hidden, cl::init(3), 98480093f4SDimitry Andric cl::desc("Potential PHI threshold per loop for PPC loop prep of update " 99480093f4SDimitry Andric "form")); 100480093f4SDimitry Andric 101480093f4SDimitry Andric static cl::opt<unsigned> MaxVarsDSForm("ppc-dsprep-max-vars", 102480093f4SDimitry Andric cl::Hidden, cl::init(3), 103480093f4SDimitry Andric cl::desc("Potential PHI threshold per loop for PPC loop prep of DS form")); 104480093f4SDimitry Andric 105480093f4SDimitry Andric static cl::opt<unsigned> MaxVarsDQForm("ppc-dqprep-max-vars", 106e8d8bef9SDimitry Andric cl::Hidden, cl::init(8), 107480093f4SDimitry Andric cl::desc("Potential PHI threshold per loop for PPC loop prep of DQ form")); 108480093f4SDimitry Andric 109480093f4SDimitry Andric 110480093f4SDimitry Andric // If would not be profitable if the common base has only one load/store, ISEL 111480093f4SDimitry Andric // should already be able to choose best load/store form based on offset for 112480093f4SDimitry Andric // single load/store. Set minimal profitable value default to 2 and make it as 113480093f4SDimitry Andric // an option. 114480093f4SDimitry Andric static cl::opt<unsigned> DispFormPrepMinThreshold("ppc-dispprep-min-threshold", 115480093f4SDimitry Andric cl::Hidden, cl::init(2), 116480093f4SDimitry Andric cl::desc("Minimal common base load/store instructions triggering DS/DQ form " 117480093f4SDimitry Andric "preparation")); 118480093f4SDimitry Andric 119480093f4SDimitry Andric STATISTIC(PHINodeAlreadyExistsUpdate, "PHI node already in pre-increment form"); 120480093f4SDimitry Andric STATISTIC(PHINodeAlreadyExistsDS, "PHI node already in DS form"); 121480093f4SDimitry Andric STATISTIC(PHINodeAlreadyExistsDQ, "PHI node already in DQ form"); 122480093f4SDimitry Andric STATISTIC(DSFormChainRewritten, "Num of DS form chain rewritten"); 123480093f4SDimitry Andric STATISTIC(DQFormChainRewritten, "Num of DQ form chain rewritten"); 124480093f4SDimitry Andric STATISTIC(UpdFormChainRewritten, "Num of update form chain rewritten"); 125480093f4SDimitry Andric 126480093f4SDimitry Andric namespace { 127480093f4SDimitry Andric struct BucketElement { 128480093f4SDimitry Andric BucketElement(const SCEVConstant *O, Instruction *I) : Offset(O), Instr(I) {} 129480093f4SDimitry Andric BucketElement(Instruction *I) : Offset(nullptr), Instr(I) {} 130480093f4SDimitry Andric 131480093f4SDimitry Andric const SCEVConstant *Offset; 132480093f4SDimitry Andric Instruction *Instr; 133480093f4SDimitry Andric }; 134480093f4SDimitry Andric 135480093f4SDimitry Andric struct Bucket { 136480093f4SDimitry Andric Bucket(const SCEV *B, Instruction *I) : BaseSCEV(B), 137480093f4SDimitry Andric Elements(1, BucketElement(I)) {} 138480093f4SDimitry Andric 139480093f4SDimitry Andric const SCEV *BaseSCEV; 140480093f4SDimitry Andric SmallVector<BucketElement, 16> Elements; 141480093f4SDimitry Andric }; 142480093f4SDimitry Andric 143480093f4SDimitry Andric // "UpdateForm" is not a real PPC instruction form, it stands for dform 144480093f4SDimitry Andric // load/store with update like ldu/stdu, or Prefetch intrinsic. 145480093f4SDimitry Andric // For DS form instructions, their displacements must be multiple of 4. 146480093f4SDimitry Andric // For DQ form instructions, their displacements must be multiple of 16. 147480093f4SDimitry Andric enum InstrForm { UpdateForm = 1, DSForm = 4, DQForm = 16 }; 148480093f4SDimitry Andric 149480093f4SDimitry Andric class PPCLoopInstrFormPrep : public FunctionPass { 150480093f4SDimitry Andric public: 151480093f4SDimitry Andric static char ID; // Pass ID, replacement for typeid 152480093f4SDimitry Andric 153480093f4SDimitry Andric PPCLoopInstrFormPrep() : FunctionPass(ID) { 154480093f4SDimitry Andric initializePPCLoopInstrFormPrepPass(*PassRegistry::getPassRegistry()); 155480093f4SDimitry Andric } 156480093f4SDimitry Andric 157480093f4SDimitry Andric PPCLoopInstrFormPrep(PPCTargetMachine &TM) : FunctionPass(ID), TM(&TM) { 158480093f4SDimitry Andric initializePPCLoopInstrFormPrepPass(*PassRegistry::getPassRegistry()); 159480093f4SDimitry Andric } 160480093f4SDimitry Andric 161480093f4SDimitry Andric void getAnalysisUsage(AnalysisUsage &AU) const override { 162480093f4SDimitry Andric AU.addPreserved<DominatorTreeWrapperPass>(); 163480093f4SDimitry Andric AU.addRequired<LoopInfoWrapperPass>(); 164480093f4SDimitry Andric AU.addPreserved<LoopInfoWrapperPass>(); 165480093f4SDimitry Andric AU.addRequired<ScalarEvolutionWrapperPass>(); 166480093f4SDimitry Andric } 167480093f4SDimitry Andric 168480093f4SDimitry Andric bool runOnFunction(Function &F) override; 169480093f4SDimitry Andric 170480093f4SDimitry Andric private: 171480093f4SDimitry Andric PPCTargetMachine *TM = nullptr; 172480093f4SDimitry Andric const PPCSubtarget *ST; 173480093f4SDimitry Andric DominatorTree *DT; 174480093f4SDimitry Andric LoopInfo *LI; 175480093f4SDimitry Andric ScalarEvolution *SE; 176480093f4SDimitry Andric bool PreserveLCSSA; 177480093f4SDimitry Andric 178480093f4SDimitry Andric /// Successful preparation number for Update/DS/DQ form in all inner most 179480093f4SDimitry Andric /// loops. One successful preparation will put one common base out of loop, 180480093f4SDimitry Andric /// this may leads to register presure like LICM does. 181480093f4SDimitry Andric /// Make sure total preparation number can be controlled by option. 182480093f4SDimitry Andric unsigned SuccPrepCount; 183480093f4SDimitry Andric 184480093f4SDimitry Andric bool runOnLoop(Loop *L); 185480093f4SDimitry Andric 186480093f4SDimitry Andric /// Check if required PHI node is already exist in Loop \p L. 187480093f4SDimitry Andric bool alreadyPrepared(Loop *L, Instruction* MemI, 188480093f4SDimitry Andric const SCEV *BasePtrStartSCEV, 189480093f4SDimitry Andric const SCEVConstant *BasePtrIncSCEV, 190480093f4SDimitry Andric InstrForm Form); 191480093f4SDimitry Andric 192480093f4SDimitry Andric /// Collect condition matched(\p isValidCandidate() returns true) 193480093f4SDimitry Andric /// candidates in Loop \p L. 194*fe6060f1SDimitry Andric SmallVector<Bucket, 16> collectCandidates( 195*fe6060f1SDimitry Andric Loop *L, 196*fe6060f1SDimitry Andric std::function<bool(const Instruction *, const Value *, const Type *)> 197480093f4SDimitry Andric isValidCandidate, 198480093f4SDimitry Andric unsigned MaxCandidateNum); 199480093f4SDimitry Andric 200480093f4SDimitry Andric /// Add a candidate to candidates \p Buckets. 201480093f4SDimitry Andric void addOneCandidate(Instruction *MemI, const SCEV *LSCEV, 202480093f4SDimitry Andric SmallVector<Bucket, 16> &Buckets, 203480093f4SDimitry Andric unsigned MaxCandidateNum); 204480093f4SDimitry Andric 205480093f4SDimitry Andric /// Prepare all candidates in \p Buckets for update form. 206480093f4SDimitry Andric bool updateFormPrep(Loop *L, SmallVector<Bucket, 16> &Buckets); 207480093f4SDimitry Andric 208480093f4SDimitry Andric /// Prepare all candidates in \p Buckets for displacement form, now for 209480093f4SDimitry Andric /// ds/dq. 210480093f4SDimitry Andric bool dispFormPrep(Loop *L, SmallVector<Bucket, 16> &Buckets, 211480093f4SDimitry Andric InstrForm Form); 212480093f4SDimitry Andric 213480093f4SDimitry Andric /// Prepare for one chain \p BucketChain, find the best base element and 214480093f4SDimitry Andric /// update all other elements in \p BucketChain accordingly. 215480093f4SDimitry Andric /// \p Form is used to find the best base element. 216480093f4SDimitry Andric /// If success, best base element must be stored as the first element of 217480093f4SDimitry Andric /// \p BucketChain. 218480093f4SDimitry Andric /// Return false if no base element found, otherwise return true. 219480093f4SDimitry Andric bool prepareBaseForDispFormChain(Bucket &BucketChain, 220480093f4SDimitry Andric InstrForm Form); 221480093f4SDimitry Andric 222480093f4SDimitry Andric /// Prepare for one chain \p BucketChain, find the best base element and 223480093f4SDimitry Andric /// update all other elements in \p BucketChain accordingly. 224480093f4SDimitry Andric /// If success, best base element must be stored as the first element of 225480093f4SDimitry Andric /// \p BucketChain. 226480093f4SDimitry Andric /// Return false if no base element found, otherwise return true. 227480093f4SDimitry Andric bool prepareBaseForUpdateFormChain(Bucket &BucketChain); 228480093f4SDimitry Andric 229480093f4SDimitry Andric /// Rewrite load/store instructions in \p BucketChain according to 230480093f4SDimitry Andric /// preparation. 231480093f4SDimitry Andric bool rewriteLoadStores(Loop *L, Bucket &BucketChain, 232480093f4SDimitry Andric SmallSet<BasicBlock *, 16> &BBChanged, 233480093f4SDimitry Andric InstrForm Form); 234480093f4SDimitry Andric }; 235480093f4SDimitry Andric 236480093f4SDimitry Andric } // end anonymous namespace 237480093f4SDimitry Andric 238480093f4SDimitry Andric char PPCLoopInstrFormPrep::ID = 0; 239480093f4SDimitry Andric static const char *name = "Prepare loop for ppc preferred instruction forms"; 240480093f4SDimitry Andric INITIALIZE_PASS_BEGIN(PPCLoopInstrFormPrep, DEBUG_TYPE, name, false, false) 241480093f4SDimitry Andric INITIALIZE_PASS_DEPENDENCY(LoopInfoWrapperPass) 242480093f4SDimitry Andric INITIALIZE_PASS_DEPENDENCY(ScalarEvolutionWrapperPass) 243480093f4SDimitry Andric INITIALIZE_PASS_END(PPCLoopInstrFormPrep, DEBUG_TYPE, name, false, false) 244480093f4SDimitry Andric 2455ffd83dbSDimitry Andric static constexpr StringRef PHINodeNameSuffix = ".phi"; 2465ffd83dbSDimitry Andric static constexpr StringRef CastNodeNameSuffix = ".cast"; 2475ffd83dbSDimitry Andric static constexpr StringRef GEPNodeIncNameSuffix = ".inc"; 2485ffd83dbSDimitry Andric static constexpr StringRef GEPNodeOffNameSuffix = ".off"; 249480093f4SDimitry Andric 250480093f4SDimitry Andric FunctionPass *llvm::createPPCLoopInstrFormPrepPass(PPCTargetMachine &TM) { 251480093f4SDimitry Andric return new PPCLoopInstrFormPrep(TM); 252480093f4SDimitry Andric } 253480093f4SDimitry Andric 254480093f4SDimitry Andric static bool IsPtrInBounds(Value *BasePtr) { 255480093f4SDimitry Andric Value *StrippedBasePtr = BasePtr; 256480093f4SDimitry Andric while (BitCastInst *BC = dyn_cast<BitCastInst>(StrippedBasePtr)) 257480093f4SDimitry Andric StrippedBasePtr = BC->getOperand(0); 258480093f4SDimitry Andric if (GetElementPtrInst *GEP = dyn_cast<GetElementPtrInst>(StrippedBasePtr)) 259480093f4SDimitry Andric return GEP->isInBounds(); 260480093f4SDimitry Andric 261480093f4SDimitry Andric return false; 262480093f4SDimitry Andric } 263480093f4SDimitry Andric 2645ffd83dbSDimitry Andric static std::string getInstrName(const Value *I, StringRef Suffix) { 265480093f4SDimitry Andric assert(I && "Invalid paramater!"); 266480093f4SDimitry Andric if (I->hasName()) 267480093f4SDimitry Andric return (I->getName() + Suffix).str(); 268480093f4SDimitry Andric else 269480093f4SDimitry Andric return ""; 270480093f4SDimitry Andric } 271480093f4SDimitry Andric 272480093f4SDimitry Andric static Value *GetPointerOperand(Value *MemI) { 273480093f4SDimitry Andric if (LoadInst *LMemI = dyn_cast<LoadInst>(MemI)) { 274480093f4SDimitry Andric return LMemI->getPointerOperand(); 275480093f4SDimitry Andric } else if (StoreInst *SMemI = dyn_cast<StoreInst>(MemI)) { 276480093f4SDimitry Andric return SMemI->getPointerOperand(); 277480093f4SDimitry Andric } else if (IntrinsicInst *IMemI = dyn_cast<IntrinsicInst>(MemI)) { 278e8d8bef9SDimitry Andric if (IMemI->getIntrinsicID() == Intrinsic::prefetch || 279e8d8bef9SDimitry Andric IMemI->getIntrinsicID() == Intrinsic::ppc_vsx_lxvp) 280480093f4SDimitry Andric return IMemI->getArgOperand(0); 281e8d8bef9SDimitry Andric if (IMemI->getIntrinsicID() == Intrinsic::ppc_vsx_stxvp) 282e8d8bef9SDimitry Andric return IMemI->getArgOperand(1); 283480093f4SDimitry Andric } 284480093f4SDimitry Andric 285480093f4SDimitry Andric return nullptr; 286480093f4SDimitry Andric } 287480093f4SDimitry Andric 288480093f4SDimitry Andric bool PPCLoopInstrFormPrep::runOnFunction(Function &F) { 289480093f4SDimitry Andric if (skipFunction(F)) 290480093f4SDimitry Andric return false; 291480093f4SDimitry Andric 292480093f4SDimitry Andric LI = &getAnalysis<LoopInfoWrapperPass>().getLoopInfo(); 293480093f4SDimitry Andric SE = &getAnalysis<ScalarEvolutionWrapperPass>().getSE(); 294480093f4SDimitry Andric auto *DTWP = getAnalysisIfAvailable<DominatorTreeWrapperPass>(); 295480093f4SDimitry Andric DT = DTWP ? &DTWP->getDomTree() : nullptr; 296480093f4SDimitry Andric PreserveLCSSA = mustPreserveAnalysisID(LCSSAID); 297480093f4SDimitry Andric ST = TM ? TM->getSubtargetImpl(F) : nullptr; 298480093f4SDimitry Andric SuccPrepCount = 0; 299480093f4SDimitry Andric 300480093f4SDimitry Andric bool MadeChange = false; 301480093f4SDimitry Andric 302480093f4SDimitry Andric for (auto I = LI->begin(), IE = LI->end(); I != IE; ++I) 303480093f4SDimitry Andric for (auto L = df_begin(*I), LE = df_end(*I); L != LE; ++L) 304480093f4SDimitry Andric MadeChange |= runOnLoop(*L); 305480093f4SDimitry Andric 306480093f4SDimitry Andric return MadeChange; 307480093f4SDimitry Andric } 308480093f4SDimitry Andric 309480093f4SDimitry Andric void PPCLoopInstrFormPrep::addOneCandidate(Instruction *MemI, const SCEV *LSCEV, 310480093f4SDimitry Andric SmallVector<Bucket, 16> &Buckets, 311480093f4SDimitry Andric unsigned MaxCandidateNum) { 312480093f4SDimitry Andric assert((MemI && GetPointerOperand(MemI)) && 313480093f4SDimitry Andric "Candidate should be a memory instruction."); 314480093f4SDimitry Andric assert(LSCEV && "Invalid SCEV for Ptr value."); 315480093f4SDimitry Andric bool FoundBucket = false; 316480093f4SDimitry Andric for (auto &B : Buckets) { 317480093f4SDimitry Andric const SCEV *Diff = SE->getMinusSCEV(LSCEV, B.BaseSCEV); 318480093f4SDimitry Andric if (const auto *CDiff = dyn_cast<SCEVConstant>(Diff)) { 319480093f4SDimitry Andric B.Elements.push_back(BucketElement(CDiff, MemI)); 320480093f4SDimitry Andric FoundBucket = true; 321480093f4SDimitry Andric break; 322480093f4SDimitry Andric } 323480093f4SDimitry Andric } 324480093f4SDimitry Andric 325480093f4SDimitry Andric if (!FoundBucket) { 326480093f4SDimitry Andric if (Buckets.size() == MaxCandidateNum) 327480093f4SDimitry Andric return; 328480093f4SDimitry Andric Buckets.push_back(Bucket(LSCEV, MemI)); 329480093f4SDimitry Andric } 330480093f4SDimitry Andric } 331480093f4SDimitry Andric 332480093f4SDimitry Andric SmallVector<Bucket, 16> PPCLoopInstrFormPrep::collectCandidates( 333480093f4SDimitry Andric Loop *L, 334*fe6060f1SDimitry Andric std::function<bool(const Instruction *, const Value *, const Type *)> 335*fe6060f1SDimitry Andric isValidCandidate, 336480093f4SDimitry Andric unsigned MaxCandidateNum) { 337480093f4SDimitry Andric SmallVector<Bucket, 16> Buckets; 338480093f4SDimitry Andric for (const auto &BB : L->blocks()) 339480093f4SDimitry Andric for (auto &J : *BB) { 340480093f4SDimitry Andric Value *PtrValue; 341*fe6060f1SDimitry Andric Type *PointerElementType; 342480093f4SDimitry Andric 343480093f4SDimitry Andric if (LoadInst *LMemI = dyn_cast<LoadInst>(&J)) { 344480093f4SDimitry Andric PtrValue = LMemI->getPointerOperand(); 345*fe6060f1SDimitry Andric PointerElementType = LMemI->getType(); 346480093f4SDimitry Andric } else if (StoreInst *SMemI = dyn_cast<StoreInst>(&J)) { 347480093f4SDimitry Andric PtrValue = SMemI->getPointerOperand(); 348*fe6060f1SDimitry Andric PointerElementType = SMemI->getValueOperand()->getType(); 349480093f4SDimitry Andric } else if (IntrinsicInst *IMemI = dyn_cast<IntrinsicInst>(&J)) { 350*fe6060f1SDimitry Andric PointerElementType = Type::getInt8Ty(J.getContext()); 351e8d8bef9SDimitry Andric if (IMemI->getIntrinsicID() == Intrinsic::prefetch || 352e8d8bef9SDimitry Andric IMemI->getIntrinsicID() == Intrinsic::ppc_vsx_lxvp) { 353480093f4SDimitry Andric PtrValue = IMemI->getArgOperand(0); 354e8d8bef9SDimitry Andric } else if (IMemI->getIntrinsicID() == Intrinsic::ppc_vsx_stxvp) { 355e8d8bef9SDimitry Andric PtrValue = IMemI->getArgOperand(1); 356480093f4SDimitry Andric } else continue; 357480093f4SDimitry Andric } else continue; 358480093f4SDimitry Andric 359480093f4SDimitry Andric unsigned PtrAddrSpace = PtrValue->getType()->getPointerAddressSpace(); 360480093f4SDimitry Andric if (PtrAddrSpace) 361480093f4SDimitry Andric continue; 362480093f4SDimitry Andric 363480093f4SDimitry Andric if (L->isLoopInvariant(PtrValue)) 364480093f4SDimitry Andric continue; 365480093f4SDimitry Andric 366480093f4SDimitry Andric const SCEV *LSCEV = SE->getSCEVAtScope(PtrValue, L); 367480093f4SDimitry Andric const SCEVAddRecExpr *LARSCEV = dyn_cast<SCEVAddRecExpr>(LSCEV); 368480093f4SDimitry Andric if (!LARSCEV || LARSCEV->getLoop() != L) 369480093f4SDimitry Andric continue; 370480093f4SDimitry Andric 371*fe6060f1SDimitry Andric if (isValidCandidate(&J, PtrValue, PointerElementType)) 372*fe6060f1SDimitry Andric addOneCandidate(&J, LSCEV, Buckets, MaxCandidateNum); 373480093f4SDimitry Andric } 374480093f4SDimitry Andric return Buckets; 375480093f4SDimitry Andric } 376480093f4SDimitry Andric 377480093f4SDimitry Andric bool PPCLoopInstrFormPrep::prepareBaseForDispFormChain(Bucket &BucketChain, 378480093f4SDimitry Andric InstrForm Form) { 379480093f4SDimitry Andric // RemainderOffsetInfo details: 380480093f4SDimitry Andric // key: value of (Offset urem DispConstraint). For DSForm, it can 381480093f4SDimitry Andric // be [0, 4). 382480093f4SDimitry Andric // first of pair: the index of first BucketElement whose remainder is equal 383480093f4SDimitry Andric // to key. For key 0, this value must be 0. 384480093f4SDimitry Andric // second of pair: number of load/stores with the same remainder. 385480093f4SDimitry Andric DenseMap<unsigned, std::pair<unsigned, unsigned>> RemainderOffsetInfo; 386480093f4SDimitry Andric 387480093f4SDimitry Andric for (unsigned j = 0, je = BucketChain.Elements.size(); j != je; ++j) { 388480093f4SDimitry Andric if (!BucketChain.Elements[j].Offset) 389480093f4SDimitry Andric RemainderOffsetInfo[0] = std::make_pair(0, 1); 390480093f4SDimitry Andric else { 391480093f4SDimitry Andric unsigned Remainder = 392480093f4SDimitry Andric BucketChain.Elements[j].Offset->getAPInt().urem(Form); 393480093f4SDimitry Andric if (RemainderOffsetInfo.find(Remainder) == RemainderOffsetInfo.end()) 394480093f4SDimitry Andric RemainderOffsetInfo[Remainder] = std::make_pair(j, 1); 395480093f4SDimitry Andric else 396480093f4SDimitry Andric RemainderOffsetInfo[Remainder].second++; 397480093f4SDimitry Andric } 398480093f4SDimitry Andric } 399480093f4SDimitry Andric // Currently we choose the most profitable base as the one which has the max 400480093f4SDimitry Andric // number of load/store with same remainder. 401480093f4SDimitry Andric // FIXME: adjust the base selection strategy according to load/store offset 402480093f4SDimitry Andric // distribution. 403480093f4SDimitry Andric // For example, if we have one candidate chain for DS form preparation, which 404480093f4SDimitry Andric // contains following load/stores with different remainders: 405480093f4SDimitry Andric // 1: 10 load/store whose remainder is 1; 406480093f4SDimitry Andric // 2: 9 load/store whose remainder is 2; 407480093f4SDimitry Andric // 3: 1 for remainder 3 and 0 for remainder 0; 408480093f4SDimitry Andric // Now we will choose the first load/store whose remainder is 1 as base and 409480093f4SDimitry Andric // adjust all other load/stores according to new base, so we will get 10 DS 410480093f4SDimitry Andric // form and 10 X form. 411480093f4SDimitry Andric // But we should be more clever, for this case we could use two bases, one for 412480093f4SDimitry Andric // remainder 1 and the other for remainder 2, thus we could get 19 DS form and 1 413480093f4SDimitry Andric // X form. 414480093f4SDimitry Andric unsigned MaxCountRemainder = 0; 415480093f4SDimitry Andric for (unsigned j = 0; j < (unsigned)Form; j++) 416480093f4SDimitry Andric if ((RemainderOffsetInfo.find(j) != RemainderOffsetInfo.end()) && 417480093f4SDimitry Andric RemainderOffsetInfo[j].second > 418480093f4SDimitry Andric RemainderOffsetInfo[MaxCountRemainder].second) 419480093f4SDimitry Andric MaxCountRemainder = j; 420480093f4SDimitry Andric 421480093f4SDimitry Andric // Abort when there are too few insts with common base. 422480093f4SDimitry Andric if (RemainderOffsetInfo[MaxCountRemainder].second < DispFormPrepMinThreshold) 423480093f4SDimitry Andric return false; 424480093f4SDimitry Andric 425480093f4SDimitry Andric // If the first value is most profitable, no needed to adjust BucketChain 426480093f4SDimitry Andric // elements as they are substracted the first value when collecting. 427480093f4SDimitry Andric if (MaxCountRemainder == 0) 428480093f4SDimitry Andric return true; 429480093f4SDimitry Andric 430480093f4SDimitry Andric // Adjust load/store to the new chosen base. 431480093f4SDimitry Andric const SCEV *Offset = 432480093f4SDimitry Andric BucketChain.Elements[RemainderOffsetInfo[MaxCountRemainder].first].Offset; 433480093f4SDimitry Andric BucketChain.BaseSCEV = SE->getAddExpr(BucketChain.BaseSCEV, Offset); 434480093f4SDimitry Andric for (auto &E : BucketChain.Elements) { 435480093f4SDimitry Andric if (E.Offset) 436480093f4SDimitry Andric E.Offset = cast<SCEVConstant>(SE->getMinusSCEV(E.Offset, Offset)); 437480093f4SDimitry Andric else 438480093f4SDimitry Andric E.Offset = cast<SCEVConstant>(SE->getNegativeSCEV(Offset)); 439480093f4SDimitry Andric } 440480093f4SDimitry Andric 441480093f4SDimitry Andric std::swap(BucketChain.Elements[RemainderOffsetInfo[MaxCountRemainder].first], 442480093f4SDimitry Andric BucketChain.Elements[0]); 443480093f4SDimitry Andric return true; 444480093f4SDimitry Andric } 445480093f4SDimitry Andric 446480093f4SDimitry Andric // FIXME: implement a more clever base choosing policy. 447480093f4SDimitry Andric // Currently we always choose an exist load/store offset. This maybe lead to 448480093f4SDimitry Andric // suboptimal code sequences. For example, for one DS chain with offsets 449480093f4SDimitry Andric // {-32769, 2003, 2007, 2011}, we choose -32769 as base offset, and left disp 450480093f4SDimitry Andric // for load/stores are {0, 34772, 34776, 34780}. Though each offset now is a 451480093f4SDimitry Andric // multipler of 4, it cannot be represented by sint16. 452480093f4SDimitry Andric bool PPCLoopInstrFormPrep::prepareBaseForUpdateFormChain(Bucket &BucketChain) { 453480093f4SDimitry Andric // We have a choice now of which instruction's memory operand we use as the 454480093f4SDimitry Andric // base for the generated PHI. Always picking the first instruction in each 455480093f4SDimitry Andric // bucket does not work well, specifically because that instruction might 456480093f4SDimitry Andric // be a prefetch (and there are no pre-increment dcbt variants). Otherwise, 457480093f4SDimitry Andric // the choice is somewhat arbitrary, because the backend will happily 458480093f4SDimitry Andric // generate direct offsets from both the pre-incremented and 459480093f4SDimitry Andric // post-incremented pointer values. Thus, we'll pick the first non-prefetch 460480093f4SDimitry Andric // instruction in each bucket, and adjust the recurrence and other offsets 461480093f4SDimitry Andric // accordingly. 462480093f4SDimitry Andric for (int j = 0, je = BucketChain.Elements.size(); j != je; ++j) { 463480093f4SDimitry Andric if (auto *II = dyn_cast<IntrinsicInst>(BucketChain.Elements[j].Instr)) 464480093f4SDimitry Andric if (II->getIntrinsicID() == Intrinsic::prefetch) 465480093f4SDimitry Andric continue; 466480093f4SDimitry Andric 467480093f4SDimitry Andric // If we'd otherwise pick the first element anyway, there's nothing to do. 468480093f4SDimitry Andric if (j == 0) 469480093f4SDimitry Andric break; 470480093f4SDimitry Andric 471480093f4SDimitry Andric // If our chosen element has no offset from the base pointer, there's 472480093f4SDimitry Andric // nothing to do. 473480093f4SDimitry Andric if (!BucketChain.Elements[j].Offset || 474480093f4SDimitry Andric BucketChain.Elements[j].Offset->isZero()) 475480093f4SDimitry Andric break; 476480093f4SDimitry Andric 477480093f4SDimitry Andric const SCEV *Offset = BucketChain.Elements[j].Offset; 478480093f4SDimitry Andric BucketChain.BaseSCEV = SE->getAddExpr(BucketChain.BaseSCEV, Offset); 479480093f4SDimitry Andric for (auto &E : BucketChain.Elements) { 480480093f4SDimitry Andric if (E.Offset) 481480093f4SDimitry Andric E.Offset = cast<SCEVConstant>(SE->getMinusSCEV(E.Offset, Offset)); 482480093f4SDimitry Andric else 483480093f4SDimitry Andric E.Offset = cast<SCEVConstant>(SE->getNegativeSCEV(Offset)); 484480093f4SDimitry Andric } 485480093f4SDimitry Andric 486480093f4SDimitry Andric std::swap(BucketChain.Elements[j], BucketChain.Elements[0]); 487480093f4SDimitry Andric break; 488480093f4SDimitry Andric } 489480093f4SDimitry Andric return true; 490480093f4SDimitry Andric } 491480093f4SDimitry Andric 492480093f4SDimitry Andric bool PPCLoopInstrFormPrep::rewriteLoadStores(Loop *L, Bucket &BucketChain, 493480093f4SDimitry Andric SmallSet<BasicBlock *, 16> &BBChanged, 494480093f4SDimitry Andric InstrForm Form) { 495480093f4SDimitry Andric bool MadeChange = false; 496480093f4SDimitry Andric const SCEVAddRecExpr *BasePtrSCEV = 497480093f4SDimitry Andric cast<SCEVAddRecExpr>(BucketChain.BaseSCEV); 498480093f4SDimitry Andric if (!BasePtrSCEV->isAffine()) 499480093f4SDimitry Andric return MadeChange; 500480093f4SDimitry Andric 501480093f4SDimitry Andric LLVM_DEBUG(dbgs() << "PIP: Transforming: " << *BasePtrSCEV << "\n"); 502480093f4SDimitry Andric 503480093f4SDimitry Andric assert(BasePtrSCEV->getLoop() == L && "AddRec for the wrong loop?"); 504480093f4SDimitry Andric 505480093f4SDimitry Andric // The instruction corresponding to the Bucket's BaseSCEV must be the first 506480093f4SDimitry Andric // in the vector of elements. 507480093f4SDimitry Andric Instruction *MemI = BucketChain.Elements.begin()->Instr; 508480093f4SDimitry Andric Value *BasePtr = GetPointerOperand(MemI); 509480093f4SDimitry Andric assert(BasePtr && "No pointer operand"); 510480093f4SDimitry Andric 511480093f4SDimitry Andric Type *I8Ty = Type::getInt8Ty(MemI->getParent()->getContext()); 512480093f4SDimitry Andric Type *I8PtrTy = Type::getInt8PtrTy(MemI->getParent()->getContext(), 513480093f4SDimitry Andric BasePtr->getType()->getPointerAddressSpace()); 514480093f4SDimitry Andric 515480093f4SDimitry Andric if (!SE->isLoopInvariant(BasePtrSCEV->getStart(), L)) 516480093f4SDimitry Andric return MadeChange; 517480093f4SDimitry Andric 518480093f4SDimitry Andric const SCEVConstant *BasePtrIncSCEV = 519480093f4SDimitry Andric dyn_cast<SCEVConstant>(BasePtrSCEV->getStepRecurrence(*SE)); 520480093f4SDimitry Andric if (!BasePtrIncSCEV) 521480093f4SDimitry Andric return MadeChange; 522480093f4SDimitry Andric 523480093f4SDimitry Andric // For some DS form load/store instructions, it can also be an update form, 524480093f4SDimitry Andric // if the stride is a multipler of 4. Use update form if prefer it. 525480093f4SDimitry Andric bool CanPreInc = (Form == UpdateForm || 526480093f4SDimitry Andric ((Form == DSForm) && !BasePtrIncSCEV->getAPInt().urem(4) && 527480093f4SDimitry Andric PreferUpdateForm)); 528480093f4SDimitry Andric const SCEV *BasePtrStartSCEV = nullptr; 529480093f4SDimitry Andric if (CanPreInc) 530480093f4SDimitry Andric BasePtrStartSCEV = 531480093f4SDimitry Andric SE->getMinusSCEV(BasePtrSCEV->getStart(), BasePtrIncSCEV); 532480093f4SDimitry Andric else 533480093f4SDimitry Andric BasePtrStartSCEV = BasePtrSCEV->getStart(); 534480093f4SDimitry Andric 535480093f4SDimitry Andric if (!isSafeToExpand(BasePtrStartSCEV, *SE)) 536480093f4SDimitry Andric return MadeChange; 537480093f4SDimitry Andric 538480093f4SDimitry Andric if (alreadyPrepared(L, MemI, BasePtrStartSCEV, BasePtrIncSCEV, Form)) 539480093f4SDimitry Andric return MadeChange; 540480093f4SDimitry Andric 541480093f4SDimitry Andric LLVM_DEBUG(dbgs() << "PIP: New start is: " << *BasePtrStartSCEV << "\n"); 542480093f4SDimitry Andric 543480093f4SDimitry Andric BasicBlock *Header = L->getHeader(); 544480093f4SDimitry Andric unsigned HeaderLoopPredCount = pred_size(Header); 545480093f4SDimitry Andric BasicBlock *LoopPredecessor = L->getLoopPredecessor(); 546480093f4SDimitry Andric 547480093f4SDimitry Andric PHINode *NewPHI = 548480093f4SDimitry Andric PHINode::Create(I8PtrTy, HeaderLoopPredCount, 549480093f4SDimitry Andric getInstrName(MemI, PHINodeNameSuffix), 550480093f4SDimitry Andric Header->getFirstNonPHI()); 551480093f4SDimitry Andric 552480093f4SDimitry Andric SCEVExpander SCEVE(*SE, Header->getModule()->getDataLayout(), "pistart"); 553480093f4SDimitry Andric Value *BasePtrStart = SCEVE.expandCodeFor(BasePtrStartSCEV, I8PtrTy, 554480093f4SDimitry Andric LoopPredecessor->getTerminator()); 555480093f4SDimitry Andric 556480093f4SDimitry Andric // Note that LoopPredecessor might occur in the predecessor list multiple 557480093f4SDimitry Andric // times, and we need to add it the right number of times. 558480093f4SDimitry Andric for (auto PI : predecessors(Header)) { 559480093f4SDimitry Andric if (PI != LoopPredecessor) 560480093f4SDimitry Andric continue; 561480093f4SDimitry Andric 562480093f4SDimitry Andric NewPHI->addIncoming(BasePtrStart, LoopPredecessor); 563480093f4SDimitry Andric } 564480093f4SDimitry Andric 565480093f4SDimitry Andric Instruction *PtrInc = nullptr; 566480093f4SDimitry Andric Instruction *NewBasePtr = nullptr; 567480093f4SDimitry Andric if (CanPreInc) { 568480093f4SDimitry Andric Instruction *InsPoint = &*Header->getFirstInsertionPt(); 569480093f4SDimitry Andric PtrInc = GetElementPtrInst::Create( 570480093f4SDimitry Andric I8Ty, NewPHI, BasePtrIncSCEV->getValue(), 571480093f4SDimitry Andric getInstrName(MemI, GEPNodeIncNameSuffix), InsPoint); 572480093f4SDimitry Andric cast<GetElementPtrInst>(PtrInc)->setIsInBounds(IsPtrInBounds(BasePtr)); 573480093f4SDimitry Andric for (auto PI : predecessors(Header)) { 574480093f4SDimitry Andric if (PI == LoopPredecessor) 575480093f4SDimitry Andric continue; 576480093f4SDimitry Andric 577480093f4SDimitry Andric NewPHI->addIncoming(PtrInc, PI); 578480093f4SDimitry Andric } 579480093f4SDimitry Andric if (PtrInc->getType() != BasePtr->getType()) 580480093f4SDimitry Andric NewBasePtr = new BitCastInst( 581480093f4SDimitry Andric PtrInc, BasePtr->getType(), 582480093f4SDimitry Andric getInstrName(PtrInc, CastNodeNameSuffix), InsPoint); 583480093f4SDimitry Andric else 584480093f4SDimitry Andric NewBasePtr = PtrInc; 585480093f4SDimitry Andric } else { 586480093f4SDimitry Andric // Note that LoopPredecessor might occur in the predecessor list multiple 587480093f4SDimitry Andric // times, and we need to make sure no more incoming value for them in PHI. 588480093f4SDimitry Andric for (auto PI : predecessors(Header)) { 589480093f4SDimitry Andric if (PI == LoopPredecessor) 590480093f4SDimitry Andric continue; 591480093f4SDimitry Andric 592480093f4SDimitry Andric // For the latch predecessor, we need to insert a GEP just before the 593480093f4SDimitry Andric // terminator to increase the address. 594480093f4SDimitry Andric BasicBlock *BB = PI; 595480093f4SDimitry Andric Instruction *InsPoint = BB->getTerminator(); 596480093f4SDimitry Andric PtrInc = GetElementPtrInst::Create( 597480093f4SDimitry Andric I8Ty, NewPHI, BasePtrIncSCEV->getValue(), 598480093f4SDimitry Andric getInstrName(MemI, GEPNodeIncNameSuffix), InsPoint); 599480093f4SDimitry Andric 600480093f4SDimitry Andric cast<GetElementPtrInst>(PtrInc)->setIsInBounds(IsPtrInBounds(BasePtr)); 601480093f4SDimitry Andric 602480093f4SDimitry Andric NewPHI->addIncoming(PtrInc, PI); 603480093f4SDimitry Andric } 604480093f4SDimitry Andric PtrInc = NewPHI; 605480093f4SDimitry Andric if (NewPHI->getType() != BasePtr->getType()) 606480093f4SDimitry Andric NewBasePtr = 607480093f4SDimitry Andric new BitCastInst(NewPHI, BasePtr->getType(), 608480093f4SDimitry Andric getInstrName(NewPHI, CastNodeNameSuffix), 609480093f4SDimitry Andric &*Header->getFirstInsertionPt()); 610480093f4SDimitry Andric else 611480093f4SDimitry Andric NewBasePtr = NewPHI; 612480093f4SDimitry Andric } 613480093f4SDimitry Andric 614e8d8bef9SDimitry Andric // Clear the rewriter cache, because values that are in the rewriter's cache 615e8d8bef9SDimitry Andric // can be deleted below, causing the AssertingVH in the cache to trigger. 616e8d8bef9SDimitry Andric SCEVE.clear(); 617e8d8bef9SDimitry Andric 618480093f4SDimitry Andric if (Instruction *IDel = dyn_cast<Instruction>(BasePtr)) 619480093f4SDimitry Andric BBChanged.insert(IDel->getParent()); 620480093f4SDimitry Andric BasePtr->replaceAllUsesWith(NewBasePtr); 621480093f4SDimitry Andric RecursivelyDeleteTriviallyDeadInstructions(BasePtr); 622480093f4SDimitry Andric 623480093f4SDimitry Andric // Keep track of the replacement pointer values we've inserted so that we 624480093f4SDimitry Andric // don't generate more pointer values than necessary. 625480093f4SDimitry Andric SmallPtrSet<Value *, 16> NewPtrs; 626480093f4SDimitry Andric NewPtrs.insert(NewBasePtr); 627480093f4SDimitry Andric 628480093f4SDimitry Andric for (auto I = std::next(BucketChain.Elements.begin()), 629480093f4SDimitry Andric IE = BucketChain.Elements.end(); I != IE; ++I) { 630480093f4SDimitry Andric Value *Ptr = GetPointerOperand(I->Instr); 631480093f4SDimitry Andric assert(Ptr && "No pointer operand"); 632480093f4SDimitry Andric if (NewPtrs.count(Ptr)) 633480093f4SDimitry Andric continue; 634480093f4SDimitry Andric 635480093f4SDimitry Andric Instruction *RealNewPtr; 636480093f4SDimitry Andric if (!I->Offset || I->Offset->getValue()->isZero()) { 637480093f4SDimitry Andric RealNewPtr = NewBasePtr; 638480093f4SDimitry Andric } else { 639480093f4SDimitry Andric Instruction *PtrIP = dyn_cast<Instruction>(Ptr); 640480093f4SDimitry Andric if (PtrIP && isa<Instruction>(NewBasePtr) && 641480093f4SDimitry Andric cast<Instruction>(NewBasePtr)->getParent() == PtrIP->getParent()) 642480093f4SDimitry Andric PtrIP = nullptr; 643480093f4SDimitry Andric else if (PtrIP && isa<PHINode>(PtrIP)) 644480093f4SDimitry Andric PtrIP = &*PtrIP->getParent()->getFirstInsertionPt(); 645480093f4SDimitry Andric else if (!PtrIP) 646480093f4SDimitry Andric PtrIP = I->Instr; 647480093f4SDimitry Andric 648480093f4SDimitry Andric GetElementPtrInst *NewPtr = GetElementPtrInst::Create( 649480093f4SDimitry Andric I8Ty, PtrInc, I->Offset->getValue(), 650480093f4SDimitry Andric getInstrName(I->Instr, GEPNodeOffNameSuffix), PtrIP); 651480093f4SDimitry Andric if (!PtrIP) 652480093f4SDimitry Andric NewPtr->insertAfter(cast<Instruction>(PtrInc)); 653480093f4SDimitry Andric NewPtr->setIsInBounds(IsPtrInBounds(Ptr)); 654480093f4SDimitry Andric RealNewPtr = NewPtr; 655480093f4SDimitry Andric } 656480093f4SDimitry Andric 657480093f4SDimitry Andric if (Instruction *IDel = dyn_cast<Instruction>(Ptr)) 658480093f4SDimitry Andric BBChanged.insert(IDel->getParent()); 659480093f4SDimitry Andric 660480093f4SDimitry Andric Instruction *ReplNewPtr; 661480093f4SDimitry Andric if (Ptr->getType() != RealNewPtr->getType()) { 662480093f4SDimitry Andric ReplNewPtr = new BitCastInst(RealNewPtr, Ptr->getType(), 663480093f4SDimitry Andric getInstrName(Ptr, CastNodeNameSuffix)); 664480093f4SDimitry Andric ReplNewPtr->insertAfter(RealNewPtr); 665480093f4SDimitry Andric } else 666480093f4SDimitry Andric ReplNewPtr = RealNewPtr; 667480093f4SDimitry Andric 668480093f4SDimitry Andric Ptr->replaceAllUsesWith(ReplNewPtr); 669480093f4SDimitry Andric RecursivelyDeleteTriviallyDeadInstructions(Ptr); 670480093f4SDimitry Andric 671480093f4SDimitry Andric NewPtrs.insert(RealNewPtr); 672480093f4SDimitry Andric } 673480093f4SDimitry Andric 674480093f4SDimitry Andric MadeChange = true; 675480093f4SDimitry Andric 676480093f4SDimitry Andric SuccPrepCount++; 677480093f4SDimitry Andric 678480093f4SDimitry Andric if (Form == DSForm && !CanPreInc) 679480093f4SDimitry Andric DSFormChainRewritten++; 680480093f4SDimitry Andric else if (Form == DQForm) 681480093f4SDimitry Andric DQFormChainRewritten++; 682480093f4SDimitry Andric else if (Form == UpdateForm || (Form == DSForm && CanPreInc)) 683480093f4SDimitry Andric UpdFormChainRewritten++; 684480093f4SDimitry Andric 685480093f4SDimitry Andric return MadeChange; 686480093f4SDimitry Andric } 687480093f4SDimitry Andric 688480093f4SDimitry Andric bool PPCLoopInstrFormPrep::updateFormPrep(Loop *L, 689480093f4SDimitry Andric SmallVector<Bucket, 16> &Buckets) { 690480093f4SDimitry Andric bool MadeChange = false; 691480093f4SDimitry Andric if (Buckets.empty()) 692480093f4SDimitry Andric return MadeChange; 693480093f4SDimitry Andric SmallSet<BasicBlock *, 16> BBChanged; 694480093f4SDimitry Andric for (auto &Bucket : Buckets) 695480093f4SDimitry Andric // The base address of each bucket is transformed into a phi and the others 696480093f4SDimitry Andric // are rewritten based on new base. 697480093f4SDimitry Andric if (prepareBaseForUpdateFormChain(Bucket)) 698480093f4SDimitry Andric MadeChange |= rewriteLoadStores(L, Bucket, BBChanged, UpdateForm); 699480093f4SDimitry Andric 700480093f4SDimitry Andric if (MadeChange) 701480093f4SDimitry Andric for (auto &BB : L->blocks()) 702480093f4SDimitry Andric if (BBChanged.count(BB)) 703480093f4SDimitry Andric DeleteDeadPHIs(BB); 704480093f4SDimitry Andric return MadeChange; 705480093f4SDimitry Andric } 706480093f4SDimitry Andric 707480093f4SDimitry Andric bool PPCLoopInstrFormPrep::dispFormPrep(Loop *L, SmallVector<Bucket, 16> &Buckets, 708480093f4SDimitry Andric InstrForm Form) { 709480093f4SDimitry Andric bool MadeChange = false; 710480093f4SDimitry Andric 711480093f4SDimitry Andric if (Buckets.empty()) 712480093f4SDimitry Andric return MadeChange; 713480093f4SDimitry Andric 714480093f4SDimitry Andric SmallSet<BasicBlock *, 16> BBChanged; 715480093f4SDimitry Andric for (auto &Bucket : Buckets) { 716480093f4SDimitry Andric if (Bucket.Elements.size() < DispFormPrepMinThreshold) 717480093f4SDimitry Andric continue; 718480093f4SDimitry Andric if (prepareBaseForDispFormChain(Bucket, Form)) 719480093f4SDimitry Andric MadeChange |= rewriteLoadStores(L, Bucket, BBChanged, Form); 720480093f4SDimitry Andric } 721480093f4SDimitry Andric 722480093f4SDimitry Andric if (MadeChange) 723480093f4SDimitry Andric for (auto &BB : L->blocks()) 724480093f4SDimitry Andric if (BBChanged.count(BB)) 725480093f4SDimitry Andric DeleteDeadPHIs(BB); 726480093f4SDimitry Andric return MadeChange; 727480093f4SDimitry Andric } 728480093f4SDimitry Andric 729480093f4SDimitry Andric // In order to prepare for the preferred instruction form, a PHI is added. 730480093f4SDimitry Andric // This function will check to see if that PHI already exists and will return 731480093f4SDimitry Andric // true if it found an existing PHI with the matched start and increment as the 732480093f4SDimitry Andric // one we wanted to create. 733480093f4SDimitry Andric bool PPCLoopInstrFormPrep::alreadyPrepared(Loop *L, Instruction* MemI, 734480093f4SDimitry Andric const SCEV *BasePtrStartSCEV, 735480093f4SDimitry Andric const SCEVConstant *BasePtrIncSCEV, 736480093f4SDimitry Andric InstrForm Form) { 737480093f4SDimitry Andric BasicBlock *BB = MemI->getParent(); 738480093f4SDimitry Andric if (!BB) 739480093f4SDimitry Andric return false; 740480093f4SDimitry Andric 741480093f4SDimitry Andric BasicBlock *PredBB = L->getLoopPredecessor(); 742480093f4SDimitry Andric BasicBlock *LatchBB = L->getLoopLatch(); 743480093f4SDimitry Andric 744480093f4SDimitry Andric if (!PredBB || !LatchBB) 745480093f4SDimitry Andric return false; 746480093f4SDimitry Andric 747480093f4SDimitry Andric // Run through the PHIs and see if we have some that looks like a preparation 748480093f4SDimitry Andric iterator_range<BasicBlock::phi_iterator> PHIIter = BB->phis(); 749480093f4SDimitry Andric for (auto & CurrentPHI : PHIIter) { 750480093f4SDimitry Andric PHINode *CurrentPHINode = dyn_cast<PHINode>(&CurrentPHI); 751480093f4SDimitry Andric if (!CurrentPHINode) 752480093f4SDimitry Andric continue; 753480093f4SDimitry Andric 754480093f4SDimitry Andric if (!SE->isSCEVable(CurrentPHINode->getType())) 755480093f4SDimitry Andric continue; 756480093f4SDimitry Andric 757480093f4SDimitry Andric const SCEV *PHISCEV = SE->getSCEVAtScope(CurrentPHINode, L); 758480093f4SDimitry Andric 759480093f4SDimitry Andric const SCEVAddRecExpr *PHIBasePtrSCEV = dyn_cast<SCEVAddRecExpr>(PHISCEV); 760480093f4SDimitry Andric if (!PHIBasePtrSCEV) 761480093f4SDimitry Andric continue; 762480093f4SDimitry Andric 763480093f4SDimitry Andric const SCEVConstant *PHIBasePtrIncSCEV = 764480093f4SDimitry Andric dyn_cast<SCEVConstant>(PHIBasePtrSCEV->getStepRecurrence(*SE)); 765480093f4SDimitry Andric if (!PHIBasePtrIncSCEV) 766480093f4SDimitry Andric continue; 767480093f4SDimitry Andric 768480093f4SDimitry Andric if (CurrentPHINode->getNumIncomingValues() == 2) { 769480093f4SDimitry Andric if ((CurrentPHINode->getIncomingBlock(0) == LatchBB && 770480093f4SDimitry Andric CurrentPHINode->getIncomingBlock(1) == PredBB) || 771480093f4SDimitry Andric (CurrentPHINode->getIncomingBlock(1) == LatchBB && 772480093f4SDimitry Andric CurrentPHINode->getIncomingBlock(0) == PredBB)) { 773480093f4SDimitry Andric if (PHIBasePtrIncSCEV == BasePtrIncSCEV) { 774480093f4SDimitry Andric // The existing PHI (CurrentPHINode) has the same start and increment 775480093f4SDimitry Andric // as the PHI that we wanted to create. 776480093f4SDimitry Andric if (Form == UpdateForm && 777480093f4SDimitry Andric PHIBasePtrSCEV->getStart() == BasePtrStartSCEV) { 778480093f4SDimitry Andric ++PHINodeAlreadyExistsUpdate; 779480093f4SDimitry Andric return true; 780480093f4SDimitry Andric } 781480093f4SDimitry Andric if (Form == DSForm || Form == DQForm) { 782480093f4SDimitry Andric const SCEVConstant *Diff = dyn_cast<SCEVConstant>( 783480093f4SDimitry Andric SE->getMinusSCEV(PHIBasePtrSCEV->getStart(), BasePtrStartSCEV)); 784480093f4SDimitry Andric if (Diff && !Diff->getAPInt().urem(Form)) { 785480093f4SDimitry Andric if (Form == DSForm) 786480093f4SDimitry Andric ++PHINodeAlreadyExistsDS; 787480093f4SDimitry Andric else 788480093f4SDimitry Andric ++PHINodeAlreadyExistsDQ; 789480093f4SDimitry Andric return true; 790480093f4SDimitry Andric } 791480093f4SDimitry Andric } 792480093f4SDimitry Andric } 793480093f4SDimitry Andric } 794480093f4SDimitry Andric } 795480093f4SDimitry Andric } 796480093f4SDimitry Andric return false; 797480093f4SDimitry Andric } 798480093f4SDimitry Andric 799480093f4SDimitry Andric bool PPCLoopInstrFormPrep::runOnLoop(Loop *L) { 800480093f4SDimitry Andric bool MadeChange = false; 801480093f4SDimitry Andric 802480093f4SDimitry Andric // Only prep. the inner-most loop 803e8d8bef9SDimitry Andric if (!L->isInnermost()) 804480093f4SDimitry Andric return MadeChange; 805480093f4SDimitry Andric 806480093f4SDimitry Andric // Return if already done enough preparation. 807480093f4SDimitry Andric if (SuccPrepCount >= MaxVarsPrep) 808480093f4SDimitry Andric return MadeChange; 809480093f4SDimitry Andric 810480093f4SDimitry Andric LLVM_DEBUG(dbgs() << "PIP: Examining: " << *L << "\n"); 811480093f4SDimitry Andric 812480093f4SDimitry Andric BasicBlock *LoopPredecessor = L->getLoopPredecessor(); 813480093f4SDimitry Andric // If there is no loop predecessor, or the loop predecessor's terminator 814480093f4SDimitry Andric // returns a value (which might contribute to determining the loop's 815480093f4SDimitry Andric // iteration space), insert a new preheader for the loop. 816480093f4SDimitry Andric if (!LoopPredecessor || 817480093f4SDimitry Andric !LoopPredecessor->getTerminator()->getType()->isVoidTy()) { 818480093f4SDimitry Andric LoopPredecessor = InsertPreheaderForLoop(L, DT, LI, nullptr, PreserveLCSSA); 819480093f4SDimitry Andric if (LoopPredecessor) 820480093f4SDimitry Andric MadeChange = true; 821480093f4SDimitry Andric } 822480093f4SDimitry Andric if (!LoopPredecessor) { 823480093f4SDimitry Andric LLVM_DEBUG(dbgs() << "PIP fails since no predecessor for current loop.\n"); 824480093f4SDimitry Andric return MadeChange; 825480093f4SDimitry Andric } 826480093f4SDimitry Andric // Check if a load/store has update form. This lambda is used by function 827480093f4SDimitry Andric // collectCandidates which can collect candidates for types defined by lambda. 828*fe6060f1SDimitry Andric auto isUpdateFormCandidate = [&](const Instruction *I, const Value *PtrValue, 829*fe6060f1SDimitry Andric const Type *PointerElementType) { 830480093f4SDimitry Andric assert((PtrValue && I) && "Invalid parameter!"); 831480093f4SDimitry Andric // There are no update forms for Altivec vector load/stores. 832*fe6060f1SDimitry Andric if (ST && ST->hasAltivec() && PointerElementType->isVectorTy()) 833480093f4SDimitry Andric return false; 834e8d8bef9SDimitry Andric // There are no update forms for P10 lxvp/stxvp intrinsic. 835e8d8bef9SDimitry Andric auto *II = dyn_cast<IntrinsicInst>(I); 836e8d8bef9SDimitry Andric if (II && ((II->getIntrinsicID() == Intrinsic::ppc_vsx_lxvp) || 837e8d8bef9SDimitry Andric II->getIntrinsicID() == Intrinsic::ppc_vsx_stxvp)) 838e8d8bef9SDimitry Andric return false; 839480093f4SDimitry Andric // See getPreIndexedAddressParts, the displacement for LDU/STDU has to 840480093f4SDimitry Andric // be 4's multiple (DS-form). For i64 loads/stores when the displacement 841480093f4SDimitry Andric // fits in a 16-bit signed field but isn't a multiple of 4, it will be 842480093f4SDimitry Andric // useless and possible to break some original well-form addressing mode 843480093f4SDimitry Andric // to make this pre-inc prep for it. 844*fe6060f1SDimitry Andric if (PointerElementType->isIntegerTy(64)) { 845480093f4SDimitry Andric const SCEV *LSCEV = SE->getSCEVAtScope(const_cast<Value *>(PtrValue), L); 846480093f4SDimitry Andric const SCEVAddRecExpr *LARSCEV = dyn_cast<SCEVAddRecExpr>(LSCEV); 847480093f4SDimitry Andric if (!LARSCEV || LARSCEV->getLoop() != L) 848480093f4SDimitry Andric return false; 849480093f4SDimitry Andric if (const SCEVConstant *StepConst = 850480093f4SDimitry Andric dyn_cast<SCEVConstant>(LARSCEV->getStepRecurrence(*SE))) { 851480093f4SDimitry Andric const APInt &ConstInt = StepConst->getValue()->getValue(); 852480093f4SDimitry Andric if (ConstInt.isSignedIntN(16) && ConstInt.srem(4) != 0) 853480093f4SDimitry Andric return false; 854480093f4SDimitry Andric } 855480093f4SDimitry Andric } 856480093f4SDimitry Andric return true; 857480093f4SDimitry Andric }; 858480093f4SDimitry Andric 859480093f4SDimitry Andric // Check if a load/store has DS form. 860*fe6060f1SDimitry Andric auto isDSFormCandidate = [](const Instruction *I, const Value *PtrValue, 861*fe6060f1SDimitry Andric const Type *PointerElementType) { 862480093f4SDimitry Andric assert((PtrValue && I) && "Invalid parameter!"); 863480093f4SDimitry Andric if (isa<IntrinsicInst>(I)) 864480093f4SDimitry Andric return false; 865480093f4SDimitry Andric return (PointerElementType->isIntegerTy(64)) || 866480093f4SDimitry Andric (PointerElementType->isFloatTy()) || 867480093f4SDimitry Andric (PointerElementType->isDoubleTy()) || 868480093f4SDimitry Andric (PointerElementType->isIntegerTy(32) && 869480093f4SDimitry Andric llvm::any_of(I->users(), 870480093f4SDimitry Andric [](const User *U) { return isa<SExtInst>(U); })); 871480093f4SDimitry Andric }; 872480093f4SDimitry Andric 873480093f4SDimitry Andric // Check if a load/store has DQ form. 874*fe6060f1SDimitry Andric auto isDQFormCandidate = [&](const Instruction *I, const Value *PtrValue, 875*fe6060f1SDimitry Andric const Type *PointerElementType) { 876480093f4SDimitry Andric assert((PtrValue && I) && "Invalid parameter!"); 877e8d8bef9SDimitry Andric // Check if it is a P10 lxvp/stxvp intrinsic. 878e8d8bef9SDimitry Andric auto *II = dyn_cast<IntrinsicInst>(I); 879e8d8bef9SDimitry Andric if (II) 880e8d8bef9SDimitry Andric return II->getIntrinsicID() == Intrinsic::ppc_vsx_lxvp || 881e8d8bef9SDimitry Andric II->getIntrinsicID() == Intrinsic::ppc_vsx_stxvp; 882e8d8bef9SDimitry Andric // Check if it is a P9 vector load/store. 883*fe6060f1SDimitry Andric return ST && ST->hasP9Vector() && (PointerElementType->isVectorTy()); 884480093f4SDimitry Andric }; 885480093f4SDimitry Andric 886480093f4SDimitry Andric // intrinsic for update form. 887480093f4SDimitry Andric SmallVector<Bucket, 16> UpdateFormBuckets = 888480093f4SDimitry Andric collectCandidates(L, isUpdateFormCandidate, MaxVarsUpdateForm); 889480093f4SDimitry Andric 890480093f4SDimitry Andric // Prepare for update form. 891480093f4SDimitry Andric if (!UpdateFormBuckets.empty()) 892480093f4SDimitry Andric MadeChange |= updateFormPrep(L, UpdateFormBuckets); 893480093f4SDimitry Andric 894480093f4SDimitry Andric // Collect buckets of comparable addresses used by loads and stores for DS 895480093f4SDimitry Andric // form. 896480093f4SDimitry Andric SmallVector<Bucket, 16> DSFormBuckets = 897480093f4SDimitry Andric collectCandidates(L, isDSFormCandidate, MaxVarsDSForm); 898480093f4SDimitry Andric 899480093f4SDimitry Andric // Prepare for DS form. 900480093f4SDimitry Andric if (!DSFormBuckets.empty()) 901480093f4SDimitry Andric MadeChange |= dispFormPrep(L, DSFormBuckets, DSForm); 902480093f4SDimitry Andric 903480093f4SDimitry Andric // Collect buckets of comparable addresses used by loads and stores for DQ 904480093f4SDimitry Andric // form. 905480093f4SDimitry Andric SmallVector<Bucket, 16> DQFormBuckets = 906480093f4SDimitry Andric collectCandidates(L, isDQFormCandidate, MaxVarsDQForm); 907480093f4SDimitry Andric 908480093f4SDimitry Andric // Prepare for DQ form. 909480093f4SDimitry Andric if (!DQFormBuckets.empty()) 910480093f4SDimitry Andric MadeChange |= dispFormPrep(L, DQFormBuckets, DQForm); 911480093f4SDimitry Andric 912480093f4SDimitry Andric return MadeChange; 913480093f4SDimitry Andric } 914