gem5  v21.0.0.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
refcnt.hh
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2017-2018 ARM Limited
3  * All rights reserved.
4  *
5  * The license below extends only to copyright in the software and shall
6  * not be construed as granting a license to any other intellectual
7  * property including but not limited to intellectual property relating
8  * to a hardware implementation of the functionality of the software
9  * licensed hereunder. You may use the software subject to the license
10  * terms below provided that you ensure that this notice is replicated
11  * unmodified and in its entirety in all distributions of the software,
12  * modified or unmodified, in source code or in binary form.
13  *
14  * Copyright (c) 2002-2005 The Regents of The University of Michigan
15  * All rights reserved.
16  *
17  * Redistribution and use in source and binary forms, with or without
18  * modification, are permitted provided that the following conditions are
19  * met: redistributions of source code must retain the above copyright
20  * notice, this list of conditions and the following disclaimer;
21  * redistributions in binary form must reproduce the above copyright
22  * notice, this list of conditions and the following disclaimer in the
23  * documentation and/or other materials provided with the distribution;
24  * neither the name of the copyright holders nor the names of its
25  * contributors may be used to endorse or promote products derived from
26  * this software without specific prior written permission.
27  *
28  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
29  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
30  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
31  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
32  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
33  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
34  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
35  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
36  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
37  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
38  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
39  */
40 
41 #ifndef __BASE_REFCNT_HH__
42 #define __BASE_REFCNT_HH__
43 
44 #include <type_traits>
45 
58 {
59  private:
60  // The reference count is mutable because one may want to
61  // reference count a const pointer. This really is OK because
62  // const is about logical constness of the object not really about
63  // strictly disallowing an object to change.
64  mutable int count;
65 
66  private:
67  // Don't allow a default copy constructor or copy operator on
68  // these objects because the default operation will copy the
69  // reference count as well and we certainly don't want that.
70  RefCounted(const RefCounted &);
72 
73  public:
81  RefCounted() : count(0) {}
82 
92  virtual ~RefCounted() {}
93 
95  void incref() const { ++count; }
96 
99  void
100  decref() const
101  {
102  if (--count <= 0)
103  delete this;
104  }
105 };
106 
122 template <class T>
124 {
125  public:
126  using PtrType = T*;
127 
128  protected:
131  static constexpr auto TisConst = std::is_const<T>::value;
132  using ConstT = typename std::conditional<TisConst,
135  friend ConstT;
136  using NonConstT = typename std::conditional<TisConst,
139  friend NonConstT;
141  T *data;
144 
151  void
152  copy(T *d)
153  {
154  data = d;
155  if (data)
156  data->incref();
157  }
158 
164  void
165  del()
166  {
167  if (data)
168  data->decref();
169  }
170 
174  void
175  set(T *d)
176  {
177  // Need to check if we're actually changing because otherwise
178  // we could delete the last reference before adding the new
179  // reference.
180  if (data != d) {
181  del();
182  copy(d);
183  }
184  }
185 
186  public:
189 
193 
196  RefCountingPtr(const RefCountingPtr &r) { copy(r.data); }
197 
202  {
203  data = r.data;
204  r.data = nullptr;
205  }
206 
207  template <bool B = TisConst>
208  RefCountingPtr(const NonConstT &r) { copy(r.data); }
209 
212 
213  // The following pointer access functions are const because they
214  // don't actually change the pointer, though the user could change
215  // what is pointed to. This is analagous to a "Foo * const".
216 
218  T *operator->() const { return data; }
219 
221  T &operator*() const { return *data; }
222 
224  T *get() const { return data; }
225 
226  template <bool B = TisConst>
228  {
229  return RefCountingPtr<const T>(*this);
230  }
231 
233  const RefCountingPtr &operator=(T *p) { set(p); return *this; }
234 
236  const RefCountingPtr &
238  {
239  return operator=(r.data);
240  }
241 
243  const RefCountingPtr &
245  {
246  /* This happens regardless of whether the pointer is the same or not,
247  * because of the move semantics, the rvalue needs to be 'destroyed'.
248  */
249  del();
250  data = r.data;
251  r.data = nullptr;
252  return *this;
253  }
254 
256  bool operator!() const { return data == 0; }
257 
259  operator bool() const { return data != 0; }
260 };
261 
263 template<class T>
264 inline bool
266 {
267  return l.get() == r.get();
268 }
269 
272 template<class T>
273 inline bool
274 operator==(const RefCountingPtr<T> &l, const T *r)
275 {
276  return l.get() == r;
277 }
278 
281 template<class T>
282 inline bool
283 operator==(const T *l, const RefCountingPtr<T> &r)
284 {
285  return l == r.get();
286 }
287 
289 template<class T>
290 inline bool
292 {
293  return l.get() != r.get();
294 }
295 
298 template<class T>
299 inline bool
300 operator!=(const RefCountingPtr<T> &l, const T *r)
301 {
302  return l.get() != r;
303 }
304 
307 template<class T>
308 inline bool
309 operator!=(const T *l, const RefCountingPtr<T> &r)
310 {
311  return l != r.get();
312 }
313 
314 #endif // __BASE_REFCNT_HH__
RefCounted::RefCounted
RefCounted()
We initialize the reference count to zero and the first object to take ownership of it must increment...
Definition: refcnt.hh:81
RefCountingPtr::NonConstT
friend NonConstT
Definition: refcnt.hh:139
RefCountingPtr::RefCountingPtr
RefCountingPtr()
Create an empty reference counting pointer.
Definition: refcnt.hh:188
RefCountingPtr::operator=
const RefCountingPtr & operator=(RefCountingPtr &&r)
Move-assign the pointer from another RefCountingPtr.
Definition: refcnt.hh:244
RefCountingPtr::ConstT
friend ConstT
Definition: refcnt.hh:135
RefCounted::count
int count
Definition: refcnt.hh:64
RefCountingPtr< MinorDynInst >::ConstT
typename std::conditional< TisConst, RefCountingPtr< MinorDynInst >, RefCountingPtr< typename std::add_const< MinorDynInst >::type > >::type ConstT
Definition: refcnt.hh:134
RefCountingPtr::operator=
const RefCountingPtr & operator=(T *p)
Assign a new value to the pointer.
Definition: refcnt.hh:233
RefCountingPtr::RefCountingPtr
RefCountingPtr(T *data)
Create a new reference counting pointer to some object (probably something newly created).
Definition: refcnt.hh:192
RefCounted::operator=
RefCounted & operator=(const RefCounted &)
RefCountingPtr::set
void set(T *d)
Drop the old reference and change it to something new.
Definition: refcnt.hh:175
RefCountingPtr::RefCountingPtr
RefCountingPtr(const NonConstT &r)
Definition: refcnt.hh:208
RefCountingPtr< MinorDynInst >::PtrType
MinorDynInst * PtrType
Definition: refcnt.hh:126
RefCountingPtr::~RefCountingPtr
~RefCountingPtr()
Destroy the pointer and any reference it may hold.
Definition: refcnt.hh:211
ArmISA::d
Bitfield< 9 > d
Definition: miscregs_types.hh:60
MipsISA::r
r
Definition: pra_constants.hh:95
RefCounted::incref
void incref() const
Increment the reference count.
Definition: refcnt.hh:95
RefCounted::~RefCounted
virtual ~RefCounted()
We make the destructor virtual because we're likely to have virtual functions on reference counted ob...
Definition: refcnt.hh:92
RefCountingPtr::RefCountingPtr
RefCountingPtr(RefCountingPtr &&r)
Move-constructor.
Definition: refcnt.hh:201
RefCountingPtr::operator*
T & operator*() const
Dereference the pointer.
Definition: refcnt.hh:221
RefCountingPtr::operator=
const RefCountingPtr & operator=(const RefCountingPtr &r)
Copy the pointer from another RefCountingPtr.
Definition: refcnt.hh:237
RefCountingPtr::del
void del()
Delete the reference to any existing object if it is non NULL.
Definition: refcnt.hh:165
RefCounted::decref
void decref() const
Decrement the reference count and destroy the object if all references are gone.
Definition: refcnt.hh:100
RefCountingPtr::RefCountingPtr
RefCountingPtr(const RefCountingPtr &r)
Create a new reference counting pointer by copying another one.
Definition: refcnt.hh:196
RefCountingPtr< MinorDynInst >::NonConstT
typename std::conditional< TisConst, RefCountingPtr< typename std::remove_const< MinorDynInst >::type >, RefCountingPtr< MinorDynInst > >::type NonConstT
Definition: refcnt.hh:138
RefCountingPtr::operator->
T * operator->() const
Access a member variable.
Definition: refcnt.hh:218
RefCounted
Derive from RefCounted if you want to enable reference counting of this class.
Definition: refcnt.hh:57
RefCountingPtr::operator!
bool operator!() const
Check if the pointer is empty.
Definition: refcnt.hh:256
RefCountingPtr::copy
void copy(T *d)
Copy a new pointer value and increment the reference count if it is a valid pointer.
Definition: refcnt.hh:152
operator!=
bool operator!=(const RefCountingPtr< T > &l, const RefCountingPtr< T > &r)
Check for inequality of two reference counting pointers.
Definition: refcnt.hh:291
operator==
bool operator==(const RefCountingPtr< T > &l, const RefCountingPtr< T > &r)
Check for equality of two reference counting pointers.
Definition: refcnt.hh:265
X86ISA::type
type
Definition: misc.hh:727
RefCountingPtr::TisConst
static constexpr auto TisConst
Convenience aliases for const/non-const versions of T w/ friendship.
Definition: refcnt.hh:131
RefCountingPtr
If you want a reference counting pointer to a mutable object, create it like this:
Definition: refcnt.hh:123
MipsISA::p
Bitfield< 0 > p
Definition: pra_constants.hh:323
MipsISA::l
Bitfield< 5 > l
Definition: pra_constants.hh:320
RefCountingPtr::data
T * data
The stored pointer.
Definition: refcnt.hh:143
RefCountingPtr::get
T * get() const
Directly access the pointer itself without taking a reference.
Definition: refcnt.hh:224

Generated on Tue Mar 23 2021 19:41:24 for gem5 by doxygen 1.8.17