blob: f75de5c22593c3588f2a2fd411d4e9f2ee53b4ff [file] [log] [blame]
Aditya Nandakumarc8ac0292019-08-06 17:18:29 +00001//===- KnownBitsTest.cpp -------------------------------------------===//
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//===----------------------------------------------------------------------===//
8
9#include "GISelMITest.h"
10#include "llvm/CodeGen/GlobalISel/GISelKnownBits.h"
11#include "llvm/CodeGen/GlobalISel/MachineIRBuilder.h"
12
13TEST_F(GISelMITest, TestKnownBitsCst) {
14 StringRef MIRString = " %3:_(s8) = G_CONSTANT i8 1\n"
15 " %4:_(s8) = COPY %3\n";
16 setUp(MIRString);
17 if (!TM)
18 return;
19 unsigned CopyReg = Copies[Copies.size() - 1];
20 MachineInstr *FinalCopy = MRI->getVRegDef(CopyReg);
21 unsigned SrcReg = FinalCopy->getOperand(1).getReg();
Daniel Sanderse55c4f32019-09-05 20:25:52 +000022 unsigned DstReg = FinalCopy->getOperand(0).getReg();
Aditya Nandakumarc8ac0292019-08-06 17:18:29 +000023 GISelKnownBits Info(*MF);
24 KnownBits Res = Info.getKnownBits(SrcReg);
25 EXPECT_EQ((uint64_t)1, Res.One.getZExtValue());
26 EXPECT_EQ((uint64_t)0xfe, Res.Zero.getZExtValue());
Daniel Sandersb276a9a2019-09-04 18:59:43 +000027
28 KnownBits Res2 = Info.getKnownBits(DstReg);
29 EXPECT_EQ(Res.One.getZExtValue(), Res2.One.getZExtValue());
30 EXPECT_EQ(Res.Zero.getZExtValue(), Res2.Zero.getZExtValue());
Aditya Nandakumarc8ac0292019-08-06 17:18:29 +000031}
Daniel Sandersb276a9a2019-09-04 18:59:43 +000032
Daniel Sandersf8032372019-09-05 20:26:02 +000033TEST_F(GISelMITest, TestKnownBitsCstWithClass) {
34 StringRef MIRString = " %10:gpr32 = MOVi32imm 1\n"
35 " %4:_(s32) = COPY %10\n";
36 setUp(MIRString);
37 if (!TM)
38 return;
39 unsigned CopyReg = Copies[Copies.size() - 1];
40 MachineInstr *FinalCopy = MRI->getVRegDef(CopyReg);
41 unsigned SrcReg = FinalCopy->getOperand(1).getReg();
42 unsigned DstReg = FinalCopy->getOperand(0).getReg();
43 GISelKnownBits Info(*MF);
44 KnownBits Res = Info.getKnownBits(SrcReg);
45 // We can't analyze %3 due to the register class constraint. We will get a
46 // default-constructed KnownBits back.
47 EXPECT_EQ((uint64_t)1, Res.getBitWidth());
48 EXPECT_EQ((uint64_t)0, Res.One.getZExtValue());
49 EXPECT_EQ((uint64_t)0, Res.Zero.getZExtValue());
50
51 KnownBits Res2 = Info.getKnownBits(DstReg);
52 // We still don't know the values due to the register class constraint but %4
53 // did reveal the size of %3.
54 EXPECT_EQ((uint64_t)32, Res2.getBitWidth());
55 EXPECT_EQ(Res.One.getZExtValue(), Res2.One.getZExtValue());
56 EXPECT_EQ(Res.Zero.getZExtValue(), Res2.Zero.getZExtValue());
57}
58
Quentin Colombet5d87b5d2020-01-24 16:15:43 -080059// Check that we are able to track bits through PHIs
60// and get the intersections of everything we know on each operand.
61TEST_F(GISelMITest, TestKnownBitsCstPHI) {
62 StringRef MIRString = " bb.10:\n"
63 " %10:_(s8) = G_CONSTANT i8 3\n"
64 " %11:_(s1) = G_IMPLICIT_DEF\n"
65 " G_BRCOND %11(s1), %bb.11\n"
66 " G_BR %bb.12\n"
67 "\n"
68 " bb.11:\n"
69 " %12:_(s8) = G_CONSTANT i8 2\n"
70 " G_BR %bb.12\n"
71 "\n"
72 " bb.12:\n"
73 " %13:_(s8) = PHI %10(s8), %bb.10, %12(s8), %bb.11\n"
74 " %14:_(s8) = COPY %13\n";
75 setUp(MIRString);
76 if (!TM)
77 return;
78 Register CopyReg = Copies[Copies.size() - 1];
79 MachineInstr *FinalCopy = MRI->getVRegDef(CopyReg);
80 Register SrcReg = FinalCopy->getOperand(1).getReg();
81 Register DstReg = FinalCopy->getOperand(0).getReg();
82 GISelKnownBits Info(*MF);
83 KnownBits Res = Info.getKnownBits(SrcReg);
84 EXPECT_EQ((uint64_t)2, Res.One.getZExtValue());
85 EXPECT_EQ((uint64_t)0xfc, Res.Zero.getZExtValue());
86
87 KnownBits Res2 = Info.getKnownBits(DstReg);
88 EXPECT_EQ(Res.One.getZExtValue(), Res2.One.getZExtValue());
89 EXPECT_EQ(Res.Zero.getZExtValue(), Res2.Zero.getZExtValue());
90}
91
92// Check that we report we know nothing when we hit a
93// non-generic register.
94// Note: this could be improved though!
95TEST_F(GISelMITest, TestKnownBitsCstPHIToNonGenericReg) {
96 StringRef MIRString = " bb.10:\n"
97 " %10:gpr32 = MOVi32imm 3\n"
98 " %11:_(s1) = G_IMPLICIT_DEF\n"
99 " G_BRCOND %11(s1), %bb.11\n"
100 " G_BR %bb.12\n"
101 "\n"
102 " bb.11:\n"
103 " %12:_(s8) = G_CONSTANT i8 2\n"
104 " G_BR %bb.12\n"
105 "\n"
106 " bb.12:\n"
107 " %13:_(s8) = PHI %10, %bb.10, %12(s8), %bb.11\n"
108 " %14:_(s8) = COPY %13\n";
109 setUp(MIRString);
110 if (!TM)
111 return;
112 Register CopyReg = Copies[Copies.size() - 1];
113 MachineInstr *FinalCopy = MRI->getVRegDef(CopyReg);
114 Register SrcReg = FinalCopy->getOperand(1).getReg();
115 Register DstReg = FinalCopy->getOperand(0).getReg();
116 GISelKnownBits Info(*MF);
117 KnownBits Res = Info.getKnownBits(SrcReg);
118 EXPECT_EQ((uint64_t)0, Res.One.getZExtValue());
119 EXPECT_EQ((uint64_t)0, Res.Zero.getZExtValue());
120
121 KnownBits Res2 = Info.getKnownBits(DstReg);
122 EXPECT_EQ(Res.One.getZExtValue(), Res2.One.getZExtValue());
123 EXPECT_EQ(Res.Zero.getZExtValue(), Res2.Zero.getZExtValue());
124}
125
Quentin Colombete4a92252020-02-20 11:27:36 -0800126// Check that we know nothing when at least one value of a PHI
127// comes from something we cannot analysis.
128// This test is not particularly interesting, it is just
129// here to cover the code that stops the analysis of PHIs
130// earlier. In that case, we would not even look at the
131// second incoming value.
132TEST_F(GISelMITest, TestKnownBitsUnknownPHI) {
133 StringRef MIRString =
134 " bb.10:\n"
135 " %10:_(s64) = COPY %0\n"
136 " %11:_(s1) = G_IMPLICIT_DEF\n"
137 " G_BRCOND %11(s1), %bb.11\n"
138 " G_BR %bb.12\n"
139 "\n"
140 " bb.11:\n"
141 " %12:_(s64) = G_CONSTANT i64 2\n"
142 " G_BR %bb.12\n"
143 "\n"
144 " bb.12:\n"
145 " %13:_(s64) = PHI %10(s64), %bb.10, %12(s64), %bb.11\n"
146 " %14:_(s64) = COPY %13\n";
147 setUp(MIRString);
148 if (!TM)
149 return;
150 Register CopyReg = Copies[Copies.size() - 1];
151 MachineInstr *FinalCopy = MRI->getVRegDef(CopyReg);
152 Register SrcReg = FinalCopy->getOperand(1).getReg();
153 Register DstReg = FinalCopy->getOperand(0).getReg();
154 GISelKnownBits Info(*MF);
155 KnownBits Res = Info.getKnownBits(SrcReg);
156 EXPECT_EQ((uint64_t)0, Res.One.getZExtValue());
157 EXPECT_EQ((uint64_t)0, Res.Zero.getZExtValue());
158
159 KnownBits Res2 = Info.getKnownBits(DstReg);
160 EXPECT_EQ(Res.One.getZExtValue(), Res2.One.getZExtValue());
161 EXPECT_EQ(Res.Zero.getZExtValue(), Res2.Zero.getZExtValue());
162}
163
Quentin Colombet5d87b5d2020-01-24 16:15:43 -0800164// Check that we manage to process PHIs that loop on themselves.
165// For now, the analysis just stops and assumes it knows nothing,
166// eventually we could teach it how to properly track phis that
167// loop back.
168TEST_F(GISelMITest, TestKnownBitsCstPHIWithLoop) {
169 StringRef MIRString =
170 " bb.10:\n"
171 " %10:_(s8) = G_CONSTANT i8 3\n"
172 " %11:_(s1) = G_IMPLICIT_DEF\n"
173 " G_BRCOND %11(s1), %bb.11\n"
174 " G_BR %bb.12\n"
175 "\n"
176 " bb.11:\n"
177 " %12:_(s8) = G_CONSTANT i8 2\n"
178 " G_BR %bb.12\n"
179 "\n"
180 " bb.12:\n"
181 " %13:_(s8) = PHI %10(s8), %bb.10, %12(s8), %bb.11, %14(s8), %bb.12\n"
182 " %14:_(s8) = COPY %13\n"
183 " G_BR %bb.12\n";
184 setUp(MIRString);
185 if (!TM)
186 return;
187 Register CopyReg = Copies[Copies.size() - 1];
188 MachineInstr *FinalCopy = MRI->getVRegDef(CopyReg);
189 Register SrcReg = FinalCopy->getOperand(1).getReg();
190 Register DstReg = FinalCopy->getOperand(0).getReg();
191 GISelKnownBits Info(*MF);
192 KnownBits Res = Info.getKnownBits(SrcReg);
193 EXPECT_EQ((uint64_t)0, Res.One.getZExtValue());
194 EXPECT_EQ((uint64_t)0, Res.Zero.getZExtValue());
195
196 KnownBits Res2 = Info.getKnownBits(DstReg);
197 EXPECT_EQ(Res.One.getZExtValue(), Res2.One.getZExtValue());
198 EXPECT_EQ(Res.Zero.getZExtValue(), Res2.Zero.getZExtValue());
199}
200
Aditya Nandakumar55371e62019-08-12 21:28:12 +0000201TEST_F(GISelMITest, TestKnownBitsPtrToIntViceVersa) {
202 StringRef MIRString = " %3:_(s16) = G_CONSTANT i16 256\n"
203 " %4:_(p0) = G_INTTOPTR %3\n"
204 " %5:_(s32) = G_PTRTOINT %4\n"
205 " %6:_(s32) = COPY %5\n";
206 setUp(MIRString);
207 if (!TM)
208 return;
209 unsigned CopyReg = Copies[Copies.size() - 1];
210 MachineInstr *FinalCopy = MRI->getVRegDef(CopyReg);
211 unsigned SrcReg = FinalCopy->getOperand(1).getReg();
212 GISelKnownBits Info(*MF);
213 KnownBits Res = Info.getKnownBits(SrcReg);
214 EXPECT_EQ(256u, Res.One.getZExtValue());
215 EXPECT_EQ(0xfffffeffu, Res.Zero.getZExtValue());
216}
Aditya Nandakumar70fdfed2019-08-13 04:32:33 +0000217TEST_F(GISelMITest, TestKnownBitsXOR) {
218 StringRef MIRString = " %3:_(s8) = G_CONSTANT i8 4\n"
219 " %4:_(s8) = G_CONSTANT i8 7\n"
220 " %5:_(s8) = G_XOR %3, %4\n"
221 " %6:_(s8) = COPY %5\n";
222 setUp(MIRString);
223 if (!TM)
224 return;
225 unsigned CopyReg = Copies[Copies.size() - 1];
226 MachineInstr *FinalCopy = MRI->getVRegDef(CopyReg);
227 unsigned SrcReg = FinalCopy->getOperand(1).getReg();
228 GISelKnownBits Info(*MF);
229 KnownBits Res = Info.getKnownBits(SrcReg);
230 EXPECT_EQ(3u, Res.One.getZExtValue());
231 EXPECT_EQ(252u, Res.Zero.getZExtValue());
232}
Aditya Nandakumarc8ac0292019-08-06 17:18:29 +0000233
234TEST_F(GISelMITest, TestKnownBits) {
235
236 StringRef MIR = " %3:_(s32) = G_TRUNC %0\n"
237 " %4:_(s32) = G_TRUNC %1\n"
238 " %5:_(s32) = G_CONSTANT i32 5\n"
239 " %6:_(s32) = G_CONSTANT i32 24\n"
240 " %7:_(s32) = G_CONSTANT i32 28\n"
241 " %14:_(p0) = G_INTTOPTR %7\n"
242 " %16:_(s32) = G_PTRTOINT %14\n"
243 " %8:_(s32) = G_SHL %3, %5\n"
244 " %9:_(s32) = G_SHL %4, %5\n"
245 " %10:_(s32) = G_OR %8, %6\n"
246 " %11:_(s32) = G_OR %9, %16\n"
247 " %12:_(s32) = G_MUL %10, %11\n"
248 " %13:_(s32) = COPY %12\n";
249 setUp(MIR);
250 if (!TM)
251 return;
252 unsigned CopyReg = Copies[Copies.size() - 1];
253 MachineInstr *FinalCopy = MRI->getVRegDef(CopyReg);
254 unsigned SrcReg = FinalCopy->getOperand(1).getReg();
255 GISelKnownBits Info(*MF);
256 KnownBits Known = Info.getKnownBits(SrcReg);
257 EXPECT_FALSE(Known.hasConflict());
258 EXPECT_EQ(0u, Known.One.getZExtValue());
259 EXPECT_EQ(31u, Known.Zero.getZExtValue());
260 APInt Zeroes = Info.getKnownZeroes(SrcReg);
261 EXPECT_EQ(Known.Zero, Zeroes);
262}
Matt Arsenaultb2b9a232019-08-29 17:24:36 +0000263
264TEST_F(GISelMITest, TestSignBitIsZero) {
Quentin Colombet49036dd2019-10-11 20:58:26 +0000265 setUp();
Matt Arsenaultb2b9a232019-08-29 17:24:36 +0000266 if (!TM)
267 return;
268
269 const LLT S32 = LLT::scalar(32);
Quentin Colombet49036dd2019-10-11 20:58:26 +0000270 auto SignBit = B.buildConstant(S32, 0x80000000);
Matt Arsenaultb2b9a232019-08-29 17:24:36 +0000271 auto Zero = B.buildConstant(S32, 0);
272
273 GISelKnownBits KnownBits(*MF);
274
275 EXPECT_TRUE(KnownBits.signBitIsZero(Zero.getReg(0)));
Matt Arsenaultb2b9a232019-08-29 17:24:36 +0000276 EXPECT_FALSE(KnownBits.signBitIsZero(SignBit.getReg(0)));
Matt Arsenaultb2b9a232019-08-29 17:24:36 +0000277}
Matt Arsenault0b093f02020-01-04 14:13:06 -0500278
279TEST_F(GISelMITest, TestNumSignBitsConstant) {
280 StringRef MIRString = " %3:_(s8) = G_CONSTANT i8 1\n"
281 " %4:_(s8) = COPY %3\n"
282
283 " %5:_(s8) = G_CONSTANT i8 -1\n"
284 " %6:_(s8) = COPY %5\n"
285
286 " %7:_(s8) = G_CONSTANT i8 127\n"
287 " %8:_(s8) = COPY %7\n"
288
289 " %9:_(s8) = G_CONSTANT i8 32\n"
290 " %10:_(s8) = COPY %9\n"
291
292 " %11:_(s8) = G_CONSTANT i8 -32\n"
293 " %12:_(s8) = COPY %11\n";
294 setUp(MIRString);
295 if (!TM)
296 return;
297 Register CopyReg1 = Copies[Copies.size() - 5];
298 Register CopyRegNeg1 = Copies[Copies.size() - 4];
299 Register CopyReg127 = Copies[Copies.size() - 3];
300 Register CopyReg32 = Copies[Copies.size() - 2];
301 Register CopyRegNeg32 = Copies[Copies.size() - 1];
302
303 GISelKnownBits Info(*MF);
304 EXPECT_EQ(7u, Info.computeNumSignBits(CopyReg1));
305 EXPECT_EQ(8u, Info.computeNumSignBits(CopyRegNeg1));
306 EXPECT_EQ(1u, Info.computeNumSignBits(CopyReg127));
307 EXPECT_EQ(2u, Info.computeNumSignBits(CopyReg32));
308 EXPECT_EQ(3u, Info.computeNumSignBits(CopyRegNeg32));
309}
310
311TEST_F(GISelMITest, TestNumSignBitsSext) {
312 StringRef MIRString = " %3:_(p0) = G_IMPLICIT_DEF\n"
313 " %4:_(s8) = G_LOAD %3 :: (load 1)\n"
314 " %5:_(s32) = G_SEXT %4\n"
315 " %6:_(s32) = COPY %5\n"
316
317 " %7:_(s8) = G_CONSTANT i8 -1\n"
318 " %8:_(s32) = G_SEXT %7\n"
319 " %9:_(s32) = COPY %8\n";
320 setUp(MIRString);
321 if (!TM)
322 return;
323 Register CopySextLoad = Copies[Copies.size() - 2];
324 Register CopySextNeg1 = Copies[Copies.size() - 1];
325
326 GISelKnownBits Info(*MF);
327 EXPECT_EQ(25u, Info.computeNumSignBits(CopySextLoad));
328 EXPECT_EQ(32u, Info.computeNumSignBits(CopySextNeg1));
329}
330
331TEST_F(GISelMITest, TestNumSignBitsTrunc) {
332 StringRef MIRString = " %3:_(p0) = G_IMPLICIT_DEF\n"
333 " %4:_(s32) = G_LOAD %3 :: (load 4)\n"
334 " %5:_(s8) = G_TRUNC %4\n"
335 " %6:_(s8) = COPY %5\n"
336
337 " %7:_(s32) = G_CONSTANT i32 -1\n"
338 " %8:_(s8) = G_TRUNC %7\n"
339 " %9:_(s8) = COPY %8\n"
340
341 " %10:_(s32) = G_CONSTANT i32 7\n"
342 " %11:_(s8) = G_TRUNC %10\n"
343 " %12:_(s8) = COPY %11\n";
344 setUp(MIRString);
345 if (!TM)
346 return;
347 Register CopyTruncLoad = Copies[Copies.size() - 3];
348 Register CopyTruncNeg1 = Copies[Copies.size() - 2];
349 Register CopyTrunc7 = Copies[Copies.size() - 1];
350
351 GISelKnownBits Info(*MF);
352 EXPECT_EQ(1u, Info.computeNumSignBits(CopyTruncLoad));
353 EXPECT_EQ(8u, Info.computeNumSignBits(CopyTruncNeg1));
354 EXPECT_EQ(5u, Info.computeNumSignBits(CopyTrunc7));
355}