18fa1e373SChandler Carruth //===- llvm/unittest/IR/BasicBlockTest.cpp - BasicBlock unit tests --------===//
28fa1e373SChandler Carruth //
32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
68fa1e373SChandler Carruth //
78fa1e373SChandler Carruth //===----------------------------------------------------------------------===//
88fa1e373SChandler Carruth 
98fa1e373SChandler Carruth #include "llvm/IR/BasicBlock.h"
108fa1e373SChandler Carruth #include "llvm/ADT/STLExtras.h"
110c2b09a9SReid Kleckner #include "llvm/AsmParser/Parser.h"
128fa1e373SChandler Carruth #include "llvm/IR/Function.h"
138fa1e373SChandler Carruth #include "llvm/IR/IRBuilder.h"
14554e6db1SArthur Eubanks #include "llvm/IR/Instructions.h"
158fa1e373SChandler Carruth #include "llvm/IR/LLVMContext.h"
168fa1e373SChandler Carruth #include "llvm/IR/Module.h"
178fa1e373SChandler Carruth #include "llvm/IR/NoFolder.h"
18554e6db1SArthur Eubanks #include "llvm/IR/Verifier.h"
190c2b09a9SReid Kleckner #include "llvm/Support/SourceMgr.h"
208fa1e373SChandler Carruth #include "gmock/gmock-matchers.h"
218fa1e373SChandler Carruth #include "gtest/gtest.h"
228fa1e373SChandler Carruth #include <memory>
238fa1e373SChandler Carruth 
248fa1e373SChandler Carruth namespace llvm {
258fa1e373SChandler Carruth namespace {
268fa1e373SChandler Carruth 
TEST(BasicBlockTest,PhiRange)278fa1e373SChandler Carruth TEST(BasicBlockTest, PhiRange) {
288fa1e373SChandler Carruth   LLVMContext Context;
298fa1e373SChandler Carruth 
308fa1e373SChandler Carruth   // Create the main block.
318fa1e373SChandler Carruth   std::unique_ptr<BasicBlock> BB(BasicBlock::Create(Context));
328fa1e373SChandler Carruth 
338fa1e373SChandler Carruth   // Create some predecessors of it.
348fa1e373SChandler Carruth   std::unique_ptr<BasicBlock> BB1(BasicBlock::Create(Context));
358fa1e373SChandler Carruth   BranchInst::Create(BB.get(), BB1.get());
368fa1e373SChandler Carruth   std::unique_ptr<BasicBlock> BB2(BasicBlock::Create(Context));
378fa1e373SChandler Carruth   BranchInst::Create(BB.get(), BB2.get());
388fa1e373SChandler Carruth 
39344c0920SMatt Arsenault   // Make sure this doesn't crash if there are no phis.
40*fb4f6057SPaul Robinson   int PhiCount = 0;
41344c0920SMatt Arsenault   for (auto &PN : BB->phis()) {
42344c0920SMatt Arsenault     (void)PN;
43*fb4f6057SPaul Robinson     PhiCount++;
44344c0920SMatt Arsenault   }
45*fb4f6057SPaul Robinson   ASSERT_EQ(PhiCount, 0) << "empty block should have no phis";
46344c0920SMatt Arsenault 
478fa1e373SChandler Carruth   // Make it a cycle.
488fa1e373SChandler Carruth   auto *BI = BranchInst::Create(BB.get(), BB.get());
498fa1e373SChandler Carruth 
508fa1e373SChandler Carruth   // Now insert some PHI nodes.
518fa1e373SChandler Carruth   auto *Int32Ty = Type::getInt32Ty(Context);
528fa1e373SChandler Carruth   auto *P1 = PHINode::Create(Int32Ty, /*NumReservedValues*/ 3, "phi.1", BI);
538fa1e373SChandler Carruth   auto *P2 = PHINode::Create(Int32Ty, /*NumReservedValues*/ 3, "phi.2", BI);
548fa1e373SChandler Carruth   auto *P3 = PHINode::Create(Int32Ty, /*NumReservedValues*/ 3, "phi.3", BI);
558fa1e373SChandler Carruth 
568fa1e373SChandler Carruth   // Some non-PHI nodes.
578fa1e373SChandler Carruth   auto *Sum = BinaryOperator::CreateAdd(P1, P2, "sum", BI);
588fa1e373SChandler Carruth 
598fa1e373SChandler Carruth   // Now wire up the incoming values that are interesting.
608fa1e373SChandler Carruth   P1->addIncoming(P2, BB.get());
618fa1e373SChandler Carruth   P2->addIncoming(P1, BB.get());
628fa1e373SChandler Carruth   P3->addIncoming(Sum, BB.get());
638fa1e373SChandler Carruth 
648fa1e373SChandler Carruth   // Finally, let's iterate them, which is the thing we're trying to test.
658fa1e373SChandler Carruth   // We'll use this to wire up the rest of the incoming values.
668fa1e373SChandler Carruth   for (auto &PN : BB->phis()) {
678fa1e373SChandler Carruth     PN.addIncoming(UndefValue::get(Int32Ty), BB1.get());
688fa1e373SChandler Carruth     PN.addIncoming(UndefValue::get(Int32Ty), BB2.get());
698fa1e373SChandler Carruth   }
708fa1e373SChandler Carruth 
718fa1e373SChandler Carruth   // Test that we can use const iterators and generally that the iterators
728fa1e373SChandler Carruth   // behave like iterators.
738fa1e373SChandler Carruth   BasicBlock::const_phi_iterator CI;
749230610fSDaniel Jasper   CI = BB->phis().begin();
758fa1e373SChandler Carruth   EXPECT_NE(CI, BB->phis().end());
768fa1e373SChandler Carruth 
7775fda2e0SVedant Kumar   // Test that filtering iterators work with basic blocks.
7875fda2e0SVedant Kumar   auto isPhi = [](Instruction &I) { return isa<PHINode>(&I); };
7975fda2e0SVedant Kumar   auto Phis = make_filter_range(*BB, isPhi);
8075fda2e0SVedant Kumar   auto ReversedPhis = reverse(make_filter_range(*BB, isPhi));
815a0872c2SVedant Kumar   EXPECT_EQ(std::distance(Phis.begin(), Phis.end()), 3);
8275fda2e0SVedant Kumar   EXPECT_EQ(&*Phis.begin(), P1);
835a0872c2SVedant Kumar   EXPECT_EQ(std::distance(ReversedPhis.begin(), ReversedPhis.end()), 3);
8475fda2e0SVedant Kumar   EXPECT_EQ(&*ReversedPhis.begin(), P3);
8575fda2e0SVedant Kumar 
868fa1e373SChandler Carruth   // And iterate a const range.
878fa1e373SChandler Carruth   for (const auto &PN : const_cast<const BasicBlock *>(BB.get())->phis()) {
888fa1e373SChandler Carruth     EXPECT_EQ(BB.get(), PN.getIncomingBlock(0));
898fa1e373SChandler Carruth     EXPECT_EQ(BB1.get(), PN.getIncomingBlock(1));
908fa1e373SChandler Carruth     EXPECT_EQ(BB2.get(), PN.getIncomingBlock(2));
918fa1e373SChandler Carruth   }
928fa1e373SChandler Carruth }
938fa1e373SChandler Carruth 
94147fc016SFlorian Hahn #define CHECK_ITERATORS(Range1, Range2)                                        \
955a0872c2SVedant Kumar   EXPECT_EQ(std::distance(Range1.begin(), Range1.end()),                       \
965a0872c2SVedant Kumar             std::distance(Range2.begin(), Range2.end()));                      \
97147fc016SFlorian Hahn   for (auto Pair : zip(Range1, Range2))                                        \
98147fc016SFlorian Hahn     EXPECT_EQ(&std::get<0>(Pair), std::get<1>(Pair));
99147fc016SFlorian Hahn 
TEST(BasicBlockTest,TestInstructionsWithoutDebug)1002342533eSFlorian Hahn TEST(BasicBlockTest, TestInstructionsWithoutDebug) {
101147fc016SFlorian Hahn   LLVMContext Ctx;
102147fc016SFlorian Hahn 
1032342533eSFlorian Hahn   Module *M = new Module("MyModule", Ctx);
104147fc016SFlorian Hahn   Type *ArgTy1[] = {Type::getInt32PtrTy(Ctx)};
105147fc016SFlorian Hahn   FunctionType *FT = FunctionType::get(Type::getVoidTy(Ctx), ArgTy1, false);
1062342533eSFlorian Hahn   Argument *V = new Argument(Type::getInt32Ty(Ctx));
1072342533eSFlorian Hahn   Function *F = Function::Create(FT, Function::ExternalLinkage, "", M);
108147fc016SFlorian Hahn 
1097976eb58SJames Y Knight   Function *DbgAddr = Intrinsic::getDeclaration(M, Intrinsic::dbg_addr);
1107976eb58SJames Y Knight   Function *DbgDeclare = Intrinsic::getDeclaration(M, Intrinsic::dbg_declare);
1117976eb58SJames Y Knight   Function *DbgValue = Intrinsic::getDeclaration(M, Intrinsic::dbg_value);
112147fc016SFlorian Hahn   Value *DIV = MetadataAsValue::get(Ctx, (Metadata *)nullptr);
113147fc016SFlorian Hahn   SmallVector<Value *, 3> Args = {DIV, DIV, DIV};
114147fc016SFlorian Hahn 
115147fc016SFlorian Hahn   BasicBlock *BB1 = BasicBlock::Create(Ctx, "", F);
116147fc016SFlorian Hahn   const BasicBlock *BBConst = BB1;
117147fc016SFlorian Hahn   IRBuilder<> Builder1(BB1);
118147fc016SFlorian Hahn 
119147fc016SFlorian Hahn   AllocaInst *Var = Builder1.CreateAlloca(Builder1.getInt8Ty());
120147fc016SFlorian Hahn   Builder1.CreateCall(DbgValue, Args);
121147fc016SFlorian Hahn   Instruction *AddInst = cast<Instruction>(Builder1.CreateAdd(V, V));
122147fc016SFlorian Hahn   Instruction *MulInst = cast<Instruction>(Builder1.CreateMul(AddInst, V));
123147fc016SFlorian Hahn   Builder1.CreateCall(DbgDeclare, Args);
124147fc016SFlorian Hahn   Instruction *SubInst = cast<Instruction>(Builder1.CreateSub(MulInst, V));
125147fc016SFlorian Hahn   Builder1.CreateCall(DbgAddr, Args);
126147fc016SFlorian Hahn 
127147fc016SFlorian Hahn   SmallVector<Instruction *, 4> Exp = {Var, AddInst, MulInst, SubInst};
128147fc016SFlorian Hahn   CHECK_ITERATORS(BB1->instructionsWithoutDebug(), Exp);
129147fc016SFlorian Hahn   CHECK_ITERATORS(BBConst->instructionsWithoutDebug(), Exp);
1302342533eSFlorian Hahn 
1312a47c77eSDavid Tellenbach   EXPECT_EQ(static_cast<size_t>(BB1->sizeWithoutDebug()), Exp.size());
1322a47c77eSDavid Tellenbach   EXPECT_EQ(static_cast<size_t>(BBConst->sizeWithoutDebug()), Exp.size());
1332a47c77eSDavid Tellenbach 
1342342533eSFlorian Hahn   delete M;
1352342533eSFlorian Hahn   delete V;
136147fc016SFlorian Hahn }
137147fc016SFlorian Hahn 
TEST(BasicBlockTest,ComesBefore)1380c2b09a9SReid Kleckner TEST(BasicBlockTest, ComesBefore) {
1390c2b09a9SReid Kleckner   const char *ModuleString = R"(define i32 @f(i32 %x) {
1400c2b09a9SReid Kleckner                                   %add = add i32 %x, 42
1410c2b09a9SReid Kleckner                                   ret i32 %add
1420c2b09a9SReid Kleckner                                 })";
1430c2b09a9SReid Kleckner   LLVMContext Ctx;
1440c2b09a9SReid Kleckner   SMDiagnostic Err;
1450c2b09a9SReid Kleckner   auto M = parseAssemblyString(ModuleString, Err, Ctx);
1460c2b09a9SReid Kleckner   ASSERT_TRUE(M.get());
1470c2b09a9SReid Kleckner 
1480c2b09a9SReid Kleckner   Function *F = M->getFunction("f");
1490c2b09a9SReid Kleckner   BasicBlock &BB = F->front();
1500c2b09a9SReid Kleckner   BasicBlock::iterator I = BB.begin();
1510c2b09a9SReid Kleckner   Instruction *Add = &*I++;
1520c2b09a9SReid Kleckner   Instruction *Ret = &*I++;
1530c2b09a9SReid Kleckner 
1540c2b09a9SReid Kleckner   // Intentionally duplicated to verify cached and uncached are the same.
1550c2b09a9SReid Kleckner   EXPECT_FALSE(BB.isInstrOrderValid());
1560c2b09a9SReid Kleckner   EXPECT_FALSE(Add->comesBefore(Add));
1570c2b09a9SReid Kleckner   EXPECT_TRUE(BB.isInstrOrderValid());
1580c2b09a9SReid Kleckner   EXPECT_FALSE(Add->comesBefore(Add));
1590c2b09a9SReid Kleckner   BB.invalidateOrders();
1600c2b09a9SReid Kleckner   EXPECT_FALSE(BB.isInstrOrderValid());
1610c2b09a9SReid Kleckner   EXPECT_TRUE(Add->comesBefore(Ret));
1620c2b09a9SReid Kleckner   EXPECT_TRUE(BB.isInstrOrderValid());
1630c2b09a9SReid Kleckner   EXPECT_TRUE(Add->comesBefore(Ret));
1640c2b09a9SReid Kleckner   BB.invalidateOrders();
1650c2b09a9SReid Kleckner   EXPECT_FALSE(Ret->comesBefore(Add));
1660c2b09a9SReid Kleckner   EXPECT_FALSE(Ret->comesBefore(Add));
1670c2b09a9SReid Kleckner   BB.invalidateOrders();
1680c2b09a9SReid Kleckner   EXPECT_FALSE(Ret->comesBefore(Ret));
1690c2b09a9SReid Kleckner   EXPECT_FALSE(Ret->comesBefore(Ret));
1700c2b09a9SReid Kleckner }
1710c2b09a9SReid Kleckner 
TEST(BasicBlockTest,EmptyPhi)172554e6db1SArthur Eubanks TEST(BasicBlockTest, EmptyPhi) {
173554e6db1SArthur Eubanks   LLVMContext Ctx;
174554e6db1SArthur Eubanks 
175554e6db1SArthur Eubanks   Module *M = new Module("MyModule", Ctx);
176554e6db1SArthur Eubanks   FunctionType *FT = FunctionType::get(Type::getVoidTy(Ctx), {}, false);
177554e6db1SArthur Eubanks   Function *F = Function::Create(FT, Function::ExternalLinkage, "", M);
178554e6db1SArthur Eubanks 
179554e6db1SArthur Eubanks   BasicBlock *BB1 = BasicBlock::Create(Ctx, "", F);
180554e6db1SArthur Eubanks   ReturnInst::Create(Ctx, BB1);
181554e6db1SArthur Eubanks 
182554e6db1SArthur Eubanks   Type *Ty = Type::getInt32PtrTy(Ctx);
183554e6db1SArthur Eubanks   BasicBlock *BB2 = BasicBlock::Create(Ctx, "", F);
184554e6db1SArthur Eubanks   PHINode::Create(Ty, 0, "", BB2);
185554e6db1SArthur Eubanks   ReturnInst::Create(Ctx, BB2);
186554e6db1SArthur Eubanks   EXPECT_FALSE(verifyModule(*M, &errs()));
187554e6db1SArthur Eubanks }
188554e6db1SArthur Eubanks 
1890c2b09a9SReid Kleckner class InstrOrderInvalidationTest : public ::testing::Test {
1900c2b09a9SReid Kleckner protected:
SetUp()1910c2b09a9SReid Kleckner   void SetUp() override {
1920c2b09a9SReid Kleckner     M.reset(new Module("MyModule", Ctx));
1930c2b09a9SReid Kleckner     Nop = Intrinsic::getDeclaration(M.get(), Intrinsic::donothing);
1940c2b09a9SReid Kleckner     FunctionType *FT = FunctionType::get(Type::getVoidTy(Ctx), {}, false);
1950c2b09a9SReid Kleckner     Function *F = Function::Create(FT, Function::ExternalLinkage, "foo", *M);
1960c2b09a9SReid Kleckner     BB = BasicBlock::Create(Ctx, "entry", F);
1970c2b09a9SReid Kleckner 
1980c2b09a9SReid Kleckner     IRBuilder<> Builder(BB);
1990c2b09a9SReid Kleckner     I1 = Builder.CreateCall(Nop);
2000c2b09a9SReid Kleckner     I2 = Builder.CreateCall(Nop);
2010c2b09a9SReid Kleckner     I3 = Builder.CreateCall(Nop);
2020c2b09a9SReid Kleckner     Ret = Builder.CreateRetVoid();
2030c2b09a9SReid Kleckner   }
2040c2b09a9SReid Kleckner 
2050c2b09a9SReid Kleckner   LLVMContext Ctx;
2060c2b09a9SReid Kleckner   std::unique_ptr<Module> M;
2070c2b09a9SReid Kleckner   Function *Nop = nullptr;
2080c2b09a9SReid Kleckner   BasicBlock *BB = nullptr;
2090c2b09a9SReid Kleckner   Instruction *I1 = nullptr;
2100c2b09a9SReid Kleckner   Instruction *I2 = nullptr;
2110c2b09a9SReid Kleckner   Instruction *I3 = nullptr;
2120c2b09a9SReid Kleckner   Instruction *Ret = nullptr;
2130c2b09a9SReid Kleckner };
2140c2b09a9SReid Kleckner 
TEST_F(InstrOrderInvalidationTest,InsertInvalidation)2150c2b09a9SReid Kleckner TEST_F(InstrOrderInvalidationTest, InsertInvalidation) {
2160c2b09a9SReid Kleckner   EXPECT_FALSE(BB->isInstrOrderValid());
2170c2b09a9SReid Kleckner   EXPECT_TRUE(I1->comesBefore(I2));
2180c2b09a9SReid Kleckner   EXPECT_TRUE(BB->isInstrOrderValid());
2190c2b09a9SReid Kleckner   EXPECT_TRUE(I2->comesBefore(I3));
2200c2b09a9SReid Kleckner   EXPECT_TRUE(I3->comesBefore(Ret));
2210c2b09a9SReid Kleckner   EXPECT_TRUE(BB->isInstrOrderValid());
2220c2b09a9SReid Kleckner 
2230c2b09a9SReid Kleckner   // Invalidate orders.
2240c2b09a9SReid Kleckner   IRBuilder<> Builder(BB, I2->getIterator());
2250c2b09a9SReid Kleckner   Instruction *I1a = Builder.CreateCall(Nop);
2260c2b09a9SReid Kleckner   EXPECT_FALSE(BB->isInstrOrderValid());
2270c2b09a9SReid Kleckner   EXPECT_TRUE(I1->comesBefore(I1a));
2280c2b09a9SReid Kleckner   EXPECT_TRUE(BB->isInstrOrderValid());
2290c2b09a9SReid Kleckner   EXPECT_TRUE(I1a->comesBefore(I2));
2300c2b09a9SReid Kleckner   EXPECT_TRUE(I2->comesBefore(I3));
2310c2b09a9SReid Kleckner   EXPECT_TRUE(I3->comesBefore(Ret));
2320c2b09a9SReid Kleckner   EXPECT_TRUE(BB->isInstrOrderValid());
2330c2b09a9SReid Kleckner }
2340c2b09a9SReid Kleckner 
TEST_F(InstrOrderInvalidationTest,SpliceInvalidation)2350c2b09a9SReid Kleckner TEST_F(InstrOrderInvalidationTest, SpliceInvalidation) {
2360c2b09a9SReid Kleckner   EXPECT_TRUE(I1->comesBefore(I2));
2370c2b09a9SReid Kleckner   EXPECT_TRUE(I2->comesBefore(I3));
2380c2b09a9SReid Kleckner   EXPECT_TRUE(I3->comesBefore(Ret));
2390c2b09a9SReid Kleckner   EXPECT_TRUE(BB->isInstrOrderValid());
2400c2b09a9SReid Kleckner 
2410c2b09a9SReid Kleckner   // Use Instruction::moveBefore, which uses splice.
2420c2b09a9SReid Kleckner   I2->moveBefore(I1);
2430c2b09a9SReid Kleckner   EXPECT_FALSE(BB->isInstrOrderValid());
2440c2b09a9SReid Kleckner 
2450c2b09a9SReid Kleckner   EXPECT_TRUE(I2->comesBefore(I1));
2460c2b09a9SReid Kleckner   EXPECT_TRUE(I1->comesBefore(I3));
2470c2b09a9SReid Kleckner   EXPECT_TRUE(I3->comesBefore(Ret));
2480c2b09a9SReid Kleckner   EXPECT_TRUE(BB->isInstrOrderValid());
2490c2b09a9SReid Kleckner }
2500c2b09a9SReid Kleckner 
TEST_F(InstrOrderInvalidationTest,RemoveNoInvalidation)2510c2b09a9SReid Kleckner TEST_F(InstrOrderInvalidationTest, RemoveNoInvalidation) {
2520c2b09a9SReid Kleckner   // Cache the instruction order.
2530c2b09a9SReid Kleckner   EXPECT_FALSE(BB->isInstrOrderValid());
2540c2b09a9SReid Kleckner   EXPECT_TRUE(I1->comesBefore(I2));
2550c2b09a9SReid Kleckner   EXPECT_TRUE(BB->isInstrOrderValid());
2560c2b09a9SReid Kleckner 
2570c2b09a9SReid Kleckner   // Removing does not invalidate instruction order.
2580c2b09a9SReid Kleckner   I2->removeFromParent();
2590c2b09a9SReid Kleckner   I2->deleteValue();
2600c2b09a9SReid Kleckner   I2 = nullptr;
2610c2b09a9SReid Kleckner   EXPECT_TRUE(BB->isInstrOrderValid());
2620c2b09a9SReid Kleckner   EXPECT_TRUE(I1->comesBefore(I3));
2630c2b09a9SReid Kleckner   EXPECT_EQ(std::next(I1->getIterator()), I3->getIterator());
2640c2b09a9SReid Kleckner }
2650c2b09a9SReid Kleckner 
TEST_F(InstrOrderInvalidationTest,EraseNoInvalidation)2660c2b09a9SReid Kleckner TEST_F(InstrOrderInvalidationTest, EraseNoInvalidation) {
2670c2b09a9SReid Kleckner   // Cache the instruction order.
2680c2b09a9SReid Kleckner   EXPECT_FALSE(BB->isInstrOrderValid());
2690c2b09a9SReid Kleckner   EXPECT_TRUE(I1->comesBefore(I2));
2700c2b09a9SReid Kleckner   EXPECT_TRUE(BB->isInstrOrderValid());
2710c2b09a9SReid Kleckner 
2720c2b09a9SReid Kleckner   // Removing does not invalidate instruction order.
2730c2b09a9SReid Kleckner   I2->eraseFromParent();
2740c2b09a9SReid Kleckner   I2 = nullptr;
2750c2b09a9SReid Kleckner   EXPECT_TRUE(BB->isInstrOrderValid());
2760c2b09a9SReid Kleckner   EXPECT_TRUE(I1->comesBefore(I3));
2770c2b09a9SReid Kleckner   EXPECT_EQ(std::next(I1->getIterator()), I3->getIterator());
2780c2b09a9SReid Kleckner }
2790c2b09a9SReid Kleckner 
2808fa1e373SChandler Carruth } // End anonymous namespace.
2818fa1e373SChandler Carruth } // End llvm namespace.
282