diff --git a/clang/include/clang/Analysis/Analyses/IntervalPartition.h b/clang/include/clang/Analysis/Analyses/IntervalPartition.h --- a/clang/include/clang/Analysis/Analyses/IntervalPartition.h +++ b/clang/include/clang/Analysis/Analyses/IntervalPartition.h @@ -6,9 +6,13 @@ // //===----------------------------------------------------------------------===// // -// This file defines functionality for partitioning a CFG into intervals. The -// concepts and implementations are based on the presentation in "Compilers" by -// Aho, Sethi and Ullman (the "dragon book"), pages 664-666. +// This file defines functionality for partitioning a CFG into intervals and +// building a weak topological order (WTO) of the nodes, based on the +// partitioning. The concepts and implementations for the graph partitioning +// are based on the presentation in "Compilers" by Aho, Sethi and Ullman (the +// "dragon book"), pages 664-666. The concepts around WTOs is taken from the +// paper "Efficient chaotic iteration strategies with widenings," by +// F. Bourdoncle ([Bourdoncle1993]). // //===----------------------------------------------------------------------===// @@ -16,34 +20,114 @@ #define LLVM_CLANG_ANALYSIS_ANALYSES_INTERVALPARTITION_H #include "clang/Analysis/CFG.h" +#include "llvm/ADT/ArrayRef.h" +#include "llvm/Support/raw_ostream.h" +#include +#include #include +#include #include namespace clang { // An interval is a strongly-connected component of the CFG along with a -// trailing acyclic structure. The _header_ of the interval is either the CFG -// entry block or has at least one predecessor outside of the interval. All -// other blocks in the interval have only predecessors also in the interval. +// trailing acyclic structure. An interval can be constructed directly from CFG +// blocks or from a graph of other intervals. The _header_ of the interval is +// either the graph's entry block or has at least one predecessor outside of the +// interval. All other blocks in the interval have only predecessors also in the +// interval. struct CFGInterval { - CFGInterval(const CFGBlock *Header) : Header(Header), Blocks({Header}) {} + CFGInterval() = default; - // The block from which the interval was constructed. Is either the CFG entry - // block or has at least one predecessor outside the interval. - const CFGBlock *Header; + template + CFGInterval(int ID, const Node *Header, std::vector Nodes) + : ID(ID), Data(NodeData{Header, std::move(Nodes)}) {} - std::set Blocks; + const std::set &preds() const { return Predecessors; } + const std::vector &succs() const { return Successors; } + + template struct NodeData { + // The block from which the interval was constructed. It is either the + // graph's entry block or has at least one predecessor outside the interval. + const Node *Header; + std::vector Nodes; + }; + using IntervalData = std::variant, NodeData>; + + int ID; + // Whether this node is the head of a feedback edge within the interval. + bool IsFeedbackHead = false; + IntervalData Data; + std::set Predecessors; // Successor blocks of the *interval*: blocks outside the interval for // reachable (in one edge) from within the interval. - std::set Successors; + std::vector Successors; +}; + +struct CFGIntervalGraph { + std::vector Intervals; }; -CFGInterval buildInterval(const CFGBlock &Header); +std::vector buildInterval(const CFGBlock *Header); -// Partitions `Cfg` into intervals and constructs a graph of the intervals, +// Partitions `Cfg` into intervals and constructs a the graph of the intervals // based on the edges between nodes in these intervals. -std::vector partitionIntoIntervals(const CFG &Cfg); +CFGIntervalGraph partitionIntoIntervals(const CFG &Cfg); + +// (Further) partitions `Graph` into intervals and constructs a the graph of the +// intervals based on the edges between nodes (themselves intervals) in these +// intervals. +CFGIntervalGraph partitionIntoIntervals(const CFGIntervalGraph &Graph); + +/// A _weak topological ordering_ (WTO) of CFG nodes provides a total order over +/// the CFG (defined in `WTOCompare`, below), which can guide the order in which +/// to visit nodes in fixpoint computations over the CFG. +/// +/// Roughly, a WTO a) groups the blocks so that loop heads are grouped with +/// their bodies and any nodes they dominate after the loop and b) orders the +/// groups topologically. As a result, the blocks in a series of loops are +/// ordered such that all nodes in loop `i` are earlier in the order than nodes +/// in loop `j`. This ordering, when combined with widening, bounds the number +/// of times a node must be visited for a dataflow algorithm to reach a +/// fixpoint. For the precise definition of a WTO and its properties, see +/// [Bourdoncle1993]. +struct WeakTopologicalOrdering { + // `Elements` represents an ordered sequence and nested WTOs induce a + // hierarchy in the ordering. When built from a limit flow graph, the first + // element of a WTO corresponds to an interval header, which is usually a loop + // head (except for the CFG entry block). + std::vector> Elements; +}; + +/// Builds a WTO of the nodes in `I`, based on the structure of the intervals. +WeakTopologicalOrdering getWTO(const CFGInterval &I); + +/// Builds a WTO from the limit flow graph of `Cfg` (derived from iteratively +/// partitioning it into intervals) if and only if it is reducible (its limit +/// flow graph has one node). Returns `nullop` when `Cfg` is not reducible. +/// +/// This WTO construction is described in Section 4.2 of [Bourdoncle1993]. +std::optional getIntervalWTO(const CFG &Cfg); + +/// Formats `WTO` as a human-readable string. +std::string formatWTO(const WeakTopologicalOrdering &WTO); + +struct WTOCompare { + WTOCompare(const WeakTopologicalOrdering &WTO); + + bool operator()(const CFGBlock *B1, const CFGBlock *B2) const { + auto It1 = BlockOrder.find(B1); + auto It2 = BlockOrder.find(B2); + + unsigned V1 = (It1 == BlockOrder.end()) ? 0 : It1->second; + unsigned V2 = (It2 == BlockOrder.end()) ? 0 : It2->second; + return V1 > V2; + } + + using BlockOrderTy = llvm::DenseMap; + BlockOrderTy BlockOrder; +}; } // namespace clang diff --git a/clang/lib/Analysis/IntervalPartition.cpp b/clang/lib/Analysis/IntervalPartition.cpp --- a/clang/lib/Analysis/IntervalPartition.cpp +++ b/clang/lib/Analysis/IntervalPartition.cpp @@ -12,20 +12,45 @@ #include "clang/Analysis/Analyses/IntervalPartition.h" #include "clang/Analysis/CFG.h" +#include "llvm/Support/ErrorHandling.h" +#include #include #include #include namespace clang { -static CFGInterval buildInterval(std::set &Partitioned, - const CFGBlock &Header) { - CFGInterval Interval(&Header); - Partitioned.insert(&Header); +namespace { +template using NodeData = CFGInterval::NodeData; +} // namespace - std::queue Worklist; - for (const CFGBlock *S : Header.succs()) - Worklist.push(S); +// Intermediate data used in constructing a CFGIntervalNode. +template struct BuildResult { + // Use a vector to maintain the insertion order. Given the expected small + // number of nodes, vector should be sufficiently efficient. + std::vector Nodes; + std::set Successors; +}; + +template +// bool inInterval(const Node *N, llvm::ArrayRef Interval) { +bool inInterval(const Node *N, std::vector &Interval) { + return std::find(Interval.begin(), Interval.end(), N) != Interval.end(); +} + +// Requires: `Node::succs()` and `Node::preds()`. +template +BuildResult buildInterval(std::set &Partitioned, + const Node *Header) { + assert(Header); + BuildResult Interval; + Interval.Nodes.push_back(Header); + Partitioned.insert(Header); + + std::queue Worklist; + for (const Node *S : Header->succs()) + if (S != nullptr) + Worklist.push(S); // Contains successors of blocks in the interval that couldn't be added to the // interval on their first encounter. This occurs when they have a predecessor @@ -34,7 +59,7 @@ // from the interval. At the end of processing the worklist, we filter out any // that ended up in the interval to produce the output set of interval // successors. - std::vector MaybeSuccessors; + std::vector MaybeSuccessors; while (!Worklist.empty()) { const auto *B = Worklist.front(); @@ -46,42 +71,56 @@ // Check whether all predecessors are in the interval, in which case `B` // is included as well. bool AllInInterval = true; - for (const CFGBlock *P : B->preds()) - if (Interval.Blocks.find(P) == Interval.Blocks.end()) { + for (const Node *P : B->preds()) + if (!inInterval(P, Interval.Nodes)) { MaybeSuccessors.push_back(B); AllInInterval = false; break; } if (AllInInterval) { - Interval.Blocks.insert(B); + Interval.Nodes.push_back(B); Partitioned.insert(B); - for (const CFGBlock *S : B->succs()) - Worklist.push(S); + for (const Node *S : B->succs()) + if (S != nullptr) + Worklist.push(S); } } // Any block successors not in the current interval are interval successors. - for (const CFGBlock *B : MaybeSuccessors) - if (Interval.Blocks.find(B) == Interval.Blocks.end()) + for (const Node *B : MaybeSuccessors) + if (!inInterval(B, Interval.Nodes)) Interval.Successors.insert(B); return Interval; } -CFGInterval buildInterval(const CFGBlock &Header) { - std::set Partitioned; - return buildInterval(Partitioned, Header); +template +void addIntervalNode(CFGIntervalGraph &Graph, + std::map &Index, + std::queue &Successors, + std::set &Partitioned, const Node *Header) { + BuildResult Data = buildInterval(Partitioned, Header); + for (const auto *S : Data.Successors) + Successors.push(S); + + // Index the nodes of the new interval. + int ID = Graph.Intervals.size(); + for (const auto *N : Data.Nodes) + Index.emplace(N, ID); + Graph.Intervals.emplace_back(ID, Header, std::move(Data.Nodes)); } -std::vector partitionIntoIntervals(const CFG &Cfg) { - std::vector Intervals; - std::set Partitioned; - auto &EntryBlock = Cfg.getEntry(); - Intervals.push_back(buildInterval(Partitioned, EntryBlock)); +// TODO: for Node = CFGInterval, we should collapse single-node intervals. +template +CFGIntervalGraph partitionIntoIntervalsImpl(const Node *EntryBlock) { + assert(EntryBlock != nullptr); + CFGIntervalGraph Graph; + // Records the ID of the header node associated with each node in the graph. + std::map Index; + std::set Partitioned; + std::queue Successors; - std::queue Successors; - for (const auto *S : Intervals[0].Successors) - Successors.push(S); + addIntervalNode(Graph, Index, Successors, Partitioned, EntryBlock); while (!Successors.empty()) { const auto *B = Successors.front(); @@ -89,14 +128,149 @@ if (Partitioned.find(B) != Partitioned.end()) continue; - // B has not been partitioned, but it has a predecessor that has. - CFGInterval I = buildInterval(Partitioned, *B); - for (const auto *S : I.Successors) - Successors.push(S); - Intervals.push_back(std::move(I)); + // B has not been partitioned, but it has a predecessor that has. Create a + // new interval from `B`. + addIntervalNode(Graph, Index, Successors, Partitioned, B); + } + + // Go back and patch up all the Intervals -- the successors and predecessors. + for (auto &I : Graph.Intervals) { + assert(std::holds_alternative>(I.Data)); + auto &Data = std::get>(I.Data); + // All external successors must be interval headers, because they + // have predecessors in multiple intervals. + for (const Node *N : Data.Header->preds()) { + auto It = Index.find(N); + if (It == Index.end()) + // Unreachable block. + continue; + if (It->second == I.ID) + // This is a feedback edge, so mark `I` the head of this edge. + I.IsFeedbackHead = true; + else if (I.Predecessors.insert(&Graph.Intervals[It->second]).second) + Graph.Intervals[It->second].Successors.push_back(&I); + } } - return Intervals; + return Graph; +} + +std::vector buildInterval(const CFGBlock *Header) { + std::set Partitioned; + return buildInterval(Partitioned, Header).Nodes; +} + +CFGIntervalGraph partitionIntoIntervals(const CFG &Cfg) { + return partitionIntoIntervalsImpl(&Cfg.getEntry()); +} + +CFGIntervalGraph partitionIntoIntervals(const CFGIntervalGraph &Graph) { + return partitionIntoIntervalsImpl(&Graph.Intervals[0]); +} + +static void addToWTO(const CFGInterval &I, WeakTopologicalOrdering &Order); + +static void addToWTO(const CFGInterval::IntervalData &D, + WeakTopologicalOrdering &Order) { + std::visit( + [&Order](auto &&D) { + for (const auto *N : D.Nodes) { + if constexpr (std::is_same_v, + NodeData>) { + Order.Elements.emplace_back(N); + } else { + static_assert(std::is_same_v, + NodeData>); + addToWTO(*N, Order); + } + } + }, + D); } +static void addToWTO(const CFGInterval &I, WeakTopologicalOrdering &Order) { + if (I.IsFeedbackHead) { + WeakTopologicalOrdering Nested; + addToWTO(I.Data, Nested); + Order.Elements.push_back(std::move(Nested)); + return; + } + addToWTO(I.Data, Order); +} + +WeakTopologicalOrdering getWTO(const CFGInterval &I) { + WeakTopologicalOrdering Order; + addToWTO(I, Order); + return Order; +} + +std::optional getIntervalWTO(const CFG &Cfg) { + // Backing storage for the allocated nodes in each graph. + std::vector Graphs; + unsigned PrevSize = Cfg.size(); + Graphs.push_back(partitionIntoIntervals(Cfg)); + unsigned Size = Graphs.back().Intervals.size(); + while (Size > 1 && Size < PrevSize) { + PrevSize = Graphs.back().Intervals.size(); + Graphs.push_back(partitionIntoIntervals(Graphs.back())); + Size = Graphs.back().Intervals.size(); + } + if (Size > 1) + // Not reducible. + return std::nullopt; + + assert(Size != 0); + return getWTO(Graphs.back().Intervals[0]); +} + +static void printWTO(const WeakTopologicalOrdering &O, llvm::raw_ostream &OS) { + bool First = true; + for (auto &E : O.Elements) { + if (!First) + OS << " "; + else + First = false; + + std::visit( + [&OS](auto &&Value) { + if constexpr (std::is_same_v, + const CFGBlock *>) { + OS << Value->getBlockID(); + } else { + static_assert(std::is_same_v, + WeakTopologicalOrdering>); + OS << "("; + printWTO(Value, OS); + OS << ")"; + } + }, + E); + } +} + +std::string formatWTO(const WeakTopologicalOrdering &O) { + std::string Result; + llvm::raw_string_ostream OS(Result); + printWTO(O, OS); + return Result; +} + +static void initWithWTO(llvm::DenseMap &BlockOrder, + const WeakTopologicalOrdering &WTO) { + for (auto &E : WTO.Elements) { + if (std::holds_alternative(E)) { + initWithWTO(BlockOrder, std::get(E)); + continue; + } + const auto *B = std::get(E); + // Calculate J first so we don't depend on order of evaluation of the + // assignment's operands. + auto J = BlockOrder.size() + 1; + BlockOrder[B] = J; + } +} + +WTOCompare::WTOCompare(const WeakTopologicalOrdering &WTO) { + initWithWTO(BlockOrder, WTO); +} } // namespace clang diff --git a/clang/unittests/Analysis/IntervalPartitionTest.cpp b/clang/unittests/Analysis/IntervalPartitionTest.cpp --- a/clang/unittests/Analysis/IntervalPartitionTest.cpp +++ b/clang/unittests/Analysis/IntervalPartitionTest.cpp @@ -8,13 +8,119 @@ #include "clang/Analysis/Analyses/IntervalPartition.h" #include "CFGBuildResult.h" +#include "clang/Analysis/CFG.h" +#include "llvm/Support/raw_ostream.h" #include "gmock/gmock.h" #include "gtest/gtest.h" +#include +#include namespace clang { -namespace analysis { + +namespace { +template using NodeData = CFGInterval::NodeData; +} // namespace + +llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, + const CFGInterval::IntervalData &D) { + OS << "Nodes{"; + if (std::holds_alternative>(D)) { + auto &BlockData = std::get>(D); + for (const auto *B : BlockData.Nodes) + OS << B->getBlockID() << ", "; + } else { + assert(std::holds_alternative>(D)); + auto &IntervalData = std::get>(D); + for (const auto *B : IntervalData.Nodes) + OS << B->ID << ", "; + } + OS << "}"; + return OS; +} + +llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, const CFGInterval &I) { + OS << "Interval{ID = " << I.ID << ", "; + OS << I.Data; + OS << ", Pre{"; + for (const auto *P : I.Predecessors) + OS << P->ID << ","; + OS << "}, Succ{"; + for (const auto *P : I.Successors) + OS << P->ID << ","; + OS << "}}\n"; + return OS; +} + +void PrintTo(const CFGInterval::IntervalData &D, std::ostream *OS) { + std::string Result; + llvm::raw_string_ostream StringOS(Result); + StringOS << D; + *OS << Result; +} + +void PrintTo(const CFGInterval &I, std::ostream *OS) { + std::string Result; + llvm::raw_string_ostream StringOS(Result); + StringOS << I; + *OS << Result; +} + namespace { +using ::clang::analysis::BuildCFG; +using ::clang::analysis::BuildResult; +using ::testing::ElementsAre; +using ::testing::Field; +using ::testing::IsEmpty; +using ::testing::Optional; +using ::testing::UnorderedElementsAre; + +MATCHER_P(BlockID, ID, "") { return arg->getBlockID == ID; } +MATCHER_P(IntervalID, ID, "") { return arg->ID == ID; } + +template +auto BlockIDs(T... IDs) { + return UnorderedElementsAre( + ::testing::Property(&CFGBlock::getBlockID, IDs)...); +} + +template +auto IntervalIDs(T... IDs) { + return UnorderedElementsAre( + ::testing::Field(&CFGInterval::ID, IDs)...); +} + +MATCHER_P3(IsBlockInterval, ID, Preds, Succs, "") { + if (!std::holds_alternative>(arg.Data)) + return false; + + return testing::Matches(ID)(arg.ID) && + testing::Matches(Preds)(arg.Predecessors) && + testing::Matches(Succs)(arg.Successors); +} + +MATCHER_P4(IsBlockInterval, ID, Nodes, Preds, Succs, "") { + if (!std::holds_alternative>(arg.Data)) + return false; + auto &IntervalData = std::get>(arg.Data); + + return testing::Matches(ID)(arg.ID) && + testing::Matches(Nodes)(IntervalData.Nodes) && + testing::Matches(Preds)(arg.Predecessors) && + testing::Matches(Succs)(arg.Successors); +} + +MATCHER_P4(IsNestedInterval, ID, Nodes, Preds, Succs, "") { + if (!std::holds_alternative>(arg.Data)) + return false; + auto &IntervalData = std::get>(arg.Data); + + return testing::Matches(ID)(arg.ID) && + testing::Matches(Nodes)(IntervalData.Nodes) && + testing::Matches(Preds)(arg.Predecessors) && + testing::Matches(Succs)(arg.Successors); +} + TEST(BuildInterval, PartitionSimpleOneInterval) { const char *Code = R"(void f() { @@ -32,8 +138,8 @@ auto &EntryBlock = cfg->getEntry(); - CFGInterval I = buildInterval(EntryBlock); - EXPECT_EQ(I.Blocks.size(), 3u); + std::vector I = buildInterval(&EntryBlock); + EXPECT_EQ(I.size(), 3u); } TEST(BuildInterval, PartitionIfThenOneInterval) { @@ -56,12 +162,10 @@ auto &EntryBlock = cfg->getEntry(); - CFGInterval I = buildInterval(EntryBlock); - EXPECT_EQ(I.Blocks.size(), 6u); + std::vector I = buildInterval(&EntryBlock); + EXPECT_EQ(I.size(), 6u); } -using ::testing::UnorderedElementsAre; - TEST(BuildInterval, PartitionWhileMultipleIntervals) { const char *Code = R"(void f() { @@ -80,11 +184,11 @@ CFGBlock *InitXBlock = *EntryBlock->succ_begin(); CFGBlock *LoopHeadBlock = *InitXBlock->succ_begin(); - CFGInterval I1 = buildInterval(*EntryBlock); - EXPECT_THAT(I1.Blocks, UnorderedElementsAre(EntryBlock, InitXBlock)); + std::vector I1 = buildInterval(EntryBlock); + EXPECT_THAT(I1, UnorderedElementsAre(EntryBlock, InitXBlock)); - CFGInterval I2 = buildInterval(*LoopHeadBlock); - EXPECT_EQ(I2.Blocks.size(), 5u); + std::vector I2 = buildInterval(LoopHeadBlock); + EXPECT_EQ(I2.size(), 5u); } TEST(PartitionIntoIntervals, PartitionIfThenOneInterval) { @@ -99,11 +203,10 @@ BuildResult Result = BuildCFG(Code); ASSERT_EQ(BuildResult::BuiltCFG, Result.getStatus()); - CFG *cfg = Result.getCFG(); - ASSERT_EQ(cfg->size(), 6u); - - auto Intervals = partitionIntoIntervals(*cfg); - EXPECT_EQ(Intervals.size(), 1u); + auto Graph = partitionIntoIntervals(*Result.getCFG()); + EXPECT_EQ(Graph.Intervals.size(), 1u); + EXPECT_THAT(Graph.Intervals, + ElementsAre(IsBlockInterval(0, IsEmpty(), IsEmpty()))); } TEST(PartitionIntoIntervals, PartitionWhileTwoIntervals) { @@ -116,11 +219,12 @@ BuildResult Result = BuildCFG(Code); ASSERT_EQ(BuildResult::BuiltCFG, Result.getStatus()); - CFG *cfg = Result.getCFG(); - ASSERT_EQ(cfg->size(), 7u); - - auto Intervals = partitionIntoIntervals(*cfg); - EXPECT_EQ(Intervals.size(), 2u); + auto Graph = partitionIntoIntervals(*Result.getCFG()); + EXPECT_THAT( + Graph.Intervals, + ElementsAre( + IsBlockInterval(0, IsEmpty(), UnorderedElementsAre(IntervalID(1))), + IsBlockInterval(1, UnorderedElementsAre(IntervalID(0)), IsEmpty()))); } TEST(PartitionIntoIntervals, PartitionNestedWhileThreeIntervals) { @@ -136,9 +240,15 @@ BuildResult Result = BuildCFG(Code); ASSERT_EQ(BuildResult::BuiltCFG, Result.getStatus()); - CFG *cfg = Result.getCFG(); - auto Intervals = partitionIntoIntervals(*cfg); - EXPECT_EQ(Intervals.size(), 3u); + auto Graph = partitionIntoIntervals(*Result.getCFG()); + EXPECT_THAT( + Graph.Intervals, + ElementsAre( + IsBlockInterval(0, IsEmpty(), UnorderedElementsAre(IntervalID(1))), + IsBlockInterval(1, UnorderedElementsAre(IntervalID(0), IntervalID(2)), + UnorderedElementsAre(IntervalID(2))), + IsBlockInterval(2, UnorderedElementsAre(IntervalID(1)), + UnorderedElementsAre(IntervalID(1))))); } TEST(PartitionIntoIntervals, PartitionSequentialWhileThreeIntervals) { @@ -154,11 +264,103 @@ BuildResult Result = BuildCFG(Code); ASSERT_EQ(BuildResult::BuiltCFG, Result.getStatus()); - CFG *cfg = Result.getCFG(); - auto Intervals = partitionIntoIntervals(*cfg); - EXPECT_EQ(Intervals.size(), 3u); + auto Graph = partitionIntoIntervals(*Result.getCFG()); + EXPECT_THAT( + Graph.Intervals, + ElementsAre( + IsBlockInterval(0, IsEmpty(), UnorderedElementsAre(IntervalID(1))), + IsBlockInterval(1, UnorderedElementsAre(IntervalID(0)), + UnorderedElementsAre(IntervalID(2))), + IsBlockInterval(2, UnorderedElementsAre(IntervalID(1)), IsEmpty()))); +} + +TEST(PartitionIntoIntervals, LimitReducibleSequentialWhile) { + const char *Code = R"(void f() { + int x = 3; + while (x >= 3) { + --x; + } + x = x + x; + int y = x; + while (y > 0) --y; + })"; + BuildResult Result = BuildCFG(Code); + ASSERT_EQ(BuildResult::BuiltCFG, Result.getStatus()); + + auto Graph = partitionIntoIntervals(*Result.getCFG()); + ASSERT_THAT( + Graph.Intervals, + ElementsAre( + IsBlockInterval(0, IsEmpty(), UnorderedElementsAre(IntervalID(1))), + IsBlockInterval(1, UnorderedElementsAre(IntervalID(0)), + UnorderedElementsAre(IntervalID(2))), + IsBlockInterval(2, UnorderedElementsAre(IntervalID(1)), IsEmpty()))); + + auto Graph2 = partitionIntoIntervals(Graph); + EXPECT_THAT(Graph2.Intervals, + ElementsAre(IsNestedInterval(0, IntervalIDs(0, 1, 2), + IsEmpty(), IsEmpty()))); + EXPECT_EQ(formatWTO(getWTO(Graph2.Intervals[0])), "9 8 (7 6 4 5) (3 2 0 1)"); +} + +TEST(PartitionIntoIntervals, LimitReducibleNestedWhile) { + const char *Code = R"(void f() { + int x = 3; + while (x >= 3) { + --x; + int y = x; + while (y > 0) --y; + } + x = x + x; + })"; + BuildResult Result = BuildCFG(Code); + ASSERT_EQ(BuildResult::BuiltCFG, Result.getStatus()); + + auto Graph = partitionIntoIntervals(*Result.getCFG()); + ASSERT_THAT( + Graph.Intervals, + ElementsAre( + IsBlockInterval(0, BlockIDs(9, 8), IsEmpty(), + UnorderedElementsAre(IntervalID(1))), + IsBlockInterval(1, BlockIDs(7, 6, 1, 0), + UnorderedElementsAre(IntervalID(0), IntervalID(2)), + UnorderedElementsAre(IntervalID(2))), + IsBlockInterval(2, BlockIDs(5, 4, 3, 2), + UnorderedElementsAre(IntervalID(1)), + UnorderedElementsAre(IntervalID(1))))); + + auto Graph2 = partitionIntoIntervals(Graph); + EXPECT_THAT(Graph2.Intervals, + ElementsAre(IsNestedInterval(0, IntervalIDs(0), IsEmpty(), + UnorderedElementsAre(IntervalID(1))), + IsNestedInterval(1, IntervalIDs(1, 2), + UnorderedElementsAre(IntervalID(0)), + IsEmpty()))); + + auto Graph3 = partitionIntoIntervals(Graph2); + EXPECT_THAT(Graph3.Intervals, + ElementsAre(IsNestedInterval(0, IntervalIDs(0, 1), IsEmpty(), + IsEmpty()))); + EXPECT_EQ(formatWTO(getWTO(Graph3.Intervals[0])), "9 8 (7 6 1 0 (5 4 2 3))"); +} + +TEST(GetWTO, NestedWhile) { + const char *Code = R"(void f() { + int x = 3; + while (x >= 3) { + --x; + int y = x; + while (y > 0) --y; + } + x = x + x; + })"; + BuildResult Result = BuildCFG(Code); + ASSERT_EQ(BuildResult::BuiltCFG, Result.getStatus()); + + auto WTO = getIntervalWTO(*Result.getCFG()); + ASSERT_NE(WTO, std::nullopt); + ASSERT_EQ(formatWTO(*WTO), "9 8 (7 6 1 0 (5 4 2 3))"); } } // namespace -} // namespace analysis } // namespace clang