clonedPtr.h
1 /* Copyright (C) 2012-2020 IBM Corp.
2  * This program is Licensed under the Apache License, Version 2.0
3  * (the "License"); you may not use this file except in compliance
4  * with the License. You may obtain a copy of the License at
5  * http://www.apache.org/licenses/LICENSE-2.0
6  * Unless required by applicable law or agreed to in writing, software
7  * distributed under the License is distributed on an "AS IS" BASIS,
8  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
9  * See the License for the specific language governing permissions and
10  * limitations under the License. See accompanying LICENSE file.
11  */
12 #ifndef HELIB_CLONEDPTR_H
13 #define HELIB_CLONEDPTR_H
14 
22 /*
23  * This seems like a very useful template class for smart
24  * pointers with "deep cloning" semantics and class hierarchies.
25  *
26  * To work, a class needs a "clone" method that creates a new copy.
27  *
28  * NOTES:
29  *
30  * this did not compile...had to change the definition of the class clone<X>
31  *
32  * added boolean operators
33  *
34  * added assign operation set_ptr(X* p) removed get(), introduced public
35  * method get_ptr()... these names chosen to make it easier to hunt down
36  * "loopholes"
37  *
38  * added noclone<X> template
39  *
40  * Made "const"-ness apply to both the pointer and the object pointed to.
41  * For this kind of "deep copy" semantics, this makes the most sense.
42  *
43  * Made a corresponding class copied_ptr, with "shallow" cloning...
44  * To bad this can't really be done with templates... C++11 defines
45  * the concept of "template aliases", which would do the trick...
46  * unfortunately, gcc does not yet implement this...
47  *
48  * Changed the name "clone" to avoid conflicts...
49  *****/
50 
51 /* For ANSI-challenged compilers, you may want to
52  * #define NO_MEMBER_TEMPLATES or explicit */
53 
54 namespace helib {
55 
61 template <typename X>
63 {
64 public:
65  static X* apply(const X* x) { return x->clone(); }
66 };
67 
73 template <typename X>
75 {
76 public:
77  static X* apply(const X* x) { return new X(*x); }
78 };
79 
80 #ifndef NO_MEMBER_TEMPLATES
81 
82 #define CLONED_PTR_TEMPLATE_MEMBERS(CLONED_PTR_TYPE) \
83  \
84  template <typename Y> \
85  CLONED_PTR_TYPE(const CLONED_PTR_TYPE<Y>& r) \
86  { \
87  copy(r.ptr); \
88  } \
89  template <typename Y> \
90  CLONED_PTR_TYPE& operator=(const CLONED_PTR_TYPE<Y>& r) \
91  { \
92  if (this != &r) { \
93  delete ptr; \
94  copy(r.ptr); \
95  } \
96  return *this; \
97  }
98 
99 #else
100 
101 #define CLONED_PTR_TEMPLATE_MEMBERS(CLONED_PTR_TYPE)
102 
103 #endif
104 
105 #define CLONED_PTR_DECLARE(CLONED_PTR_TYPE, CLONED_PTR_INIT) \
106  \
107  template <typename X, typename Cloner = CLONED_PTR_INIT<X>> \
108  class CLONED_PTR_TYPE \
109  { \
110  public: \
111  typedef X element_type; \
112  \
113  explicit CLONED_PTR_TYPE(X* p = 0) : ptr(p) {} \
114  ~CLONED_PTR_TYPE() { delete ptr; } \
115  CLONED_PTR_TYPE(const CLONED_PTR_TYPE& r) { copy(r.ptr); } \
116  \
117  CLONED_PTR_TYPE& operator=(const CLONED_PTR_TYPE& r) \
118  { \
119  if (this != &r) { \
120  delete ptr; \
121  copy(r.ptr); \
122  } \
123  return *this; \
124  } \
125  \
126  void set_ptr(X* p) \
127  { \
128  if (ptr != p) { \
129  delete ptr; \
130  ptr = p; \
131  } \
132  } \
133  \
134  CLONED_PTR_TEMPLATE_MEMBERS(CLONED_PTR_TYPE) \
135  \
136  const X& operator*() const { return *ptr; } \
137  X& operator*() { return *ptr; } \
138  \
139  const X* operator->() const { return ptr; } \
140  X* operator->() { return ptr; } \
141  \
142  bool null() const { return ptr == 0; } \
143  \
144  const X* get_ptr() const { return ptr; } \
145  X* get_ptr() { return ptr; } \
146  \
147  void swap(CLONED_PTR_TYPE& r) \
148  { \
149  X* tmp; \
150  tmp = r.ptr; \
151  r.ptr = ptr; \
152  ptr = tmp; \
153  } \
154  \
155  private: \
156  X* ptr; \
157  \
158  void copy(X* p) { ptr = (p ? Cloner::apply(p) : 0); } \
159  };
160 
161 // declare the template class cloned_ptr<X>
162 CLONED_PTR_DECLARE(cloned_ptr, deep_clone)
163 
164 // declare the template class copied_ptr<X>
165 CLONED_PTR_DECLARE(copied_ptr, shallow_clone)
166 
167 // template <class X> using copied_ptr = cloned_ptr<X, noclone<X> >;
168 
169 template <typename X, typename Cloner>
170 void swap(cloned_ptr<X, Cloner>& x, cloned_ptr<X, Cloner>& y)
171 {
172  x.swap(y);
173 }
174 
175 template <typename X, typename Cloner>
176 void swap(copied_ptr<X, Cloner>& x, copied_ptr<X, Cloner>& y)
177 {
178  x.swap(y);
179 }
180 
181 } // namespace helib
182 
183 #endif // ifndef HELIB_CLONEDPTR_H
static X * apply(const X *x)
Definition: clonedPtr.h:65
Definition: apiAttributes.h:21
Deep copy: initialize with clone.
Definition: clonedPtr.h:63
static X * apply(const X *x)
Definition: clonedPtr.h:77
Shallow copy: initialize with copy constructor.
Definition: clonedPtr.h:75
void swap(cloned_ptr< X, Cloner > &x, cloned_ptr< X, Cloner > &y)
Definition: clonedPtr.h:170