summaryrefslogtreecommitdiff
path: root/unittests/ADT/polymorphic_ptr_test.cpp
blob: bd5d83879a83fa5cb419f83808e5bbf01c7523f7 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
//===- llvm/unittest/ADT/polymorphic_ptr.h - polymorphic_ptr<T> tests -----===//
//
//                     The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//

#include "gtest/gtest.h"
#include "llvm/ADT/polymorphic_ptr.h"
#include "llvm/Support/raw_ostream.h"

using namespace llvm;

namespace {

struct S {
  S(int x) : x(x) {}
  S *clone() { return new S(*this); }
  int x;
};

// A function that forces the return of a copy.
template <typename T>
T dummy_copy(const T &arg) { return arg; }

TEST(polymorphic_ptr_test, Basic) {
  polymorphic_ptr<S> null;
  EXPECT_FALSE((bool)null);
  EXPECT_TRUE(!null);
  EXPECT_EQ((S*)0, null.get());

  S *s = new S(42);
  polymorphic_ptr<S> p(s);
  EXPECT_TRUE((bool)p);
  EXPECT_FALSE(!p);
  EXPECT_TRUE(p != null);
  EXPECT_FALSE(p == null);
  EXPECT_TRUE(p == s);
  EXPECT_TRUE(s == p);
  EXPECT_FALSE(p != s);
  EXPECT_FALSE(s != p);
  EXPECT_EQ(s, &*p);
  EXPECT_EQ(s, p.operator->());
  EXPECT_EQ(s, p.get());
  EXPECT_EQ(42, p->x);

  EXPECT_EQ(s, p.take());
  EXPECT_FALSE((bool)p);
  EXPECT_TRUE(!p);
  p = s;
  EXPECT_TRUE((bool)p);
  EXPECT_FALSE(!p);
  EXPECT_EQ(s, &*p);
  EXPECT_EQ(s, p.operator->());
  EXPECT_EQ(s, p.get());
  EXPECT_EQ(42, p->x);

  polymorphic_ptr<S> p2((llvm_move(p)));
#if !LLVM_HAS_RVALUE_REFERENCES
  // 'p' may not have been moved from in C++98, fake it for the test.
  p2 = p.take();
#endif
  EXPECT_FALSE((bool)p);
  EXPECT_TRUE(!p);
  EXPECT_TRUE((bool)p2);
  EXPECT_FALSE(!p2);
  EXPECT_EQ(s, &*p2);

  using std::swap;
  swap(p, p2);
  EXPECT_TRUE((bool)p);
  EXPECT_FALSE(!p);
  EXPECT_EQ(s, &*p);
  EXPECT_FALSE((bool)p2);
  EXPECT_TRUE(!p2);

  // Force copies and that everything survives.
  polymorphic_ptr<S> p3 = dummy_copy(polymorphic_ptr<S>(p));
  EXPECT_TRUE((bool)p3);
  EXPECT_FALSE(!p3);
  EXPECT_NE(s, &*p3);
  EXPECT_EQ(42, p3->x);

  // Force copies of null without trying to dereference anything.
  polymorphic_ptr<S> null_copy = dummy_copy(polymorphic_ptr<S>(null));
  EXPECT_FALSE((bool)null_copy);
  EXPECT_TRUE(!null_copy);
  EXPECT_EQ(null, null_copy);
}

struct Base {
  virtual ~Base() {}
  virtual Base *clone() = 0;
  virtual StringRef name() { return "Base"; }
};

struct DerivedA : Base {
  virtual DerivedA *clone() { return new DerivedA(); }
  virtual StringRef name() { return "DerivedA"; }
};
struct DerivedB : Base {
  virtual DerivedB *clone() { return new DerivedB(); }
  virtual StringRef name() { return "DerivedB"; }
};

TEST(polymorphic_ptr_test, Polymorphism) {
  polymorphic_ptr<Base> a(new DerivedA());
  polymorphic_ptr<Base> b(new DerivedB());

  EXPECT_EQ("DerivedA", a->name());
  EXPECT_EQ("DerivedB", b->name());

  polymorphic_ptr<Base> copy = dummy_copy(a);
  EXPECT_NE(a, copy);
  EXPECT_EQ("DerivedA", copy->name());

  copy = dummy_copy(b);
  EXPECT_NE(b, copy);
  EXPECT_EQ("DerivedB", copy->name());

  // Test creating a copy out of a temporary directly.
  copy = dummy_copy<polymorphic_ptr<Base> >(new DerivedA());
  EXPECT_NE(a, copy);
  EXPECT_EQ("DerivedA", copy->name());
}

}