1 //===- llvm/unittest/ADT/PointerIntPairTest.cpp - Unit tests --------------===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 
10 #include "llvm/ADT/PointerIntPair.h"
11 #include "gtest/gtest.h"
12 #include <limits>
13 using namespace llvm;
14 
15 namespace {
16 
TEST(PointerIntPairTest,GetSet)17 TEST(PointerIntPairTest, GetSet) {
18   struct S {
19     int i;
20   };
21   S s;
22 
23   PointerIntPair<S *, 2> Pair(&s, 1U);
24   EXPECT_EQ(&s, Pair.getPointer());
25   EXPECT_EQ(1U, Pair.getInt());
26 
27   Pair.setInt(2);
28   EXPECT_EQ(&s, Pair.getPointer());
29   EXPECT_EQ(2U, Pair.getInt());
30 
31   Pair.setPointer(nullptr);
32   EXPECT_EQ(nullptr, Pair.getPointer());
33   EXPECT_EQ(2U, Pair.getInt());
34 
35   Pair.setPointerAndInt(&s, 3U);
36   EXPECT_EQ(&s, Pair.getPointer());
37   EXPECT_EQ(3U, Pair.getInt());
38 
39   // Make sure that we can perform all of our operations on enum classes.
40   //
41   // The concern is that enum classes are only explicitly convertible to
42   // integers. This means that if we assume in PointerIntPair this, a
43   // compilation error will result. This group of tests exercises the enum class
44   // code to make sure that we do not run into such issues in the future.
45   enum class E : unsigned {
46     Case1,
47     Case2,
48     Case3,
49   };
50   PointerIntPair<S *, 2, E> Pair2(&s, E::Case1);
51   EXPECT_EQ(&s, Pair2.getPointer());
52   EXPECT_EQ(E::Case1, Pair2.getInt());
53 
54   Pair2.setInt(E::Case2);
55   EXPECT_EQ(&s, Pair2.getPointer());
56   EXPECT_EQ(E::Case2, Pair2.getInt());
57 
58   Pair2.setPointer(nullptr);
59   EXPECT_EQ(nullptr, Pair2.getPointer());
60   EXPECT_EQ(E::Case2, Pair2.getInt());
61 
62   Pair2.setPointerAndInt(&s, E::Case3);
63   EXPECT_EQ(&s, Pair2.getPointer());
64   EXPECT_EQ(E::Case3, Pair2.getInt());
65 }
66 
TEST(PointerIntPairTest,DefaultInitialize)67 TEST(PointerIntPairTest, DefaultInitialize) {
68   PointerIntPair<float *, 2> Pair;
69   EXPECT_EQ(nullptr, Pair.getPointer());
70   EXPECT_EQ(0U, Pair.getInt());
71 }
72 
TEST(PointerIntPairTest,ManyUnusedBits)73 TEST(PointerIntPairTest, ManyUnusedBits) {
74   // In real code this would be a word-sized integer limited to 31 bits.
75   struct Fixnum31 {
76     uintptr_t Value;
77   };
78   class FixnumPointerTraits {
79   public:
80     static inline void *getAsVoidPointer(Fixnum31 Num) {
81       return reinterpret_cast<void *>(Num.Value << NumLowBitsAvailable);
82     }
83     static inline Fixnum31 getFromVoidPointer(void *P) {
84       // In real code this would assert that the value is in range.
85       return { reinterpret_cast<uintptr_t>(P) >> NumLowBitsAvailable };
86     }
87     enum { NumLowBitsAvailable = std::numeric_limits<uintptr_t>::digits - 31 };
88   };
89 
90   PointerIntPair<Fixnum31, 1, bool, FixnumPointerTraits> pair;
91   EXPECT_EQ((uintptr_t)0, pair.getPointer().Value);
92   EXPECT_FALSE(pair.getInt());
93 
94   pair.setPointerAndInt({ 0x7FFFFFFF }, true );
95   EXPECT_EQ((uintptr_t)0x7FFFFFFF, pair.getPointer().Value);
96   EXPECT_TRUE(pair.getInt());
97 
98   EXPECT_EQ(FixnumPointerTraits::NumLowBitsAvailable - 1,
99             PointerLikeTypeTraits<decltype(pair)>::NumLowBitsAvailable);
100 }
101 
102 } // end anonymous namespace
103