13ca95b02SDimitry Andric //===-------- LoopDataPrefetch.cpp - Loop Data Prefetching Pass -----------===//
23ca95b02SDimitry Andric //
33ca95b02SDimitry Andric // The LLVM Compiler Infrastructure
43ca95b02SDimitry Andric //
53ca95b02SDimitry Andric // This file is distributed under the University of Illinois Open Source
63ca95b02SDimitry Andric // License. See LICENSE.TXT for details.
73ca95b02SDimitry Andric //
83ca95b02SDimitry Andric //===----------------------------------------------------------------------===//
93ca95b02SDimitry Andric //
103ca95b02SDimitry Andric // This file implements a Loop Data Prefetching Pass.
113ca95b02SDimitry Andric //
123ca95b02SDimitry Andric //===----------------------------------------------------------------------===//
133ca95b02SDimitry Andric
14d88c1a5aSDimitry Andric #include "llvm/Transforms/Scalar/LoopDataPrefetch.h"
15d88c1a5aSDimitry Andric
163ca95b02SDimitry Andric #define DEBUG_TYPE "loop-data-prefetch"
173ca95b02SDimitry Andric #include "llvm/ADT/DepthFirstIterator.h"
183ca95b02SDimitry Andric #include "llvm/ADT/Statistic.h"
193ca95b02SDimitry Andric #include "llvm/Analysis/AssumptionCache.h"
203ca95b02SDimitry Andric #include "llvm/Analysis/CodeMetrics.h"
213ca95b02SDimitry Andric #include "llvm/Analysis/LoopInfo.h"
222cab237bSDimitry Andric #include "llvm/Analysis/OptimizationRemarkEmitter.h"
233ca95b02SDimitry Andric #include "llvm/Analysis/ScalarEvolution.h"
243ca95b02SDimitry Andric #include "llvm/Analysis/ScalarEvolutionExpander.h"
253ca95b02SDimitry Andric #include "llvm/Analysis/ScalarEvolutionExpressions.h"
263ca95b02SDimitry Andric #include "llvm/Analysis/TargetTransformInfo.h"
273ca95b02SDimitry Andric #include "llvm/IR/CFG.h"
283ca95b02SDimitry Andric #include "llvm/IR/Dominators.h"
293ca95b02SDimitry Andric #include "llvm/IR/Function.h"
303ca95b02SDimitry Andric #include "llvm/IR/Module.h"
313ca95b02SDimitry Andric #include "llvm/Support/CommandLine.h"
323ca95b02SDimitry Andric #include "llvm/Support/Debug.h"
33d88c1a5aSDimitry Andric #include "llvm/Transforms/Scalar.h"
343ca95b02SDimitry Andric #include "llvm/Transforms/Utils/BasicBlockUtils.h"
353ca95b02SDimitry Andric #include "llvm/Transforms/Utils/ValueMapper.h"
363ca95b02SDimitry Andric using namespace llvm;
373ca95b02SDimitry Andric
383ca95b02SDimitry Andric // By default, we limit this to creating 16 PHIs (which is a little over half
393ca95b02SDimitry Andric // of the allocatable register set).
403ca95b02SDimitry Andric static cl::opt<bool>
413ca95b02SDimitry Andric PrefetchWrites("loop-prefetch-writes", cl::Hidden, cl::init(false),
423ca95b02SDimitry Andric cl::desc("Prefetch write addresses"));
433ca95b02SDimitry Andric
443ca95b02SDimitry Andric static cl::opt<unsigned>
453ca95b02SDimitry Andric PrefetchDistance("prefetch-distance",
463ca95b02SDimitry Andric cl::desc("Number of instructions to prefetch ahead"),
473ca95b02SDimitry Andric cl::Hidden);
483ca95b02SDimitry Andric
493ca95b02SDimitry Andric static cl::opt<unsigned>
503ca95b02SDimitry Andric MinPrefetchStride("min-prefetch-stride",
513ca95b02SDimitry Andric cl::desc("Min stride to add prefetches"), cl::Hidden);
523ca95b02SDimitry Andric
533ca95b02SDimitry Andric static cl::opt<unsigned> MaxPrefetchIterationsAhead(
543ca95b02SDimitry Andric "max-prefetch-iters-ahead",
553ca95b02SDimitry Andric cl::desc("Max number of iterations to prefetch ahead"), cl::Hidden);
563ca95b02SDimitry Andric
573ca95b02SDimitry Andric STATISTIC(NumPrefetches, "Number of prefetches inserted");
583ca95b02SDimitry Andric
593ca95b02SDimitry Andric namespace {
603ca95b02SDimitry Andric
61d88c1a5aSDimitry Andric /// Loop prefetch implementation class.
62d88c1a5aSDimitry Andric class LoopDataPrefetch {
633ca95b02SDimitry Andric public:
LoopDataPrefetch(AssumptionCache * AC,LoopInfo * LI,ScalarEvolution * SE,const TargetTransformInfo * TTI,OptimizationRemarkEmitter * ORE)64d88c1a5aSDimitry Andric LoopDataPrefetch(AssumptionCache *AC, LoopInfo *LI, ScalarEvolution *SE,
65d88c1a5aSDimitry Andric const TargetTransformInfo *TTI,
66d88c1a5aSDimitry Andric OptimizationRemarkEmitter *ORE)
67d88c1a5aSDimitry Andric : AC(AC), LI(LI), SE(SE), TTI(TTI), ORE(ORE) {}
683ca95b02SDimitry Andric
69d88c1a5aSDimitry Andric bool run();
703ca95b02SDimitry Andric
713ca95b02SDimitry Andric private:
723ca95b02SDimitry Andric bool runOnLoop(Loop *L);
733ca95b02SDimitry Andric
74*4ba319b5SDimitry Andric /// Check if the stride of the accesses is large enough to
753ca95b02SDimitry Andric /// warrant a prefetch.
763ca95b02SDimitry Andric bool isStrideLargeEnough(const SCEVAddRecExpr *AR);
773ca95b02SDimitry Andric
getMinPrefetchStride()783ca95b02SDimitry Andric unsigned getMinPrefetchStride() {
793ca95b02SDimitry Andric if (MinPrefetchStride.getNumOccurrences() > 0)
803ca95b02SDimitry Andric return MinPrefetchStride;
813ca95b02SDimitry Andric return TTI->getMinPrefetchStride();
823ca95b02SDimitry Andric }
833ca95b02SDimitry Andric
getPrefetchDistance()843ca95b02SDimitry Andric unsigned getPrefetchDistance() {
853ca95b02SDimitry Andric if (PrefetchDistance.getNumOccurrences() > 0)
863ca95b02SDimitry Andric return PrefetchDistance;
873ca95b02SDimitry Andric return TTI->getPrefetchDistance();
883ca95b02SDimitry Andric }
893ca95b02SDimitry Andric
getMaxPrefetchIterationsAhead()903ca95b02SDimitry Andric unsigned getMaxPrefetchIterationsAhead() {
913ca95b02SDimitry Andric if (MaxPrefetchIterationsAhead.getNumOccurrences() > 0)
923ca95b02SDimitry Andric return MaxPrefetchIterationsAhead;
933ca95b02SDimitry Andric return TTI->getMaxPrefetchIterationsAhead();
943ca95b02SDimitry Andric }
953ca95b02SDimitry Andric
963ca95b02SDimitry Andric AssumptionCache *AC;
973ca95b02SDimitry Andric LoopInfo *LI;
983ca95b02SDimitry Andric ScalarEvolution *SE;
993ca95b02SDimitry Andric const TargetTransformInfo *TTI;
100d88c1a5aSDimitry Andric OptimizationRemarkEmitter *ORE;
101d88c1a5aSDimitry Andric };
102d88c1a5aSDimitry Andric
103d88c1a5aSDimitry Andric /// Legacy class for inserting loop data prefetches.
104d88c1a5aSDimitry Andric class LoopDataPrefetchLegacyPass : public FunctionPass {
105d88c1a5aSDimitry Andric public:
106d88c1a5aSDimitry Andric static char ID; // Pass ID, replacement for typeid
LoopDataPrefetchLegacyPass()107d88c1a5aSDimitry Andric LoopDataPrefetchLegacyPass() : FunctionPass(ID) {
108d88c1a5aSDimitry Andric initializeLoopDataPrefetchLegacyPassPass(*PassRegistry::getPassRegistry());
109d88c1a5aSDimitry Andric }
110d88c1a5aSDimitry Andric
getAnalysisUsage(AnalysisUsage & AU) const111d88c1a5aSDimitry Andric void getAnalysisUsage(AnalysisUsage &AU) const override {
112d88c1a5aSDimitry Andric AU.addRequired<AssumptionCacheTracker>();
113d88c1a5aSDimitry Andric AU.addPreserved<DominatorTreeWrapperPass>();
114d88c1a5aSDimitry Andric AU.addRequired<LoopInfoWrapperPass>();
115d88c1a5aSDimitry Andric AU.addPreserved<LoopInfoWrapperPass>();
116d88c1a5aSDimitry Andric AU.addRequired<OptimizationRemarkEmitterWrapperPass>();
117d88c1a5aSDimitry Andric AU.addRequired<ScalarEvolutionWrapperPass>();
1182cab237bSDimitry Andric AU.addPreserved<ScalarEvolutionWrapperPass>();
119d88c1a5aSDimitry Andric AU.addRequired<TargetTransformInfoWrapperPass>();
120d88c1a5aSDimitry Andric }
121d88c1a5aSDimitry Andric
122d88c1a5aSDimitry Andric bool runOnFunction(Function &F) override;
1233ca95b02SDimitry Andric };
1243ca95b02SDimitry Andric }
1253ca95b02SDimitry Andric
126d88c1a5aSDimitry Andric char LoopDataPrefetchLegacyPass::ID = 0;
127d88c1a5aSDimitry Andric INITIALIZE_PASS_BEGIN(LoopDataPrefetchLegacyPass, "loop-data-prefetch",
1283ca95b02SDimitry Andric "Loop Data Prefetch", false, false)
INITIALIZE_PASS_DEPENDENCY(AssumptionCacheTracker)1293ca95b02SDimitry Andric INITIALIZE_PASS_DEPENDENCY(AssumptionCacheTracker)
1303ca95b02SDimitry Andric INITIALIZE_PASS_DEPENDENCY(TargetTransformInfoWrapperPass)
1313ca95b02SDimitry Andric INITIALIZE_PASS_DEPENDENCY(LoopInfoWrapperPass)
132d88c1a5aSDimitry Andric INITIALIZE_PASS_DEPENDENCY(OptimizationRemarkEmitterWrapperPass)
1333ca95b02SDimitry Andric INITIALIZE_PASS_DEPENDENCY(ScalarEvolutionWrapperPass)
134d88c1a5aSDimitry Andric INITIALIZE_PASS_END(LoopDataPrefetchLegacyPass, "loop-data-prefetch",
1353ca95b02SDimitry Andric "Loop Data Prefetch", false, false)
1363ca95b02SDimitry Andric
137d88c1a5aSDimitry Andric FunctionPass *llvm::createLoopDataPrefetchPass() {
138d88c1a5aSDimitry Andric return new LoopDataPrefetchLegacyPass();
139d88c1a5aSDimitry Andric }
1403ca95b02SDimitry Andric
isStrideLargeEnough(const SCEVAddRecExpr * AR)1413ca95b02SDimitry Andric bool LoopDataPrefetch::isStrideLargeEnough(const SCEVAddRecExpr *AR) {
1423ca95b02SDimitry Andric unsigned TargetMinStride = getMinPrefetchStride();
1433ca95b02SDimitry Andric // No need to check if any stride goes.
1443ca95b02SDimitry Andric if (TargetMinStride <= 1)
1453ca95b02SDimitry Andric return true;
1463ca95b02SDimitry Andric
1473ca95b02SDimitry Andric const auto *ConstStride = dyn_cast<SCEVConstant>(AR->getStepRecurrence(*SE));
1483ca95b02SDimitry Andric // If MinStride is set, don't prefetch unless we can ensure that stride is
1493ca95b02SDimitry Andric // larger.
1503ca95b02SDimitry Andric if (!ConstStride)
1513ca95b02SDimitry Andric return false;
1523ca95b02SDimitry Andric
1533ca95b02SDimitry Andric unsigned AbsStride = std::abs(ConstStride->getAPInt().getSExtValue());
1543ca95b02SDimitry Andric return TargetMinStride <= AbsStride;
1553ca95b02SDimitry Andric }
1563ca95b02SDimitry Andric
run(Function & F,FunctionAnalysisManager & AM)157d88c1a5aSDimitry Andric PreservedAnalyses LoopDataPrefetchPass::run(Function &F,
158d88c1a5aSDimitry Andric FunctionAnalysisManager &AM) {
159d88c1a5aSDimitry Andric LoopInfo *LI = &AM.getResult<LoopAnalysis>(F);
160d88c1a5aSDimitry Andric ScalarEvolution *SE = &AM.getResult<ScalarEvolutionAnalysis>(F);
161d88c1a5aSDimitry Andric AssumptionCache *AC = &AM.getResult<AssumptionAnalysis>(F);
162d88c1a5aSDimitry Andric OptimizationRemarkEmitter *ORE =
163d88c1a5aSDimitry Andric &AM.getResult<OptimizationRemarkEmitterAnalysis>(F);
164d88c1a5aSDimitry Andric const TargetTransformInfo *TTI = &AM.getResult<TargetIRAnalysis>(F);
165d88c1a5aSDimitry Andric
166d88c1a5aSDimitry Andric LoopDataPrefetch LDP(AC, LI, SE, TTI, ORE);
167d88c1a5aSDimitry Andric bool Changed = LDP.run();
168d88c1a5aSDimitry Andric
169d88c1a5aSDimitry Andric if (Changed) {
170d88c1a5aSDimitry Andric PreservedAnalyses PA;
171d88c1a5aSDimitry Andric PA.preserve<DominatorTreeAnalysis>();
172d88c1a5aSDimitry Andric PA.preserve<LoopAnalysis>();
173d88c1a5aSDimitry Andric return PA;
174d88c1a5aSDimitry Andric }
175d88c1a5aSDimitry Andric
176d88c1a5aSDimitry Andric return PreservedAnalyses::all();
177d88c1a5aSDimitry Andric }
178d88c1a5aSDimitry Andric
runOnFunction(Function & F)179d88c1a5aSDimitry Andric bool LoopDataPrefetchLegacyPass::runOnFunction(Function &F) {
1803ca95b02SDimitry Andric if (skipFunction(F))
1813ca95b02SDimitry Andric return false;
1823ca95b02SDimitry Andric
183d88c1a5aSDimitry Andric LoopInfo *LI = &getAnalysis<LoopInfoWrapperPass>().getLoopInfo();
184d88c1a5aSDimitry Andric ScalarEvolution *SE = &getAnalysis<ScalarEvolutionWrapperPass>().getSE();
185d88c1a5aSDimitry Andric AssumptionCache *AC =
186d88c1a5aSDimitry Andric &getAnalysis<AssumptionCacheTracker>().getAssumptionCache(F);
187d88c1a5aSDimitry Andric OptimizationRemarkEmitter *ORE =
188d88c1a5aSDimitry Andric &getAnalysis<OptimizationRemarkEmitterWrapperPass>().getORE();
189d88c1a5aSDimitry Andric const TargetTransformInfo *TTI =
190d88c1a5aSDimitry Andric &getAnalysis<TargetTransformInfoWrapperPass>().getTTI(F);
1913ca95b02SDimitry Andric
192d88c1a5aSDimitry Andric LoopDataPrefetch LDP(AC, LI, SE, TTI, ORE);
193d88c1a5aSDimitry Andric return LDP.run();
194d88c1a5aSDimitry Andric }
195d88c1a5aSDimitry Andric
run()196d88c1a5aSDimitry Andric bool LoopDataPrefetch::run() {
1973ca95b02SDimitry Andric // If PrefetchDistance is not set, don't run the pass. This gives an
1983ca95b02SDimitry Andric // opportunity for targets to run this pass for selected subtargets only
1993ca95b02SDimitry Andric // (whose TTI sets PrefetchDistance).
2003ca95b02SDimitry Andric if (getPrefetchDistance() == 0)
2013ca95b02SDimitry Andric return false;
2023ca95b02SDimitry Andric assert(TTI->getCacheLineSize() && "Cache line size is not set for target");
2033ca95b02SDimitry Andric
2043ca95b02SDimitry Andric bool MadeChange = false;
2053ca95b02SDimitry Andric
2063ca95b02SDimitry Andric for (Loop *I : *LI)
2073ca95b02SDimitry Andric for (auto L = df_begin(I), LE = df_end(I); L != LE; ++L)
2083ca95b02SDimitry Andric MadeChange |= runOnLoop(*L);
2093ca95b02SDimitry Andric
2103ca95b02SDimitry Andric return MadeChange;
2113ca95b02SDimitry Andric }
2123ca95b02SDimitry Andric
runOnLoop(Loop * L)2133ca95b02SDimitry Andric bool LoopDataPrefetch::runOnLoop(Loop *L) {
2143ca95b02SDimitry Andric bool MadeChange = false;
2153ca95b02SDimitry Andric
2163ca95b02SDimitry Andric // Only prefetch in the inner-most loop
2173ca95b02SDimitry Andric if (!L->empty())
2183ca95b02SDimitry Andric return MadeChange;
2193ca95b02SDimitry Andric
2203ca95b02SDimitry Andric SmallPtrSet<const Value *, 32> EphValues;
2213ca95b02SDimitry Andric CodeMetrics::collectEphemeralValues(L, AC, EphValues);
2223ca95b02SDimitry Andric
2233ca95b02SDimitry Andric // Calculate the number of iterations ahead to prefetch
2243ca95b02SDimitry Andric CodeMetrics Metrics;
225d88c1a5aSDimitry Andric for (const auto BB : L->blocks()) {
2263ca95b02SDimitry Andric // If the loop already has prefetches, then assume that the user knows
2273ca95b02SDimitry Andric // what they are doing and don't add any more.
228d88c1a5aSDimitry Andric for (auto &I : *BB)
229d88c1a5aSDimitry Andric if (CallInst *CI = dyn_cast<CallInst>(&I))
2303ca95b02SDimitry Andric if (Function *F = CI->getCalledFunction())
2313ca95b02SDimitry Andric if (F->getIntrinsicID() == Intrinsic::prefetch)
2323ca95b02SDimitry Andric return MadeChange;
2333ca95b02SDimitry Andric
234d88c1a5aSDimitry Andric Metrics.analyzeBasicBlock(BB, *TTI, EphValues);
2353ca95b02SDimitry Andric }
2363ca95b02SDimitry Andric unsigned LoopSize = Metrics.NumInsts;
2373ca95b02SDimitry Andric if (!LoopSize)
2383ca95b02SDimitry Andric LoopSize = 1;
2393ca95b02SDimitry Andric
2403ca95b02SDimitry Andric unsigned ItersAhead = getPrefetchDistance() / LoopSize;
2413ca95b02SDimitry Andric if (!ItersAhead)
2423ca95b02SDimitry Andric ItersAhead = 1;
2433ca95b02SDimitry Andric
2443ca95b02SDimitry Andric if (ItersAhead > getMaxPrefetchIterationsAhead())
2453ca95b02SDimitry Andric return MadeChange;
2463ca95b02SDimitry Andric
247*4ba319b5SDimitry Andric LLVM_DEBUG(dbgs() << "Prefetching " << ItersAhead
2483ca95b02SDimitry Andric << " iterations ahead (loop size: " << LoopSize << ") in "
249d88c1a5aSDimitry Andric << L->getHeader()->getParent()->getName() << ": " << *L);
2503ca95b02SDimitry Andric
2513ca95b02SDimitry Andric SmallVector<std::pair<Instruction *, const SCEVAddRecExpr *>, 16> PrefLoads;
252d88c1a5aSDimitry Andric for (const auto BB : L->blocks()) {
253d88c1a5aSDimitry Andric for (auto &I : *BB) {
2543ca95b02SDimitry Andric Value *PtrValue;
2553ca95b02SDimitry Andric Instruction *MemI;
2563ca95b02SDimitry Andric
257d88c1a5aSDimitry Andric if (LoadInst *LMemI = dyn_cast<LoadInst>(&I)) {
2583ca95b02SDimitry Andric MemI = LMemI;
2593ca95b02SDimitry Andric PtrValue = LMemI->getPointerOperand();
260d88c1a5aSDimitry Andric } else if (StoreInst *SMemI = dyn_cast<StoreInst>(&I)) {
2613ca95b02SDimitry Andric if (!PrefetchWrites) continue;
2623ca95b02SDimitry Andric MemI = SMemI;
2633ca95b02SDimitry Andric PtrValue = SMemI->getPointerOperand();
2643ca95b02SDimitry Andric } else continue;
2653ca95b02SDimitry Andric
2663ca95b02SDimitry Andric unsigned PtrAddrSpace = PtrValue->getType()->getPointerAddressSpace();
2673ca95b02SDimitry Andric if (PtrAddrSpace)
2683ca95b02SDimitry Andric continue;
2693ca95b02SDimitry Andric
2703ca95b02SDimitry Andric if (L->isLoopInvariant(PtrValue))
2713ca95b02SDimitry Andric continue;
2723ca95b02SDimitry Andric
2733ca95b02SDimitry Andric const SCEV *LSCEV = SE->getSCEV(PtrValue);
2743ca95b02SDimitry Andric const SCEVAddRecExpr *LSCEVAddRec = dyn_cast<SCEVAddRecExpr>(LSCEV);
2753ca95b02SDimitry Andric if (!LSCEVAddRec)
2763ca95b02SDimitry Andric continue;
2773ca95b02SDimitry Andric
278*4ba319b5SDimitry Andric // Check if the stride of the accesses is large enough to warrant a
2793ca95b02SDimitry Andric // prefetch.
2803ca95b02SDimitry Andric if (!isStrideLargeEnough(LSCEVAddRec))
2813ca95b02SDimitry Andric continue;
2823ca95b02SDimitry Andric
2833ca95b02SDimitry Andric // We don't want to double prefetch individual cache lines. If this load
2843ca95b02SDimitry Andric // is known to be within one cache line of some other load that has
2853ca95b02SDimitry Andric // already been prefetched, then don't prefetch this one as well.
2863ca95b02SDimitry Andric bool DupPref = false;
2873ca95b02SDimitry Andric for (const auto &PrefLoad : PrefLoads) {
2883ca95b02SDimitry Andric const SCEV *PtrDiff = SE->getMinusSCEV(LSCEVAddRec, PrefLoad.second);
2893ca95b02SDimitry Andric if (const SCEVConstant *ConstPtrDiff =
2903ca95b02SDimitry Andric dyn_cast<SCEVConstant>(PtrDiff)) {
2913ca95b02SDimitry Andric int64_t PD = std::abs(ConstPtrDiff->getValue()->getSExtValue());
2923ca95b02SDimitry Andric if (PD < (int64_t) TTI->getCacheLineSize()) {
2933ca95b02SDimitry Andric DupPref = true;
2943ca95b02SDimitry Andric break;
2953ca95b02SDimitry Andric }
2963ca95b02SDimitry Andric }
2973ca95b02SDimitry Andric }
2983ca95b02SDimitry Andric if (DupPref)
2993ca95b02SDimitry Andric continue;
3003ca95b02SDimitry Andric
3013ca95b02SDimitry Andric const SCEV *NextLSCEV = SE->getAddExpr(LSCEVAddRec, SE->getMulExpr(
3023ca95b02SDimitry Andric SE->getConstant(LSCEVAddRec->getType(), ItersAhead),
3033ca95b02SDimitry Andric LSCEVAddRec->getStepRecurrence(*SE)));
3043ca95b02SDimitry Andric if (!isSafeToExpand(NextLSCEV, *SE))
3053ca95b02SDimitry Andric continue;
3063ca95b02SDimitry Andric
3073ca95b02SDimitry Andric PrefLoads.push_back(std::make_pair(MemI, LSCEVAddRec));
3083ca95b02SDimitry Andric
309d88c1a5aSDimitry Andric Type *I8Ptr = Type::getInt8PtrTy(BB->getContext(), PtrAddrSpace);
310d88c1a5aSDimitry Andric SCEVExpander SCEVE(*SE, I.getModule()->getDataLayout(), "prefaddr");
3113ca95b02SDimitry Andric Value *PrefPtrValue = SCEVE.expandCodeFor(NextLSCEV, I8Ptr, MemI);
3123ca95b02SDimitry Andric
3133ca95b02SDimitry Andric IRBuilder<> Builder(MemI);
314d88c1a5aSDimitry Andric Module *M = BB->getParent()->getParent();
315d88c1a5aSDimitry Andric Type *I32 = Type::getInt32Ty(BB->getContext());
3163ca95b02SDimitry Andric Value *PrefetchFunc = Intrinsic::getDeclaration(M, Intrinsic::prefetch);
3173ca95b02SDimitry Andric Builder.CreateCall(
3183ca95b02SDimitry Andric PrefetchFunc,
3193ca95b02SDimitry Andric {PrefPtrValue,
3203ca95b02SDimitry Andric ConstantInt::get(I32, MemI->mayReadFromMemory() ? 0 : 1),
3213ca95b02SDimitry Andric ConstantInt::get(I32, 3), ConstantInt::get(I32, 1)});
3223ca95b02SDimitry Andric ++NumPrefetches;
323*4ba319b5SDimitry Andric LLVM_DEBUG(dbgs() << " Access: " << *PtrValue << ", SCEV: " << *LSCEV
3243ca95b02SDimitry Andric << "\n");
3252cab237bSDimitry Andric ORE->emit([&]() {
3262cab237bSDimitry Andric return OptimizationRemark(DEBUG_TYPE, "Prefetched", MemI)
3272cab237bSDimitry Andric << "prefetched memory access";
3282cab237bSDimitry Andric });
3293ca95b02SDimitry Andric
3303ca95b02SDimitry Andric MadeChange = true;
3313ca95b02SDimitry Andric }
3323ca95b02SDimitry Andric }
3333ca95b02SDimitry Andric
3343ca95b02SDimitry Andric return MadeChange;
3353ca95b02SDimitry Andric }
3363ca95b02SDimitry Andric
337