1 //===- SparsePropagation.cpp - Unit tests for the generic solver ----------===//
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
7 //===----------------------------------------------------------------------===//
9 #include "llvm/Analysis/SparsePropagation.h"
10 #include "llvm/ADT/PointerIntPair.h"
11 #include "llvm/IR/IRBuilder.h"
12 #include "gtest/gtest.h"
16 /// To enable interprocedural analysis, we assign LLVM values to the following
17 /// groups. The register group represents SSA registers, the return group
18 /// represents the return values of functions, and the memory group represents
19 /// in-memory values. An LLVM Value can technically be in more than one group.
20 /// It's necessary to distinguish these groups so we can, for example, track a
21 /// global variable separately from the value stored at its location.
22 enum class IPOGrouping
{ Register
, Return
, Memory
};
24 /// Our LatticeKeys are PointerIntPairs composed of LLVM values and groupings.
25 /// The PointerIntPair header provides a DenseMapInfo specialization, so using
26 /// these as LatticeKeys is fine.
27 using TestLatticeKey
= PointerIntPair
<Value
*, 2, IPOGrouping
>;
31 /// A specialization of LatticeKeyInfo for TestLatticeKeys. The generic solver
32 /// must translate between LatticeKeys and LLVM Values when adding Values to
33 /// its work list and inspecting the state of control-flow related values.
34 template <> struct LatticeKeyInfo
<TestLatticeKey
> {
35 static inline Value
*getValueFromLatticeKey(TestLatticeKey Key
) {
36 return Key
.getPointer();
38 static inline TestLatticeKey
getLatticeKeyFromValue(Value
*V
) {
39 return TestLatticeKey(V
, IPOGrouping::Register
);
45 /// This class defines a simple test lattice value that could be used for
46 /// solving problems similar to constant propagation. The value is maintained
47 /// as a PointerIntPair.
48 class TestLatticeVal
{
50 /// The states of the lattices value. Only the ConstantVal state is
51 /// interesting; the rest are special states used by the generic solver. The
52 /// UntrackedVal state differs from the other three in that the generic
53 /// solver uses it to avoid doing unnecessary work. In particular, when a
54 /// value moves to the UntrackedVal state, it's users are not notified.
55 enum TestLatticeStateTy
{
62 TestLatticeVal() : LatticeVal(nullptr, UndefinedVal
) {}
63 TestLatticeVal(Constant
*C
, TestLatticeStateTy State
)
64 : LatticeVal(C
, State
) {}
66 /// Return true if this lattice value is in the Constant state. This is used
67 /// for checking the solver results.
68 bool isConstant() const { return LatticeVal
.getInt() == ConstantVal
; }
70 /// Return true if this lattice value is in the Overdefined state. This is
71 /// used for checking the solver results.
72 bool isOverdefined() const { return LatticeVal
.getInt() == OverdefinedVal
; }
74 bool operator==(const TestLatticeVal
&RHS
) const {
75 return LatticeVal
== RHS
.LatticeVal
;
78 bool operator!=(const TestLatticeVal
&RHS
) const {
79 return LatticeVal
!= RHS
.LatticeVal
;
83 /// A simple lattice value type for problems similar to constant propagation.
84 /// It holds the constant value and the lattice state.
85 PointerIntPair
<const Constant
*, 2, TestLatticeStateTy
> LatticeVal
;
88 /// This class defines a simple test lattice function that could be used for
89 /// solving problems similar to constant propagation. The test lattice differs
90 /// from a "real" lattice in a few ways. First, it initializes all return
91 /// values, values stored in global variables, and arguments in the undefined
92 /// state. This means that there are no limitations on what we can track
93 /// interprocedurally. For simplicity, all global values in the tests will be
94 /// given internal linkage, since this is not something this lattice function
95 /// tracks. Second, it only handles the few instructions necessary for the
98 : public AbstractLatticeFunction
<TestLatticeKey
, TestLatticeVal
> {
100 /// Construct a new test lattice function with special values for the
101 /// Undefined, Overdefined, and Untracked states.
103 : AbstractLatticeFunction(
104 TestLatticeVal(nullptr, TestLatticeVal::UndefinedVal
),
105 TestLatticeVal(nullptr, TestLatticeVal::OverdefinedVal
),
106 TestLatticeVal(nullptr, TestLatticeVal::UntrackedVal
)) {}
108 /// Compute and return a TestLatticeVal for the given TestLatticeKey. For the
109 /// test analysis, a LatticeKey will begin in the undefined state, unless it
110 /// represents an LLVM Constant in the register grouping.
111 TestLatticeVal
ComputeLatticeVal(TestLatticeKey Key
) override
{
112 if (Key
.getInt() == IPOGrouping::Register
)
113 if (auto *C
= dyn_cast
<Constant
>(Key
.getPointer()))
114 return TestLatticeVal(C
, TestLatticeVal::ConstantVal
);
115 return getUndefVal();
118 /// Merge the two given lattice values. This merge should be equivalent to
119 /// what is done for constant propagation. That is, the resulting lattice
120 /// value is constant only if the two given lattice values are constant and
121 /// hold the same value.
122 TestLatticeVal
MergeValues(TestLatticeVal X
, TestLatticeVal Y
) override
{
123 if (X
== getUntrackedVal() || Y
== getUntrackedVal())
124 return getUntrackedVal();
125 if (X
== getOverdefinedVal() || Y
== getOverdefinedVal())
126 return getOverdefinedVal();
127 if (X
== getUndefVal() && Y
== getUndefVal())
128 return getUndefVal();
129 if (X
== getUndefVal())
131 if (Y
== getUndefVal())
135 return getOverdefinedVal();
138 /// Compute the lattice values that change as a result of executing the given
139 /// instruction. We only handle the few instructions needed for the tests.
140 void ComputeInstructionState(
141 Instruction
&I
, DenseMap
<TestLatticeKey
, TestLatticeVal
> &ChangedValues
,
142 SparseSolver
<TestLatticeKey
, TestLatticeVal
> &SS
) override
{
143 switch (I
.getOpcode()) {
144 case Instruction::Call
:
145 return visitCallBase(cast
<CallBase
>(I
), ChangedValues
, SS
);
146 case Instruction::Ret
:
147 return visitReturn(*cast
<ReturnInst
>(&I
), ChangedValues
, SS
);
148 case Instruction::Store
:
149 return visitStore(*cast
<StoreInst
>(&I
), ChangedValues
, SS
);
151 return visitInst(I
, ChangedValues
, SS
);
156 /// Handle call sites. The state of a called function's argument is the merge
157 /// of the current formal argument state with the call site's corresponding
158 /// actual argument state. The call site state is the merge of the call site
159 /// state with the returned value state of the called function.
160 void visitCallBase(CallBase
&I
,
161 DenseMap
<TestLatticeKey
, TestLatticeVal
> &ChangedValues
,
162 SparseSolver
<TestLatticeKey
, TestLatticeVal
> &SS
) {
163 Function
*F
= I
.getCalledFunction();
164 auto RegI
= TestLatticeKey(&I
, IPOGrouping::Register
);
166 ChangedValues
[RegI
] = getOverdefinedVal();
169 SS
.MarkBlockExecutable(&F
->front());
170 for (Argument
&A
: F
->args()) {
171 auto RegFormal
= TestLatticeKey(&A
, IPOGrouping::Register
);
173 TestLatticeKey(I
.getArgOperand(A
.getArgNo()), IPOGrouping::Register
);
174 ChangedValues
[RegFormal
] =
175 MergeValues(SS
.getValueState(RegFormal
), SS
.getValueState(RegActual
));
177 auto RetF
= TestLatticeKey(F
, IPOGrouping::Return
);
178 ChangedValues
[RegI
] =
179 MergeValues(SS
.getValueState(RegI
), SS
.getValueState(RetF
));
182 /// Handle return instructions. The function's return state is the merge of
183 /// the returned value state and the function's current return state.
184 void visitReturn(ReturnInst
&I
,
185 DenseMap
<TestLatticeKey
, TestLatticeVal
> &ChangedValues
,
186 SparseSolver
<TestLatticeKey
, TestLatticeVal
> &SS
) {
187 Function
*F
= I
.getParent()->getParent();
188 if (F
->getReturnType()->isVoidTy())
190 auto RegR
= TestLatticeKey(I
.getReturnValue(), IPOGrouping::Register
);
191 auto RetF
= TestLatticeKey(F
, IPOGrouping::Return
);
192 ChangedValues
[RetF
] =
193 MergeValues(SS
.getValueState(RegR
), SS
.getValueState(RetF
));
196 /// Handle store instructions. If the pointer operand of the store is a
197 /// global variable, we attempt to track the value. The global variable state
198 /// is the merge of the stored value state with the current global variable
200 void visitStore(StoreInst
&I
,
201 DenseMap
<TestLatticeKey
, TestLatticeVal
> &ChangedValues
,
202 SparseSolver
<TestLatticeKey
, TestLatticeVal
> &SS
) {
203 auto *GV
= dyn_cast
<GlobalVariable
>(I
.getPointerOperand());
206 auto RegVal
= TestLatticeKey(I
.getValueOperand(), IPOGrouping::Register
);
207 auto MemPtr
= TestLatticeKey(GV
, IPOGrouping::Memory
);
208 ChangedValues
[MemPtr
] =
209 MergeValues(SS
.getValueState(RegVal
), SS
.getValueState(MemPtr
));
212 /// Handle all other instructions. All other instructions are marked
214 void visitInst(Instruction
&I
,
215 DenseMap
<TestLatticeKey
, TestLatticeVal
> &ChangedValues
,
216 SparseSolver
<TestLatticeKey
, TestLatticeVal
> &SS
) {
217 auto RegI
= TestLatticeKey(&I
, IPOGrouping::Register
);
218 ChangedValues
[RegI
] = getOverdefinedVal();
222 /// This class defines the common data used for all of the tests. The tests
223 /// should add code to the module and then run the solver.
224 class SparsePropagationTest
: public testing::Test
{
229 TestLatticeFunc Lattice
;
230 SparseSolver
<TestLatticeKey
, TestLatticeVal
> Solver
;
233 SparsePropagationTest()
234 : M("", Context
), Builder(Context
), Solver(&Lattice
) {}
238 /// Test that we mark discovered functions executable.
240 /// define internal void @f() {
245 /// define internal void @g() {
250 /// For this test, we initially mark "f" executable, and the solver discovers
251 /// "g" because of the call in "f". The mutually recursive call in "g" also
252 /// tests that we don't add a block to the basic block work list if it is
253 /// already executable. Doing so would put the solver into an infinite loop.
254 TEST_F(SparsePropagationTest
, MarkBlockExecutable
) {
255 Function
*F
= Function::Create(FunctionType::get(Builder
.getVoidTy(), false),
256 GlobalValue::InternalLinkage
, "f", &M
);
257 Function
*G
= Function::Create(FunctionType::get(Builder
.getVoidTy(), false),
258 GlobalValue::InternalLinkage
, "g", &M
);
259 BasicBlock
*FEntry
= BasicBlock::Create(Context
, "", F
);
260 BasicBlock
*GEntry
= BasicBlock::Create(Context
, "", G
);
261 Builder
.SetInsertPoint(FEntry
);
262 Builder
.CreateCall(G
);
263 Builder
.CreateRetVoid();
264 Builder
.SetInsertPoint(GEntry
);
265 Builder
.CreateCall(F
);
266 Builder
.CreateRetVoid();
268 Solver
.MarkBlockExecutable(FEntry
);
271 EXPECT_TRUE(Solver
.isBlockExecutable(GEntry
));
274 /// Test that we propagate information through global variables.
276 /// @gv = internal global i64
278 /// define internal void @f() {
279 /// store i64 1, i64* @gv
283 /// define internal void @g() {
284 /// store i64 1, i64* @gv
288 /// For this test, we initially mark both "f" and "g" executable, and the
289 /// solver computes the lattice state of the global variable as constant.
290 TEST_F(SparsePropagationTest
, GlobalVariableConstant
) {
291 Function
*F
= Function::Create(FunctionType::get(Builder
.getVoidTy(), false),
292 GlobalValue::InternalLinkage
, "f", &M
);
293 Function
*G
= Function::Create(FunctionType::get(Builder
.getVoidTy(), false),
294 GlobalValue::InternalLinkage
, "g", &M
);
296 new GlobalVariable(M
, Builder
.getInt64Ty(), false,
297 GlobalValue::InternalLinkage
, nullptr, "gv");
298 BasicBlock
*FEntry
= BasicBlock::Create(Context
, "", F
);
299 BasicBlock
*GEntry
= BasicBlock::Create(Context
, "", G
);
300 Builder
.SetInsertPoint(FEntry
);
301 Builder
.CreateStore(Builder
.getInt64(1), GV
);
302 Builder
.CreateRetVoid();
303 Builder
.SetInsertPoint(GEntry
);
304 Builder
.CreateStore(Builder
.getInt64(1), GV
);
305 Builder
.CreateRetVoid();
307 Solver
.MarkBlockExecutable(FEntry
);
308 Solver
.MarkBlockExecutable(GEntry
);
311 auto MemGV
= TestLatticeKey(GV
, IPOGrouping::Memory
);
312 EXPECT_TRUE(Solver
.getExistingValueState(MemGV
).isConstant());
315 /// Test that we propagate information through global variables.
317 /// @gv = internal global i64
319 /// define internal void @f() {
320 /// store i64 0, i64* @gv
324 /// define internal void @g() {
325 /// store i64 1, i64* @gv
329 /// For this test, we initially mark both "f" and "g" executable, and the
330 /// solver computes the lattice state of the global variable as overdefined.
331 TEST_F(SparsePropagationTest
, GlobalVariableOverDefined
) {
332 Function
*F
= Function::Create(FunctionType::get(Builder
.getVoidTy(), false),
333 GlobalValue::InternalLinkage
, "f", &M
);
334 Function
*G
= Function::Create(FunctionType::get(Builder
.getVoidTy(), false),
335 GlobalValue::InternalLinkage
, "g", &M
);
337 new GlobalVariable(M
, Builder
.getInt64Ty(), false,
338 GlobalValue::InternalLinkage
, nullptr, "gv");
339 BasicBlock
*FEntry
= BasicBlock::Create(Context
, "", F
);
340 BasicBlock
*GEntry
= BasicBlock::Create(Context
, "", G
);
341 Builder
.SetInsertPoint(FEntry
);
342 Builder
.CreateStore(Builder
.getInt64(0), GV
);
343 Builder
.CreateRetVoid();
344 Builder
.SetInsertPoint(GEntry
);
345 Builder
.CreateStore(Builder
.getInt64(1), GV
);
346 Builder
.CreateRetVoid();
348 Solver
.MarkBlockExecutable(FEntry
);
349 Solver
.MarkBlockExecutable(GEntry
);
352 auto MemGV
= TestLatticeKey(GV
, IPOGrouping::Memory
);
353 EXPECT_TRUE(Solver
.getExistingValueState(MemGV
).isOverdefined());
356 /// Test that we propagate information through function returns.
358 /// define internal i64 @f(i1* %cond) {
360 /// %0 = load i1, i1* %cond
361 /// br i1 %0, label %then, label %else
370 /// For this test, we initially mark "f" executable, and the solver computes
371 /// the return value of the function as constant.
372 TEST_F(SparsePropagationTest
, FunctionDefined
) {
374 Function::Create(FunctionType::get(Builder
.getInt64Ty(),
375 {PointerType::get(Context
, 0)}, false),
376 GlobalValue::InternalLinkage
, "f", &M
);
377 BasicBlock
*If
= BasicBlock::Create(Context
, "if", F
);
378 BasicBlock
*Then
= BasicBlock::Create(Context
, "then", F
);
379 BasicBlock
*Else
= BasicBlock::Create(Context
, "else", F
);
380 F
->arg_begin()->setName("cond");
381 Builder
.SetInsertPoint(If
);
382 LoadInst
*Cond
= Builder
.CreateLoad(Type::getInt1Ty(Context
), F
->arg_begin());
383 Builder
.CreateCondBr(Cond
, Then
, Else
);
384 Builder
.SetInsertPoint(Then
);
385 Builder
.CreateRet(Builder
.getInt64(1));
386 Builder
.SetInsertPoint(Else
);
387 Builder
.CreateRet(Builder
.getInt64(1));
389 Solver
.MarkBlockExecutable(If
);
392 auto RetF
= TestLatticeKey(F
, IPOGrouping::Return
);
393 EXPECT_TRUE(Solver
.getExistingValueState(RetF
).isConstant());
396 /// Test that we propagate information through function returns.
398 /// define internal i64 @f(i1* %cond) {
400 /// %0 = load i1, i1* %cond
401 /// br i1 %0, label %then, label %else
410 /// For this test, we initially mark "f" executable, and the solver computes
411 /// the return value of the function as overdefined.
412 TEST_F(SparsePropagationTest
, FunctionOverDefined
) {
414 Function::Create(FunctionType::get(Builder
.getInt64Ty(),
415 {PointerType::get(Context
, 0)}, false),
416 GlobalValue::InternalLinkage
, "f", &M
);
417 BasicBlock
*If
= BasicBlock::Create(Context
, "if", F
);
418 BasicBlock
*Then
= BasicBlock::Create(Context
, "then", F
);
419 BasicBlock
*Else
= BasicBlock::Create(Context
, "else", F
);
420 F
->arg_begin()->setName("cond");
421 Builder
.SetInsertPoint(If
);
422 LoadInst
*Cond
= Builder
.CreateLoad(Type::getInt1Ty(Context
), F
->arg_begin());
423 Builder
.CreateCondBr(Cond
, Then
, Else
);
424 Builder
.SetInsertPoint(Then
);
425 Builder
.CreateRet(Builder
.getInt64(0));
426 Builder
.SetInsertPoint(Else
);
427 Builder
.CreateRet(Builder
.getInt64(1));
429 Solver
.MarkBlockExecutable(If
);
432 auto RetF
= TestLatticeKey(F
, IPOGrouping::Return
);
433 EXPECT_TRUE(Solver
.getExistingValueState(RetF
).isOverdefined());
436 /// Test that we propagate information through arguments.
438 /// define internal void @f() {
439 /// call void @g(i64 0, i64 1)
440 /// call void @g(i64 1, i64 1)
444 /// define internal void @g(i64 %a, i64 %b) {
448 /// For this test, we initially mark "f" executable, and the solver discovers
449 /// "g" because of the calls in "f". The solver computes the state of argument
450 /// "a" as overdefined and the state of "b" as constant.
452 /// In addition, this test demonstrates that ComputeInstructionState can alter
453 /// the state of multiple lattice values, in addition to the one associated
454 /// with the instruction definition. Each call instruction in this test updates
455 /// the state of arguments "a" and "b".
456 TEST_F(SparsePropagationTest
, ComputeInstructionState
) {
457 Function
*F
= Function::Create(FunctionType::get(Builder
.getVoidTy(), false),
458 GlobalValue::InternalLinkage
, "f", &M
);
459 Function
*G
= Function::Create(
460 FunctionType::get(Builder
.getVoidTy(),
461 {Builder
.getInt64Ty(), Builder
.getInt64Ty()}, false),
462 GlobalValue::InternalLinkage
, "g", &M
);
463 Argument
*A
= G
->arg_begin();
464 Argument
*B
= std::next(G
->arg_begin());
467 BasicBlock
*FEntry
= BasicBlock::Create(Context
, "", F
);
468 BasicBlock
*GEntry
= BasicBlock::Create(Context
, "", G
);
469 Builder
.SetInsertPoint(FEntry
);
470 Builder
.CreateCall(G
, {Builder
.getInt64(0), Builder
.getInt64(1)});
471 Builder
.CreateCall(G
, {Builder
.getInt64(1), Builder
.getInt64(1)});
472 Builder
.CreateRetVoid();
473 Builder
.SetInsertPoint(GEntry
);
474 Builder
.CreateRetVoid();
476 Solver
.MarkBlockExecutable(FEntry
);
479 auto RegA
= TestLatticeKey(A
, IPOGrouping::Register
);
480 auto RegB
= TestLatticeKey(B
, IPOGrouping::Register
);
481 EXPECT_TRUE(Solver
.getExistingValueState(RegA
).isOverdefined());
482 EXPECT_TRUE(Solver
.getExistingValueState(RegB
).isConstant());
485 /// Test that we can handle exceptional terminator instructions.
487 /// declare internal void @p()
489 /// declare internal void @g()
491 /// define internal void @f() personality ptr @p {
494 /// to label %exit unwind label %catch.pad
497 /// %0 = catchswitch within none [label %catch.body] unwind to caller
500 /// %1 = catchpad within %0 []
501 /// catchret from %1 to label %exit
507 /// For this test, we initially mark the entry block executable. The solver
508 /// then discovers the rest of the blocks in the function are executable.
509 TEST_F(SparsePropagationTest
, ExceptionalTerminatorInsts
) {
510 Function
*P
= Function::Create(FunctionType::get(Builder
.getVoidTy(), false),
511 GlobalValue::InternalLinkage
, "p", &M
);
512 Function
*G
= Function::Create(FunctionType::get(Builder
.getVoidTy(), false),
513 GlobalValue::InternalLinkage
, "g", &M
);
514 Function
*F
= Function::Create(FunctionType::get(Builder
.getVoidTy(), false),
515 GlobalValue::InternalLinkage
, "f", &M
);
516 F
->setPersonalityFn(P
);
517 BasicBlock
*Entry
= BasicBlock::Create(Context
, "entry", F
);
518 BasicBlock
*Pad
= BasicBlock::Create(Context
, "catch.pad", F
);
519 BasicBlock
*Body
= BasicBlock::Create(Context
, "catch.body", F
);
520 BasicBlock
*Exit
= BasicBlock::Create(Context
, "exit", F
);
521 Builder
.SetInsertPoint(Entry
);
522 Builder
.CreateInvoke(G
, Exit
, Pad
);
523 Builder
.SetInsertPoint(Pad
);
524 CatchSwitchInst
*CatchSwitch
=
525 Builder
.CreateCatchSwitch(ConstantTokenNone::get(Context
), nullptr, 1);
526 CatchSwitch
->addHandler(Body
);
527 Builder
.SetInsertPoint(Body
);
528 CatchPadInst
*CatchPad
= Builder
.CreateCatchPad(CatchSwitch
, {});
529 Builder
.CreateCatchRet(CatchPad
, Exit
);
530 Builder
.SetInsertPoint(Exit
);
531 Builder
.CreateRetVoid();
533 Solver
.MarkBlockExecutable(Entry
);
536 EXPECT_TRUE(Solver
.isBlockExecutable(Pad
));
537 EXPECT_TRUE(Solver
.isBlockExecutable(Body
));
538 EXPECT_TRUE(Solver
.isBlockExecutable(Exit
));