xref: /llvm-project/clang/unittests/Analysis/FlowSensitive/MapLatticeTest.cpp (revision 29d35ece8249f2d8a51437a5c008e6bf63da9874)
14dcc47aaSYitzhak Mandelbaum #include "clang/Analysis/FlowSensitive/MapLattice.h"
24dcc47aaSYitzhak Mandelbaum #include "clang/Analysis/FlowSensitive/DataflowLattice.h"
34dcc47aaSYitzhak Mandelbaum #include "llvm/Support/Error.h"
44dcc47aaSYitzhak Mandelbaum #include "gmock/gmock.h"
54dcc47aaSYitzhak Mandelbaum #include "gtest/gtest.h"
64dcc47aaSYitzhak Mandelbaum #include <ostream>
74dcc47aaSYitzhak Mandelbaum 
84dcc47aaSYitzhak Mandelbaum using namespace clang;
94dcc47aaSYitzhak Mandelbaum using namespace dataflow;
104dcc47aaSYitzhak Mandelbaum 
114dcc47aaSYitzhak Mandelbaum namespace {
124dcc47aaSYitzhak Mandelbaum // A simple lattice for basic tests.
134dcc47aaSYitzhak Mandelbaum class BooleanLattice {
144dcc47aaSYitzhak Mandelbaum public:
BooleanLattice()154dcc47aaSYitzhak Mandelbaum   BooleanLattice() : Value(false) {}
BooleanLattice(bool B)164dcc47aaSYitzhak Mandelbaum   explicit BooleanLattice(bool B) : Value(B) {}
174dcc47aaSYitzhak Mandelbaum 
bottom()184dcc47aaSYitzhak Mandelbaum   static BooleanLattice bottom() { return BooleanLattice(false); }
194dcc47aaSYitzhak Mandelbaum 
top()204dcc47aaSYitzhak Mandelbaum   static BooleanLattice top() { return BooleanLattice(true); }
214dcc47aaSYitzhak Mandelbaum 
join(BooleanLattice Other)224dcc47aaSYitzhak Mandelbaum   LatticeJoinEffect join(BooleanLattice Other) {
234dcc47aaSYitzhak Mandelbaum     auto Prev = Value;
244dcc47aaSYitzhak Mandelbaum     Value = Value || Other.Value;
254dcc47aaSYitzhak Mandelbaum     return Prev == Value ? LatticeJoinEffect::Unchanged
264dcc47aaSYitzhak Mandelbaum                          : LatticeJoinEffect::Changed;
274dcc47aaSYitzhak Mandelbaum   }
284dcc47aaSYitzhak Mandelbaum 
operator ==(BooleanLattice LHS,BooleanLattice RHS)294dcc47aaSYitzhak Mandelbaum   friend bool operator==(BooleanLattice LHS, BooleanLattice RHS) {
304dcc47aaSYitzhak Mandelbaum     return LHS.Value == RHS.Value;
314dcc47aaSYitzhak Mandelbaum   }
324dcc47aaSYitzhak Mandelbaum 
operator !=(BooleanLattice LHS,BooleanLattice RHS)334dcc47aaSYitzhak Mandelbaum   friend bool operator!=(BooleanLattice LHS, BooleanLattice RHS) {
344dcc47aaSYitzhak Mandelbaum     return LHS.Value != RHS.Value;
354dcc47aaSYitzhak Mandelbaum   }
364dcc47aaSYitzhak Mandelbaum 
operator <<(std::ostream & Os,const BooleanLattice & B)374dcc47aaSYitzhak Mandelbaum   friend std::ostream &operator<<(std::ostream &Os, const BooleanLattice &B) {
384dcc47aaSYitzhak Mandelbaum     Os << B.Value;
394dcc47aaSYitzhak Mandelbaum     return Os;
404dcc47aaSYitzhak Mandelbaum   }
414dcc47aaSYitzhak Mandelbaum 
value() const424dcc47aaSYitzhak Mandelbaum   bool value() const { return Value; }
434dcc47aaSYitzhak Mandelbaum 
444dcc47aaSYitzhak Mandelbaum private:
454dcc47aaSYitzhak Mandelbaum   bool Value;
464dcc47aaSYitzhak Mandelbaum };
474dcc47aaSYitzhak Mandelbaum } // namespace
484dcc47aaSYitzhak Mandelbaum 
494dcc47aaSYitzhak Mandelbaum static constexpr int Key1 = 0;
504dcc47aaSYitzhak Mandelbaum static constexpr int Key2 = 1;
514dcc47aaSYitzhak Mandelbaum 
524dcc47aaSYitzhak Mandelbaum namespace {
53*29d35eceSEric Li using ::testing::_;
544dcc47aaSYitzhak Mandelbaum using ::testing::Pair;
554dcc47aaSYitzhak Mandelbaum using ::testing::UnorderedElementsAre;
564dcc47aaSYitzhak Mandelbaum 
TEST(MapLatticeTest,InsertWorks)574dcc47aaSYitzhak Mandelbaum TEST(MapLatticeTest, InsertWorks) {
584dcc47aaSYitzhak Mandelbaum   MapLattice<int, BooleanLattice> Lattice;
59*29d35eceSEric Li   EXPECT_THAT(Lattice.insert({Key1, BooleanLattice(false)}), Pair(_, true));
60*29d35eceSEric Li   EXPECT_THAT(Lattice.insert({Key2, BooleanLattice(false)}), Pair(_, true));
61*29d35eceSEric Li 
62*29d35eceSEric Li   // Insertion fails on collision.
63*29d35eceSEric Li   EXPECT_THAT(Lattice.insert({Key1, BooleanLattice(false)}), Pair(_, false));
64*29d35eceSEric Li   EXPECT_THAT(Lattice.insert({Key2, BooleanLattice(false)}), Pair(_, false));
654dcc47aaSYitzhak Mandelbaum 
664dcc47aaSYitzhak Mandelbaum   EXPECT_THAT(Lattice, UnorderedElementsAre(Pair(Key1, BooleanLattice(false)),
674dcc47aaSYitzhak Mandelbaum                                             Pair(Key2, BooleanLattice(false))));
684dcc47aaSYitzhak Mandelbaum }
694dcc47aaSYitzhak Mandelbaum 
TEST(MapLatticeTest,ComparisonWorks)704dcc47aaSYitzhak Mandelbaum TEST(MapLatticeTest, ComparisonWorks) {
714dcc47aaSYitzhak Mandelbaum   MapLattice<int, BooleanLattice> Lattice1;
724dcc47aaSYitzhak Mandelbaum   Lattice1.insert({Key1, BooleanLattice(true)});
734dcc47aaSYitzhak Mandelbaum   Lattice1.insert({Key2, BooleanLattice(false)});
744dcc47aaSYitzhak Mandelbaum   MapLattice<int, BooleanLattice> Lattice2 = Lattice1;
754dcc47aaSYitzhak Mandelbaum   EXPECT_EQ(Lattice1, Lattice2);
764dcc47aaSYitzhak Mandelbaum 
774dcc47aaSYitzhak Mandelbaum   Lattice2.find(Key2)->second = BooleanLattice(true);
784dcc47aaSYitzhak Mandelbaum   EXPECT_NE(Lattice1, Lattice2);
794dcc47aaSYitzhak Mandelbaum }
804dcc47aaSYitzhak Mandelbaum 
TEST(MapLatticeTest,JoinChange)814dcc47aaSYitzhak Mandelbaum TEST(MapLatticeTest, JoinChange) {
824dcc47aaSYitzhak Mandelbaum   MapLattice<int, BooleanLattice> Lattice1;
834dcc47aaSYitzhak Mandelbaum   Lattice1.insert({Key1, BooleanLattice(false)});
844dcc47aaSYitzhak Mandelbaum   Lattice1.insert({Key2, BooleanLattice(false)});
854dcc47aaSYitzhak Mandelbaum 
864dcc47aaSYitzhak Mandelbaum   MapLattice<int, BooleanLattice> Lattice2;
874dcc47aaSYitzhak Mandelbaum   Lattice2.insert({Key1, BooleanLattice(true)});
884dcc47aaSYitzhak Mandelbaum   Lattice2.insert({Key2, BooleanLattice(true)});
894dcc47aaSYitzhak Mandelbaum 
904dcc47aaSYitzhak Mandelbaum   ASSERT_THAT(Lattice1,
914dcc47aaSYitzhak Mandelbaum               UnorderedElementsAre(Pair(Key1, BooleanLattice(false)),
924dcc47aaSYitzhak Mandelbaum                                    Pair(Key2, BooleanLattice(false))));
934dcc47aaSYitzhak Mandelbaum 
944dcc47aaSYitzhak Mandelbaum   ASSERT_EQ(Lattice1.join(Lattice2), LatticeJoinEffect::Changed);
954dcc47aaSYitzhak Mandelbaum   EXPECT_THAT(Lattice1, UnorderedElementsAre(Pair(Key1, BooleanLattice(true)),
964dcc47aaSYitzhak Mandelbaum                                              Pair(Key2, BooleanLattice(true))));
974dcc47aaSYitzhak Mandelbaum }
984dcc47aaSYitzhak Mandelbaum 
TEST(MapLatticeTest,JoinEqNoChange)994dcc47aaSYitzhak Mandelbaum TEST(MapLatticeTest, JoinEqNoChange) {
1004dcc47aaSYitzhak Mandelbaum   MapLattice<int, BooleanLattice> Lattice;
1014dcc47aaSYitzhak Mandelbaum   Lattice.insert({Key1, BooleanLattice(false)});
1024dcc47aaSYitzhak Mandelbaum   Lattice.insert({Key2, BooleanLattice(false)});
1034dcc47aaSYitzhak Mandelbaum 
1044dcc47aaSYitzhak Mandelbaum   ASSERT_EQ(Lattice.join(Lattice), LatticeJoinEffect::Unchanged);
1054dcc47aaSYitzhak Mandelbaum   EXPECT_THAT(Lattice, UnorderedElementsAre(Pair(Key1, BooleanLattice(false)),
1064dcc47aaSYitzhak Mandelbaum                                             Pair(Key2, BooleanLattice(false))));
1074dcc47aaSYitzhak Mandelbaum }
1084dcc47aaSYitzhak Mandelbaum 
TEST(MapLatticeTest,JoinLtNoChange)1094dcc47aaSYitzhak Mandelbaum TEST(MapLatticeTest, JoinLtNoChange) {
1104dcc47aaSYitzhak Mandelbaum   MapLattice<int, BooleanLattice> Lattice1;
1114dcc47aaSYitzhak Mandelbaum   Lattice1.insert({Key1, BooleanLattice(false)});
1124dcc47aaSYitzhak Mandelbaum   Lattice1.insert({Key2, BooleanLattice(false)});
1134dcc47aaSYitzhak Mandelbaum 
1144dcc47aaSYitzhak Mandelbaum   MapLattice<int, BooleanLattice> Lattice2;
1154dcc47aaSYitzhak Mandelbaum   Lattice2.insert({Key1, BooleanLattice(true)});
1164dcc47aaSYitzhak Mandelbaum   Lattice2.insert({Key2, BooleanLattice(true)});
1174dcc47aaSYitzhak Mandelbaum 
1184dcc47aaSYitzhak Mandelbaum   ASSERT_THAT(Lattice1,
1194dcc47aaSYitzhak Mandelbaum               UnorderedElementsAre(Pair(Key1, BooleanLattice(false)),
1204dcc47aaSYitzhak Mandelbaum                                    Pair(Key2, BooleanLattice(false))));
1214dcc47aaSYitzhak Mandelbaum 
1224dcc47aaSYitzhak Mandelbaum   ASSERT_THAT(Lattice2, UnorderedElementsAre(Pair(Key1, BooleanLattice(true)),
1234dcc47aaSYitzhak Mandelbaum                                              Pair(Key2, BooleanLattice(true))));
1244dcc47aaSYitzhak Mandelbaum 
1254dcc47aaSYitzhak Mandelbaum   ASSERT_EQ(Lattice2.join(Lattice1), LatticeJoinEffect::Unchanged);
1264dcc47aaSYitzhak Mandelbaum   EXPECT_THAT(Lattice2, UnorderedElementsAre(Pair(Key1, BooleanLattice(true)),
1274dcc47aaSYitzhak Mandelbaum                                              Pair(Key2, BooleanLattice(true))));
1284dcc47aaSYitzhak Mandelbaum }
1294dcc47aaSYitzhak Mandelbaum 
TEST(MapLatticeTest,JoinDifferentDomainsProducesUnion)1304dcc47aaSYitzhak Mandelbaum TEST(MapLatticeTest, JoinDifferentDomainsProducesUnion) {
1314dcc47aaSYitzhak Mandelbaum   MapLattice<int, BooleanLattice> Lattice1;
1324dcc47aaSYitzhak Mandelbaum   Lattice1.insert({Key1, BooleanLattice(true)});
1334dcc47aaSYitzhak Mandelbaum   MapLattice<int, BooleanLattice> Lattice2;
1344dcc47aaSYitzhak Mandelbaum   Lattice2.insert({Key2, BooleanLattice(true)});
1354dcc47aaSYitzhak Mandelbaum 
1364dcc47aaSYitzhak Mandelbaum   ASSERT_EQ(Lattice1.join(Lattice2), LatticeJoinEffect::Changed);
1374dcc47aaSYitzhak Mandelbaum   EXPECT_THAT(Lattice1, UnorderedElementsAre(Pair(Key1, BooleanLattice(true)),
1384dcc47aaSYitzhak Mandelbaum                                              Pair(Key2, BooleanLattice(true))));
1394dcc47aaSYitzhak Mandelbaum }
1404dcc47aaSYitzhak Mandelbaum 
TEST(MapLatticeTest,FindWorks)1414dcc47aaSYitzhak Mandelbaum TEST(MapLatticeTest, FindWorks) {
1424dcc47aaSYitzhak Mandelbaum   MapLattice<int, BooleanLattice> Lattice;
1434dcc47aaSYitzhak Mandelbaum   Lattice.insert({Key1, BooleanLattice(true)});
1444dcc47aaSYitzhak Mandelbaum   Lattice.insert({Key2, BooleanLattice(false)});
1454dcc47aaSYitzhak Mandelbaum 
1464dcc47aaSYitzhak Mandelbaum   auto It = Lattice.find(Key1);
1474dcc47aaSYitzhak Mandelbaum   ASSERT_NE(It, Lattice.end());
1484dcc47aaSYitzhak Mandelbaum   EXPECT_EQ(It->second, BooleanLattice(true));
1494dcc47aaSYitzhak Mandelbaum 
1504dcc47aaSYitzhak Mandelbaum   It = Lattice.find(Key2);
1514dcc47aaSYitzhak Mandelbaum   ASSERT_NE(It, Lattice.end());
1524dcc47aaSYitzhak Mandelbaum   EXPECT_EQ(It->second, BooleanLattice(false));
1534dcc47aaSYitzhak Mandelbaum }
1544dcc47aaSYitzhak Mandelbaum 
TEST(MapLatticeTest,ContainsWorks)1554dcc47aaSYitzhak Mandelbaum TEST(MapLatticeTest, ContainsWorks) {
1564dcc47aaSYitzhak Mandelbaum   MapLattice<int, BooleanLattice> Lattice;
1574dcc47aaSYitzhak Mandelbaum   Lattice.insert({Key1, BooleanLattice(true)});
1584dcc47aaSYitzhak Mandelbaum   EXPECT_TRUE(Lattice.contains(Key1));
1594dcc47aaSYitzhak Mandelbaum   EXPECT_FALSE(Lattice.contains(Key2));
1604dcc47aaSYitzhak Mandelbaum }
1614dcc47aaSYitzhak Mandelbaum } // namespace
162