• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #ifndef BASE_MEMORY_REF_COUNTED_H_
6 #define BASE_MEMORY_REF_COUNTED_H_
7 #pragma once
8 
9 #include "base/atomic_ref_count.h"
10 #include "base/base_api.h"
11 #include "base/threading/thread_collision_warner.h"
12 
13 namespace base {
14 
15 namespace subtle {
16 
17 class BASE_API RefCountedBase {
18  public:
ImplementsThreadSafeReferenceCounting()19   static bool ImplementsThreadSafeReferenceCounting() { return false; }
20 
HasOneRef()21   bool HasOneRef() const { return ref_count_ == 1; }
22 
23  protected:
24   RefCountedBase();
25   ~RefCountedBase();
26 
27   void AddRef() const;
28 
29   // Returns true if the object should self-delete.
30   bool Release() const;
31 
32  private:
33   mutable int ref_count_;
34 #ifndef NDEBUG
35   mutable bool in_dtor_;
36 #endif
37 
38   DFAKE_MUTEX(add_release_);
39 
40   DISALLOW_COPY_AND_ASSIGN(RefCountedBase);
41 };
42 
43 class BASE_API RefCountedThreadSafeBase {
44  public:
ImplementsThreadSafeReferenceCounting()45   static bool ImplementsThreadSafeReferenceCounting() { return true; }
46 
47   bool HasOneRef() const;
48 
49  protected:
50   RefCountedThreadSafeBase();
51   ~RefCountedThreadSafeBase();
52 
53   void AddRef() const;
54 
55   // Returns true if the object should self-delete.
56   bool Release() const;
57 
58  private:
59   mutable AtomicRefCount ref_count_;
60 #ifndef NDEBUG
61   mutable bool in_dtor_;
62 #endif
63 
64   DISALLOW_COPY_AND_ASSIGN(RefCountedThreadSafeBase);
65 };
66 
67 }  // namespace subtle
68 
69 //
70 // A base class for reference counted classes.  Otherwise, known as a cheap
71 // knock-off of WebKit's RefCounted<T> class.  To use this guy just extend your
72 // class from it like so:
73 //
74 //   class MyFoo : public base::RefCounted<MyFoo> {
75 //    ...
76 //    private:
77 //     friend class base::RefCounted<MyFoo>;
78 //     ~MyFoo();
79 //   };
80 //
81 // You should always make your destructor private, to avoid any code deleting
82 // the object accidently while there are references to it.
83 template <class T>
84 class RefCounted : public subtle::RefCountedBase {
85  public:
RefCounted()86   RefCounted() { }
~RefCounted()87   ~RefCounted() { }
88 
AddRef()89   void AddRef() const {
90     subtle::RefCountedBase::AddRef();
91   }
92 
Release()93   void Release() const {
94     if (subtle::RefCountedBase::Release()) {
95       delete static_cast<const T*>(this);
96     }
97   }
98 
99  private:
100   DISALLOW_COPY_AND_ASSIGN(RefCounted<T>);
101 };
102 
103 // Forward declaration.
104 template <class T, typename Traits> class RefCountedThreadSafe;
105 
106 // Default traits for RefCountedThreadSafe<T>.  Deletes the object when its ref
107 // count reaches 0.  Overload to delete it on a different thread etc.
108 template<typename T>
109 struct DefaultRefCountedThreadSafeTraits {
DestructDefaultRefCountedThreadSafeTraits110   static void Destruct(const T* x) {
111     // Delete through RefCountedThreadSafe to make child classes only need to be
112     // friend with RefCountedThreadSafe instead of this struct, which is an
113     // implementation detail.
114     RefCountedThreadSafe<T,
115                          DefaultRefCountedThreadSafeTraits>::DeleteInternal(x);
116   }
117 };
118 
119 //
120 // A thread-safe variant of RefCounted<T>
121 //
122 //   class MyFoo : public base::RefCountedThreadSafe<MyFoo> {
123 //    ...
124 //   };
125 //
126 // If you're using the default trait, then you should add compile time
127 // asserts that no one else is deleting your object.  i.e.
128 //    private:
129 //     friend class base::RefCountedThreadSafe<MyFoo>;
130 //     ~MyFoo();
131 template <class T, typename Traits = DefaultRefCountedThreadSafeTraits<T> >
132 class RefCountedThreadSafe : public subtle::RefCountedThreadSafeBase {
133  public:
RefCountedThreadSafe()134   RefCountedThreadSafe() { }
~RefCountedThreadSafe()135   ~RefCountedThreadSafe() { }
136 
AddRef()137   void AddRef() const {
138     subtle::RefCountedThreadSafeBase::AddRef();
139   }
140 
Release()141   void Release() const {
142     if (subtle::RefCountedThreadSafeBase::Release()) {
143       Traits::Destruct(static_cast<const T*>(this));
144     }
145   }
146 
147  private:
148   friend struct DefaultRefCountedThreadSafeTraits<T>;
149   static void DeleteInternal(const T* x) { delete x; }
150 
151   DISALLOW_COPY_AND_ASSIGN(RefCountedThreadSafe);
152 };
153 
154 //
155 // A wrapper for some piece of data so we can place other things in
156 // scoped_refptrs<>.
157 //
158 template<typename T>
159 class RefCountedData : public base::RefCounted< base::RefCountedData<T> > {
160  public:
161   RefCountedData() : data() {}
162   RefCountedData(const T& in_value) : data(in_value) {}
163 
164   T data;
165 };
166 
167 }  // namespace base
168 
169 //
170 // A smart pointer class for reference counted objects.  Use this class instead
171 // of calling AddRef and Release manually on a reference counted object to
172 // avoid common memory leaks caused by forgetting to Release an object
173 // reference.  Sample usage:
174 //
175 //   class MyFoo : public RefCounted<MyFoo> {
176 //    ...
177 //   };
178 //
179 //   void some_function() {
180 //     scoped_refptr<MyFoo> foo = new MyFoo();
181 //     foo->Method(param);
182 //     // |foo| is released when this function returns
183 //   }
184 //
185 //   void some_other_function() {
186 //     scoped_refptr<MyFoo> foo = new MyFoo();
187 //     ...
188 //     foo = NULL;  // explicitly releases |foo|
189 //     ...
190 //     if (foo)
191 //       foo->Method(param);
192 //   }
193 //
194 // The above examples show how scoped_refptr<T> acts like a pointer to T.
195 // Given two scoped_refptr<T> classes, it is also possible to exchange
196 // references between the two objects, like so:
197 //
198 //   {
199 //     scoped_refptr<MyFoo> a = new MyFoo();
200 //     scoped_refptr<MyFoo> b;
201 //
202 //     b.swap(a);
203 //     // now, |b| references the MyFoo object, and |a| references NULL.
204 //   }
205 //
206 // To make both |a| and |b| in the above example reference the same MyFoo
207 // object, simply use the assignment operator:
208 //
209 //   {
210 //     scoped_refptr<MyFoo> a = new MyFoo();
211 //     scoped_refptr<MyFoo> b;
212 //
213 //     b = a;
214 //     // now, |a| and |b| each own a reference to the same MyFoo object.
215 //   }
216 //
217 template <class T>
218 class scoped_refptr {
219  public:
220   scoped_refptr() : ptr_(NULL) {
221   }
222 
223   scoped_refptr(T* p) : ptr_(p) {
224     if (ptr_)
225       ptr_->AddRef();
226   }
227 
228   scoped_refptr(const scoped_refptr<T>& r) : ptr_(r.ptr_) {
229     if (ptr_)
230       ptr_->AddRef();
231   }
232 
233   template <typename U>
234   scoped_refptr(const scoped_refptr<U>& r) : ptr_(r.get()) {
235     if (ptr_)
236       ptr_->AddRef();
237   }
238 
239   ~scoped_refptr() {
240     if (ptr_)
241       ptr_->Release();
242   }
243 
244   T* get() const { return ptr_; }
245   operator T*() const { return ptr_; }
246   T* operator->() const { return ptr_; }
247 
248   // Release a pointer.
249   // The return value is the current pointer held by this object.
250   // If this object holds a NULL pointer, the return value is NULL.
251   // After this operation, this object will hold a NULL pointer,
252   // and will not own the object any more.
253   T* release() {
254     T* retVal = ptr_;
255     ptr_ = NULL;
256     return retVal;
257   }
258 
259   scoped_refptr<T>& operator=(T* p) {
260     // AddRef first so that self assignment should work
261     if (p)
262       p->AddRef();
263     T* old_ptr = ptr_;
264     ptr_ = p;
265     if (old_ptr)
266       old_ptr ->Release();
267     return *this;
268   }
269 
270   scoped_refptr<T>& operator=(const scoped_refptr<T>& r) {
271     return *this = r.ptr_;
272   }
273 
274   template <typename U>
275   scoped_refptr<T>& operator=(const scoped_refptr<U>& r) {
276     return *this = r.get();
277   }
278 
279   void swap(T** pp) {
280     T* p = ptr_;
281     ptr_ = *pp;
282     *pp = p;
283   }
284 
285   void swap(scoped_refptr<T>& r) {
286     swap(&r.ptr_);
287   }
288 
289  protected:
290   T* ptr_;
291 };
292 
293 // Handy utility for creating a scoped_refptr<T> out of a T* explicitly without
294 // having to retype all the template arguments
295 template <typename T>
296 scoped_refptr<T> make_scoped_refptr(T* t) {
297   return scoped_refptr<T>(t);
298 }
299 
300 #endif  // BASE_MEMORY_REF_COUNTED_H_
301