Run DCE after a LoopFlatten test to reduce spurious output [nfc]
[llvm-project.git] / llvm / unittests / Transforms / Utils / LoopUtilsTest.cpp
blob8272c304ba288485256c473336efbb039b32da8f
1 //===- LoopUtilsTest.cpp - Unit tests for LoopUtils -----------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
9 #include "llvm/Transforms/Utils/LoopUtils.h"
10 #include "llvm/Analysis/AssumptionCache.h"
11 #include "llvm/Analysis/LoopInfo.h"
12 #include "llvm/Analysis/ScalarEvolution.h"
13 #include "llvm/Analysis/TargetLibraryInfo.h"
14 #include "llvm/AsmParser/Parser.h"
15 #include "llvm/IR/Dominators.h"
16 #include "llvm/Support/SourceMgr.h"
17 #include "gtest/gtest.h"
19 using namespace llvm;
21 static std::unique_ptr<Module> parseIR(LLVMContext &C, const char *IR) {
22 SMDiagnostic Err;
23 std::unique_ptr<Module> Mod = parseAssemblyString(IR, Err, C);
24 if (!Mod)
25 Err.print("LoopUtilsTests", errs());
26 return Mod;
29 static void run(Module &M, StringRef FuncName,
30 function_ref<void(Function &F, DominatorTree &DT,
31 ScalarEvolution &SE, LoopInfo &LI)>
32 Test) {
33 Function *F = M.getFunction(FuncName);
34 DominatorTree DT(*F);
35 TargetLibraryInfoImpl TLII;
36 TargetLibraryInfo TLI(TLII);
37 AssumptionCache AC(*F);
38 LoopInfo LI(DT);
39 ScalarEvolution SE(*F, TLI, AC, DT, LI);
40 Test(*F, DT, SE, LI);
43 TEST(LoopUtils, DeleteDeadLoopNest) {
44 LLVMContext C;
45 std::unique_ptr<Module> M =
46 parseIR(C, "define void @foo() {\n"
47 "entry:\n"
48 " br label %for.i\n"
49 "for.i:\n"
50 " %i = phi i64 [ 0, %entry ], [ %inc.i, %for.i.latch ]\n"
51 " br label %for.j\n"
52 "for.j:\n"
53 " %j = phi i64 [ 0, %for.i ], [ %inc.j, %for.j ]\n"
54 " %inc.j = add nsw i64 %j, 1\n"
55 " %cmp.j = icmp slt i64 %inc.j, 100\n"
56 " br i1 %cmp.j, label %for.j, label %for.k.preheader\n"
57 "for.k.preheader:\n"
58 " br label %for.k\n"
59 "for.k:\n"
60 " %k = phi i64 [ %inc.k, %for.k ], [ 0, %for.k.preheader ]\n"
61 " %inc.k = add nsw i64 %k, 1\n"
62 " %cmp.k = icmp slt i64 %inc.k, 100\n"
63 " br i1 %cmp.k, label %for.k, label %for.i.latch\n"
64 "for.i.latch:\n"
65 " %inc.i = add nsw i64 %i, 1\n"
66 " %cmp.i = icmp slt i64 %inc.i, 100\n"
67 " br i1 %cmp.i, label %for.i, label %for.end\n"
68 "for.end:\n"
69 " ret void\n"
70 "}\n");
72 run(*M, "foo",
73 [&](Function &F, DominatorTree &DT, ScalarEvolution &SE, LoopInfo &LI) {
74 assert(LI.begin() != LI.end() && "Expecting loops in function F");
75 Loop *L = *LI.begin();
76 assert(L && L->getName() == "for.i" && "Expecting loop for.i");
78 deleteDeadLoop(L, &DT, &SE, &LI);
80 assert(DT.verify(DominatorTree::VerificationLevel::Fast) &&
81 "Expecting valid dominator tree");
82 LI.verify(DT);
83 assert(LI.begin() == LI.end() &&
84 "Expecting no loops left in function F");
85 SE.verify();
87 Function::iterator FI = F.begin();
88 BasicBlock *Entry = &*(FI++);
89 assert(Entry->getName() == "entry" && "Expecting BasicBlock entry");
90 const BranchInst *BI = dyn_cast<BranchInst>(Entry->getTerminator());
91 assert(BI && "Expecting valid branch instruction");
92 EXPECT_EQ(BI->getNumSuccessors(), (unsigned)1);
93 EXPECT_EQ(BI->getSuccessor(0)->getName(), "for.end");
94 });
97 TEST(LoopUtils, IsKnownPositiveInLoopTest) {
98 LLVMContext C;
99 std::unique_ptr<Module> M =
100 parseIR(C, "define void @foo(i32 %n, i1 %c) {\n"
101 "entry:\n"
102 " %is.positive = icmp sgt i32 %n, 0\n"
103 " br i1 %is.positive, label %loop, label %exit\n"
104 "loop:\n"
105 " br i1 %c, label %loop, label %exit\n"
106 "exit:\n"
107 " ret void\n"
108 "}\n");
110 run(*M, "foo",
111 [&](Function &F, DominatorTree &DT, ScalarEvolution &SE, LoopInfo &LI) {
112 assert(LI.begin() != LI.end() && "Expecting loops in function F");
113 Loop *L = *LI.begin();
114 assert(L && L->getName() == "loop" && "Expecting loop 'loop'");
115 auto *Arg = F.getArg(0);
116 auto *ArgSCEV = SE.getSCEV(Arg);
117 EXPECT_EQ(isKnownPositiveInLoop(ArgSCEV, L, SE), true);
121 TEST(LoopUtils, IsKnownNonPositiveInLoopTest) {
122 LLVMContext C;
123 std::unique_ptr<Module> M =
124 parseIR(C, "define void @foo(i32 %n, i1 %c) {\n"
125 "entry:\n"
126 " %is.non.positive = icmp sle i32 %n, 0\n"
127 " br i1 %is.non.positive, label %loop, label %exit\n"
128 "loop:\n"
129 " br i1 %c, label %loop, label %exit\n"
130 "exit:\n"
131 " ret void\n"
132 "}\n");
134 run(*M, "foo",
135 [&](Function &F, DominatorTree &DT, ScalarEvolution &SE, LoopInfo &LI) {
136 assert(LI.begin() != LI.end() && "Expecting loops in function F");
137 Loop *L = *LI.begin();
138 assert(L && L->getName() == "loop" && "Expecting loop 'loop'");
139 auto *Arg = F.getArg(0);
140 auto *ArgSCEV = SE.getSCEV(Arg);
141 EXPECT_EQ(isKnownNonPositiveInLoop(ArgSCEV, L, SE), true);