containers/smart_ptr.hpp

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
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
#ifndef STLPLUS_SMART_PTR
#define STLPLUS_SMART_PTR
////////////////////////////////////////////////////////////////////////////////
 
//   Author:    Andy Rushton
//   Copyright: (c) Southampton University 1999-2004
//              (c) Andy Rushton           2004 onwards
//   License:   BSD License, see ../docs/license.html
 
//   A smart pointer is a memory-managing pointer to an object. If you like, it
//   is a zero-dimensional container.
 
//   Assignment of smart pointers result in multiple aliases of the same object.
//   The term alias is used to differentiate from conventional pointers because
//   the semantics are different.
 
//   Aliases can be turned into copies if the pointed-to class supports copying.
 
//   The base class is smart_ptr_base which defines the common interface. Then
//   there are three subclasses which have the same interface but different copy
//   semantics:
 
//   - smart_ptr        for simple types and classes which have copy constructors
//   - smart_ptr_clone  for polymorphic class hierarchies which are copied using a clone method
//   - smart_ptr_nocopy for any class that cannot or should not be copied
 
////////////////////////////////////////////////////////////////////////////////
#include "containers_fixes.hpp"
#include "exceptions.hpp"
#include "copy_functors.hpp"
#include <map>
#include <string>
 
namespace stlplus
{
 
  ////////////////////////////////////////////////////////////////////////////////
  // internals
 
  template<typename T> class smart_ptr_holder;
 
  ////////////////////////////////////////////////////////////////////////////////
  // Base class
  ////////////////////////////////////////////////////////////////////////////////
 
  template<typename T, typename C>
  class smart_ptr_base
  {
  public:
    //////////////////////////////////////////////////////////////////////////////
    // member type definitions
 
    typedef T value_type;
    typedef T& reference;
    typedef const T& const_reference;
    typedef C value_copy;
 
    //////////////////////////////////////////////////////////////////////////////
    // constructors and destructors
 
    // create a null pointer
    smart_ptr_base(void);
 
    // create a pointer containing a dynamically created object
    // Note: the object must be allocated *by the user* with new
    // constructor form - must be called in the form smart_ptr_base<type> x(new type(args))
    explicit smart_ptr_base(T* data);
 
    // copy constructor implements aliasing so no copy is made
    // note that the copy constructor should NOT be explicit, as this breaks
    // the returning of pointer objects from functions (at least within GCC 4.4)
    smart_ptr_base(const smart_ptr_base<T,C>& r);
 
    // assignment operator - required, else the output of GCC suffers segmentation faults
    smart_ptr_base<T,C>& operator=(const smart_ptr_base<T,C>& r);
 
    // destructor decrements the reference count and delete only when the last reference is destroyed
    ~smart_ptr_base(void);
 
    //////////////////////////////////////////////////////////////////////////////
    // logical tests to see if there is anything contained in the pointer since it can be null
 
    // there are two forms:explicit and implicit
    // implicit: if(!r) or if(r)
    // explicit: if(r.null()) or if(r.present())
    operator bool(void) const;
    bool operator!(void) const;
    bool present(void) const;
    bool null(void) const;
 
    //////////////////////////////////////////////////////////////////////////////
    // dereference operators and functions
 
    // dereference the smart pointer to get the object - use in the form *p1
    // exceptions: null_dereference
    T& operator*(void) ;
    // exceptions: null_dereference
    const T& operator*(void) const ;
 
    // used as a prefix to a member access to the contained object e.g. p1->print() calls T::print()
    // exceptions: null_dereference
    T* operator->(void) ;
    // exceptions: null_dereference
    const T* operator->(void) const ;
 
    //////////////////////////////////////////////////////////////////////////////
    // explicit function forms of the above assignment and dereference operators
 
    // get the value
    // exceptions: null_dereference
    T& value(void) ;
    // exceptions: null_dereference
    const T& value(void) const ;
 
    // set the pointer
    // deletes the previous pointer and adopts the passed pointer instead
    // Note: the object must be allocated *by the user* with new
    // Warning: it is very easy to break the memory management with this operation
    void set(T* data = 0);
    // get the pointer
    T* pointer(void);
    const T* pointer(void) const;
 
    //////////////////////////////////////////////////////////////////////////////
    // functions to manage aliases
 
    // make this an alias of the passed object
    void alias(const smart_ptr_base<T,C>&);
 
    // test whether two pointers point to the same object(known as aliasing the object)
    // used in the form if(a.aliases(b))
    bool aliases(const smart_ptr_base<T,C>&) const;
 
    // find the number of aliases - used when you need to know whether an
    // object is still referred to from elsewhere (rare!)
    unsigned alias_count(void) const;
 
    // delete the object and make the pointer null - does not make it unique
    // first, so all other pointers to this will be null too
    void clear(void);
 
    // make the pointer unique and null in one step - does not affect other
    // pointers that were pointing to the same object
    void clear_unique(void);
 
    //////////////////////////////////////////////////////////////////////////////
    // functions that involve copying
 
    // these functions use the copy functor passed as the template parameter C
    // to copy the object with the right copy semantics. If the copy functor
    // is no_copy, an exception will be thrown.
 
    // create a pointer containing a *copy* of the object using the template parameter C
    // this copy is taken because the pointer class maintains a dynamically allocated object
    // and the T& may not be (usually is not) dynamically allocated
    // exceptions: illegal_copy
    explicit smart_ptr_base(const T& data) ;
 
    // set the value - note that this does a copy using the C template parameter
    // exceptions: illegal_copy
    void set_value(const T& data) ;
 
    // make this pointer unique with respect to any other references to the same object
    // if this pointer is already unique, it does nothing - otherwise it copies the object
    // exceptions: illegal_copy
    void make_unique(void) ;
 
    // make this pointer a unique copy of the parameter
    // useful for expressions like p1.copy(p2) which makes p1 a pointer to a unique copy of the contents of p2
    // exceptions: illegal_copy
    void copy(const smart_ptr_base<T,C>&) ;
 
  protected:
    smart_ptr_holder<T>* m_holder;
 
  public:
    // internal use only - had to make them public because they need to be
    // accessed by routines that could not be made friends
    smart_ptr_holder<T>* _handle(void) const;
    void _make_alias(smart_ptr_holder<T>* handle);
  };
 
  ////////////////////////////////////////////////////////////////////////////////
  // smart_ptr        for simple types and classes which have copy constructors
 
  template <typename T>
  class smart_ptr : public smart_ptr_base<T, constructor_copy<T> >
  {
  public:
    smart_ptr(void) {}
    explicit smart_ptr(const T& data) : smart_ptr_base<T, constructor_copy<T> >(data) {}
    explicit smart_ptr(T* data) : smart_ptr_base<T, constructor_copy<T> >(data) {}
    smart_ptr<T>& operator=(const T& data) {this->set_value(data); return *this;}
    smart_ptr<T>& operator=(T* data) {this->set(data); return *this;}
    ~smart_ptr(void) {}
  };
 
  ////////////////////////////////////////////////////////////////////////////////
  // smart_ptr_clone  for polymorphic class hierarchies which have a clone method
 
  template <typename T>
  class smart_ptr_clone : public smart_ptr_base<T, clone_copy<T> >
  {
  public:
    smart_ptr_clone(void) {}
    explicit smart_ptr_clone(const T& data) : smart_ptr_base<T, clone_copy<T> >(data) {}
    explicit smart_ptr_clone(T* data) : smart_ptr_base<T, clone_copy<T> >(data) {}
    smart_ptr_clone<T>& operator=(const T& data) {this->set_value(data); return *this;}
    smart_ptr_clone<T>& operator=(T* data) {this->set(data); return *this;}
    ~smart_ptr_clone(void) {}
  };
 
  ////////////////////////////////////////////////////////////////////////////////
  // smart_ptr_nocopy for any class that cannot or should not be copied
 
  template <typename T>
  class smart_ptr_nocopy : public smart_ptr_base<T, no_copy<T> >
  {
  public:
    smart_ptr_nocopy(void) {}
    explicit smart_ptr_nocopy(T* data) : smart_ptr_base<T, no_copy<T> >(data) {}
    smart_ptr_nocopy<T>& operator=(T* data) {this->set(data); return *this;}
    ~smart_ptr_nocopy(void) {}
  };
 
  ////////////////////////////////////////////////////////////////////////////////
 
} // end namespace stlplus
 
#include "smart_ptr.tpp"
#endif