xref: /aosp_15_r20/external/llvm/unittests/Analysis/ValueTrackingTest.cpp (revision 9880d6810fe72a1726cb53787c6711e909410d58)
1*9880d681SAndroid Build Coastguard Worker //===- ValueTrackingTest.cpp - ValueTracking tests ------------------------===//
2*9880d681SAndroid Build Coastguard Worker //
3*9880d681SAndroid Build Coastguard Worker //                     The LLVM Compiler Infrastructure
4*9880d681SAndroid Build Coastguard Worker //
5*9880d681SAndroid Build Coastguard Worker // This file is distributed under the University of Illinois Open Source
6*9880d681SAndroid Build Coastguard Worker // License. See LICENSE.TXT for details.
7*9880d681SAndroid Build Coastguard Worker //
8*9880d681SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
9*9880d681SAndroid Build Coastguard Worker 
10*9880d681SAndroid Build Coastguard Worker #include "llvm/Analysis/ValueTracking.h"
11*9880d681SAndroid Build Coastguard Worker #include "llvm/AsmParser/Parser.h"
12*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/Function.h"
13*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/InstIterator.h"
14*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/LLVMContext.h"
15*9880d681SAndroid Build Coastguard Worker #include "llvm/IR/Module.h"
16*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/ErrorHandling.h"
17*9880d681SAndroid Build Coastguard Worker #include "llvm/Support/SourceMgr.h"
18*9880d681SAndroid Build Coastguard Worker #include "gtest/gtest.h"
19*9880d681SAndroid Build Coastguard Worker 
20*9880d681SAndroid Build Coastguard Worker using namespace llvm;
21*9880d681SAndroid Build Coastguard Worker 
22*9880d681SAndroid Build Coastguard Worker namespace {
23*9880d681SAndroid Build Coastguard Worker 
24*9880d681SAndroid Build Coastguard Worker class MatchSelectPatternTest : public testing::Test {
25*9880d681SAndroid Build Coastguard Worker protected:
parseAssembly(const char * Assembly)26*9880d681SAndroid Build Coastguard Worker   void parseAssembly(const char *Assembly) {
27*9880d681SAndroid Build Coastguard Worker     SMDiagnostic Error;
28*9880d681SAndroid Build Coastguard Worker     M = parseAssemblyString(Assembly, Error, Context);
29*9880d681SAndroid Build Coastguard Worker 
30*9880d681SAndroid Build Coastguard Worker     std::string errMsg;
31*9880d681SAndroid Build Coastguard Worker     raw_string_ostream os(errMsg);
32*9880d681SAndroid Build Coastguard Worker     Error.print("", os);
33*9880d681SAndroid Build Coastguard Worker 
34*9880d681SAndroid Build Coastguard Worker     // A failure here means that the test itself is buggy.
35*9880d681SAndroid Build Coastguard Worker     if (!M)
36*9880d681SAndroid Build Coastguard Worker       report_fatal_error(os.str());
37*9880d681SAndroid Build Coastguard Worker 
38*9880d681SAndroid Build Coastguard Worker     Function *F = M->getFunction("test");
39*9880d681SAndroid Build Coastguard Worker     if (F == nullptr)
40*9880d681SAndroid Build Coastguard Worker       report_fatal_error("Test must have a function named @test");
41*9880d681SAndroid Build Coastguard Worker 
42*9880d681SAndroid Build Coastguard Worker     A = nullptr;
43*9880d681SAndroid Build Coastguard Worker     for (inst_iterator I = inst_begin(F), E = inst_end(F); I != E; ++I) {
44*9880d681SAndroid Build Coastguard Worker       if (I->hasName()) {
45*9880d681SAndroid Build Coastguard Worker         if (I->getName() == "A")
46*9880d681SAndroid Build Coastguard Worker           A = &*I;
47*9880d681SAndroid Build Coastguard Worker       }
48*9880d681SAndroid Build Coastguard Worker     }
49*9880d681SAndroid Build Coastguard Worker     if (A == nullptr)
50*9880d681SAndroid Build Coastguard Worker       report_fatal_error("@test must have an instruction %A");
51*9880d681SAndroid Build Coastguard Worker   }
52*9880d681SAndroid Build Coastguard Worker 
expectPattern(const SelectPatternResult & P)53*9880d681SAndroid Build Coastguard Worker   void expectPattern(const SelectPatternResult &P) {
54*9880d681SAndroid Build Coastguard Worker     Value *LHS, *RHS;
55*9880d681SAndroid Build Coastguard Worker     Instruction::CastOps CastOp;
56*9880d681SAndroid Build Coastguard Worker     SelectPatternResult R = matchSelectPattern(A, LHS, RHS, &CastOp);
57*9880d681SAndroid Build Coastguard Worker     EXPECT_EQ(P.Flavor, R.Flavor);
58*9880d681SAndroid Build Coastguard Worker     EXPECT_EQ(P.NaNBehavior, R.NaNBehavior);
59*9880d681SAndroid Build Coastguard Worker     EXPECT_EQ(P.Ordered, R.Ordered);
60*9880d681SAndroid Build Coastguard Worker   }
61*9880d681SAndroid Build Coastguard Worker 
62*9880d681SAndroid Build Coastguard Worker   LLVMContext Context;
63*9880d681SAndroid Build Coastguard Worker   std::unique_ptr<Module> M;
64*9880d681SAndroid Build Coastguard Worker   Instruction *A, *B;
65*9880d681SAndroid Build Coastguard Worker };
66*9880d681SAndroid Build Coastguard Worker 
67*9880d681SAndroid Build Coastguard Worker }
68*9880d681SAndroid Build Coastguard Worker 
TEST_F(MatchSelectPatternTest,SimpleFMin)69*9880d681SAndroid Build Coastguard Worker TEST_F(MatchSelectPatternTest, SimpleFMin) {
70*9880d681SAndroid Build Coastguard Worker   parseAssembly(
71*9880d681SAndroid Build Coastguard Worker       "define float @test(float %a) {\n"
72*9880d681SAndroid Build Coastguard Worker       "  %1 = fcmp ult float %a, 5.0\n"
73*9880d681SAndroid Build Coastguard Worker       "  %A = select i1 %1, float %a, float 5.0\n"
74*9880d681SAndroid Build Coastguard Worker       "  ret float %A\n"
75*9880d681SAndroid Build Coastguard Worker       "}\n");
76*9880d681SAndroid Build Coastguard Worker   expectPattern({SPF_FMINNUM, SPNB_RETURNS_NAN, false});
77*9880d681SAndroid Build Coastguard Worker }
78*9880d681SAndroid Build Coastguard Worker 
TEST_F(MatchSelectPatternTest,SimpleFMax)79*9880d681SAndroid Build Coastguard Worker TEST_F(MatchSelectPatternTest, SimpleFMax) {
80*9880d681SAndroid Build Coastguard Worker   parseAssembly(
81*9880d681SAndroid Build Coastguard Worker       "define float @test(float %a) {\n"
82*9880d681SAndroid Build Coastguard Worker       "  %1 = fcmp ogt float %a, 5.0\n"
83*9880d681SAndroid Build Coastguard Worker       "  %A = select i1 %1, float %a, float 5.0\n"
84*9880d681SAndroid Build Coastguard Worker       "  ret float %A\n"
85*9880d681SAndroid Build Coastguard Worker       "}\n");
86*9880d681SAndroid Build Coastguard Worker   expectPattern({SPF_FMAXNUM, SPNB_RETURNS_OTHER, true});
87*9880d681SAndroid Build Coastguard Worker }
88*9880d681SAndroid Build Coastguard Worker 
TEST_F(MatchSelectPatternTest,SwappedFMax)89*9880d681SAndroid Build Coastguard Worker TEST_F(MatchSelectPatternTest, SwappedFMax) {
90*9880d681SAndroid Build Coastguard Worker   parseAssembly(
91*9880d681SAndroid Build Coastguard Worker       "define float @test(float %a) {\n"
92*9880d681SAndroid Build Coastguard Worker       "  %1 = fcmp olt float 5.0, %a\n"
93*9880d681SAndroid Build Coastguard Worker       "  %A = select i1 %1, float %a, float 5.0\n"
94*9880d681SAndroid Build Coastguard Worker       "  ret float %A\n"
95*9880d681SAndroid Build Coastguard Worker       "}\n");
96*9880d681SAndroid Build Coastguard Worker   expectPattern({SPF_FMAXNUM, SPNB_RETURNS_OTHER, false});
97*9880d681SAndroid Build Coastguard Worker }
98*9880d681SAndroid Build Coastguard Worker 
TEST_F(MatchSelectPatternTest,SwappedFMax2)99*9880d681SAndroid Build Coastguard Worker TEST_F(MatchSelectPatternTest, SwappedFMax2) {
100*9880d681SAndroid Build Coastguard Worker   parseAssembly(
101*9880d681SAndroid Build Coastguard Worker       "define float @test(float %a) {\n"
102*9880d681SAndroid Build Coastguard Worker       "  %1 = fcmp olt float %a, 5.0\n"
103*9880d681SAndroid Build Coastguard Worker       "  %A = select i1 %1, float 5.0, float %a\n"
104*9880d681SAndroid Build Coastguard Worker       "  ret float %A\n"
105*9880d681SAndroid Build Coastguard Worker       "}\n");
106*9880d681SAndroid Build Coastguard Worker   expectPattern({SPF_FMAXNUM, SPNB_RETURNS_NAN, false});
107*9880d681SAndroid Build Coastguard Worker }
108*9880d681SAndroid Build Coastguard Worker 
TEST_F(MatchSelectPatternTest,SwappedFMax3)109*9880d681SAndroid Build Coastguard Worker TEST_F(MatchSelectPatternTest, SwappedFMax3) {
110*9880d681SAndroid Build Coastguard Worker   parseAssembly(
111*9880d681SAndroid Build Coastguard Worker       "define float @test(float %a) {\n"
112*9880d681SAndroid Build Coastguard Worker       "  %1 = fcmp ult float %a, 5.0\n"
113*9880d681SAndroid Build Coastguard Worker       "  %A = select i1 %1, float 5.0, float %a\n"
114*9880d681SAndroid Build Coastguard Worker       "  ret float %A\n"
115*9880d681SAndroid Build Coastguard Worker       "}\n");
116*9880d681SAndroid Build Coastguard Worker   expectPattern({SPF_FMAXNUM, SPNB_RETURNS_OTHER, true});
117*9880d681SAndroid Build Coastguard Worker }
118*9880d681SAndroid Build Coastguard Worker 
TEST_F(MatchSelectPatternTest,FastFMin)119*9880d681SAndroid Build Coastguard Worker TEST_F(MatchSelectPatternTest, FastFMin) {
120*9880d681SAndroid Build Coastguard Worker   parseAssembly(
121*9880d681SAndroid Build Coastguard Worker       "define float @test(float %a) {\n"
122*9880d681SAndroid Build Coastguard Worker       "  %1 = fcmp nnan olt float %a, 5.0\n"
123*9880d681SAndroid Build Coastguard Worker       "  %A = select i1 %1, float %a, float 5.0\n"
124*9880d681SAndroid Build Coastguard Worker       "  ret float %A\n"
125*9880d681SAndroid Build Coastguard Worker       "}\n");
126*9880d681SAndroid Build Coastguard Worker   expectPattern({SPF_FMINNUM, SPNB_RETURNS_ANY, false});
127*9880d681SAndroid Build Coastguard Worker }
128*9880d681SAndroid Build Coastguard Worker 
TEST_F(MatchSelectPatternTest,FMinConstantZero)129*9880d681SAndroid Build Coastguard Worker TEST_F(MatchSelectPatternTest, FMinConstantZero) {
130*9880d681SAndroid Build Coastguard Worker   parseAssembly(
131*9880d681SAndroid Build Coastguard Worker       "define float @test(float %a) {\n"
132*9880d681SAndroid Build Coastguard Worker       "  %1 = fcmp ole float %a, 0.0\n"
133*9880d681SAndroid Build Coastguard Worker       "  %A = select i1 %1, float %a, float 0.0\n"
134*9880d681SAndroid Build Coastguard Worker       "  ret float %A\n"
135*9880d681SAndroid Build Coastguard Worker       "}\n");
136*9880d681SAndroid Build Coastguard Worker   // This shouldn't be matched, as %a could be -0.0.
137*9880d681SAndroid Build Coastguard Worker   expectPattern({SPF_UNKNOWN, SPNB_NA, false});
138*9880d681SAndroid Build Coastguard Worker }
139*9880d681SAndroid Build Coastguard Worker 
TEST_F(MatchSelectPatternTest,FMinConstantZeroNsz)140*9880d681SAndroid Build Coastguard Worker TEST_F(MatchSelectPatternTest, FMinConstantZeroNsz) {
141*9880d681SAndroid Build Coastguard Worker   parseAssembly(
142*9880d681SAndroid Build Coastguard Worker       "define float @test(float %a) {\n"
143*9880d681SAndroid Build Coastguard Worker       "  %1 = fcmp nsz ole float %a, 0.0\n"
144*9880d681SAndroid Build Coastguard Worker       "  %A = select i1 %1, float %a, float 0.0\n"
145*9880d681SAndroid Build Coastguard Worker       "  ret float %A\n"
146*9880d681SAndroid Build Coastguard Worker       "}\n");
147*9880d681SAndroid Build Coastguard Worker   // But this should be, because we've ignored signed zeroes.
148*9880d681SAndroid Build Coastguard Worker   expectPattern({SPF_FMINNUM, SPNB_RETURNS_OTHER, true});
149*9880d681SAndroid Build Coastguard Worker }
150*9880d681SAndroid Build Coastguard Worker 
TEST_F(MatchSelectPatternTest,DoubleCastU)151*9880d681SAndroid Build Coastguard Worker TEST_F(MatchSelectPatternTest, DoubleCastU) {
152*9880d681SAndroid Build Coastguard Worker   parseAssembly(
153*9880d681SAndroid Build Coastguard Worker       "define i32 @test(i8 %a, i8 %b) {\n"
154*9880d681SAndroid Build Coastguard Worker       "  %1 = icmp ult i8 %a, %b\n"
155*9880d681SAndroid Build Coastguard Worker       "  %2 = zext i8 %a to i32\n"
156*9880d681SAndroid Build Coastguard Worker       "  %3 = zext i8 %b to i32\n"
157*9880d681SAndroid Build Coastguard Worker       "  %A = select i1 %1, i32 %2, i32 %3\n"
158*9880d681SAndroid Build Coastguard Worker       "  ret i32 %A\n"
159*9880d681SAndroid Build Coastguard Worker       "}\n");
160*9880d681SAndroid Build Coastguard Worker   // We should be able to look through the situation where we cast both operands
161*9880d681SAndroid Build Coastguard Worker   // to the select.
162*9880d681SAndroid Build Coastguard Worker   expectPattern({SPF_UMIN, SPNB_NA, false});
163*9880d681SAndroid Build Coastguard Worker }
164*9880d681SAndroid Build Coastguard Worker 
TEST_F(MatchSelectPatternTest,DoubleCastS)165*9880d681SAndroid Build Coastguard Worker TEST_F(MatchSelectPatternTest, DoubleCastS) {
166*9880d681SAndroid Build Coastguard Worker   parseAssembly(
167*9880d681SAndroid Build Coastguard Worker       "define i32 @test(i8 %a, i8 %b) {\n"
168*9880d681SAndroid Build Coastguard Worker       "  %1 = icmp slt i8 %a, %b\n"
169*9880d681SAndroid Build Coastguard Worker       "  %2 = sext i8 %a to i32\n"
170*9880d681SAndroid Build Coastguard Worker       "  %3 = sext i8 %b to i32\n"
171*9880d681SAndroid Build Coastguard Worker       "  %A = select i1 %1, i32 %2, i32 %3\n"
172*9880d681SAndroid Build Coastguard Worker       "  ret i32 %A\n"
173*9880d681SAndroid Build Coastguard Worker       "}\n");
174*9880d681SAndroid Build Coastguard Worker   // We should be able to look through the situation where we cast both operands
175*9880d681SAndroid Build Coastguard Worker   // to the select.
176*9880d681SAndroid Build Coastguard Worker   expectPattern({SPF_SMIN, SPNB_NA, false});
177*9880d681SAndroid Build Coastguard Worker }
178*9880d681SAndroid Build Coastguard Worker 
TEST_F(MatchSelectPatternTest,DoubleCastBad)179*9880d681SAndroid Build Coastguard Worker TEST_F(MatchSelectPatternTest, DoubleCastBad) {
180*9880d681SAndroid Build Coastguard Worker   parseAssembly(
181*9880d681SAndroid Build Coastguard Worker       "define i32 @test(i8 %a, i8 %b) {\n"
182*9880d681SAndroid Build Coastguard Worker       "  %1 = icmp ult i8 %a, %b\n"
183*9880d681SAndroid Build Coastguard Worker       "  %2 = zext i8 %a to i32\n"
184*9880d681SAndroid Build Coastguard Worker       "  %3 = sext i8 %b to i32\n"
185*9880d681SAndroid Build Coastguard Worker       "  %A = select i1 %1, i32 %2, i32 %3\n"
186*9880d681SAndroid Build Coastguard Worker       "  ret i32 %A\n"
187*9880d681SAndroid Build Coastguard Worker       "}\n");
188*9880d681SAndroid Build Coastguard Worker   // The cast types here aren't the same, so we cannot match an UMIN.
189*9880d681SAndroid Build Coastguard Worker   expectPattern({SPF_UNKNOWN, SPNB_NA, false});
190*9880d681SAndroid Build Coastguard Worker }
191