]> git.ipfire.org Git - thirdparty/squid.git/blob - src/base/RefCount.h
Implemented c++11 move operations for RefCount
[thirdparty/squid.git] / src / base / RefCount.h
1 /*
2 * Copyright (C) 1996-2015 The Squid Software Foundation and contributors
3 *
4 * Squid software is distributed under GPLv2+ license and includes
5 * contributions from numerous individuals and organizations.
6 * Please see the COPYING and CONTRIBUTORS files for details.
7 */
8
9 /* DEBUG: section -- Refcount allocator */
10
11 #ifndef SQUID_REFCOUNT_H_
12 #define SQUID_REFCOUNT_H_
13
14 // reference counting requires the Lock API on base classes
15 #include "base/Lock.h"
16
17 #include <iostream>
18
19 /**
20 * Template for Reference Counting pointers.
21 *
22 * Objects of type 'C' must inherit from 'RefCountable' in base/Lock.h
23 * which provides the locking interface used by reference counting.
24 */
25 template <class C>
26 class RefCount
27 {
28
29 public:
30 RefCount () : p_ (NULL) {}
31
32 RefCount (C const *p) : p_(p) { reference (*this); }
33
34 ~RefCount() {
35 dereference();
36 }
37
38 RefCount (const RefCount &p) : p_(p.p_) {
39 reference (p);
40 }
41
42 RefCount (RefCount &&p) : p_(std::move(p.p_)) {
43 p.p_=NULL;
44 }
45
46 RefCount& operator = (const RefCount& p) {
47 // DO NOT CHANGE THE ORDER HERE!!!
48 // This preserves semantics on self assignment
49 C const *newP_ = p.p_;
50 reference(p);
51 dereference(newP_);
52 return *this;
53 }
54
55 RefCount& operator = (RefCount&& p) {
56 p_ = std::move(p.p_);
57 p.p_ = NULL;
58 return *this;
59 }
60
61 bool operator !() const { return !p_; }
62
63 C * operator-> () const {return const_cast<C *>(p_); }
64
65 C & operator * () const {return *const_cast<C *>(p_); }
66
67 C const * getRaw() const {return p_; }
68
69 C * getRaw() {return const_cast<C *>(p_); }
70
71 bool operator == (const RefCount& p) const {
72 return p.p_ == p_;
73 }
74
75 bool operator != (const RefCount &p) const {
76 return p.p_ != p_;
77 }
78
79 private:
80 void dereference(C const *newP = NULL) {
81 /* Setting p_ first is important:
82 * we may be freed ourselves as a result of
83 * delete p_;
84 */
85 C const (*tempP_) (p_);
86 p_ = newP;
87
88 if (tempP_ && tempP_->unlock() == 0)
89 delete tempP_;
90 }
91
92 void reference (const RefCount& p) {
93 if (p.p_)
94 p.p_->lock();
95 }
96
97 C const *p_;
98
99 };
100
101 template <class C>
102 inline std::ostream &operator <<(std::ostream &os, const RefCount<C> &p)
103 {
104 if (p != NULL)
105 return os << p.getRaw() << '*' << p->LockCount();
106 else
107 return os << "NULL";
108 }
109
110 #endif /* SQUID_REFCOUNT_H_ */
111