1bcbd26bfSFlorian Hahn //=== ScalarEvolutionExpanderTest.cpp - ScalarEvolutionExpander unit tests ===//
2bcbd26bfSFlorian Hahn //
3bcbd26bfSFlorian Hahn // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4bcbd26bfSFlorian Hahn // See https://llvm.org/LICENSE.txt for license information.
5bcbd26bfSFlorian Hahn // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6bcbd26bfSFlorian Hahn //
7bcbd26bfSFlorian Hahn //===----------------------------------------------------------------------===//
8bcbd26bfSFlorian Hahn
9bcbd26bfSFlorian Hahn #include "llvm/Transforms/Utils/ScalarEvolutionExpander.h"
10bcbd26bfSFlorian Hahn #include "llvm/ADT/SmallVector.h"
11bcbd26bfSFlorian Hahn #include "llvm/Analysis/AssumptionCache.h"
12bcbd26bfSFlorian Hahn #include "llvm/Analysis/LoopInfo.h"
13bcbd26bfSFlorian Hahn #include "llvm/Analysis/ScalarEvolutionExpressions.h"
14bcbd26bfSFlorian Hahn #include "llvm/Analysis/TargetLibraryInfo.h"
15bcbd26bfSFlorian Hahn #include "llvm/AsmParser/Parser.h"
16bcbd26bfSFlorian Hahn #include "llvm/IR/Constants.h"
17bcbd26bfSFlorian Hahn #include "llvm/IR/Dominators.h"
18bcbd26bfSFlorian Hahn #include "llvm/IR/GlobalVariable.h"
19bcbd26bfSFlorian Hahn #include "llvm/IR/IRBuilder.h"
20bcbd26bfSFlorian Hahn #include "llvm/IR/InstIterator.h"
21bcbd26bfSFlorian Hahn #include "llvm/IR/LLVMContext.h"
22bcbd26bfSFlorian Hahn #include "llvm/IR/Module.h"
231d8f2e52SFlorian Hahn #include "llvm/IR/PatternMatch.h"
24bcbd26bfSFlorian Hahn #include "llvm/IR/Verifier.h"
25ba7a92c0SNico Weber #include "llvm/Support/SourceMgr.h"
26bcbd26bfSFlorian Hahn #include "gtest/gtest.h"
27bcbd26bfSFlorian Hahn
28bcbd26bfSFlorian Hahn namespace llvm {
29bcbd26bfSFlorian Hahn
301d8f2e52SFlorian Hahn using namespace PatternMatch;
311d8f2e52SFlorian Hahn
32bcbd26bfSFlorian Hahn // We use this fixture to ensure that we clean up ScalarEvolution before
33bcbd26bfSFlorian Hahn // deleting the PassManager.
34bcbd26bfSFlorian Hahn class ScalarEvolutionExpanderTest : public testing::Test {
35bcbd26bfSFlorian Hahn protected:
36bcbd26bfSFlorian Hahn LLVMContext Context;
37bcbd26bfSFlorian Hahn Module M;
38bcbd26bfSFlorian Hahn TargetLibraryInfoImpl TLII;
39bcbd26bfSFlorian Hahn TargetLibraryInfo TLI;
40bcbd26bfSFlorian Hahn
41bcbd26bfSFlorian Hahn std::unique_ptr<AssumptionCache> AC;
42bcbd26bfSFlorian Hahn std::unique_ptr<DominatorTree> DT;
43bcbd26bfSFlorian Hahn std::unique_ptr<LoopInfo> LI;
44bcbd26bfSFlorian Hahn
ScalarEvolutionExpanderTest()45bcbd26bfSFlorian Hahn ScalarEvolutionExpanderTest() : M("", Context), TLII(), TLI(TLII) {}
46bcbd26bfSFlorian Hahn
buildSE(Function & F)47bcbd26bfSFlorian Hahn ScalarEvolution buildSE(Function &F) {
48bcbd26bfSFlorian Hahn AC.reset(new AssumptionCache(F));
49bcbd26bfSFlorian Hahn DT.reset(new DominatorTree(F));
50bcbd26bfSFlorian Hahn LI.reset(new LoopInfo(*DT));
51bcbd26bfSFlorian Hahn return ScalarEvolution(F, TLI, *AC, *DT, *LI);
52bcbd26bfSFlorian Hahn }
53bcbd26bfSFlorian Hahn
runWithSE(Module & M,StringRef FuncName,function_ref<void (Function & F,LoopInfo & LI,ScalarEvolution & SE)> Test)54bcbd26bfSFlorian Hahn void runWithSE(
55bcbd26bfSFlorian Hahn Module &M, StringRef FuncName,
56bcbd26bfSFlorian Hahn function_ref<void(Function &F, LoopInfo &LI, ScalarEvolution &SE)> Test) {
57bcbd26bfSFlorian Hahn auto *F = M.getFunction(FuncName);
58bcbd26bfSFlorian Hahn ASSERT_NE(F, nullptr) << "Could not find " << FuncName;
59bcbd26bfSFlorian Hahn ScalarEvolution SE = buildSE(*F);
60bcbd26bfSFlorian Hahn Test(*F, *LI, SE);
61bcbd26bfSFlorian Hahn }
62bcbd26bfSFlorian Hahn };
63bcbd26bfSFlorian Hahn
GetInstByName(Function & F,StringRef Name)64bcbd26bfSFlorian Hahn static Instruction &GetInstByName(Function &F, StringRef Name) {
65bcbd26bfSFlorian Hahn for (auto &I : instructions(F))
66bcbd26bfSFlorian Hahn if (I.getName() == Name)
67bcbd26bfSFlorian Hahn return I;
68bcbd26bfSFlorian Hahn llvm_unreachable("Could not find instructions!");
69bcbd26bfSFlorian Hahn }
70bcbd26bfSFlorian Hahn
TEST_F(ScalarEvolutionExpanderTest,ExpandPtrTypeSCEV)71bcbd26bfSFlorian Hahn TEST_F(ScalarEvolutionExpanderTest, ExpandPtrTypeSCEV) {
72bcbd26bfSFlorian Hahn // It is to test the fix for PR30213. It exercises the branch in scev
73bcbd26bfSFlorian Hahn // expansion when the value in ValueOffsetPair is a ptr and the offset
74bcbd26bfSFlorian Hahn // is not divisible by the elem type size of value.
75bcbd26bfSFlorian Hahn auto *I8Ty = Type::getInt8Ty(Context);
76bcbd26bfSFlorian Hahn auto *I8PtrTy = Type::getInt8PtrTy(Context);
77bcbd26bfSFlorian Hahn auto *I32Ty = Type::getInt32Ty(Context);
78bcbd26bfSFlorian Hahn auto *I32PtrTy = Type::getInt32PtrTy(Context);
79bcbd26bfSFlorian Hahn FunctionType *FTy =
80bcbd26bfSFlorian Hahn FunctionType::get(Type::getVoidTy(Context), std::vector<Type *>(), false);
81bcbd26bfSFlorian Hahn Function *F = Function::Create(FTy, Function::ExternalLinkage, "f", M);
82bcbd26bfSFlorian Hahn BasicBlock *EntryBB = BasicBlock::Create(Context, "entry", F);
83bcbd26bfSFlorian Hahn BasicBlock *LoopBB = BasicBlock::Create(Context, "loop", F);
84bcbd26bfSFlorian Hahn BasicBlock *ExitBB = BasicBlock::Create(Context, "exit", F);
85bcbd26bfSFlorian Hahn BranchInst::Create(LoopBB, EntryBB);
86bcbd26bfSFlorian Hahn ReturnInst::Create(Context, nullptr, ExitBB);
87bcbd26bfSFlorian Hahn
88bcbd26bfSFlorian Hahn // loop: ; preds = %loop, %entry
89bcbd26bfSFlorian Hahn // %alloca = alloca i32
90bcbd26bfSFlorian Hahn // %gep0 = getelementptr i32, i32* %alloca, i32 1
91bcbd26bfSFlorian Hahn // %bitcast1 = bitcast i32* %gep0 to i8*
92bcbd26bfSFlorian Hahn // %gep1 = getelementptr i8, i8* %bitcast1, i32 1
93bcbd26bfSFlorian Hahn // %gep2 = getelementptr i8, i8* undef, i32 1
94bcbd26bfSFlorian Hahn // %cmp = icmp ult i8* undef, %bitcast1
95bcbd26bfSFlorian Hahn // %select = select i1 %cmp, i8* %gep1, i8* %gep2
96bcbd26bfSFlorian Hahn // %bitcast2 = bitcast i8* %select to i32*
97bcbd26bfSFlorian Hahn // br i1 undef, label %loop, label %exit
98bcbd26bfSFlorian Hahn
99bcbd26bfSFlorian Hahn const DataLayout &DL = F->getParent()->getDataLayout();
100bcbd26bfSFlorian Hahn BranchInst *Br = BranchInst::Create(
101bcbd26bfSFlorian Hahn LoopBB, ExitBB, UndefValue::get(Type::getInt1Ty(Context)), LoopBB);
102bcbd26bfSFlorian Hahn AllocaInst *Alloca =
103bcbd26bfSFlorian Hahn new AllocaInst(I32Ty, DL.getAllocaAddrSpace(), "alloca", Br);
104bcbd26bfSFlorian Hahn ConstantInt *Ci32 = ConstantInt::get(Context, APInt(32, 1));
105bcbd26bfSFlorian Hahn GetElementPtrInst *Gep0 =
106bcbd26bfSFlorian Hahn GetElementPtrInst::Create(I32Ty, Alloca, Ci32, "gep0", Br);
107bcbd26bfSFlorian Hahn CastInst *CastA =
108bcbd26bfSFlorian Hahn CastInst::CreateBitOrPointerCast(Gep0, I8PtrTy, "bitcast1", Br);
109bcbd26bfSFlorian Hahn GetElementPtrInst *Gep1 =
110bcbd26bfSFlorian Hahn GetElementPtrInst::Create(I8Ty, CastA, Ci32, "gep1", Br);
111bcbd26bfSFlorian Hahn GetElementPtrInst *Gep2 = GetElementPtrInst::Create(
112bcbd26bfSFlorian Hahn I8Ty, UndefValue::get(I8PtrTy), Ci32, "gep2", Br);
113bcbd26bfSFlorian Hahn CmpInst *Cmp = CmpInst::Create(Instruction::ICmp, CmpInst::ICMP_ULT,
114bcbd26bfSFlorian Hahn UndefValue::get(I8PtrTy), CastA, "cmp", Br);
115bcbd26bfSFlorian Hahn SelectInst *Sel = SelectInst::Create(Cmp, Gep1, Gep2, "select", Br);
116bcbd26bfSFlorian Hahn CastInst *CastB =
117bcbd26bfSFlorian Hahn CastInst::CreateBitOrPointerCast(Sel, I32PtrTy, "bitcast2", Br);
118bcbd26bfSFlorian Hahn
119bcbd26bfSFlorian Hahn ScalarEvolution SE = buildSE(*F);
120bcbd26bfSFlorian Hahn auto *S = SE.getSCEV(CastB);
1218a567e5fSEli Friedman EXPECT_TRUE(isa<SCEVUnknown>(S));
122bcbd26bfSFlorian Hahn }
123bcbd26bfSFlorian Hahn
124bcbd26bfSFlorian Hahn // Make sure that SCEV doesn't introduce illegal ptrtoint/inttoptr instructions
TEST_F(ScalarEvolutionExpanderTest,SCEVZeroExtendExprNonIntegral)125bcbd26bfSFlorian Hahn TEST_F(ScalarEvolutionExpanderTest, SCEVZeroExtendExprNonIntegral) {
126bcbd26bfSFlorian Hahn /*
127bcbd26bfSFlorian Hahn * Create the following code:
128bcbd26bfSFlorian Hahn * func(i64 addrspace(10)* %arg)
129bcbd26bfSFlorian Hahn * top:
130bcbd26bfSFlorian Hahn * br label %L.ph
131bcbd26bfSFlorian Hahn * L.ph:
132bcbd26bfSFlorian Hahn * br label %L
133bcbd26bfSFlorian Hahn * L:
134bcbd26bfSFlorian Hahn * %phi = phi i64 [i64 0, %L.ph], [ %add, %L2 ]
135bcbd26bfSFlorian Hahn * %add = add i64 %phi2, 1
136bcbd26bfSFlorian Hahn * br i1 undef, label %post, label %L2
137bcbd26bfSFlorian Hahn * post:
138bcbd26bfSFlorian Hahn * %gepbase = getelementptr i64 addrspace(10)* %arg, i64 1
139bcbd26bfSFlorian Hahn * #= %gep = getelementptr i64 addrspace(10)* %gepbase, i64 %add =#
140bcbd26bfSFlorian Hahn * ret void
141bcbd26bfSFlorian Hahn *
142bcbd26bfSFlorian Hahn * We will create the appropriate SCEV expression for %gep and expand it,
143bcbd26bfSFlorian Hahn * then check that no inttoptr/ptrtoint instructions got inserted.
144bcbd26bfSFlorian Hahn */
145bcbd26bfSFlorian Hahn
146bcbd26bfSFlorian Hahn // Create a module with non-integral pointers in it's datalayout
147bcbd26bfSFlorian Hahn Module NIM("nonintegral", Context);
148bcbd26bfSFlorian Hahn std::string DataLayout = M.getDataLayoutStr();
149bcbd26bfSFlorian Hahn if (!DataLayout.empty())
150bcbd26bfSFlorian Hahn DataLayout += "-";
151bcbd26bfSFlorian Hahn DataLayout += "ni:10";
152bcbd26bfSFlorian Hahn NIM.setDataLayout(DataLayout);
153bcbd26bfSFlorian Hahn
154bcbd26bfSFlorian Hahn Type *T_int1 = Type::getInt1Ty(Context);
155bcbd26bfSFlorian Hahn Type *T_int64 = Type::getInt64Ty(Context);
156bcbd26bfSFlorian Hahn Type *T_pint64 = T_int64->getPointerTo(10);
157bcbd26bfSFlorian Hahn
158bcbd26bfSFlorian Hahn FunctionType *FTy =
159bcbd26bfSFlorian Hahn FunctionType::get(Type::getVoidTy(Context), {T_pint64}, false);
160bcbd26bfSFlorian Hahn Function *F = Function::Create(FTy, Function::ExternalLinkage, "foo", NIM);
161bcbd26bfSFlorian Hahn
162bcbd26bfSFlorian Hahn Argument *Arg = &*F->arg_begin();
163bcbd26bfSFlorian Hahn
164bcbd26bfSFlorian Hahn BasicBlock *Top = BasicBlock::Create(Context, "top", F);
165bcbd26bfSFlorian Hahn BasicBlock *LPh = BasicBlock::Create(Context, "L.ph", F);
166bcbd26bfSFlorian Hahn BasicBlock *L = BasicBlock::Create(Context, "L", F);
167bcbd26bfSFlorian Hahn BasicBlock *Post = BasicBlock::Create(Context, "post", F);
168bcbd26bfSFlorian Hahn
169bcbd26bfSFlorian Hahn IRBuilder<> Builder(Top);
170bcbd26bfSFlorian Hahn Builder.CreateBr(LPh);
171bcbd26bfSFlorian Hahn
172bcbd26bfSFlorian Hahn Builder.SetInsertPoint(LPh);
173bcbd26bfSFlorian Hahn Builder.CreateBr(L);
174bcbd26bfSFlorian Hahn
175bcbd26bfSFlorian Hahn Builder.SetInsertPoint(L);
176bcbd26bfSFlorian Hahn PHINode *Phi = Builder.CreatePHI(T_int64, 2);
177bcbd26bfSFlorian Hahn Value *Add = Builder.CreateAdd(Phi, ConstantInt::get(T_int64, 1), "add");
178bcbd26bfSFlorian Hahn Builder.CreateCondBr(UndefValue::get(T_int1), L, Post);
179bcbd26bfSFlorian Hahn Phi->addIncoming(ConstantInt::get(T_int64, 0), LPh);
180bcbd26bfSFlorian Hahn Phi->addIncoming(Add, L);
181bcbd26bfSFlorian Hahn
182bcbd26bfSFlorian Hahn Builder.SetInsertPoint(Post);
183bcbd26bfSFlorian Hahn Value *GepBase =
184bcbd26bfSFlorian Hahn Builder.CreateGEP(T_int64, Arg, ConstantInt::get(T_int64, 1));
185bcbd26bfSFlorian Hahn Instruction *Ret = Builder.CreateRetVoid();
186bcbd26bfSFlorian Hahn
187bcbd26bfSFlorian Hahn ScalarEvolution SE = buildSE(*F);
188bcbd26bfSFlorian Hahn auto *AddRec =
189bcbd26bfSFlorian Hahn SE.getAddRecExpr(SE.getUnknown(GepBase), SE.getConstant(T_int64, 1),
190bcbd26bfSFlorian Hahn LI->getLoopFor(L), SCEV::FlagNUW);
191bcbd26bfSFlorian Hahn
192bcbd26bfSFlorian Hahn SCEVExpander Exp(SE, NIM.getDataLayout(), "expander");
193bcbd26bfSFlorian Hahn Exp.disableCanonicalMode();
194bcbd26bfSFlorian Hahn Exp.expandCodeFor(AddRec, T_pint64, Ret);
195bcbd26bfSFlorian Hahn
196bcbd26bfSFlorian Hahn // Make sure none of the instructions inserted were inttoptr/ptrtoint.
197bcbd26bfSFlorian Hahn // The verifier will check this.
198bcbd26bfSFlorian Hahn EXPECT_FALSE(verifyFunction(*F, &errs()));
199bcbd26bfSFlorian Hahn }
200bcbd26bfSFlorian Hahn
201bcbd26bfSFlorian Hahn // Check that we can correctly identify the points at which the SCEV of the
202bcbd26bfSFlorian Hahn // AddRec can be expanded.
TEST_F(ScalarEvolutionExpanderTest,SCEVExpanderIsSafeToExpandAt)203bcbd26bfSFlorian Hahn TEST_F(ScalarEvolutionExpanderTest, SCEVExpanderIsSafeToExpandAt) {
204bcbd26bfSFlorian Hahn /*
205bcbd26bfSFlorian Hahn * Create the following code:
206bcbd26bfSFlorian Hahn * func(i64 addrspace(10)* %arg)
207bcbd26bfSFlorian Hahn * top:
208bcbd26bfSFlorian Hahn * br label %L.ph
209bcbd26bfSFlorian Hahn * L.ph:
210bcbd26bfSFlorian Hahn * br label %L
211bcbd26bfSFlorian Hahn * L:
212bcbd26bfSFlorian Hahn * %phi = phi i64 [i64 0, %L.ph], [ %add, %L2 ]
213bcbd26bfSFlorian Hahn * %add = add i64 %phi2, 1
214bcbd26bfSFlorian Hahn * %cond = icmp slt i64 %add, 1000; then becomes 2000.
215bcbd26bfSFlorian Hahn * br i1 %cond, label %post, label %L2
216bcbd26bfSFlorian Hahn * post:
217bcbd26bfSFlorian Hahn * ret void
218bcbd26bfSFlorian Hahn *
219bcbd26bfSFlorian Hahn */
220bcbd26bfSFlorian Hahn
221bcbd26bfSFlorian Hahn // Create a module with non-integral pointers in it's datalayout
222bcbd26bfSFlorian Hahn Module NIM("nonintegral", Context);
223bcbd26bfSFlorian Hahn std::string DataLayout = M.getDataLayoutStr();
224bcbd26bfSFlorian Hahn if (!DataLayout.empty())
225bcbd26bfSFlorian Hahn DataLayout += "-";
226bcbd26bfSFlorian Hahn DataLayout += "ni:10";
227bcbd26bfSFlorian Hahn NIM.setDataLayout(DataLayout);
228bcbd26bfSFlorian Hahn
229bcbd26bfSFlorian Hahn Type *T_int64 = Type::getInt64Ty(Context);
230bcbd26bfSFlorian Hahn Type *T_pint64 = T_int64->getPointerTo(10);
231bcbd26bfSFlorian Hahn
232bcbd26bfSFlorian Hahn FunctionType *FTy =
233bcbd26bfSFlorian Hahn FunctionType::get(Type::getVoidTy(Context), {T_pint64}, false);
234bcbd26bfSFlorian Hahn Function *F = Function::Create(FTy, Function::ExternalLinkage, "foo", NIM);
235bcbd26bfSFlorian Hahn
236bcbd26bfSFlorian Hahn BasicBlock *Top = BasicBlock::Create(Context, "top", F);
237bcbd26bfSFlorian Hahn BasicBlock *LPh = BasicBlock::Create(Context, "L.ph", F);
238bcbd26bfSFlorian Hahn BasicBlock *L = BasicBlock::Create(Context, "L", F);
239bcbd26bfSFlorian Hahn BasicBlock *Post = BasicBlock::Create(Context, "post", F);
240bcbd26bfSFlorian Hahn
241bcbd26bfSFlorian Hahn IRBuilder<> Builder(Top);
242bcbd26bfSFlorian Hahn Builder.CreateBr(LPh);
243bcbd26bfSFlorian Hahn
244bcbd26bfSFlorian Hahn Builder.SetInsertPoint(LPh);
245bcbd26bfSFlorian Hahn Builder.CreateBr(L);
246bcbd26bfSFlorian Hahn
247bcbd26bfSFlorian Hahn Builder.SetInsertPoint(L);
248bcbd26bfSFlorian Hahn PHINode *Phi = Builder.CreatePHI(T_int64, 2);
249bcbd26bfSFlorian Hahn auto *Add = cast<Instruction>(
250bcbd26bfSFlorian Hahn Builder.CreateAdd(Phi, ConstantInt::get(T_int64, 1), "add"));
251bcbd26bfSFlorian Hahn auto *Limit = ConstantInt::get(T_int64, 1000);
252bcbd26bfSFlorian Hahn auto *Cond = cast<Instruction>(
253bcbd26bfSFlorian Hahn Builder.CreateICmp(ICmpInst::ICMP_SLT, Add, Limit, "cond"));
254bcbd26bfSFlorian Hahn Builder.CreateCondBr(Cond, L, Post);
255bcbd26bfSFlorian Hahn Phi->addIncoming(ConstantInt::get(T_int64, 0), LPh);
256bcbd26bfSFlorian Hahn Phi->addIncoming(Add, L);
257bcbd26bfSFlorian Hahn
258bcbd26bfSFlorian Hahn Builder.SetInsertPoint(Post);
259f75564adSFlorian Hahn Instruction *Ret = Builder.CreateRetVoid();
260bcbd26bfSFlorian Hahn
261bcbd26bfSFlorian Hahn ScalarEvolution SE = buildSE(*F);
262*dcf4b733SNikita Popov SCEVExpander Exp(SE, M.getDataLayout(), "expander");
263bcbd26bfSFlorian Hahn const SCEV *S = SE.getSCEV(Phi);
264bcbd26bfSFlorian Hahn EXPECT_TRUE(isa<SCEVAddRecExpr>(S));
265bcbd26bfSFlorian Hahn const SCEVAddRecExpr *AR = cast<SCEVAddRecExpr>(S);
266bcbd26bfSFlorian Hahn EXPECT_TRUE(AR->isAffine());
267*dcf4b733SNikita Popov EXPECT_FALSE(Exp.isSafeToExpandAt(AR, Top->getTerminator()));
268*dcf4b733SNikita Popov EXPECT_FALSE(Exp.isSafeToExpandAt(AR, LPh->getTerminator()));
269*dcf4b733SNikita Popov EXPECT_TRUE(Exp.isSafeToExpandAt(AR, L->getTerminator()));
270*dcf4b733SNikita Popov EXPECT_TRUE(Exp.isSafeToExpandAt(AR, Post->getTerminator()));
271f75564adSFlorian Hahn
272f75564adSFlorian Hahn EXPECT_TRUE(LI->getLoopFor(L)->isLCSSAForm(*DT));
273f75564adSFlorian Hahn Exp.expandCodeFor(SE.getSCEV(Add), nullptr, Ret);
274f75564adSFlorian Hahn EXPECT_TRUE(LI->getLoopFor(L)->isLCSSAForm(*DT));
275bcbd26bfSFlorian Hahn }
276bcbd26bfSFlorian Hahn
277bcbd26bfSFlorian Hahn // Check that SCEV expander does not use the nuw instruction
278bcbd26bfSFlorian Hahn // for expansion.
TEST_F(ScalarEvolutionExpanderTest,SCEVExpanderNUW)279bcbd26bfSFlorian Hahn TEST_F(ScalarEvolutionExpanderTest, SCEVExpanderNUW) {
280bcbd26bfSFlorian Hahn /*
281bcbd26bfSFlorian Hahn * Create the following code:
282bcbd26bfSFlorian Hahn * func(i64 %a)
283bcbd26bfSFlorian Hahn * entry:
284bcbd26bfSFlorian Hahn * br false, label %exit, label %body
285bcbd26bfSFlorian Hahn * body:
286bcbd26bfSFlorian Hahn * %s1 = add i64 %a, -1
287bcbd26bfSFlorian Hahn * br label %exit
288bcbd26bfSFlorian Hahn * exit:
289bcbd26bfSFlorian Hahn * %s = add nuw i64 %a, -1
290bcbd26bfSFlorian Hahn * ret %s
291bcbd26bfSFlorian Hahn */
292bcbd26bfSFlorian Hahn
293bcbd26bfSFlorian Hahn // Create a module.
294bcbd26bfSFlorian Hahn Module M("SCEVExpanderNUW", Context);
295bcbd26bfSFlorian Hahn
296bcbd26bfSFlorian Hahn Type *T_int64 = Type::getInt64Ty(Context);
297bcbd26bfSFlorian Hahn
298bcbd26bfSFlorian Hahn FunctionType *FTy =
299bcbd26bfSFlorian Hahn FunctionType::get(Type::getVoidTy(Context), {T_int64}, false);
300bcbd26bfSFlorian Hahn Function *F = Function::Create(FTy, Function::ExternalLinkage, "func", M);
301bcbd26bfSFlorian Hahn Argument *Arg = &*F->arg_begin();
302bcbd26bfSFlorian Hahn ConstantInt *C = ConstantInt::get(Context, APInt(64, -1));
303bcbd26bfSFlorian Hahn
304bcbd26bfSFlorian Hahn BasicBlock *Entry = BasicBlock::Create(Context, "entry", F);
305bcbd26bfSFlorian Hahn BasicBlock *Body = BasicBlock::Create(Context, "body", F);
306bcbd26bfSFlorian Hahn BasicBlock *Exit = BasicBlock::Create(Context, "exit", F);
307bcbd26bfSFlorian Hahn
308bcbd26bfSFlorian Hahn IRBuilder<> Builder(Entry);
309bcbd26bfSFlorian Hahn ConstantInt *Cond = ConstantInt::get(Context, APInt(1, 0));
310bcbd26bfSFlorian Hahn Builder.CreateCondBr(Cond, Exit, Body);
311bcbd26bfSFlorian Hahn
312bcbd26bfSFlorian Hahn Builder.SetInsertPoint(Body);
313bcbd26bfSFlorian Hahn auto *S1 = cast<Instruction>(Builder.CreateAdd(Arg, C, "add"));
314bcbd26bfSFlorian Hahn Builder.CreateBr(Exit);
315bcbd26bfSFlorian Hahn
316bcbd26bfSFlorian Hahn Builder.SetInsertPoint(Exit);
317bcbd26bfSFlorian Hahn auto *S2 = cast<Instruction>(Builder.CreateAdd(Arg, C, "add"));
318bcbd26bfSFlorian Hahn S2->setHasNoUnsignedWrap(true);
319bcbd26bfSFlorian Hahn auto *R = cast<Instruction>(Builder.CreateRetVoid());
320bcbd26bfSFlorian Hahn
321bcbd26bfSFlorian Hahn ScalarEvolution SE = buildSE(*F);
322bcbd26bfSFlorian Hahn const SCEV *S = SE.getSCEV(S1);
323bcbd26bfSFlorian Hahn EXPECT_TRUE(isa<SCEVAddExpr>(S));
324bcbd26bfSFlorian Hahn SCEVExpander Exp(SE, M.getDataLayout(), "expander");
325bcbd26bfSFlorian Hahn auto *I = cast<Instruction>(Exp.expandCodeFor(S, nullptr, R));
326bcbd26bfSFlorian Hahn EXPECT_FALSE(I->hasNoUnsignedWrap());
327bcbd26bfSFlorian Hahn }
328bcbd26bfSFlorian Hahn
329bcbd26bfSFlorian Hahn // Check that SCEV expander does not use the nsw instruction
330bcbd26bfSFlorian Hahn // for expansion.
TEST_F(ScalarEvolutionExpanderTest,SCEVExpanderNSW)331bcbd26bfSFlorian Hahn TEST_F(ScalarEvolutionExpanderTest, SCEVExpanderNSW) {
332bcbd26bfSFlorian Hahn /*
333bcbd26bfSFlorian Hahn * Create the following code:
334bcbd26bfSFlorian Hahn * func(i64 %a)
335bcbd26bfSFlorian Hahn * entry:
336bcbd26bfSFlorian Hahn * br false, label %exit, label %body
337bcbd26bfSFlorian Hahn * body:
338bcbd26bfSFlorian Hahn * %s1 = add i64 %a, -1
339bcbd26bfSFlorian Hahn * br label %exit
340bcbd26bfSFlorian Hahn * exit:
341bcbd26bfSFlorian Hahn * %s = add nsw i64 %a, -1
342bcbd26bfSFlorian Hahn * ret %s
343bcbd26bfSFlorian Hahn */
344bcbd26bfSFlorian Hahn
345bcbd26bfSFlorian Hahn // Create a module.
346bcbd26bfSFlorian Hahn Module M("SCEVExpanderNSW", Context);
347bcbd26bfSFlorian Hahn
348bcbd26bfSFlorian Hahn Type *T_int64 = Type::getInt64Ty(Context);
349bcbd26bfSFlorian Hahn
350bcbd26bfSFlorian Hahn FunctionType *FTy =
351bcbd26bfSFlorian Hahn FunctionType::get(Type::getVoidTy(Context), {T_int64}, false);
352bcbd26bfSFlorian Hahn Function *F = Function::Create(FTy, Function::ExternalLinkage, "func", M);
353bcbd26bfSFlorian Hahn Argument *Arg = &*F->arg_begin();
354bcbd26bfSFlorian Hahn ConstantInt *C = ConstantInt::get(Context, APInt(64, -1));
355bcbd26bfSFlorian Hahn
356bcbd26bfSFlorian Hahn BasicBlock *Entry = BasicBlock::Create(Context, "entry", F);
357bcbd26bfSFlorian Hahn BasicBlock *Body = BasicBlock::Create(Context, "body", F);
358bcbd26bfSFlorian Hahn BasicBlock *Exit = BasicBlock::Create(Context, "exit", F);
359bcbd26bfSFlorian Hahn
360bcbd26bfSFlorian Hahn IRBuilder<> Builder(Entry);
361bcbd26bfSFlorian Hahn ConstantInt *Cond = ConstantInt::get(Context, APInt(1, 0));
362bcbd26bfSFlorian Hahn Builder.CreateCondBr(Cond, Exit, Body);
363bcbd26bfSFlorian Hahn
364bcbd26bfSFlorian Hahn Builder.SetInsertPoint(Body);
365bcbd26bfSFlorian Hahn auto *S1 = cast<Instruction>(Builder.CreateAdd(Arg, C, "add"));
366bcbd26bfSFlorian Hahn Builder.CreateBr(Exit);
367bcbd26bfSFlorian Hahn
368bcbd26bfSFlorian Hahn Builder.SetInsertPoint(Exit);
369bcbd26bfSFlorian Hahn auto *S2 = cast<Instruction>(Builder.CreateAdd(Arg, C, "add"));
370bcbd26bfSFlorian Hahn S2->setHasNoSignedWrap(true);
371bcbd26bfSFlorian Hahn auto *R = cast<Instruction>(Builder.CreateRetVoid());
372bcbd26bfSFlorian Hahn
373bcbd26bfSFlorian Hahn ScalarEvolution SE = buildSE(*F);
374bcbd26bfSFlorian Hahn const SCEV *S = SE.getSCEV(S1);
375bcbd26bfSFlorian Hahn EXPECT_TRUE(isa<SCEVAddExpr>(S));
376bcbd26bfSFlorian Hahn SCEVExpander Exp(SE, M.getDataLayout(), "expander");
377bcbd26bfSFlorian Hahn auto *I = cast<Instruction>(Exp.expandCodeFor(S, nullptr, R));
378bcbd26bfSFlorian Hahn EXPECT_FALSE(I->hasNoSignedWrap());
379bcbd26bfSFlorian Hahn }
380bcbd26bfSFlorian Hahn
381bcbd26bfSFlorian Hahn // Check that SCEV does not save the SCEV -> V
382bcbd26bfSFlorian Hahn // mapping of SCEV differ from V in NUW flag.
TEST_F(ScalarEvolutionExpanderTest,SCEVCacheNUW)383bcbd26bfSFlorian Hahn TEST_F(ScalarEvolutionExpanderTest, SCEVCacheNUW) {
384bcbd26bfSFlorian Hahn /*
385bcbd26bfSFlorian Hahn * Create the following code:
386bcbd26bfSFlorian Hahn * func(i64 %a)
387bcbd26bfSFlorian Hahn * entry:
388bcbd26bfSFlorian Hahn * %s1 = add i64 %a, -1
389bcbd26bfSFlorian Hahn * %s2 = add nuw i64 %a, -1
390bcbd26bfSFlorian Hahn * br label %exit
391bcbd26bfSFlorian Hahn * exit:
392bcbd26bfSFlorian Hahn * ret %s
393bcbd26bfSFlorian Hahn */
394bcbd26bfSFlorian Hahn
395bcbd26bfSFlorian Hahn // Create a module.
396bcbd26bfSFlorian Hahn Module M("SCEVCacheNUW", Context);
397bcbd26bfSFlorian Hahn
398bcbd26bfSFlorian Hahn Type *T_int64 = Type::getInt64Ty(Context);
399bcbd26bfSFlorian Hahn
400bcbd26bfSFlorian Hahn FunctionType *FTy =
401bcbd26bfSFlorian Hahn FunctionType::get(Type::getVoidTy(Context), {T_int64}, false);
402bcbd26bfSFlorian Hahn Function *F = Function::Create(FTy, Function::ExternalLinkage, "func", M);
403bcbd26bfSFlorian Hahn Argument *Arg = &*F->arg_begin();
404bcbd26bfSFlorian Hahn ConstantInt *C = ConstantInt::get(Context, APInt(64, -1));
405bcbd26bfSFlorian Hahn
406bcbd26bfSFlorian Hahn BasicBlock *Entry = BasicBlock::Create(Context, "entry", F);
407bcbd26bfSFlorian Hahn BasicBlock *Exit = BasicBlock::Create(Context, "exit", F);
408bcbd26bfSFlorian Hahn
409bcbd26bfSFlorian Hahn IRBuilder<> Builder(Entry);
410bcbd26bfSFlorian Hahn auto *S1 = cast<Instruction>(Builder.CreateAdd(Arg, C, "add"));
411bcbd26bfSFlorian Hahn auto *S2 = cast<Instruction>(Builder.CreateAdd(Arg, C, "add"));
412bcbd26bfSFlorian Hahn S2->setHasNoUnsignedWrap(true);
413bcbd26bfSFlorian Hahn Builder.CreateBr(Exit);
414bcbd26bfSFlorian Hahn
415bcbd26bfSFlorian Hahn Builder.SetInsertPoint(Exit);
416bcbd26bfSFlorian Hahn auto *R = cast<Instruction>(Builder.CreateRetVoid());
417bcbd26bfSFlorian Hahn
418bcbd26bfSFlorian Hahn ScalarEvolution SE = buildSE(*F);
419bcbd26bfSFlorian Hahn // Get S2 first to move it to cache.
420bcbd26bfSFlorian Hahn const SCEV *SC2 = SE.getSCEV(S2);
421bcbd26bfSFlorian Hahn EXPECT_TRUE(isa<SCEVAddExpr>(SC2));
422bcbd26bfSFlorian Hahn // Now get S1.
423bcbd26bfSFlorian Hahn const SCEV *SC1 = SE.getSCEV(S1);
424bcbd26bfSFlorian Hahn EXPECT_TRUE(isa<SCEVAddExpr>(SC1));
425bcbd26bfSFlorian Hahn // Expand for S1, it should use S1 not S2 in spite S2
426bcbd26bfSFlorian Hahn // first in the cache.
427bcbd26bfSFlorian Hahn SCEVExpander Exp(SE, M.getDataLayout(), "expander");
428bcbd26bfSFlorian Hahn auto *I = cast<Instruction>(Exp.expandCodeFor(SC1, nullptr, R));
429bcbd26bfSFlorian Hahn EXPECT_FALSE(I->hasNoUnsignedWrap());
430bcbd26bfSFlorian Hahn }
431bcbd26bfSFlorian Hahn
432bcbd26bfSFlorian Hahn // Check that SCEV does not save the SCEV -> V
433bcbd26bfSFlorian Hahn // mapping of SCEV differ from V in NSW flag.
TEST_F(ScalarEvolutionExpanderTest,SCEVCacheNSW)434bcbd26bfSFlorian Hahn TEST_F(ScalarEvolutionExpanderTest, SCEVCacheNSW) {
435bcbd26bfSFlorian Hahn /*
436bcbd26bfSFlorian Hahn * Create the following code:
437bcbd26bfSFlorian Hahn * func(i64 %a)
438bcbd26bfSFlorian Hahn * entry:
439bcbd26bfSFlorian Hahn * %s1 = add i64 %a, -1
440bcbd26bfSFlorian Hahn * %s2 = add nsw i64 %a, -1
441bcbd26bfSFlorian Hahn * br label %exit
442bcbd26bfSFlorian Hahn * exit:
443bcbd26bfSFlorian Hahn * ret %s
444bcbd26bfSFlorian Hahn */
445bcbd26bfSFlorian Hahn
446bcbd26bfSFlorian Hahn // Create a module.
447bcbd26bfSFlorian Hahn Module M("SCEVCacheNUW", Context);
448bcbd26bfSFlorian Hahn
449bcbd26bfSFlorian Hahn Type *T_int64 = Type::getInt64Ty(Context);
450bcbd26bfSFlorian Hahn
451bcbd26bfSFlorian Hahn FunctionType *FTy =
452bcbd26bfSFlorian Hahn FunctionType::get(Type::getVoidTy(Context), {T_int64}, false);
453bcbd26bfSFlorian Hahn Function *F = Function::Create(FTy, Function::ExternalLinkage, "func", M);
454bcbd26bfSFlorian Hahn Argument *Arg = &*F->arg_begin();
455bcbd26bfSFlorian Hahn ConstantInt *C = ConstantInt::get(Context, APInt(64, -1));
456bcbd26bfSFlorian Hahn
457bcbd26bfSFlorian Hahn BasicBlock *Entry = BasicBlock::Create(Context, "entry", F);
458bcbd26bfSFlorian Hahn BasicBlock *Exit = BasicBlock::Create(Context, "exit", F);
459bcbd26bfSFlorian Hahn
460bcbd26bfSFlorian Hahn IRBuilder<> Builder(Entry);
461bcbd26bfSFlorian Hahn auto *S1 = cast<Instruction>(Builder.CreateAdd(Arg, C, "add"));
462bcbd26bfSFlorian Hahn auto *S2 = cast<Instruction>(Builder.CreateAdd(Arg, C, "add"));
463bcbd26bfSFlorian Hahn S2->setHasNoSignedWrap(true);
464bcbd26bfSFlorian Hahn Builder.CreateBr(Exit);
465bcbd26bfSFlorian Hahn
466bcbd26bfSFlorian Hahn Builder.SetInsertPoint(Exit);
467bcbd26bfSFlorian Hahn auto *R = cast<Instruction>(Builder.CreateRetVoid());
468bcbd26bfSFlorian Hahn
469bcbd26bfSFlorian Hahn ScalarEvolution SE = buildSE(*F);
470bcbd26bfSFlorian Hahn // Get S2 first to move it to cache.
471bcbd26bfSFlorian Hahn const SCEV *SC2 = SE.getSCEV(S2);
472bcbd26bfSFlorian Hahn EXPECT_TRUE(isa<SCEVAddExpr>(SC2));
473bcbd26bfSFlorian Hahn // Now get S1.
474bcbd26bfSFlorian Hahn const SCEV *SC1 = SE.getSCEV(S1);
475bcbd26bfSFlorian Hahn EXPECT_TRUE(isa<SCEVAddExpr>(SC1));
476bcbd26bfSFlorian Hahn // Expand for S1, it should use S1 not S2 in spite S2
477bcbd26bfSFlorian Hahn // first in the cache.
478bcbd26bfSFlorian Hahn SCEVExpander Exp(SE, M.getDataLayout(), "expander");
479bcbd26bfSFlorian Hahn auto *I = cast<Instruction>(Exp.expandCodeFor(SC1, nullptr, R));
480bcbd26bfSFlorian Hahn EXPECT_FALSE(I->hasNoSignedWrap());
481bcbd26bfSFlorian Hahn }
482bcbd26bfSFlorian Hahn
TEST_F(ScalarEvolutionExpanderTest,SCEVExpandInsertCanonicalIV)483bcbd26bfSFlorian Hahn TEST_F(ScalarEvolutionExpanderTest, SCEVExpandInsertCanonicalIV) {
484bcbd26bfSFlorian Hahn LLVMContext C;
485bcbd26bfSFlorian Hahn SMDiagnostic Err;
486bcbd26bfSFlorian Hahn
487bcbd26bfSFlorian Hahn // Expand the addrec produced by GetAddRec into a loop without a canonical IV.
488bcbd26bfSFlorian Hahn // SCEVExpander will insert one.
489bcbd26bfSFlorian Hahn auto TestNoCanonicalIV =
490bcbd26bfSFlorian Hahn [&](std::function<const SCEV *(ScalarEvolution & SE, Loop * L)>
491bcbd26bfSFlorian Hahn GetAddRec) {
492bcbd26bfSFlorian Hahn std::unique_ptr<Module> M = parseAssemblyString(
493bcbd26bfSFlorian Hahn "define i32 @test(i32 %limit) { "
494bcbd26bfSFlorian Hahn "entry: "
495bcbd26bfSFlorian Hahn " br label %loop "
496bcbd26bfSFlorian Hahn "loop: "
497bcbd26bfSFlorian Hahn " %i = phi i32 [ 1, %entry ], [ %i.inc, %loop ] "
498bcbd26bfSFlorian Hahn " %i.inc = add nsw i32 %i, 1 "
499bcbd26bfSFlorian Hahn " %cont = icmp slt i32 %i.inc, %limit "
500bcbd26bfSFlorian Hahn " br i1 %cont, label %loop, label %exit "
501bcbd26bfSFlorian Hahn "exit: "
502bcbd26bfSFlorian Hahn " ret i32 %i.inc "
503bcbd26bfSFlorian Hahn "}",
504bcbd26bfSFlorian Hahn Err, C);
505bcbd26bfSFlorian Hahn
506bcbd26bfSFlorian Hahn assert(M && "Could not parse module?");
507bcbd26bfSFlorian Hahn assert(!verifyModule(*M) && "Must have been well formed!");
508bcbd26bfSFlorian Hahn
509bcbd26bfSFlorian Hahn runWithSE(
510bcbd26bfSFlorian Hahn *M, "test", [&](Function &F, LoopInfo &LI, ScalarEvolution &SE) {
511bcbd26bfSFlorian Hahn auto &I = GetInstByName(F, "i");
512bcbd26bfSFlorian Hahn auto *Loop = LI.getLoopFor(I.getParent());
513bcbd26bfSFlorian Hahn EXPECT_FALSE(Loop->getCanonicalInductionVariable());
514bcbd26bfSFlorian Hahn
515bcbd26bfSFlorian Hahn auto *AR = GetAddRec(SE, Loop);
516bcbd26bfSFlorian Hahn unsigned ExpectedCanonicalIVWidth =
517bcbd26bfSFlorian Hahn SE.getTypeSizeInBits(AR->getType());
518bcbd26bfSFlorian Hahn
519bcbd26bfSFlorian Hahn SCEVExpander Exp(SE, M->getDataLayout(), "expander");
520bcbd26bfSFlorian Hahn auto *InsertAt = I.getNextNode();
521bcbd26bfSFlorian Hahn Exp.expandCodeFor(AR, nullptr, InsertAt);
522bcbd26bfSFlorian Hahn PHINode *CanonicalIV = Loop->getCanonicalInductionVariable();
523bcbd26bfSFlorian Hahn unsigned CanonicalIVBitWidth =
524bcbd26bfSFlorian Hahn cast<IntegerType>(CanonicalIV->getType())->getBitWidth();
525bcbd26bfSFlorian Hahn EXPECT_EQ(CanonicalIVBitWidth, ExpectedCanonicalIVWidth);
526bcbd26bfSFlorian Hahn });
527bcbd26bfSFlorian Hahn };
528bcbd26bfSFlorian Hahn
529bcbd26bfSFlorian Hahn // Expand the addrec produced by GetAddRec into a loop with a canonical IV
530bcbd26bfSFlorian Hahn // which is narrower than addrec type.
531bcbd26bfSFlorian Hahn // SCEVExpander will insert a canonical IV of a wider type to expand the
532bcbd26bfSFlorian Hahn // addrec.
533bcbd26bfSFlorian Hahn auto TestNarrowCanonicalIV = [&](std::function<const SCEV *(
534bcbd26bfSFlorian Hahn ScalarEvolution & SE, Loop * L)>
535bcbd26bfSFlorian Hahn GetAddRec) {
536bcbd26bfSFlorian Hahn std::unique_ptr<Module> M = parseAssemblyString(
537bcbd26bfSFlorian Hahn "define i32 @test(i32 %limit) { "
538bcbd26bfSFlorian Hahn "entry: "
539bcbd26bfSFlorian Hahn " br label %loop "
540bcbd26bfSFlorian Hahn "loop: "
541bcbd26bfSFlorian Hahn " %i = phi i32 [ 1, %entry ], [ %i.inc, %loop ] "
542bcbd26bfSFlorian Hahn " %canonical.iv = phi i8 [ 0, %entry ], [ %canonical.iv.inc, %loop ] "
543bcbd26bfSFlorian Hahn " %i.inc = add nsw i32 %i, 1 "
544bcbd26bfSFlorian Hahn " %canonical.iv.inc = add i8 %canonical.iv, 1 "
545bcbd26bfSFlorian Hahn " %cont = icmp slt i32 %i.inc, %limit "
546bcbd26bfSFlorian Hahn " br i1 %cont, label %loop, label %exit "
547bcbd26bfSFlorian Hahn "exit: "
548bcbd26bfSFlorian Hahn " ret i32 %i.inc "
549bcbd26bfSFlorian Hahn "}",
550bcbd26bfSFlorian Hahn Err, C);
551bcbd26bfSFlorian Hahn
552bcbd26bfSFlorian Hahn assert(M && "Could not parse module?");
553bcbd26bfSFlorian Hahn assert(!verifyModule(*M) && "Must have been well formed!");
554bcbd26bfSFlorian Hahn
555bcbd26bfSFlorian Hahn runWithSE(*M, "test", [&](Function &F, LoopInfo &LI, ScalarEvolution &SE) {
556bcbd26bfSFlorian Hahn auto &I = GetInstByName(F, "i");
557bcbd26bfSFlorian Hahn
558bcbd26bfSFlorian Hahn auto *LoopHeaderBB = I.getParent();
559bcbd26bfSFlorian Hahn auto *Loop = LI.getLoopFor(LoopHeaderBB);
560bcbd26bfSFlorian Hahn PHINode *CanonicalIV = Loop->getCanonicalInductionVariable();
561bcbd26bfSFlorian Hahn EXPECT_EQ(CanonicalIV, &GetInstByName(F, "canonical.iv"));
562bcbd26bfSFlorian Hahn
563bcbd26bfSFlorian Hahn auto *AR = GetAddRec(SE, Loop);
564bcbd26bfSFlorian Hahn
565bcbd26bfSFlorian Hahn unsigned ExpectedCanonicalIVWidth = SE.getTypeSizeInBits(AR->getType());
566bcbd26bfSFlorian Hahn unsigned CanonicalIVBitWidth =
567bcbd26bfSFlorian Hahn cast<IntegerType>(CanonicalIV->getType())->getBitWidth();
568bcbd26bfSFlorian Hahn EXPECT_LT(CanonicalIVBitWidth, ExpectedCanonicalIVWidth);
569bcbd26bfSFlorian Hahn
570bcbd26bfSFlorian Hahn SCEVExpander Exp(SE, M->getDataLayout(), "expander");
571bcbd26bfSFlorian Hahn auto *InsertAt = I.getNextNode();
572bcbd26bfSFlorian Hahn Exp.expandCodeFor(AR, nullptr, InsertAt);
573bcbd26bfSFlorian Hahn
574bcbd26bfSFlorian Hahn // Loop over all of the PHI nodes, looking for the new canonical indvar.
575bcbd26bfSFlorian Hahn PHINode *NewCanonicalIV = nullptr;
576bcbd26bfSFlorian Hahn for (BasicBlock::iterator i = LoopHeaderBB->begin(); isa<PHINode>(i);
577bcbd26bfSFlorian Hahn ++i) {
578bcbd26bfSFlorian Hahn PHINode *PN = cast<PHINode>(i);
579bcbd26bfSFlorian Hahn if (PN == &I || PN == CanonicalIV)
580bcbd26bfSFlorian Hahn continue;
581bcbd26bfSFlorian Hahn // We expect that the only PHI added is the new canonical IV
582bcbd26bfSFlorian Hahn EXPECT_FALSE(NewCanonicalIV);
583bcbd26bfSFlorian Hahn NewCanonicalIV = PN;
584bcbd26bfSFlorian Hahn }
585bcbd26bfSFlorian Hahn
586bcbd26bfSFlorian Hahn // Check that NewCanonicalIV is a canonical IV, i.e {0,+,1}
587bcbd26bfSFlorian Hahn BasicBlock *Incoming = nullptr, *Backedge = nullptr;
588bcbd26bfSFlorian Hahn EXPECT_TRUE(Loop->getIncomingAndBackEdge(Incoming, Backedge));
589bcbd26bfSFlorian Hahn auto *Start = NewCanonicalIV->getIncomingValueForBlock(Incoming);
590bcbd26bfSFlorian Hahn EXPECT_TRUE(isa<ConstantInt>(Start));
591bcbd26bfSFlorian Hahn EXPECT_TRUE(dyn_cast<ConstantInt>(Start)->isZero());
592bcbd26bfSFlorian Hahn auto *Next = NewCanonicalIV->getIncomingValueForBlock(Backedge);
593bcbd26bfSFlorian Hahn EXPECT_TRUE(isa<BinaryOperator>(Next));
594bcbd26bfSFlorian Hahn auto *NextBinOp = dyn_cast<BinaryOperator>(Next);
595bcbd26bfSFlorian Hahn EXPECT_EQ(NextBinOp->getOpcode(), Instruction::Add);
596bcbd26bfSFlorian Hahn EXPECT_EQ(NextBinOp->getOperand(0), NewCanonicalIV);
597bcbd26bfSFlorian Hahn auto *Step = NextBinOp->getOperand(1);
598bcbd26bfSFlorian Hahn EXPECT_TRUE(isa<ConstantInt>(Step));
599bcbd26bfSFlorian Hahn EXPECT_TRUE(dyn_cast<ConstantInt>(Step)->isOne());
600bcbd26bfSFlorian Hahn
601bcbd26bfSFlorian Hahn unsigned NewCanonicalIVBitWidth =
602bcbd26bfSFlorian Hahn cast<IntegerType>(NewCanonicalIV->getType())->getBitWidth();
603bcbd26bfSFlorian Hahn EXPECT_EQ(NewCanonicalIVBitWidth, ExpectedCanonicalIVWidth);
604bcbd26bfSFlorian Hahn });
605bcbd26bfSFlorian Hahn };
606bcbd26bfSFlorian Hahn
607bcbd26bfSFlorian Hahn // Expand the addrec produced by GetAddRec into a loop with a canonical IV
608bcbd26bfSFlorian Hahn // of addrec width.
609bcbd26bfSFlorian Hahn // To expand the addrec SCEVExpander should use the existing canonical IV.
610bcbd26bfSFlorian Hahn auto TestMatchingCanonicalIV =
611bcbd26bfSFlorian Hahn [&](std::function<const SCEV *(ScalarEvolution & SE, Loop * L)> GetAddRec,
612bcbd26bfSFlorian Hahn unsigned ARBitWidth) {
613bcbd26bfSFlorian Hahn auto ARBitWidthTypeStr = "i" + std::to_string(ARBitWidth);
614bcbd26bfSFlorian Hahn std::unique_ptr<Module> M = parseAssemblyString(
615bcbd26bfSFlorian Hahn "define i32 @test(i32 %limit) { "
616bcbd26bfSFlorian Hahn "entry: "
617bcbd26bfSFlorian Hahn " br label %loop "
618bcbd26bfSFlorian Hahn "loop: "
619bcbd26bfSFlorian Hahn " %i = phi i32 [ 1, %entry ], [ %i.inc, %loop ] "
620bcbd26bfSFlorian Hahn " %canonical.iv = phi " +
621bcbd26bfSFlorian Hahn ARBitWidthTypeStr +
622bcbd26bfSFlorian Hahn " [ 0, %entry ], [ %canonical.iv.inc, %loop ] "
623bcbd26bfSFlorian Hahn " %i.inc = add nsw i32 %i, 1 "
624bcbd26bfSFlorian Hahn " %canonical.iv.inc = add " +
625bcbd26bfSFlorian Hahn ARBitWidthTypeStr +
626bcbd26bfSFlorian Hahn " %canonical.iv, 1 "
627bcbd26bfSFlorian Hahn " %cont = icmp slt i32 %i.inc, %limit "
628bcbd26bfSFlorian Hahn " br i1 %cont, label %loop, label %exit "
629bcbd26bfSFlorian Hahn "exit: "
630bcbd26bfSFlorian Hahn " ret i32 %i.inc "
631bcbd26bfSFlorian Hahn "}",
632bcbd26bfSFlorian Hahn Err, C);
633bcbd26bfSFlorian Hahn
634bcbd26bfSFlorian Hahn assert(M && "Could not parse module?");
635bcbd26bfSFlorian Hahn assert(!verifyModule(*M) && "Must have been well formed!");
636bcbd26bfSFlorian Hahn
637bcbd26bfSFlorian Hahn runWithSE(
638bcbd26bfSFlorian Hahn *M, "test", [&](Function &F, LoopInfo &LI, ScalarEvolution &SE) {
639bcbd26bfSFlorian Hahn auto &I = GetInstByName(F, "i");
640bcbd26bfSFlorian Hahn auto &CanonicalIV = GetInstByName(F, "canonical.iv");
641bcbd26bfSFlorian Hahn
642bcbd26bfSFlorian Hahn auto *LoopHeaderBB = I.getParent();
643bcbd26bfSFlorian Hahn auto *Loop = LI.getLoopFor(LoopHeaderBB);
644bcbd26bfSFlorian Hahn EXPECT_EQ(&CanonicalIV, Loop->getCanonicalInductionVariable());
645bcbd26bfSFlorian Hahn unsigned CanonicalIVBitWidth =
646bcbd26bfSFlorian Hahn cast<IntegerType>(CanonicalIV.getType())->getBitWidth();
647bcbd26bfSFlorian Hahn
648bcbd26bfSFlorian Hahn auto *AR = GetAddRec(SE, Loop);
649bcbd26bfSFlorian Hahn EXPECT_EQ(ARBitWidth, SE.getTypeSizeInBits(AR->getType()));
650bcbd26bfSFlorian Hahn EXPECT_EQ(CanonicalIVBitWidth, ARBitWidth);
651bcbd26bfSFlorian Hahn
652bcbd26bfSFlorian Hahn SCEVExpander Exp(SE, M->getDataLayout(), "expander");
653bcbd26bfSFlorian Hahn auto *InsertAt = I.getNextNode();
654bcbd26bfSFlorian Hahn Exp.expandCodeFor(AR, nullptr, InsertAt);
655bcbd26bfSFlorian Hahn
656bcbd26bfSFlorian Hahn // Loop over all of the PHI nodes, looking if a new canonical
657bcbd26bfSFlorian Hahn // indvar was introduced.
658bcbd26bfSFlorian Hahn PHINode *NewCanonicalIV = nullptr;
659bcbd26bfSFlorian Hahn for (BasicBlock::iterator i = LoopHeaderBB->begin();
660bcbd26bfSFlorian Hahn isa<PHINode>(i); ++i) {
661bcbd26bfSFlorian Hahn PHINode *PN = cast<PHINode>(i);
662bcbd26bfSFlorian Hahn if (PN == &I || PN == &CanonicalIV)
663bcbd26bfSFlorian Hahn continue;
664bcbd26bfSFlorian Hahn NewCanonicalIV = PN;
665bcbd26bfSFlorian Hahn }
666bcbd26bfSFlorian Hahn EXPECT_FALSE(NewCanonicalIV);
667bcbd26bfSFlorian Hahn });
668bcbd26bfSFlorian Hahn };
669bcbd26bfSFlorian Hahn
670bcbd26bfSFlorian Hahn unsigned ARBitWidth = 16;
671bcbd26bfSFlorian Hahn Type *ARType = IntegerType::get(C, ARBitWidth);
672bcbd26bfSFlorian Hahn
673bcbd26bfSFlorian Hahn // Expand {5,+,1}
674bcbd26bfSFlorian Hahn auto GetAR2 = [&](ScalarEvolution &SE, Loop *L) -> const SCEV * {
675bcbd26bfSFlorian Hahn return SE.getAddRecExpr(SE.getConstant(APInt(ARBitWidth, 5)),
676bcbd26bfSFlorian Hahn SE.getOne(ARType), L, SCEV::FlagAnyWrap);
677bcbd26bfSFlorian Hahn };
678bcbd26bfSFlorian Hahn TestNoCanonicalIV(GetAR2);
679bcbd26bfSFlorian Hahn TestNarrowCanonicalIV(GetAR2);
680bcbd26bfSFlorian Hahn TestMatchingCanonicalIV(GetAR2, ARBitWidth);
681bcbd26bfSFlorian Hahn }
682bcbd26bfSFlorian Hahn
TEST_F(ScalarEvolutionExpanderTest,SCEVExpanderShlNSW)683bcbd26bfSFlorian Hahn TEST_F(ScalarEvolutionExpanderTest, SCEVExpanderShlNSW) {
684bcbd26bfSFlorian Hahn
685bcbd26bfSFlorian Hahn auto checkOneCase = [this](std::string &&str) {
686bcbd26bfSFlorian Hahn LLVMContext C;
687bcbd26bfSFlorian Hahn SMDiagnostic Err;
688bcbd26bfSFlorian Hahn std::unique_ptr<Module> M = parseAssemblyString(str, Err, C);
689bcbd26bfSFlorian Hahn
690bcbd26bfSFlorian Hahn assert(M && "Could not parse module?");
691bcbd26bfSFlorian Hahn assert(!verifyModule(*M) && "Must have been well formed!");
692bcbd26bfSFlorian Hahn
693bcbd26bfSFlorian Hahn Function *F = M->getFunction("f");
694bcbd26bfSFlorian Hahn ASSERT_NE(F, nullptr) << "Could not find function 'f'";
695bcbd26bfSFlorian Hahn
696bcbd26bfSFlorian Hahn BasicBlock &Entry = F->getEntryBlock();
697bcbd26bfSFlorian Hahn LoadInst *Load = cast<LoadInst>(&Entry.front());
698bcbd26bfSFlorian Hahn BinaryOperator *And = cast<BinaryOperator>(*Load->user_begin());
699bcbd26bfSFlorian Hahn
700bcbd26bfSFlorian Hahn ScalarEvolution SE = buildSE(*F);
701bcbd26bfSFlorian Hahn const SCEV *AndSCEV = SE.getSCEV(And);
702bcbd26bfSFlorian Hahn EXPECT_TRUE(isa<SCEVMulExpr>(AndSCEV));
703bcbd26bfSFlorian Hahn EXPECT_TRUE(cast<SCEVMulExpr>(AndSCEV)->hasNoSignedWrap());
704bcbd26bfSFlorian Hahn
705bcbd26bfSFlorian Hahn SCEVExpander Exp(SE, M->getDataLayout(), "expander");
706bcbd26bfSFlorian Hahn auto *I = cast<Instruction>(Exp.expandCodeFor(AndSCEV, nullptr, And));
707bcbd26bfSFlorian Hahn EXPECT_EQ(I->getOpcode(), Instruction::Shl);
708bcbd26bfSFlorian Hahn EXPECT_FALSE(I->hasNoSignedWrap());
709bcbd26bfSFlorian Hahn };
710bcbd26bfSFlorian Hahn
711bcbd26bfSFlorian Hahn checkOneCase("define void @f(i16* %arrayidx) { "
712bcbd26bfSFlorian Hahn " %1 = load i16, i16* %arrayidx "
713bcbd26bfSFlorian Hahn " %2 = and i16 %1, -32768 "
714bcbd26bfSFlorian Hahn " ret void "
715bcbd26bfSFlorian Hahn "} ");
716bcbd26bfSFlorian Hahn
717bcbd26bfSFlorian Hahn checkOneCase("define void @f(i8* %arrayidx) { "
718bcbd26bfSFlorian Hahn " %1 = load i8, i8* %arrayidx "
719bcbd26bfSFlorian Hahn " %2 = and i8 %1, -128 "
720bcbd26bfSFlorian Hahn " ret void "
721bcbd26bfSFlorian Hahn "} ");
722bcbd26bfSFlorian Hahn }
723bcbd26bfSFlorian Hahn
724bcbd26bfSFlorian Hahn // Test expansion of nested addrecs in CanonicalMode.
725bcbd26bfSFlorian Hahn // Expanding nested addrecs in canonical mode requiers a canonical IV of a
726bcbd26bfSFlorian Hahn // type wider than the type of the addrec itself. Currently, SCEVExpander
727bcbd26bfSFlorian Hahn // just falls back to literal mode for nested addrecs.
TEST_F(ScalarEvolutionExpanderTest,SCEVExpandNonAffineAddRec)728bcbd26bfSFlorian Hahn TEST_F(ScalarEvolutionExpanderTest, SCEVExpandNonAffineAddRec) {
729bcbd26bfSFlorian Hahn LLVMContext C;
730bcbd26bfSFlorian Hahn SMDiagnostic Err;
731bcbd26bfSFlorian Hahn
732bcbd26bfSFlorian Hahn // Expand the addrec produced by GetAddRec into a loop without a canonical IV.
733bcbd26bfSFlorian Hahn auto TestNoCanonicalIV =
734bcbd26bfSFlorian Hahn [&](std::function<const SCEVAddRecExpr *(ScalarEvolution & SE, Loop * L)>
735bcbd26bfSFlorian Hahn GetAddRec) {
736bcbd26bfSFlorian Hahn std::unique_ptr<Module> M = parseAssemblyString(
737bcbd26bfSFlorian Hahn "define i32 @test(i32 %limit) { "
738bcbd26bfSFlorian Hahn "entry: "
739bcbd26bfSFlorian Hahn " br label %loop "
740bcbd26bfSFlorian Hahn "loop: "
741bcbd26bfSFlorian Hahn " %i = phi i32 [ 1, %entry ], [ %i.inc, %loop ] "
742bcbd26bfSFlorian Hahn " %i.inc = add nsw i32 %i, 1 "
743bcbd26bfSFlorian Hahn " %cont = icmp slt i32 %i.inc, %limit "
744bcbd26bfSFlorian Hahn " br i1 %cont, label %loop, label %exit "
745bcbd26bfSFlorian Hahn "exit: "
746bcbd26bfSFlorian Hahn " ret i32 %i.inc "
747bcbd26bfSFlorian Hahn "}",
748bcbd26bfSFlorian Hahn Err, C);
749bcbd26bfSFlorian Hahn
750bcbd26bfSFlorian Hahn assert(M && "Could not parse module?");
751bcbd26bfSFlorian Hahn assert(!verifyModule(*M) && "Must have been well formed!");
752bcbd26bfSFlorian Hahn
753bcbd26bfSFlorian Hahn runWithSE(*M, "test",
754bcbd26bfSFlorian Hahn [&](Function &F, LoopInfo &LI, ScalarEvolution &SE) {
755bcbd26bfSFlorian Hahn auto &I = GetInstByName(F, "i");
756bcbd26bfSFlorian Hahn auto *Loop = LI.getLoopFor(I.getParent());
757bcbd26bfSFlorian Hahn EXPECT_FALSE(Loop->getCanonicalInductionVariable());
758bcbd26bfSFlorian Hahn
759bcbd26bfSFlorian Hahn auto *AR = GetAddRec(SE, Loop);
760bcbd26bfSFlorian Hahn EXPECT_FALSE(AR->isAffine());
761bcbd26bfSFlorian Hahn
762bcbd26bfSFlorian Hahn SCEVExpander Exp(SE, M->getDataLayout(), "expander");
763bcbd26bfSFlorian Hahn auto *InsertAt = I.getNextNode();
764bcbd26bfSFlorian Hahn Value *V = Exp.expandCodeFor(AR, nullptr, InsertAt);
765bcbd26bfSFlorian Hahn auto *ExpandedAR = SE.getSCEV(V);
766bcbd26bfSFlorian Hahn // Check that the expansion happened literally.
767bcbd26bfSFlorian Hahn EXPECT_EQ(AR, ExpandedAR);
768bcbd26bfSFlorian Hahn });
769bcbd26bfSFlorian Hahn };
770bcbd26bfSFlorian Hahn
771bcbd26bfSFlorian Hahn // Expand the addrec produced by GetAddRec into a loop with a canonical IV
772bcbd26bfSFlorian Hahn // which is narrower than addrec type.
773bcbd26bfSFlorian Hahn auto TestNarrowCanonicalIV = [&](std::function<const SCEVAddRecExpr *(
774bcbd26bfSFlorian Hahn ScalarEvolution & SE, Loop * L)>
775bcbd26bfSFlorian Hahn GetAddRec) {
776bcbd26bfSFlorian Hahn std::unique_ptr<Module> M = parseAssemblyString(
777bcbd26bfSFlorian Hahn "define i32 @test(i32 %limit) { "
778bcbd26bfSFlorian Hahn "entry: "
779bcbd26bfSFlorian Hahn " br label %loop "
780bcbd26bfSFlorian Hahn "loop: "
781bcbd26bfSFlorian Hahn " %i = phi i32 [ 1, %entry ], [ %i.inc, %loop ] "
782bcbd26bfSFlorian Hahn " %canonical.iv = phi i8 [ 0, %entry ], [ %canonical.iv.inc, %loop ] "
783bcbd26bfSFlorian Hahn " %i.inc = add nsw i32 %i, 1 "
784bcbd26bfSFlorian Hahn " %canonical.iv.inc = add i8 %canonical.iv, 1 "
785bcbd26bfSFlorian Hahn " %cont = icmp slt i32 %i.inc, %limit "
786bcbd26bfSFlorian Hahn " br i1 %cont, label %loop, label %exit "
787bcbd26bfSFlorian Hahn "exit: "
788bcbd26bfSFlorian Hahn " ret i32 %i.inc "
789bcbd26bfSFlorian Hahn "}",
790bcbd26bfSFlorian Hahn Err, C);
791bcbd26bfSFlorian Hahn
792bcbd26bfSFlorian Hahn assert(M && "Could not parse module?");
793bcbd26bfSFlorian Hahn assert(!verifyModule(*M) && "Must have been well formed!");
794bcbd26bfSFlorian Hahn
795bcbd26bfSFlorian Hahn runWithSE(*M, "test", [&](Function &F, LoopInfo &LI, ScalarEvolution &SE) {
796bcbd26bfSFlorian Hahn auto &I = GetInstByName(F, "i");
797bcbd26bfSFlorian Hahn
798bcbd26bfSFlorian Hahn auto *LoopHeaderBB = I.getParent();
799bcbd26bfSFlorian Hahn auto *Loop = LI.getLoopFor(LoopHeaderBB);
800bcbd26bfSFlorian Hahn PHINode *CanonicalIV = Loop->getCanonicalInductionVariable();
801bcbd26bfSFlorian Hahn EXPECT_EQ(CanonicalIV, &GetInstByName(F, "canonical.iv"));
802bcbd26bfSFlorian Hahn
803bcbd26bfSFlorian Hahn auto *AR = GetAddRec(SE, Loop);
804bcbd26bfSFlorian Hahn EXPECT_FALSE(AR->isAffine());
805bcbd26bfSFlorian Hahn
806bcbd26bfSFlorian Hahn unsigned ExpectedCanonicalIVWidth = SE.getTypeSizeInBits(AR->getType());
807bcbd26bfSFlorian Hahn unsigned CanonicalIVBitWidth =
808bcbd26bfSFlorian Hahn cast<IntegerType>(CanonicalIV->getType())->getBitWidth();
809bcbd26bfSFlorian Hahn EXPECT_LT(CanonicalIVBitWidth, ExpectedCanonicalIVWidth);
810bcbd26bfSFlorian Hahn
811bcbd26bfSFlorian Hahn SCEVExpander Exp(SE, M->getDataLayout(), "expander");
812bcbd26bfSFlorian Hahn auto *InsertAt = I.getNextNode();
813bcbd26bfSFlorian Hahn Value *V = Exp.expandCodeFor(AR, nullptr, InsertAt);
814bcbd26bfSFlorian Hahn auto *ExpandedAR = SE.getSCEV(V);
815bcbd26bfSFlorian Hahn // Check that the expansion happened literally.
816bcbd26bfSFlorian Hahn EXPECT_EQ(AR, ExpandedAR);
817bcbd26bfSFlorian Hahn });
818bcbd26bfSFlorian Hahn };
819bcbd26bfSFlorian Hahn
820bcbd26bfSFlorian Hahn // Expand the addrec produced by GetAddRec into a loop with a canonical IV
821bcbd26bfSFlorian Hahn // of addrec width.
822bcbd26bfSFlorian Hahn auto TestMatchingCanonicalIV =
823bcbd26bfSFlorian Hahn [&](std::function<const SCEVAddRecExpr *(ScalarEvolution & SE, Loop * L)>
824bcbd26bfSFlorian Hahn GetAddRec,
825bcbd26bfSFlorian Hahn unsigned ARBitWidth) {
826bcbd26bfSFlorian Hahn auto ARBitWidthTypeStr = "i" + std::to_string(ARBitWidth);
827bcbd26bfSFlorian Hahn std::unique_ptr<Module> M = parseAssemblyString(
828bcbd26bfSFlorian Hahn "define i32 @test(i32 %limit) { "
829bcbd26bfSFlorian Hahn "entry: "
830bcbd26bfSFlorian Hahn " br label %loop "
831bcbd26bfSFlorian Hahn "loop: "
832bcbd26bfSFlorian Hahn " %i = phi i32 [ 1, %entry ], [ %i.inc, %loop ] "
833bcbd26bfSFlorian Hahn " %canonical.iv = phi " +
834bcbd26bfSFlorian Hahn ARBitWidthTypeStr +
835bcbd26bfSFlorian Hahn " [ 0, %entry ], [ %canonical.iv.inc, %loop ] "
836bcbd26bfSFlorian Hahn " %i.inc = add nsw i32 %i, 1 "
837bcbd26bfSFlorian Hahn " %canonical.iv.inc = add " +
838bcbd26bfSFlorian Hahn ARBitWidthTypeStr +
839bcbd26bfSFlorian Hahn " %canonical.iv, 1 "
840bcbd26bfSFlorian Hahn " %cont = icmp slt i32 %i.inc, %limit "
841bcbd26bfSFlorian Hahn " br i1 %cont, label %loop, label %exit "
842bcbd26bfSFlorian Hahn "exit: "
843bcbd26bfSFlorian Hahn " ret i32 %i.inc "
844bcbd26bfSFlorian Hahn "}",
845bcbd26bfSFlorian Hahn Err, C);
846bcbd26bfSFlorian Hahn
847bcbd26bfSFlorian Hahn assert(M && "Could not parse module?");
848bcbd26bfSFlorian Hahn assert(!verifyModule(*M) && "Must have been well formed!");
849bcbd26bfSFlorian Hahn
850bcbd26bfSFlorian Hahn runWithSE(
851bcbd26bfSFlorian Hahn *M, "test", [&](Function &F, LoopInfo &LI, ScalarEvolution &SE) {
852bcbd26bfSFlorian Hahn auto &I = GetInstByName(F, "i");
853bcbd26bfSFlorian Hahn auto &CanonicalIV = GetInstByName(F, "canonical.iv");
854bcbd26bfSFlorian Hahn
855bcbd26bfSFlorian Hahn auto *LoopHeaderBB = I.getParent();
856bcbd26bfSFlorian Hahn auto *Loop = LI.getLoopFor(LoopHeaderBB);
857bcbd26bfSFlorian Hahn EXPECT_EQ(&CanonicalIV, Loop->getCanonicalInductionVariable());
858bcbd26bfSFlorian Hahn unsigned CanonicalIVBitWidth =
859bcbd26bfSFlorian Hahn cast<IntegerType>(CanonicalIV.getType())->getBitWidth();
860bcbd26bfSFlorian Hahn
861bcbd26bfSFlorian Hahn auto *AR = GetAddRec(SE, Loop);
862bcbd26bfSFlorian Hahn EXPECT_FALSE(AR->isAffine());
863bcbd26bfSFlorian Hahn EXPECT_EQ(ARBitWidth, SE.getTypeSizeInBits(AR->getType()));
864bcbd26bfSFlorian Hahn EXPECT_EQ(CanonicalIVBitWidth, ARBitWidth);
865bcbd26bfSFlorian Hahn
866bcbd26bfSFlorian Hahn SCEVExpander Exp(SE, M->getDataLayout(), "expander");
867bcbd26bfSFlorian Hahn auto *InsertAt = I.getNextNode();
868bcbd26bfSFlorian Hahn Value *V = Exp.expandCodeFor(AR, nullptr, InsertAt);
869bcbd26bfSFlorian Hahn auto *ExpandedAR = SE.getSCEV(V);
870bcbd26bfSFlorian Hahn // Check that the expansion happened literally.
871bcbd26bfSFlorian Hahn EXPECT_EQ(AR, ExpandedAR);
872bcbd26bfSFlorian Hahn });
873bcbd26bfSFlorian Hahn };
874bcbd26bfSFlorian Hahn
875bcbd26bfSFlorian Hahn unsigned ARBitWidth = 16;
876bcbd26bfSFlorian Hahn Type *ARType = IntegerType::get(C, ARBitWidth);
877bcbd26bfSFlorian Hahn
878bcbd26bfSFlorian Hahn // Expand {5,+,1,+,1}
879bcbd26bfSFlorian Hahn auto GetAR3 = [&](ScalarEvolution &SE, Loop *L) -> const SCEVAddRecExpr * {
880bcbd26bfSFlorian Hahn SmallVector<const SCEV *, 3> Ops = {SE.getConstant(APInt(ARBitWidth, 5)),
881bcbd26bfSFlorian Hahn SE.getOne(ARType), SE.getOne(ARType)};
882bcbd26bfSFlorian Hahn return cast<SCEVAddRecExpr>(SE.getAddRecExpr(Ops, L, SCEV::FlagAnyWrap));
883bcbd26bfSFlorian Hahn };
884bcbd26bfSFlorian Hahn TestNoCanonicalIV(GetAR3);
885bcbd26bfSFlorian Hahn TestNarrowCanonicalIV(GetAR3);
886bcbd26bfSFlorian Hahn TestMatchingCanonicalIV(GetAR3, ARBitWidth);
887bcbd26bfSFlorian Hahn
888bcbd26bfSFlorian Hahn // Expand {5,+,1,+,1,+,1}
889bcbd26bfSFlorian Hahn auto GetAR4 = [&](ScalarEvolution &SE, Loop *L) -> const SCEVAddRecExpr * {
890bcbd26bfSFlorian Hahn SmallVector<const SCEV *, 4> Ops = {SE.getConstant(APInt(ARBitWidth, 5)),
891bcbd26bfSFlorian Hahn SE.getOne(ARType), SE.getOne(ARType),
892bcbd26bfSFlorian Hahn SE.getOne(ARType)};
893bcbd26bfSFlorian Hahn return cast<SCEVAddRecExpr>(SE.getAddRecExpr(Ops, L, SCEV::FlagAnyWrap));
894bcbd26bfSFlorian Hahn };
895bcbd26bfSFlorian Hahn TestNoCanonicalIV(GetAR4);
896bcbd26bfSFlorian Hahn TestNarrowCanonicalIV(GetAR4);
897bcbd26bfSFlorian Hahn TestMatchingCanonicalIV(GetAR4, ARBitWidth);
898bcbd26bfSFlorian Hahn
899bcbd26bfSFlorian Hahn // Expand {5,+,1,+,1,+,1,+,1}
900bcbd26bfSFlorian Hahn auto GetAR5 = [&](ScalarEvolution &SE, Loop *L) -> const SCEVAddRecExpr * {
901bcbd26bfSFlorian Hahn SmallVector<const SCEV *, 5> Ops = {SE.getConstant(APInt(ARBitWidth, 5)),
902bcbd26bfSFlorian Hahn SE.getOne(ARType), SE.getOne(ARType),
903bcbd26bfSFlorian Hahn SE.getOne(ARType), SE.getOne(ARType)};
904bcbd26bfSFlorian Hahn return cast<SCEVAddRecExpr>(SE.getAddRecExpr(Ops, L, SCEV::FlagAnyWrap));
905bcbd26bfSFlorian Hahn };
906bcbd26bfSFlorian Hahn TestNoCanonicalIV(GetAR5);
907bcbd26bfSFlorian Hahn TestNarrowCanonicalIV(GetAR5);
908bcbd26bfSFlorian Hahn TestMatchingCanonicalIV(GetAR5, ARBitWidth);
909bcbd26bfSFlorian Hahn }
910bcbd26bfSFlorian Hahn
TEST_F(ScalarEvolutionExpanderTest,ExpandNonIntegralPtrWithNullBase)9111d8f2e52SFlorian Hahn TEST_F(ScalarEvolutionExpanderTest, ExpandNonIntegralPtrWithNullBase) {
9121d8f2e52SFlorian Hahn LLVMContext C;
9131d8f2e52SFlorian Hahn SMDiagnostic Err;
9141d8f2e52SFlorian Hahn
9151d8f2e52SFlorian Hahn std::unique_ptr<Module> M =
9161d8f2e52SFlorian Hahn parseAssemblyString("target datalayout = "
9171d8f2e52SFlorian Hahn "\"e-m:e-p270:32:32-p271:32:32-p272:64:64-i64:64-f80:"
9181d8f2e52SFlorian Hahn "128-n8:16:32:64-S128-ni:1-p2:32:8:8:32-ni:2\""
9191d8f2e52SFlorian Hahn "define float addrspace(1)* @test(i64 %offset) { "
9201d8f2e52SFlorian Hahn " %ptr = getelementptr inbounds float, float "
9211d8f2e52SFlorian Hahn "addrspace(1)* null, i64 %offset"
9221d8f2e52SFlorian Hahn " ret float addrspace(1)* %ptr"
9231d8f2e52SFlorian Hahn "}",
9241d8f2e52SFlorian Hahn Err, C);
9251d8f2e52SFlorian Hahn
9261d8f2e52SFlorian Hahn assert(M && "Could not parse module?");
9271d8f2e52SFlorian Hahn assert(!verifyModule(*M) && "Must have been well formed!");
9281d8f2e52SFlorian Hahn
9291d8f2e52SFlorian Hahn runWithSE(*M, "test", [&](Function &F, LoopInfo &LI, ScalarEvolution &SE) {
9301d8f2e52SFlorian Hahn auto &I = GetInstByName(F, "ptr");
9311d8f2e52SFlorian Hahn auto PtrPlus1 =
9321d8f2e52SFlorian Hahn SE.getAddExpr(SE.getSCEV(&I), SE.getConstant(I.getType(), 1));
9331d8f2e52SFlorian Hahn SCEVExpander Exp(SE, M->getDataLayout(), "expander");
9341d8f2e52SFlorian Hahn
9351d8f2e52SFlorian Hahn Value *V = Exp.expandCodeFor(PtrPlus1, I.getType(), &I);
9361d8f2e52SFlorian Hahn I.replaceAllUsesWith(V);
9371d8f2e52SFlorian Hahn
93878b8ce40SRoman Lebedev // Check that the expander created:
93978b8ce40SRoman Lebedev // define float addrspace(1)* @test(i64 %off) {
94078b8ce40SRoman Lebedev // %scevgep = getelementptr float, float addrspace(1)* null, i64 %off
94178b8ce40SRoman Lebedev // %scevgep1 = bitcast float addrspace(1)* %scevgep to i8 addrspace(1)*
94278b8ce40SRoman Lebedev // %uglygep = getelementptr i8, i8 addrspace(1)* %scevgep1, i64 1
94378b8ce40SRoman Lebedev // %uglygep2 = bitcast i8 addrspace(1)* %uglygep to float addrspace(1)*
94478b8ce40SRoman Lebedev // %ptr = getelementptr inbounds float, float addrspace(1)* null, i64 %off
94578b8ce40SRoman Lebedev // ret float addrspace(1)* %uglygep2
94678b8ce40SRoman Lebedev // }
94778b8ce40SRoman Lebedev
9481d8f2e52SFlorian Hahn auto *Cast = dyn_cast<BitCastInst>(V);
9491d8f2e52SFlorian Hahn EXPECT_TRUE(Cast);
9501d8f2e52SFlorian Hahn EXPECT_EQ(Cast->getType(), I.getType());
9511d8f2e52SFlorian Hahn auto *GEP = dyn_cast<GetElementPtrInst>(Cast->getOperand(0));
9521d8f2e52SFlorian Hahn EXPECT_TRUE(GEP);
95378b8ce40SRoman Lebedev EXPECT_TRUE(match(GEP->getOperand(1), m_SpecificInt(1)));
95478b8ce40SRoman Lebedev auto *Cast1 = dyn_cast<BitCastInst>(GEP->getPointerOperand());
95578b8ce40SRoman Lebedev EXPECT_TRUE(Cast1);
95678b8ce40SRoman Lebedev auto *GEP1 = dyn_cast<GetElementPtrInst>(Cast1->getOperand(0));
95778b8ce40SRoman Lebedev EXPECT_TRUE(GEP1);
95878b8ce40SRoman Lebedev EXPECT_TRUE(cast<Constant>(GEP1->getPointerOperand())->isNullValue());
95978b8ce40SRoman Lebedev EXPECT_EQ(GEP1->getOperand(1), &*F.arg_begin());
96078b8ce40SRoman Lebedev EXPECT_EQ(cast<PointerType>(GEP1->getPointerOperand()->getType())
9611d8f2e52SFlorian Hahn ->getAddressSpace(),
9621d8f2e52SFlorian Hahn cast<PointerType>(I.getType())->getAddressSpace());
9631d8f2e52SFlorian Hahn EXPECT_FALSE(verifyFunction(F, &errs()));
9641d8f2e52SFlorian Hahn });
9651d8f2e52SFlorian Hahn }
9661d8f2e52SFlorian Hahn
967bcbd26bfSFlorian Hahn } // end namespace llvm
968