blob: 72d713beacc88147a57e2d433efa7ebb34c016a1 [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
126// Check that we manage to process PHIs that loop on themselves.
127// For now, the analysis just stops and assumes it knows nothing,
128// eventually we could teach it how to properly track phis that
129// loop back.
130TEST_F(GISelMITest, TestKnownBitsCstPHIWithLoop) {
131 StringRef MIRString =
132 " bb.10:\n"
133 " %10:_(s8) = G_CONSTANT i8 3\n"
134 " %11:_(s1) = G_IMPLICIT_DEF\n"
135 " G_BRCOND %11(s1), %bb.11\n"
136 " G_BR %bb.12\n"
137 "\n"
138 " bb.11:\n"
139 " %12:_(s8) = G_CONSTANT i8 2\n"
140 " G_BR %bb.12\n"
141 "\n"
142 " bb.12:\n"
143 " %13:_(s8) = PHI %10(s8), %bb.10, %12(s8), %bb.11, %14(s8), %bb.12\n"
144 " %14:_(s8) = COPY %13\n"
145 " G_BR %bb.12\n";
146 setUp(MIRString);
147 if (!TM)
148 return;
149 Register CopyReg = Copies[Copies.size() - 1];
150 MachineInstr *FinalCopy = MRI->getVRegDef(CopyReg);
151 Register SrcReg = FinalCopy->getOperand(1).getReg();
152 Register DstReg = FinalCopy->getOperand(0).getReg();
153 GISelKnownBits Info(*MF);
154 KnownBits Res = Info.getKnownBits(SrcReg);
155 EXPECT_EQ((uint64_t)0, Res.One.getZExtValue());
156 EXPECT_EQ((uint64_t)0, Res.Zero.getZExtValue());
157
158 KnownBits Res2 = Info.getKnownBits(DstReg);
159 EXPECT_EQ(Res.One.getZExtValue(), Res2.One.getZExtValue());
160 EXPECT_EQ(Res.Zero.getZExtValue(), Res2.Zero.getZExtValue());
161}
162
Aditya Nandakumar55371e62019-08-12 21:28:12 +0000163TEST_F(GISelMITest, TestKnownBitsPtrToIntViceVersa) {
164 StringRef MIRString = " %3:_(s16) = G_CONSTANT i16 256\n"
165 " %4:_(p0) = G_INTTOPTR %3\n"
166 " %5:_(s32) = G_PTRTOINT %4\n"
167 " %6:_(s32) = COPY %5\n";
168 setUp(MIRString);
169 if (!TM)
170 return;
171 unsigned CopyReg = Copies[Copies.size() - 1];
172 MachineInstr *FinalCopy = MRI->getVRegDef(CopyReg);
173 unsigned SrcReg = FinalCopy->getOperand(1).getReg();
174 GISelKnownBits Info(*MF);
175 KnownBits Res = Info.getKnownBits(SrcReg);
176 EXPECT_EQ(256u, Res.One.getZExtValue());
177 EXPECT_EQ(0xfffffeffu, Res.Zero.getZExtValue());
178}
Aditya Nandakumar70fdfed2019-08-13 04:32:33 +0000179TEST_F(GISelMITest, TestKnownBitsXOR) {
180 StringRef MIRString = " %3:_(s8) = G_CONSTANT i8 4\n"
181 " %4:_(s8) = G_CONSTANT i8 7\n"
182 " %5:_(s8) = G_XOR %3, %4\n"
183 " %6:_(s8) = COPY %5\n";
184 setUp(MIRString);
185 if (!TM)
186 return;
187 unsigned CopyReg = Copies[Copies.size() - 1];
188 MachineInstr *FinalCopy = MRI->getVRegDef(CopyReg);
189 unsigned SrcReg = FinalCopy->getOperand(1).getReg();
190 GISelKnownBits Info(*MF);
191 KnownBits Res = Info.getKnownBits(SrcReg);
192 EXPECT_EQ(3u, Res.One.getZExtValue());
193 EXPECT_EQ(252u, Res.Zero.getZExtValue());
194}
Aditya Nandakumarc8ac0292019-08-06 17:18:29 +0000195
196TEST_F(GISelMITest, TestKnownBits) {
197
198 StringRef MIR = " %3:_(s32) = G_TRUNC %0\n"
199 " %4:_(s32) = G_TRUNC %1\n"
200 " %5:_(s32) = G_CONSTANT i32 5\n"
201 " %6:_(s32) = G_CONSTANT i32 24\n"
202 " %7:_(s32) = G_CONSTANT i32 28\n"
203 " %14:_(p0) = G_INTTOPTR %7\n"
204 " %16:_(s32) = G_PTRTOINT %14\n"
205 " %8:_(s32) = G_SHL %3, %5\n"
206 " %9:_(s32) = G_SHL %4, %5\n"
207 " %10:_(s32) = G_OR %8, %6\n"
208 " %11:_(s32) = G_OR %9, %16\n"
209 " %12:_(s32) = G_MUL %10, %11\n"
210 " %13:_(s32) = COPY %12\n";
211 setUp(MIR);
212 if (!TM)
213 return;
214 unsigned CopyReg = Copies[Copies.size() - 1];
215 MachineInstr *FinalCopy = MRI->getVRegDef(CopyReg);
216 unsigned SrcReg = FinalCopy->getOperand(1).getReg();
217 GISelKnownBits Info(*MF);
218 KnownBits Known = Info.getKnownBits(SrcReg);
219 EXPECT_FALSE(Known.hasConflict());
220 EXPECT_EQ(0u, Known.One.getZExtValue());
221 EXPECT_EQ(31u, Known.Zero.getZExtValue());
222 APInt Zeroes = Info.getKnownZeroes(SrcReg);
223 EXPECT_EQ(Known.Zero, Zeroes);
224}
Matt Arsenaultb2b9a232019-08-29 17:24:36 +0000225
226TEST_F(GISelMITest, TestSignBitIsZero) {
Quentin Colombet49036dd2019-10-11 20:58:26 +0000227 setUp();
Matt Arsenaultb2b9a232019-08-29 17:24:36 +0000228 if (!TM)
229 return;
230
231 const LLT S32 = LLT::scalar(32);
Quentin Colombet49036dd2019-10-11 20:58:26 +0000232 auto SignBit = B.buildConstant(S32, 0x80000000);
Matt Arsenaultb2b9a232019-08-29 17:24:36 +0000233 auto Zero = B.buildConstant(S32, 0);
234
235 GISelKnownBits KnownBits(*MF);
236
237 EXPECT_TRUE(KnownBits.signBitIsZero(Zero.getReg(0)));
Matt Arsenaultb2b9a232019-08-29 17:24:36 +0000238 EXPECT_FALSE(KnownBits.signBitIsZero(SignBit.getReg(0)));
Matt Arsenaultb2b9a232019-08-29 17:24:36 +0000239}
Matt Arsenault0b093f02020-01-04 14:13:06 -0500240
241TEST_F(GISelMITest, TestNumSignBitsConstant) {
242 StringRef MIRString = " %3:_(s8) = G_CONSTANT i8 1\n"
243 " %4:_(s8) = COPY %3\n"
244
245 " %5:_(s8) = G_CONSTANT i8 -1\n"
246 " %6:_(s8) = COPY %5\n"
247
248 " %7:_(s8) = G_CONSTANT i8 127\n"
249 " %8:_(s8) = COPY %7\n"
250
251 " %9:_(s8) = G_CONSTANT i8 32\n"
252 " %10:_(s8) = COPY %9\n"
253
254 " %11:_(s8) = G_CONSTANT i8 -32\n"
255 " %12:_(s8) = COPY %11\n";
256 setUp(MIRString);
257 if (!TM)
258 return;
259 Register CopyReg1 = Copies[Copies.size() - 5];
260 Register CopyRegNeg1 = Copies[Copies.size() - 4];
261 Register CopyReg127 = Copies[Copies.size() - 3];
262 Register CopyReg32 = Copies[Copies.size() - 2];
263 Register CopyRegNeg32 = Copies[Copies.size() - 1];
264
265 GISelKnownBits Info(*MF);
266 EXPECT_EQ(7u, Info.computeNumSignBits(CopyReg1));
267 EXPECT_EQ(8u, Info.computeNumSignBits(CopyRegNeg1));
268 EXPECT_EQ(1u, Info.computeNumSignBits(CopyReg127));
269 EXPECT_EQ(2u, Info.computeNumSignBits(CopyReg32));
270 EXPECT_EQ(3u, Info.computeNumSignBits(CopyRegNeg32));
271}
272
273TEST_F(GISelMITest, TestNumSignBitsSext) {
274 StringRef MIRString = " %3:_(p0) = G_IMPLICIT_DEF\n"
275 " %4:_(s8) = G_LOAD %3 :: (load 1)\n"
276 " %5:_(s32) = G_SEXT %4\n"
277 " %6:_(s32) = COPY %5\n"
278
279 " %7:_(s8) = G_CONSTANT i8 -1\n"
280 " %8:_(s32) = G_SEXT %7\n"
281 " %9:_(s32) = COPY %8\n";
282 setUp(MIRString);
283 if (!TM)
284 return;
285 Register CopySextLoad = Copies[Copies.size() - 2];
286 Register CopySextNeg1 = Copies[Copies.size() - 1];
287
288 GISelKnownBits Info(*MF);
289 EXPECT_EQ(25u, Info.computeNumSignBits(CopySextLoad));
290 EXPECT_EQ(32u, Info.computeNumSignBits(CopySextNeg1));
291}
292
293TEST_F(GISelMITest, TestNumSignBitsTrunc) {
294 StringRef MIRString = " %3:_(p0) = G_IMPLICIT_DEF\n"
295 " %4:_(s32) = G_LOAD %3 :: (load 4)\n"
296 " %5:_(s8) = G_TRUNC %4\n"
297 " %6:_(s8) = COPY %5\n"
298
299 " %7:_(s32) = G_CONSTANT i32 -1\n"
300 " %8:_(s8) = G_TRUNC %7\n"
301 " %9:_(s8) = COPY %8\n"
302
303 " %10:_(s32) = G_CONSTANT i32 7\n"
304 " %11:_(s8) = G_TRUNC %10\n"
305 " %12:_(s8) = COPY %11\n";
306 setUp(MIRString);
307 if (!TM)
308 return;
309 Register CopyTruncLoad = Copies[Copies.size() - 3];
310 Register CopyTruncNeg1 = Copies[Copies.size() - 2];
311 Register CopyTrunc7 = Copies[Copies.size() - 1];
312
313 GISelKnownBits Info(*MF);
314 EXPECT_EQ(1u, Info.computeNumSignBits(CopyTruncLoad));
315 EXPECT_EQ(8u, Info.computeNumSignBits(CopyTruncNeg1));
316 EXPECT_EQ(5u, Info.computeNumSignBits(CopyTrunc7));
317}