llvm-project/clang/unittests/Analysis/CFGDominatorTree.cpp

195 lines
6.4 KiB
C++

//===- unittests/Analysis/CFGDominatorTree.cpp - CFG tests ----------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "CFGBuildResult.h"
#include "clang/Analysis/Analyses/Dominators.h"
#include "gtest/gtest.h"
namespace clang {
namespace analysis {
namespace {
TEST(CFGDominatorTree, DomTree) {
const char *Code = R"(enum Kind {
A
};
void f() {
switch(Kind{}) {
case A:
break;
}
})";
BuildResult Result = BuildCFG(Code);
EXPECT_EQ(BuildResult::BuiltCFG, Result.getStatus());
// [B3 (ENTRY)] -> [B1] -> [B2] -> [B0 (EXIT)]
// switch case A
CFG *cfg = Result.getCFG();
// Basic correctness checks.
EXPECT_EQ(cfg->size(), 4u);
CFGBlock *ExitBlock = *cfg->begin();
EXPECT_EQ(ExitBlock, &cfg->getExit());
CFGBlock *SwitchBlock = *(cfg->begin() + 1);
CFGBlock *CaseABlock = *(cfg->begin() + 2);
CFGBlock *EntryBlock = *(cfg->begin() + 3);
EXPECT_EQ(EntryBlock, &cfg->getEntry());
// Test the dominator tree.
CFGDomTree Dom;
Dom.buildDominatorTree(cfg);
EXPECT_TRUE(Dom.dominates(ExitBlock, ExitBlock));
EXPECT_FALSE(Dom.properlyDominates(ExitBlock, ExitBlock));
EXPECT_TRUE(Dom.dominates(CaseABlock, ExitBlock));
EXPECT_TRUE(Dom.dominates(SwitchBlock, ExitBlock));
EXPECT_TRUE(Dom.dominates(EntryBlock, ExitBlock));
EXPECT_TRUE(Dom.dominates(CaseABlock, CaseABlock));
EXPECT_FALSE(Dom.properlyDominates(CaseABlock, CaseABlock));
EXPECT_TRUE(Dom.dominates(SwitchBlock, CaseABlock));
EXPECT_TRUE(Dom.dominates(EntryBlock, CaseABlock));
EXPECT_TRUE(Dom.dominates(SwitchBlock, SwitchBlock));
EXPECT_FALSE(Dom.properlyDominates(SwitchBlock, SwitchBlock));
EXPECT_TRUE(Dom.dominates(EntryBlock, SwitchBlock));
EXPECT_TRUE(Dom.dominates(EntryBlock, EntryBlock));
EXPECT_FALSE(Dom.properlyDominates(EntryBlock, EntryBlock));
// Test the post dominator tree.
CFGPostDomTree PostDom;
PostDom.buildDominatorTree(cfg);
EXPECT_TRUE(PostDom.dominates(ExitBlock, EntryBlock));
EXPECT_TRUE(PostDom.dominates(CaseABlock, EntryBlock));
EXPECT_TRUE(PostDom.dominates(SwitchBlock, EntryBlock));
EXPECT_TRUE(PostDom.dominates(EntryBlock, EntryBlock));
EXPECT_FALSE(Dom.properlyDominates(EntryBlock, EntryBlock));
EXPECT_TRUE(PostDom.dominates(ExitBlock, SwitchBlock));
EXPECT_TRUE(PostDom.dominates(CaseABlock, SwitchBlock));
EXPECT_TRUE(PostDom.dominates(SwitchBlock, SwitchBlock));
EXPECT_FALSE(Dom.properlyDominates(SwitchBlock, SwitchBlock));
EXPECT_TRUE(PostDom.dominates(ExitBlock, CaseABlock));
EXPECT_TRUE(PostDom.dominates(CaseABlock, CaseABlock));
EXPECT_FALSE(Dom.properlyDominates(CaseABlock, CaseABlock));
EXPECT_TRUE(PostDom.dominates(ExitBlock, ExitBlock));
EXPECT_FALSE(Dom.properlyDominates(ExitBlock, ExitBlock));
// Tests for the post dominator tree's virtual root.
EXPECT_TRUE(PostDom.dominates(nullptr, EntryBlock));
EXPECT_TRUE(PostDom.dominates(nullptr, SwitchBlock));
EXPECT_TRUE(PostDom.dominates(nullptr, CaseABlock));
EXPECT_TRUE(PostDom.dominates(nullptr, ExitBlock));
}
TEST(CFGDominatorTree, ControlDependency) {
const char *Code = R"(bool coin();
void funcWithBranch() {
int x = 0;
if (coin()) {
if (coin()) {
x = 5;
}
int j = 10 / x;
(void)j;
}
};)";
BuildResult Result = BuildCFG(Code);
EXPECT_EQ(BuildResult::BuiltCFG, Result.getStatus());
// 1st if 2nd if
// [B5 (ENTRY)] -> [B4] -> [B3] -> [B2] -> [B1] -> [B0 (EXIT)]
// \ \ / /
// \ -------------> /
// ------------------------------>
CFG *cfg = Result.getCFG();
// Basic correctness checks.
EXPECT_EQ(cfg->size(), 6u);
CFGBlock *ExitBlock = *cfg->begin();
EXPECT_EQ(ExitBlock, &cfg->getExit());
CFGBlock *NullDerefBlock = *(cfg->begin() + 1);
CFGBlock *SecondThenBlock = *(cfg->begin() + 2);
CFGBlock *SecondIfBlock = *(cfg->begin() + 3);
CFGBlock *FirstIfBlock = *(cfg->begin() + 4);
CFGBlock *EntryBlock = *(cfg->begin() + 5);
EXPECT_EQ(EntryBlock, &cfg->getEntry());
ControlDependencyCalculator Control(cfg);
EXPECT_TRUE(Control.isControlDependent(SecondThenBlock, SecondIfBlock));
EXPECT_TRUE(Control.isControlDependent(SecondIfBlock, FirstIfBlock));
EXPECT_FALSE(Control.isControlDependent(NullDerefBlock, SecondIfBlock));
}
TEST(CFGDominatorTree, ControlDependencyWithLoops) {
const char *Code = R"(int test3() {
int x,y,z;
x = y = z = 1;
if (x > 0) {
while (x >= 0){
while (y >= x) {
x = x-1;
y = y/2;
}
}
}
z = y;
return 0;
})";
BuildResult Result = BuildCFG(Code);
EXPECT_EQ(BuildResult::BuiltCFG, Result.getStatus());
// <- [B2] <-
// / \
// [B8 (ENTRY)] -> [B7] -> [B6] ---> [B5] -> [B4] -> [B3]
// \ | \ /
// \ | <-------------
// \ \
// --------> [B1] -> [B0 (EXIT)]
CFG *cfg = Result.getCFG();
ControlDependencyCalculator Control(cfg);
auto GetBlock = [cfg] (unsigned Index) -> CFGBlock * {
assert(Index < cfg->size());
return *(cfg->begin() + Index);
};
// While not immediately obvious, the second block in fact post dominates the
// fifth, hence B5 is not a control dependency of 2.
EXPECT_FALSE(Control.isControlDependent(GetBlock(5), GetBlock(2)));
}
} // namespace
} // namespace analysis
} // namespace clang