blocxx
COWIntrusiveReference.hpp
Go to the documentation of this file.
1 /*******************************************************************************
2 * Copyright (C) 2005, Vintela, Inc. All rights reserved.
3 * Copyright (C) 2006, Novell, Inc. All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
7 *
8 * * Redistributions of source code must retain the above copyright notice,
9 * this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * * Neither the name of
14 * Vintela, Inc.,
15 * nor Novell, Inc.,
16 * nor the names of its contributors or employees may be used to
17 * endorse or promote products derived from this software without
18 * specific prior written permission.
19 *
20 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
21 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
24 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
25 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
26 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
27 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
28 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
29 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
30 * POSSIBILITY OF SUCH DAMAGE.
31 *******************************************************************************/
32 
33 
34 //
35 // Copyright (c) 2001, 2002 Peter Dimov
36 //
37 // Permission to copy, use, modify, sell and distribute this software
38 // is granted provided this copyright notice appears in all copies.
39 // This software is provided "as is" without express or implied
40 // warranty, and with no claim as to its suitability for any purpose.
41 //
42 
48 #ifndef BLOCXX_COW_INTRUSIVE_REFERENCE_HPP_INCLUDE_GUARD_
49 #define BLOCXX_COW_INTRUSIVE_REFERENCE_HPP_INCLUDE_GUARD_
50 
51 #include "blocxx/BLOCXX_config.h"
53 
54 namespace BLOCXX_NAMESPACE
55 {
56 
83 template<class T> class COWIntrusiveReference
84 {
85 private:
87 public:
88  typedef T element_type;
89 
95  {
96  }
104  COWIntrusiveReference(T * p, bool addRef = true): m_pObj(p)
105  {
106  if (m_pObj != 0 && addRef) COWIntrusiveReferenceAddRef(m_pObj);
107  }
118  template<class U> COWIntrusiveReference(COWIntrusiveReference<U> const & rhs): m_pObj(rhs.m_pObj)
119  {
121  }
131  {
133  }
140  {
142  }
143 
156  {
157  this_type(rhs).swap(*this);
158  return *this;
159  }
171  {
172  this_type(rhs).swap(*this);
173  return *this;
174  }
184  {
185  this_type(rhs).swap(*this);
186  return *this;
187  }
192  const T * getPtr() const
193  {
194  return m_pObj;
195  }
196 
201  const T & operator*() const
202  {
203 #ifdef BLOCXX_CHECK_NULL_REFERENCES
204  ReferenceHelpers::checkNull(this);
205  ReferenceHelpers::checkNull(m_pObj);
206 #endif
207  return *m_pObj;
208  }
209 
214  const T * operator->() const
215  {
216 #ifdef BLOCXX_CHECK_NULL_REFERENCES
217  ReferenceHelpers::checkNull(this);
218  ReferenceHelpers::checkNull(m_pObj);
219 #endif
220  return m_pObj;
221  }
222 
227  T & operator*()
228  {
229 #ifdef BLOCXX_CHECK_NULL_REFERENCES
230  ReferenceHelpers::checkNull(this);
231  ReferenceHelpers::checkNull(m_pObj);
232 #endif
233  getWriteLock();
234  return *m_pObj;
235  }
236 
242  {
243 #ifdef BLOCXX_CHECK_NULL_REFERENCES
244  ReferenceHelpers::checkNull(this);
245  ReferenceHelpers::checkNull(m_pObj);
246 #endif
247  getWriteLock();
248  return m_pObj;
249  }
250 
252  operator unspecified_bool_type () const
253  {
254  return m_pObj == 0? 0: &this_type::m_pObj;
255  }
256 
262  bool operator! () const
263  {
264  return m_pObj == 0;
265  }
266 
268  {
269  T * tmp = m_pObj;
270  m_pObj = rhs.m_pObj;
271  rhs.m_pObj = tmp;
272  }
273 
274 #if !defined(__GNUC__) || __GNUC__ > 2 // causes gcc 2.95 to ICE
275  /* This is so the templated constructor will work */
276  template <class U> friend class COWIntrusiveReference;
277 private:
278 #endif
279 
286  {
287  if ((m_pObj != 0) && !COWIntrusiveReferenceUnique(m_pObj))
288  {
290  }
291  }
292 
293 
294  T * m_pObj;
295 };
296 template<class T, class U> inline bool operator==(COWIntrusiveReference<T> const & a, COWIntrusiveReference<U> const & b)
297 {
298  return a.getPtr() == b.getPtr();
299 }
300 template<class T, class U> inline bool operator!=(COWIntrusiveReference<T> const & a, COWIntrusiveReference<U> const & b)
301 {
302  return a.getPtr() != b.getPtr();
303 }
304 template<class T> inline bool operator==(COWIntrusiveReference<T> const & a, const T * b)
305 {
306  return a.getPtr() == b;
307 }
308 template<class T> inline bool operator!=(COWIntrusiveReference<T> const & a, const T * b)
309 {
310  return a.getPtr() != b;
311 }
312 template<class T> inline bool operator==(const T * a, COWIntrusiveReference<T> const & b)
313 {
314  return a == b.getPtr();
315 }
316 template<class T> inline bool operator!=(const T * a, COWIntrusiveReference<T> const & b)
317 {
318  return a != b.getPtr();
319 }
320 #if __GNUC__ == 2 && __GNUC_MINOR__ <= 96
321 // Resolve the ambiguity between our op!= and the one in rel_ops
322 template<class T> inline bool operator!=(COWIntrusiveReference<T> const & a, COWIntrusiveReference<T> const & b)
323 {
324  return a.getPtr() != b.getPtr();
325 }
326 #endif
327 template<class T> inline bool operator<(COWIntrusiveReference<T> const & a, COWIntrusiveReference<T> const & b)
328 {
329  return a.getPtr() < b.getPtr();
330 }
331 template<class T> void swap(COWIntrusiveReference<T> & lhs, COWIntrusiveReference<T> & rhs)
332 {
333  lhs.swap(rhs);
334 }
335 
336 } // end namespace BLOCXX_NAMESPACE
337 
338 #endif
339