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