blob: f3a27b8d2509b3f9093d36356be6ce6f88574ea2 [file] [log] [blame]
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +00001//===---- llvm/unittest/IR/PatternMatch.cpp - PatternMatch unit tests ----===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "llvm/ADT/STLExtras.h"
11#include "llvm/Analysis/ValueTracking.h"
12#include "llvm/IR/BasicBlock.h"
13#include "llvm/IR/Constants.h"
14#include "llvm/IR/DataLayout.h"
15#include "llvm/IR/DerivedTypes.h"
Stephen Hines36b56882014-04-23 16:57:46 -070016#include "llvm/IR/Function.h"
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000017#include "llvm/IR/IRBuilder.h"
Stephen Hines36b56882014-04-23 16:57:46 -070018#include "llvm/IR/Instructions.h"
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000019#include "llvm/IR/LLVMContext.h"
20#include "llvm/IR/MDBuilder.h"
Stephen Hines36b56882014-04-23 16:57:46 -070021#include "llvm/IR/Module.h"
22#include "llvm/IR/NoFolder.h"
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000023#include "llvm/IR/Operator.h"
Stephen Hines36b56882014-04-23 16:57:46 -070024#include "llvm/IR/PatternMatch.h"
25#include "llvm/IR/Type.h"
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000026#include "gtest/gtest.h"
27
Stephen Hines36b56882014-04-23 16:57:46 -070028using namespace llvm;
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000029using namespace llvm::PatternMatch;
30
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000031namespace {
32
Stephen Hines36b56882014-04-23 16:57:46 -070033struct PatternMatchTest : ::testing::Test {
34 LLVMContext Ctx;
35 std::unique_ptr<Module> M;
36 Function *F;
37 BasicBlock *BB;
38 IRBuilder<true, NoFolder> IRB;
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000039
Stephen Hines36b56882014-04-23 16:57:46 -070040 PatternMatchTest()
41 : M(new Module("PatternMatchTestModule", Ctx)),
42 F(Function::Create(
43 FunctionType::get(Type::getVoidTy(Ctx), /* IsVarArg */ false),
44 Function::ExternalLinkage, "f", M.get())),
45 BB(BasicBlock::Create(Ctx, "entry", F)), IRB(BB) {}
46};
47
48TEST_F(PatternMatchTest, OneUse) {
49 // Build up a little tree of values:
50 //
51 // One = (1 + 2) + 42
52 // Two = One + 42
53 // Leaf = (Two + 8) + (Two + 13)
54 Value *One = IRB.CreateAdd(IRB.CreateAdd(IRB.getInt32(1), IRB.getInt32(2)),
55 IRB.getInt32(42));
56 Value *Two = IRB.CreateAdd(One, IRB.getInt32(42));
57 Value *Leaf = IRB.CreateAdd(IRB.CreateAdd(Two, IRB.getInt32(8)),
58 IRB.CreateAdd(Two, IRB.getInt32(13)));
59 Value *V;
60
61 EXPECT_TRUE(m_OneUse(m_Value(V)).match(One));
62 EXPECT_EQ(One, V);
63
64 EXPECT_FALSE(m_OneUse(m_Value()).match(Two));
65 EXPECT_FALSE(m_OneUse(m_Value()).match(Leaf));
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000066}
67
Stephen Hines36b56882014-04-23 16:57:46 -070068TEST_F(PatternMatchTest, FloatingPointOrderedMin) {
69 Type *FltTy = IRB.getFloatTy();
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000070 Value *L = ConstantFP::get(FltTy, 1.0);
71 Value *R = ConstantFP::get(FltTy, 2.0);
Stephen Hines36b56882014-04-23 16:57:46 -070072 Value *MatchL, *MatchR;
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000073
74 // Test OLT.
Stephen Hines36b56882014-04-23 16:57:46 -070075 EXPECT_TRUE(m_OrdFMin(m_Value(MatchL), m_Value(MatchR))
76 .match(IRB.CreateSelect(IRB.CreateFCmpOLT(L, R), L, R)));
77 EXPECT_EQ(L, MatchL);
78 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000079
80 // Test OLE.
Stephen Hines36b56882014-04-23 16:57:46 -070081 EXPECT_TRUE(m_OrdFMin(m_Value(MatchL), m_Value(MatchR))
82 .match(IRB.CreateSelect(IRB.CreateFCmpOLE(L, R), L, R)));
83 EXPECT_EQ(L, MatchL);
84 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000085
86 // Test no match on OGE.
Stephen Hines36b56882014-04-23 16:57:46 -070087 EXPECT_FALSE(m_OrdFMin(m_Value(MatchL), m_Value(MatchR))
88 .match(IRB.CreateSelect(IRB.CreateFCmpOGE(L, R), L, R)));
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000089
90 // Test no match on OGT.
Stephen Hines36b56882014-04-23 16:57:46 -070091 EXPECT_FALSE(m_OrdFMin(m_Value(MatchL), m_Value(MatchR))
92 .match(IRB.CreateSelect(IRB.CreateFCmpOGT(L, R), L, R)));
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000093
94 // Test match on OGE with inverted select.
Stephen Hines36b56882014-04-23 16:57:46 -070095 EXPECT_TRUE(m_OrdFMin(m_Value(MatchL), m_Value(MatchR))
96 .match(IRB.CreateSelect(IRB.CreateFCmpOGE(L, R), R, L)));
97 EXPECT_EQ(L, MatchL);
98 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +000099
100 // Test match on OGT with inverted select.
Stephen Hines36b56882014-04-23 16:57:46 -0700101 EXPECT_TRUE(m_OrdFMin(m_Value(MatchL), m_Value(MatchR))
102 .match(IRB.CreateSelect(IRB.CreateFCmpOGT(L, R), R, L)));
103 EXPECT_EQ(L, MatchL);
104 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000105}
106
Stephen Hines36b56882014-04-23 16:57:46 -0700107TEST_F(PatternMatchTest, FloatingPointOrderedMax) {
108 Type *FltTy = IRB.getFloatTy();
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000109 Value *L = ConstantFP::get(FltTy, 1.0);
110 Value *R = ConstantFP::get(FltTy, 2.0);
Stephen Hines36b56882014-04-23 16:57:46 -0700111 Value *MatchL, *MatchR;
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000112
113 // Test OGT.
Stephen Hines36b56882014-04-23 16:57:46 -0700114 EXPECT_TRUE(m_OrdFMax(m_Value(MatchL), m_Value(MatchR))
115 .match(IRB.CreateSelect(IRB.CreateFCmpOGT(L, R), L, R)));
116 EXPECT_EQ(L, MatchL);
117 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000118
119 // Test OGE.
Stephen Hines36b56882014-04-23 16:57:46 -0700120 EXPECT_TRUE(m_OrdFMax(m_Value(MatchL), m_Value(MatchR))
121 .match(IRB.CreateSelect(IRB.CreateFCmpOGE(L, R), L, R)));
122 EXPECT_EQ(L, MatchL);
123 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000124
125 // Test no match on OLE.
Stephen Hines36b56882014-04-23 16:57:46 -0700126 EXPECT_FALSE(m_OrdFMax(m_Value(MatchL), m_Value(MatchR))
127 .match(IRB.CreateSelect(IRB.CreateFCmpOLE(L, R), L, R)));
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000128
129 // Test no match on OLT.
Stephen Hines36b56882014-04-23 16:57:46 -0700130 EXPECT_FALSE(m_OrdFMax(m_Value(MatchL), m_Value(MatchR))
131 .match(IRB.CreateSelect(IRB.CreateFCmpOLT(L, R), L, R)));
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000132
133 // Test match on OLE with inverted select.
Stephen Hines36b56882014-04-23 16:57:46 -0700134 EXPECT_TRUE(m_OrdFMax(m_Value(MatchL), m_Value(MatchR))
135 .match(IRB.CreateSelect(IRB.CreateFCmpOLE(L, R), R, L)));
136 EXPECT_EQ(L, MatchL);
137 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000138
139 // Test match on OLT with inverted select.
Stephen Hines36b56882014-04-23 16:57:46 -0700140 EXPECT_TRUE(m_OrdFMax(m_Value(MatchL), m_Value(MatchR))
141 .match(IRB.CreateSelect(IRB.CreateFCmpOLT(L, R), R, L)));
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000142 EXPECT_EQ(L, MatchL);
143 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000144}
145
Stephen Hines36b56882014-04-23 16:57:46 -0700146TEST_F(PatternMatchTest, FloatingPointUnorderedMin) {
147 Type *FltTy = IRB.getFloatTy();
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000148 Value *L = ConstantFP::get(FltTy, 1.0);
149 Value *R = ConstantFP::get(FltTy, 2.0);
Stephen Hines36b56882014-04-23 16:57:46 -0700150 Value *MatchL, *MatchR;
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000151
152 // Test ULT.
Stephen Hines36b56882014-04-23 16:57:46 -0700153 EXPECT_TRUE(m_UnordFMin(m_Value(MatchL), m_Value(MatchR))
154 .match(IRB.CreateSelect(IRB.CreateFCmpULT(L, R), L, R)));
155 EXPECT_EQ(L, MatchL);
156 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000157
158 // Test ULE.
Stephen Hines36b56882014-04-23 16:57:46 -0700159 EXPECT_TRUE(m_UnordFMin(m_Value(MatchL), m_Value(MatchR))
160 .match(IRB.CreateSelect(IRB.CreateFCmpULE(L, R), L, R)));
161 EXPECT_EQ(L, MatchL);
162 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000163
164 // Test no match on UGE.
Stephen Hines36b56882014-04-23 16:57:46 -0700165 EXPECT_FALSE(m_UnordFMin(m_Value(MatchL), m_Value(MatchR))
166 .match(IRB.CreateSelect(IRB.CreateFCmpUGE(L, R), L, R)));
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000167
168 // Test no match on UGT.
Stephen Hines36b56882014-04-23 16:57:46 -0700169 EXPECT_FALSE(m_UnordFMin(m_Value(MatchL), m_Value(MatchR))
170 .match(IRB.CreateSelect(IRB.CreateFCmpUGT(L, R), L, R)));
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000171
172 // Test match on UGE with inverted select.
Stephen Hines36b56882014-04-23 16:57:46 -0700173 EXPECT_TRUE(m_UnordFMin(m_Value(MatchL), m_Value(MatchR))
174 .match(IRB.CreateSelect(IRB.CreateFCmpUGE(L, R), R, L)));
175 EXPECT_EQ(L, MatchL);
176 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000177
178 // Test match on UGT with inverted select.
Stephen Hines36b56882014-04-23 16:57:46 -0700179 EXPECT_TRUE(m_UnordFMin(m_Value(MatchL), m_Value(MatchR))
180 .match(IRB.CreateSelect(IRB.CreateFCmpUGT(L, R), R, L)));
181 EXPECT_EQ(L, MatchL);
182 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000183}
184
Stephen Hines36b56882014-04-23 16:57:46 -0700185TEST_F(PatternMatchTest, FloatingPointUnorderedMax) {
186 Type *FltTy = IRB.getFloatTy();
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000187 Value *L = ConstantFP::get(FltTy, 1.0);
188 Value *R = ConstantFP::get(FltTy, 2.0);
Stephen Hines36b56882014-04-23 16:57:46 -0700189 Value *MatchL, *MatchR;
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000190
191 // Test UGT.
Stephen Hines36b56882014-04-23 16:57:46 -0700192 EXPECT_TRUE(m_UnordFMax(m_Value(MatchL), m_Value(MatchR))
193 .match(IRB.CreateSelect(IRB.CreateFCmpUGT(L, R), L, R)));
194 EXPECT_EQ(L, MatchL);
195 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000196
197 // Test UGE.
Stephen Hines36b56882014-04-23 16:57:46 -0700198 EXPECT_TRUE(m_UnordFMax(m_Value(MatchL), m_Value(MatchR))
199 .match(IRB.CreateSelect(IRB.CreateFCmpUGE(L, R), L, R)));
200 EXPECT_EQ(L, MatchL);
201 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000202
203 // Test no match on ULE.
Stephen Hines36b56882014-04-23 16:57:46 -0700204 EXPECT_FALSE(m_UnordFMax(m_Value(MatchL), m_Value(MatchR))
205 .match(IRB.CreateSelect(IRB.CreateFCmpULE(L, R), L, R)));
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000206
207 // Test no match on ULT.
Stephen Hines36b56882014-04-23 16:57:46 -0700208 EXPECT_FALSE(m_UnordFMax(m_Value(MatchL), m_Value(MatchR))
209 .match(IRB.CreateSelect(IRB.CreateFCmpULT(L, R), L, R)));
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000210
211 // Test match on ULE with inverted select.
Stephen Hines36b56882014-04-23 16:57:46 -0700212 EXPECT_TRUE(m_UnordFMax(m_Value(MatchL), m_Value(MatchR))
213 .match(IRB.CreateSelect(IRB.CreateFCmpULE(L, R), R, L)));
214 EXPECT_EQ(L, MatchL);
215 EXPECT_EQ(R, MatchR);
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000216
217 // Test match on ULT with inverted select.
Stephen Hines36b56882014-04-23 16:57:46 -0700218 EXPECT_TRUE(m_UnordFMax(m_Value(MatchL), m_Value(MatchR))
219 .match(IRB.CreateSelect(IRB.CreateFCmpULT(L, R), R, L)));
220 EXPECT_EQ(L, MatchL);
221 EXPECT_EQ(R, MatchR);
222}
223
224TEST_F(PatternMatchTest, OverflowingBinOps) {
225 Value *L = IRB.getInt32(1);
226 Value *R = IRB.getInt32(2);
227 Value *MatchL, *MatchR;
228
229 EXPECT_TRUE(
230 m_NSWAdd(m_Value(MatchL), m_Value(MatchR)).match(IRB.CreateNSWAdd(L, R)));
231 EXPECT_EQ(L, MatchL);
232 EXPECT_EQ(R, MatchR);
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -0700233 MatchL = MatchR = nullptr;
Stephen Hines36b56882014-04-23 16:57:46 -0700234 EXPECT_TRUE(
235 m_NSWSub(m_Value(MatchL), m_Value(MatchR)).match(IRB.CreateNSWSub(L, R)));
236 EXPECT_EQ(L, MatchL);
237 EXPECT_EQ(R, MatchR);
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -0700238 MatchL = MatchR = nullptr;
Stephen Hines36b56882014-04-23 16:57:46 -0700239 EXPECT_TRUE(
240 m_NSWMul(m_Value(MatchL), m_Value(MatchR)).match(IRB.CreateNSWMul(L, R)));
241 EXPECT_EQ(L, MatchL);
242 EXPECT_EQ(R, MatchR);
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -0700243 MatchL = MatchR = nullptr;
Stephen Hines36b56882014-04-23 16:57:46 -0700244 EXPECT_TRUE(m_NSWShl(m_Value(MatchL), m_Value(MatchR)).match(
245 IRB.CreateShl(L, R, "", /* NUW */ false, /* NSW */ true)));
246 EXPECT_EQ(L, MatchL);
247 EXPECT_EQ(R, MatchR);
248
249 EXPECT_TRUE(
250 m_NUWAdd(m_Value(MatchL), m_Value(MatchR)).match(IRB.CreateNUWAdd(L, R)));
251 EXPECT_EQ(L, MatchL);
252 EXPECT_EQ(R, MatchR);
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -0700253 MatchL = MatchR = nullptr;
Stephen Hines36b56882014-04-23 16:57:46 -0700254 EXPECT_TRUE(
255 m_NUWSub(m_Value(MatchL), m_Value(MatchR)).match(IRB.CreateNUWSub(L, R)));
256 EXPECT_EQ(L, MatchL);
257 EXPECT_EQ(R, MatchR);
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -0700258 MatchL = MatchR = nullptr;
Stephen Hines36b56882014-04-23 16:57:46 -0700259 EXPECT_TRUE(
260 m_NUWMul(m_Value(MatchL), m_Value(MatchR)).match(IRB.CreateNUWMul(L, R)));
261 EXPECT_EQ(L, MatchL);
262 EXPECT_EQ(R, MatchR);
Stephen Hinesc6a4f5e2014-07-21 00:45:20 -0700263 MatchL = MatchR = nullptr;
Stephen Hines36b56882014-04-23 16:57:46 -0700264 EXPECT_TRUE(m_NUWShl(m_Value(MatchL), m_Value(MatchR)).match(
265 IRB.CreateShl(L, R, "", /* NUW */ true, /* NSW */ false)));
266 EXPECT_EQ(L, MatchL);
267 EXPECT_EQ(R, MatchR);
268
269 EXPECT_FALSE(m_NSWAdd(m_Value(), m_Value()).match(IRB.CreateAdd(L, R)));
270 EXPECT_FALSE(m_NSWAdd(m_Value(), m_Value()).match(IRB.CreateNUWAdd(L, R)));
271 EXPECT_FALSE(m_NSWAdd(m_Value(), m_Value()).match(IRB.CreateNSWSub(L, R)));
272 EXPECT_FALSE(m_NSWSub(m_Value(), m_Value()).match(IRB.CreateSub(L, R)));
273 EXPECT_FALSE(m_NSWSub(m_Value(), m_Value()).match(IRB.CreateNUWSub(L, R)));
274 EXPECT_FALSE(m_NSWSub(m_Value(), m_Value()).match(IRB.CreateNSWAdd(L, R)));
275 EXPECT_FALSE(m_NSWMul(m_Value(), m_Value()).match(IRB.CreateMul(L, R)));
276 EXPECT_FALSE(m_NSWMul(m_Value(), m_Value()).match(IRB.CreateNUWMul(L, R)));
277 EXPECT_FALSE(m_NSWMul(m_Value(), m_Value()).match(IRB.CreateNSWAdd(L, R)));
278 EXPECT_FALSE(m_NSWShl(m_Value(), m_Value()).match(IRB.CreateShl(L, R)));
279 EXPECT_FALSE(m_NSWShl(m_Value(), m_Value()).match(
280 IRB.CreateShl(L, R, "", /* NUW */ true, /* NSW */ false)));
281 EXPECT_FALSE(m_NSWShl(m_Value(), m_Value()).match(IRB.CreateNSWAdd(L, R)));
282
283 EXPECT_FALSE(m_NUWAdd(m_Value(), m_Value()).match(IRB.CreateAdd(L, R)));
284 EXPECT_FALSE(m_NUWAdd(m_Value(), m_Value()).match(IRB.CreateNSWAdd(L, R)));
285 EXPECT_FALSE(m_NUWAdd(m_Value(), m_Value()).match(IRB.CreateNUWSub(L, R)));
286 EXPECT_FALSE(m_NUWSub(m_Value(), m_Value()).match(IRB.CreateSub(L, R)));
287 EXPECT_FALSE(m_NUWSub(m_Value(), m_Value()).match(IRB.CreateNSWSub(L, R)));
288 EXPECT_FALSE(m_NUWSub(m_Value(), m_Value()).match(IRB.CreateNUWAdd(L, R)));
289 EXPECT_FALSE(m_NUWMul(m_Value(), m_Value()).match(IRB.CreateMul(L, R)));
290 EXPECT_FALSE(m_NUWMul(m_Value(), m_Value()).match(IRB.CreateNSWMul(L, R)));
291 EXPECT_FALSE(m_NUWMul(m_Value(), m_Value()).match(IRB.CreateNUWAdd(L, R)));
292 EXPECT_FALSE(m_NUWShl(m_Value(), m_Value()).match(IRB.CreateShl(L, R)));
293 EXPECT_FALSE(m_NUWShl(m_Value(), m_Value()).match(
294 IRB.CreateShl(L, R, "", /* NUW */ false, /* NSW */ true)));
295 EXPECT_FALSE(m_NUWShl(m_Value(), m_Value()).match(IRB.CreateNUWAdd(L, R)));
Arnold Schwaighofere79d92c2013-05-05 01:54:46 +0000296}
297
298} // anonymous namespace.