Rietveld Code Review Tool
Help | Bug tracker | Discussion group | Source code

Delta Between Two Patch Sets: compiled/intrusive_ptr.h

Issue 29333474: Issue 4125 - [emscripten] Convert filter classes to C++ (Closed)
Left Patch Set: Optimized hash lookup performance a bit Created Feb. 8, 2016, 7:11 p.m.
Right Patch Set: Addressed comments from Patch Set 28 Created March 21, 2017, 10:04 a.m.
Left:
Right:
Use n/p to move between diff chunks; N/P to move between comments.
Jump to:
Left: Side by side diff | Download
Right: Side by side diff | Download
« no previous file with change/comment | « compiled/debug.h ('k') | compiled/shell.js » ('j') | no next file with change/comment »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
LEFTRIGHT
1 // Parts of this code have been copied from boost/smart_ptr/intrusive_ptr.hpp. 1 // Parts of this code have been copied from boost/smart_ptr/intrusive_ptr.hpp.
2 // 2 //
3 // Copyright (c) 2001, 2002 Peter Dimov 3 // Copyright (c) 2001, 2002 Peter Dimov
4 // 4 //
5 // Distributed under the Boost Software License, Version 1.0. (See 5 // Distributed under the Boost Software License, Version 1.0. (See
6 // accompanying file LICENSE_1_0.txt or copy at 6 // accompanying file LICENSE_1_0.txt or copy at
7 // http://www.boost.org/LICENSE_1_0.txt) 7 // http://www.boost.org/LICENSE_1_0.txt)
8 8
9 #pragma once 9 #pragma once
sergei 2016/02/17 12:55:05 Yahoo! Finally we start it! I am looking forward t
10 10
11 #include <algorithm> 11 #include <algorithm>
12 #include <type_traits> 12 #include <type_traits>
13 13
14 #include "debug.h"
15
14 class ref_counted 16 class ref_counted
15 { 17 {
16 public: 18 public:
17 void AddRef() 19 void AddRef()
18 { 20 {
19 mRefCount++; 21 mRefCount++;
20 } 22 }
21 23
22 void ReleaseRef() 24 void ReleaseRef()
23 { 25 {
26 assert(mRefCount > 0, u"Unexpected zero or negative reference count"_str);
24 if (--mRefCount == 0) 27 if (--mRefCount == 0)
25 delete this; 28 delete this;
26 } 29 }
27 30
28 protected: 31 protected:
29 ref_counted() 32 ref_counted()
30 : mRefCount(0) 33 : mRefCount(1)
31 { 34 {
32 } 35 }
33 36
34 // We need this virtual destructor, otherwise pointers to ref_counted and
35 // pointers to derived classes won't have the same value (e.g. converting from
36 // Filter* to ref_counted* decreases pointer value by 4).
37 virtual ~ref_counted() 37 virtual ~ref_counted()
38 { 38 {
39 assert(mRefCount == 0, u"Destroying a ref-counted object with a non-zero ref erence count"_str);
39 } 40 }
40 41
41 private: 42 private:
42 int mRefCount; 43 int mRefCount;
43 }; 44 };
44 45
45 template<typename T, 46 template<typename T,
46 class = typename std::enable_if<std::is_base_of<ref_counted,T>::value>::type > 47 class = typename std::enable_if<std::is_base_of<ref_counted,T>::value>::type >
47 class intrusive_ptr 48 class intrusive_ptr
48 { 49 {
49 public: 50 public:
50 intrusive_ptr() 51 explicit intrusive_ptr()
51 : mPointer(nullptr) 52 : mPointer(nullptr)
52 { 53 {
53 } 54 }
54 55
55 intrusive_ptr(T* pointer) 56 explicit intrusive_ptr(T* pointer)
56 : mPointer(pointer) 57 : mPointer(pointer)
58 {
59 // Raw pointers always had their reference count increased by whatever gave
60 // us the pointer so we don't need to do it here.
61 }
62
63 intrusive_ptr(const intrusive_ptr& other)
64 : mPointer(other.mPointer)
57 { 65 {
58 if (mPointer) 66 if (mPointer)
59 mPointer->AddRef(); 67 mPointer->AddRef();
60 } 68 }
61 69
62 intrusive_ptr(const intrusive_ptr& other)
63 : mPointer(other.mPointer)
64 {
65 if (mPointer)
66 mPointer->AddRef();
67 }
68
69 intrusive_ptr(intrusive_ptr&& other) 70 intrusive_ptr(intrusive_ptr&& other)
70 : mPointer(other.mPointer) 71 : mPointer(other.mPointer)
71 { 72 {
72 other.mPointer = nullptr; 73 other.mPointer = nullptr;
73 } 74 }
74 75
75 ~intrusive_ptr() 76 ~intrusive_ptr()
76 { 77 {
77 if (mPointer) 78 if (mPointer)
78 mPointer->ReleaseRef(); 79 mPointer->ReleaseRef();
79 } 80 }
80 81
81 intrusive_ptr& operator=(intrusive_ptr& other) 82 intrusive_ptr& operator=(intrusive_ptr& other)
82 { 83 {
83 intrusive_ptr(other).swap(*this); 84 intrusive_ptr(other).swap(*this);
84 return *this; 85 return *this;
85 } 86 }
86 87
87 intrusive_ptr& operator=(intrusive_ptr&& other) 88 intrusive_ptr& operator=(intrusive_ptr&& other)
88 { 89 {
90 intrusive_ptr(std::move(other)).swap(*this);
91 return *this;
92 }
93
94 intrusive_ptr& operator=(T* other)
95 {
89 intrusive_ptr(other).swap(*this); 96 intrusive_ptr(other).swap(*this);
sergei 2016/02/17 12:55:07 JIC, `other` is just a reference here, I guess, to
Wladimir Palant 2016/02/18 16:07:14 I'll just trust you on this one, currently I have
90 return *this; 97 return *this;
91 } 98 }
92 99
93 intrusive_ptr& operator=(T* other) 100 void reset()
101 {
102 intrusive_ptr().swap(*this);
103 }
104
105 void reset(T* other)
94 { 106 {
95 intrusive_ptr(other).swap(*this); 107 intrusive_ptr(other).swap(*this);
96 return *this; 108 }
97 } 109
98 110 const T* get() const
99 void reset() 111 {
100 { 112 return mPointer;
101 intrusive_ptr().swap(*this); 113 }
102 } 114
103 115 T* get()
104 void reset(T* other) 116 {
105 { 117 return mPointer;
106 intrusive_ptr(other).swap(*this); 118 }
107 } 119
108 120 const T& operator*() const
109 T* get() const
sergei 2016/02/17 12:55:06 Although it's a common practice, I'm personally no
Wladimir Palant 2016/02/18 16:07:13 Done, here and for the methods below. None of this
110 {
111 return mPointer;
112 }
113
114 T& operator*() const
115 { 121 {
116 return *mPointer; 122 return *mPointer;
117 } 123 }
118 124
119 T* operator->() const 125 T& operator*()
120 { 126 {
121 return mPointer; 127 return *mPointer;
122 } 128 }
123 129
124 operator bool() const 130 const T* operator->() const
131 {
132 return mPointer;
133 }
134
135 T* operator->()
136 {
137 return mPointer;
138 }
139
140 explicit operator bool() const
125 { 141 {
126 return mPointer != nullptr; 142 return mPointer != nullptr;
127 } 143 }
128 144
129 operator T*() const
130 {
131 return mPointer;
132 }
133
134 bool operator!() const 145 bool operator!() const
135 { 146 {
136 return mPointer == nullptr; 147 return mPointer == nullptr;
148 }
149
150 T* release()
151 {
152 T* result = mPointer;
153 mPointer = nullptr;
154 return result;
137 } 155 }
138 156
139 void swap(intrusive_ptr& other) 157 void swap(intrusive_ptr& other)
140 { 158 {
141 std::swap(mPointer, other.mPointer); 159 std::swap(mPointer, other.mPointer);
142 } 160 }
143 161
144 private: 162 private:
145 T* mPointer; 163 T* mPointer;
146 }; 164 };
(...skipping 26 matching lines...) Expand all
173 inline bool operator==(const T* a, const intrusive_ptr<U>& b) 191 inline bool operator==(const T* a, const intrusive_ptr<U>& b)
174 { 192 {
175 return a == b.get(); 193 return a == b.get();
176 } 194 }
177 195
178 template<typename T, typename U> 196 template<typename T, typename U>
179 inline bool operator!=(const T* a, intrusive_ptr<U> const& b) 197 inline bool operator!=(const T* a, intrusive_ptr<U> const& b)
180 { 198 {
181 return a != b.get(); 199 return a != b.get();
182 } 200 }
LEFTRIGHT

Powered by Google App Engine
This is Rietveld