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